一. 什么是Lambda 5Jp@n .
所谓Lambda,简单的说就是快速的小函数生成。 /jC0[%~jV
在C++中,STL的很多算法都要求使用者提供一个函数对象。例如for_each函数,会要求用户提供一个表明“行为”的函数对象。以vector<bool>为例,如果想使用for_each对其中的各元素全部赋值为true,一般需要这么一个函数对象, R5X<8(4p
]Q-ON&/
#PVgx9T=_
IJD'0/R'c
class filler Axk
p
{ w)&] k#r
public : |D$U{5}Mv
void operator ()( bool & i) const {i = true ;} Sl:Qq!
} ; N1\u~%AT"
]8htJ]<|Q
C;oP"K]4=
这样实现不但麻烦,而且不直观。而如果使用lambda,则允许用户使用一种直观和见解的方式来处理这个问题。以boost.lambda为例,刚才的问题可以这么解决: )U>q><
+VdYT6{p
) Y\} ,O
NlU:e}zGR
for_each(v.begin(), v.end(), _1 = true ); 16ke CG\
J}i$ny_3OB
$T^O3 8$
那么下面,就让我们来实现一个lambda库。 8|d lt$
j08G-_Gjn
:V HJD
uB
6`e!Q
二. 战前分析 tJUMLn?
首先要说明的是,我并没有读过boost.lambda或其他任何lambda库的代码,因此如代码有雷同,纯属巧合。 U/&?rY^|
开始实现以前,首先要分析出大致的实现手法。先让我们来看几段使用Lambda的代码 $ZK4Ps -$
GTYGm
D(~6h,=m
for_each(v.begin(), v.end(), _1 = 1 ); |LcN_,}6
/* --------------------------------------------- */ 8/-GrdyE
vector < int *> vp( 10 ); \kzxt/Ow
transform(v.begin(), v.end(), vp.begin(), & _1); G( nT.\
/* --------------------------------------------- */ LdU, 32
sort(vp.begin(), vp.end(), * _1 > * _2); >
9JzYI^
/* --------------------------------------------- */ _Eq:Qbw#
int b = * find_if(v.begin, v.end(), _1 >= 3 && _1 < 5 ); \$VtwVQ,b
/* --------------------------------------------- */ |C=^:@}ri?
for_each(vp.begin(), vp.end(), cout << * _1 << ' \n ' ); X3!btxa%t
/* --------------------------------------------- */ bRLmJt98P
for_each(vp.begin(), vp.end(), cout << constant( ' \n ' ) << * _1); lR{eO~'~V
jzI\Q{[m'
~~;fWM '
GJy><'J,!>
看了之后,我们可以思考一些问题: 00%$?Fyk
1._1, _2是什么? 1#(,Bq4
显然_1和_2都满足C++对于标识符的要求,可见_1和_2都是对象。 >J 3N,f
2._1 = 1是在做什么? w]"Y1J(i
既然_1是一个对象,那么_1的类必然重载了operator=(int)。那么operator=返回什么呢?该函数所返回的对象被传入for_each的第3个参数,可见其返回了一个函数对象。现在整个流程就很清楚了。_1 = 1调用了operator=,其返回了一个函数对象,该函数对象能够将参数1赋值为1。 [LL"86D
Ok,回答了这两个问题之后,我们的思路就很清晰了。如果要实现operator=,那么至少要实现2个类,一个用于产生_1的对象,另一个用于代表operator=返回的函数对象。 zO9$fU
M_T$\z;,
"2'nLQ""q
三. 动工 [uc;M6o}?
首先实现一个能够范型的进行赋值的函数对象类: j
&,vju
'#4ya=Ww
Z&s+*&TM
;T"}dJel#
template < typename T > 6IPhy.8
class assignment ^KF
{ $*xnq%A
T value; Z#w1,n88
public : I =qd\
assignment( const T & v) : value(v) {} W5
fO1F
template < typename T2 > R|$=Pfg~4
T2 & operator ()(T2 & rhs) const { return rhs = value; } 6b-d#H/1Y
} ; Z:,HB]&;9
>P>.j+o/
q}ZZqYk
其中operator()被声明为模版函数以支持不同类型之间的赋值。 "o<:[c9/
然后我们就可以书写_1的类来返回assignment 9V.)=*0hp
k#JFDw\
I?4J69'
V F6OC4 K
class holder 7T_g?!sdMh
{ $Fc*^8$ryC
public : 42Gr0+Mb
template < typename T > qoB
assignment < T > operator = ( const T & t) const ^5 F-7R8Q
{ {KeHqM}e
return assignment < T > (t); EK@yzJ%
} #n{wK+lz
} ; <C+:hsS=
{8@?9Z9R{
.Z8 x!!Q*
由于该类是一个空类,因此我们可以在其后放心大胆的写上: udp&