(原創) 如何使for_each()傳回值? (C/C++) (STL)

本文介绍STL中for_each算法的高级用法,演示如何利用其返回值特性和函数对象保持状态,实现每打印n个元素换行并计算总和的功能。通过自定义函数对象,展示了C++中函数对象相较于全局函数的优势。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Abstract
for_each()是STL中少數可以回傳值的algorithm,此範例在展示for_each()此特殊功能與function object可以留住state的特性。

Introduction
function object與global function的差別除了function object可以傳入參數外,還可以不使用static就可以留住state。

一個簡單的需求,想要每印n個數字就換行,並且統計出所有iterator的和,所以function object必須能留住state才知道目前印了幾個數字,且統計sum為多少。

Sample Code

 1 ExpandedBlockStart.gif ContractedBlock.gif /**/ /* 
 2InBlock.gif(C) OOMusou 2007 http://oomusou.cnblogs.com
 3InBlock.gif
 4InBlock.gifFilename    : GenericAlgo_for_each_state_returnValue.cpp
 5InBlock.gifCompiler    : Visual C++ 8.0 / BCB 6.0 / gcc 3.4.2 / ISO C++
 6InBlock.gifDescription : Demo how to use for_each to return value and remain state.
 7InBlock.gifRelease     : 05/13/2007 1.0
 8InBlock.gif              05/15/2007 2.0
 9ExpandedBlockEnd.gif*/

10 None.gif
11 None.gif#include  < iostream >
12 None.gif#include  < algorithm >
13 None.gif#include  < vector >
14 None.gif
15 None.gif using   namespace  std;
16 None.gif
17 ExpandedBlockStart.gifContractedBlock.gif class  printElem  dot.gif {
18InBlock.gifprivate:
19InBlock.gif  int _n;
20InBlock.gif  int _cnt;
21InBlock.gif  int _sum;
22InBlock.gif  
23InBlock.gifpublic:
24ExpandedSubBlockStart.gifContractedSubBlock.gif  printElem(int n = 5) : _n(n), _cnt(0), _sum(0dot.gif{}
25InBlock.gif
26ExpandedSubBlockStart.gifContractedSubBlock.gif  void operator() (int elem) dot.gif{
27InBlock.gif    ++_cnt;
28InBlock.gif    _sum += elem;
29InBlock.gif    
30InBlock.gif    cout << elem;
31InBlock.gif    
32InBlock.gif    (_cnt % _n) ? cout << " " : cout << endl;
33ExpandedSubBlockEnd.gif  }

34InBlock.gif  
35ExpandedSubBlockStart.gifContractedSubBlock.gif  operator int() dot.gif{
36InBlock.gif    return _sum;
37ExpandedSubBlockEnd.gif  }

38ExpandedBlockEnd.gif}
;
39 None.gif
40 ExpandedBlockStart.gifContractedBlock.gif int  main()  dot.gif {
41InBlock.gif  vector<int> ivec;
42InBlock.gif  
43InBlock.gif  for(int i = 0; i != 20++i) 
44InBlock.gif    ivec.push_back(i);
45InBlock.gif    
46InBlock.gif  int sum = for_each(ivec.begin(), ivec.end(), printElem(5));
47InBlock.gif  
48InBlock.gif  cout << "sum is " << sum << endl;
49ExpandedBlockEnd.gif}



執行結果

None.gif 0   1   2   3   4
None.gif
5   6   7   8   9
None.gif
10   11   12   13   14
None.gif
15   16   17   18   19
None.gifsum is 
190


17行

None.gif private :
None.gif  
int  _n;
None.gif  
int  _cnt;
None.gif  
int  _sum;


_n為設定幾個字跳行
_cnt統計目前已經印了幾個字
_sum統計目前加總結果

29行

None.gif if  (_cnt  %  _n) 
None.gif  cout 
<<  elem  <<   "   " ;
None.gif
else
None.gif  cout 
<<  elem  <<  endl;


若每印n個字,就加印換行

35行

ExpandedBlockStart.gif ContractedBlock.gif operator   int ()  dot.gif {
InBlock.gif  
return _sum;
ExpandedBlockEnd.gif}


為了讓for_each()能傳回值,特別改寫operator int(),讓function object能夠傳回值。

46行

None.gif int  sum  =  for_each(ivec.begin(), ivec.end(), printElem( 5 ));


這樣for_each()就能風風光光的每n個字就換行,還可以順便加總結果。

Conclusion
STL真的很神奇,以上的程式想一行一行翻成C#還真的做不到呢!!

See Also
(原創) 如何正確的使用迴圈(使用for_each)? (C/C++) (STL) (template)

Reference
Nicolai M. Josuttis,The C++ Standard Library : A Tutorial and Referencd,Addison Wesley,1999

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值