欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C++11 算法std::copy_if 與 std::copy_n詳解

 更新時(shí)間:2025年07月11日 09:17:44   作者:碼事漫談  
C++11新增std::copy_if與std::copy_n,前者條件篩選復(fù)制,后者固定數(shù)量復(fù)制,提升代碼可讀性與性能,適用于數(shù)據(jù)過濾及批量處理,需注意范圍重疊與目標(biāo)空間不足等陷阱,本文給大家介紹C++11算法std::copy_if 與 std::copy_n詳解,感興趣的朋友一起看看吧

引言

C++11 標(biāo)準(zhǔn)為算法庫帶來了諸多增強(qiáng),其中 std::copy_ifstd::copy_n 作為 std::copy 的補(bǔ)充,為元素復(fù)制操作提供了更精細(xì)的控制。這兩個(gè)算法不僅簡(jiǎn)化了代碼邏輯,還提升了可讀性和性能。本文將深入探討這兩個(gè)算法的實(shí)現(xiàn)細(xì)節(jié)、使用場(chǎng)景及最佳實(shí)踐,幫助開發(fā)者在實(shí)際項(xiàng)目中正確高效地應(yīng)用它們。

std::copy_if:條件篩選復(fù)制

函數(shù)原型

template< class InputIt, class OutputIt, class UnaryPred >
OutputIt copy_if( InputIt first, InputIt last, OutputIt d_first, UnaryPred pred );

核心功能

std::copy_if 從輸入范圍 [first, last) 中復(fù)制滿足謂詞 pred 的元素到目標(biāo)范圍(始于 d_first),并保持元素的相對(duì)順序。該算法在 C++11 中引入,是對(duì)傳統(tǒng) std::copy 的條件化擴(kuò)展。

參數(shù)解析

  • first/last:輸入迭代器對(duì),定義源元素范圍。
  • d_first:輸出迭代器,指向目標(biāo)范圍的起始位置。
  • pred:一元謂詞函數(shù)(可調(diào)用對(duì)象),返回 bool 類型,用于判斷元素是否應(yīng)被復(fù)制。

注意:謂詞 pred 不得修改輸入元素,其參數(shù)類型通常為 const T&。

返回值

返回目標(biāo)范圍中最后一個(gè)被復(fù)制元素的下一個(gè)位置迭代器,便于后續(xù)操作(如繼續(xù)添加元素)。

實(shí)現(xiàn)邏輯

cppreference 提供的參考實(shí)現(xiàn)清晰展示了其工作原理:

template<class InputIt, class OutputIt, class UnaryPred>
OutputIt copy_if(InputIt first, InputIt last, OutputIt d_first, UnaryPred pred) {
    for (; first != last; ++first) {
        if (pred(*first)) {
            *d_first = *first;
            ++d_first;
        }
    }
    return d_first;
}

循環(huán)遍歷輸入范圍,對(duì)每個(gè)元素應(yīng)用謂詞判斷,滿足條件則復(fù)制到目標(biāo)位置并移動(dòng)目標(biāo)迭代器。

示例:篩選容器中的偶數(shù)

#include <algorithm>
#include <vector>
#include <iostream>
int main() {
    std::vector<int> src = {1, 2, 3, 4, 5, 6, 7, 8, 9};
    std::vector<int> dest;
    // 預(yù)留空間以避免多次擴(kuò)容(性能優(yōu)化)
    dest.reserve(src.size());
    // 復(fù)制所有偶數(shù)
    std::copy_if(src.begin(), src.end(), std::back_inserter(dest),
                 [](int x) { return x % 2 == 0; });
    // 輸出結(jié)果:2 4 6 8 
    for (int num : dest) {
        std::cout << num << " ";
    }
}

注意事項(xiàng)

  1. 范圍重疊:若目標(biāo)范圍與輸入范圍重疊,行為未定義。此時(shí)應(yīng)考慮 std::copy_backward
  2. 謂詞副作用:謂詞函數(shù)不得修改輸入元素,否則可能導(dǎo)致未定義行為。
  3. 性能考量:對(duì)于大型容器,提前調(diào)用 reserve 為目標(biāo)容器分配空間可避免多次內(nèi)存分配。

std::copy_n:固定數(shù)量復(fù)制

函數(shù)原型

template< class InputIt, class Size, class OutputIt >
OutputIt copy_n( InputIt first, Size count, OutputIt result );

核心功能

std::copy_n 從起始位置 first 復(fù)制恰好 count 個(gè)元素到目標(biāo)范圍(始于 result)。該算法同樣在 C++11 中引入,填補(bǔ)了傳統(tǒng) std::copy 無法指定復(fù)制數(shù)量的空白。

參數(shù)解析

  • first:輸入迭代器,指向源范圍的起始位置。
  • count:要復(fù)制的元素?cái)?shù)量(若為負(fù)數(shù),行為未定義)。
  • result:輸出迭代器,指向目標(biāo)范圍的起始位置。

返回值

返回目標(biāo)范圍中最后一個(gè)被復(fù)制元素的下一個(gè)位置迭代器(若 count 為 0,則返回 result)。

實(shí)現(xiàn)邏輯

參考實(shí)現(xiàn)如下:

template<class InputIt, class Size, class OutputIt>
constexpr OutputIt copy_n(InputIt first, Size count, OutputIt result) {
    if (count > 0) {
        *result = *first;
        ++result;
        for (Size i = 1; i != count; ++i, (void)++result) {
            *result = *++first;
        }
    }
    return result;
}

首先處理 count > 0 的情況,復(fù)制首個(gè)元素后循環(huán)復(fù)制剩余 count-1 個(gè)元素。

示例:復(fù)制前 N 個(gè)元素

#include <algorithm>
#include <vector>
#include <numeric>
#include <iostream>
int main() {
    std::vector<int> src(100);
    std::iota(src.begin(), src.end(), 1); // 填充 1~100
    std::vector<int> dest(5);
    // 復(fù)制前 5 個(gè)元素(1,2,3,4,5)
    std::copy_n(src.begin(), 5, dest.begin());
    // 輸出結(jié)果:1 2 3 4 5 
    for (int num : dest) {
        std::cout << num << " ";
    }
}

注意事項(xiàng)

  1. 目標(biāo)空間不足:若目標(biāo)容器容量小于 count,會(huì)導(dǎo)致緩沖區(qū)溢出(未定義行為)。
  2. 負(fù)數(shù) count:標(biāo)準(zhǔn)明確規(guī)定 count 為負(fù)數(shù)時(shí)行為未定義,實(shí)際使用中應(yīng)確保其非負(fù)。
  3. 迭代器類型:輸入迭代器只需滿足 LegacyInputIterator,但隨機(jī)訪問迭代器可提升性能(支持 first + i 直接訪問)。

對(duì)比分析與應(yīng)用場(chǎng)景

功能差異

特性std::copy_ifstd::copy_n
核心邏輯條件篩選復(fù)制固定數(shù)量復(fù)制
關(guān)鍵參數(shù)謂詞函數(shù) pred元素?cái)?shù)量 count
元素?cái)?shù)量取決于謂詞匹配結(jié)果嚴(yán)格等于 count(若源足夠)
順序保證保持源范圍中的相對(duì)順序按源范圍順序復(fù)制

性能對(duì)比

  • std::copy_if:需對(duì)每個(gè)元素執(zhí)行謂詞判斷,時(shí)間復(fù)雜度為 O(N)(N 為輸入范圍大?。?,但實(shí)際復(fù)制次數(shù)可能小于 N。
  • std::copy_n:時(shí)間復(fù)雜度為 O(count),無額外判斷開銷,適合已知復(fù)制數(shù)量的場(chǎng)景。

優(yōu)化提示:當(dāng)源迭代器為 LegacyContiguousIterator(如 std::vector::iterator)且元素類型為 TriviallyCopyable 時(shí),編譯器可能將 std::copy_n 優(yōu)化為 memmove,大幅提升性能。

典型應(yīng)用場(chǎng)景

std::copy_if 適用場(chǎng)景

  • 數(shù)據(jù)過濾:從容器中提取滿足特定條件的元素(如篩選日志中的錯(cuò)誤信息)。
  • 數(shù)據(jù)清洗:移除無效數(shù)據(jù)(如空字符串、負(fù)數(shù)等)。
  • 條件轉(zhuǎn)換:結(jié)合 std::back_inserter 動(dòng)態(tài)構(gòu)建新容器。

std::copy_n 適用場(chǎng)景

  • 批量數(shù)據(jù)處理:讀取固定大小的數(shù)據(jù)包(如網(wǎng)絡(luò)通信中的報(bào)文頭)。
  • 截?cái)?截取:獲取容器的前 N 個(gè)元素(如分頁顯示前 10 條記錄)。
  • 定長(zhǎng)緩沖區(qū)填充:向固定大小的數(shù)組中復(fù)制數(shù)據(jù)。

最佳實(shí)踐與常見陷阱

1. 避免目標(biāo)容器空間不足

問題:使用 std::copy_n 時(shí),若目標(biāo)容器大小小于 count,會(huì)導(dǎo)致未定義行為。
解決方案:提前確保目標(biāo)容器有足夠空間,或使用 std::back_inserter 自動(dòng)擴(kuò)容。

// 錯(cuò)誤示例:目標(biāo)容器大小不足
std::vector<int> dest(3);
std::copy_n(src.begin(), 5, dest.begin()); // 緩沖區(qū)溢出!
// 正確示例:使用 back_inserter
std::vector<int> dest;
std::copy_n(src.begin(), 5, std::back_inserter(dest)); // 自動(dòng)擴(kuò)容

2. 謂詞函數(shù)的設(shè)計(jì)

問題:謂詞函數(shù)修改輸入元素或有副作用。
解決方案:確保謂詞為純函數(shù),僅依賴輸入?yún)?shù)且無副作用。

// 錯(cuò)誤示例:謂詞修改輸入元素
std::copy_if(src.begin(), src.end(), dest.begin(),
             [](int& x) { return x++ > 5; }); // 修改了 x
// 正確示例:純函數(shù)謂詞
std::copy_if(src.begin(), src.end(), dest.begin(),
             [](int x) { return x > 5; }); // 僅讀取 x

3. 處理重疊范圍

問題:源范圍與目標(biāo)范圍重疊時(shí)使用 std::copy_ifstd::copy_n。
解決方案:若需復(fù)制到右側(cè)重疊區(qū)域,使用 std::copy_backward;若需條件復(fù)制,手動(dòng)實(shí)現(xiàn)安全邏輯。

4. 與其他算法的配合

結(jié)合 std::distancestd::copy_n 可實(shí)現(xiàn)動(dòng)態(tài)數(shù)量復(fù)制:

// 復(fù)制兩個(gè)迭代器之間的元素(等價(jià)于 std::copy)
auto n = std::distance(first, last);
std::copy_n(first, n, result);

總結(jié)

std::copy_ifstd::copy_n 作為 C++11 引入的算法,為元素復(fù)制提供了更靈活的選擇。前者擅長(zhǎng)條件篩選,后者專注固定數(shù)量復(fù)制,二者相輔相成,可大幅簡(jiǎn)化代碼并提升可讀性。實(shí)際使用中,需注意目標(biāo)容器空間、迭代器類型及范圍重疊等問題,結(jié)合具體場(chǎng)景選擇合適的算法。

現(xiàn)代 C++ 倡導(dǎo)使用標(biāo)準(zhǔn)算法而非手動(dòng)循環(huán),這不僅能減少錯(cuò)誤,還能讓代碼更具表達(dá)力。掌握這些算法的細(xì)節(jié),將有助于寫出更高效、更優(yōu)雅的 C++ 代碼。

參考資料

  1. cppreference.com - std::copy_if
  2. cppreference.com - std::copy_n
  3. ISO/IEC 14882:2011 (C++11 Standard), § 25.3.1]

到此這篇關(guān)于C++11 算法詳解:std::copy_if 與 std::copy_n的文章就介紹到這了,更多相關(guān)C++ std::copy_if 與 std::copy_n內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • C語言左旋轉(zhuǎn)字符串與翻轉(zhuǎn)字符串中單詞順序的方法

    C語言左旋轉(zhuǎn)字符串與翻轉(zhuǎn)字符串中單詞順序的方法

    這篇文章主要介紹了C語言左旋轉(zhuǎn)字符串與翻轉(zhuǎn)字符串中單詞順序的方法,給出了相關(guān)的兩道算法題目作為例子,需要的朋友可以參考下
    2016-02-02
  • 基于C語言中野指針的深入解析

    基于C語言中野指針的深入解析

    “野指針”不是NULL指針,是指向“垃圾”內(nèi)存的指針。人們一般不會(huì)錯(cuò)用NULL指針,因?yàn)橛胕f語句很容易判斷。但是“野指針”是很危險(xiǎn)的,if語句對(duì)它不起作用
    2013-07-07
  • C語言實(shí)現(xiàn)停車場(chǎng)項(xiàng)目

    C語言實(shí)現(xiàn)停車場(chǎng)項(xiàng)目

    這篇文章主要為大家詳細(xì)介紹了C語言實(shí)現(xiàn)停車場(chǎng)項(xiàng)目,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-03-03
  • C++淺析內(nèi)存分區(qū)模型概念與示例

    C++淺析內(nèi)存分區(qū)模型概念與示例

    在了解內(nèi)存分區(qū)之前,我們先來聊一聊為什么要進(jìn)行內(nèi)存分區(qū)。在進(jìn)行了內(nèi)存分區(qū)之后,在不同的區(qū)域存放的數(shù)據(jù),會(huì)有不同的生命周期,從而會(huì)讓程序員的編程變得更加靈活
    2022-09-09
  • C語言實(shí)現(xiàn)2048小游戲

    C語言實(shí)現(xiàn)2048小游戲

    這篇文章主要為大家詳細(xì)介紹了C語言實(shí)現(xiàn)2048小游戲,注釋清晰,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2018-05-05
  • C語言+EasyX實(shí)現(xiàn)數(shù)字雨效果

    C語言+EasyX實(shí)現(xiàn)數(shù)字雨效果

    這篇文章主要為大家詳細(xì)介紹了C語言+EasyX實(shí)現(xiàn)數(shù)字雨效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2018-11-11
  • C語言實(shí)現(xiàn)按行讀寫文件

    C語言實(shí)現(xiàn)按行讀寫文件

    這篇文章主要為大家詳細(xì)介紹了C語言實(shí)現(xiàn)按行讀寫文件,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2019-11-11
  • 從頭學(xué)習(xí)C語言之二維數(shù)組

    從頭學(xué)習(xí)C語言之二維數(shù)組

    這篇文章主要為大家詳細(xì)介紹了C語言之二維數(shù)組,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助
    2022-01-01
  • C語言實(shí)現(xiàn)可增容動(dòng)態(tài)通訊錄詳細(xì)過程

    C語言實(shí)現(xiàn)可增容動(dòng)態(tài)通訊錄詳細(xì)過程

    這篇文章主要為大家介紹了C語言實(shí)現(xiàn)簡(jiǎn)易通訊錄的完整流程,此通訊錄還可以增容,并且每個(gè)環(huán)節(jié)都有完整代碼,有需要的朋友可以借鑒參考下,希望能夠有所幫助
    2022-05-05
  • C/C++內(nèi)存管理基礎(chǔ)與面試

    C/C++內(nèi)存管理基礎(chǔ)與面試

    本章主要介紹C語言與C++的內(nèi)存管理,以C++的內(nèi)存分布作為引入,介紹C++不同于C語言的內(nèi)存管理方式(new?delete對(duì)比?malloc?free),感興趣的朋友來看看吧
    2022-07-07

最新評(píng)論