返回列表 发新帖

探索c#之函数创建和闭包

480.6k 9
swmozowtfl 发表于 2015-7-10 20:46:02|湖北 | 查看全部 阅读模式
阅读目录:
3 q+ X! F; w( y" t7 N动态创建函数
3 X# b: t4 D4 l9 W" {9 R3 L& L匿名函数不足之处
- h* s4 K* V/ l2 L) f& Y理解c#中的闭包: N% R2 F7 m& t) ?: ?! u$ G
闭包的优点
  o( w* f( ~/ i+ R5 |动态创建函数
$ v- U# ^0 n; m  b5 v大多数同学,都或多或少的使用过。回顾下c#中动态创建函数的进化:# L& V" C- P7 ~6 }9 X" s
c# 1.0中:
# b, K6 Z+ X; ~( ?( n0 }- Bpublic delegate string dynamicfunction(string name);
# O: m8 y0 Q, y5 r' g- Upublic static dynamicfunction getdynamicfunction()5 b$ F8 K: ?9 h' T, ^8 t% R
{; a- i! P$ n, x# {$ p9 E0 \
return getname;' Q& \2 Z+ r' X& j* w1 _
}! n$ x& H/ B5 i( T
static string getname(string name)
9 {2 v5 L: M2 Z' T/ i) D% R{
3 l0 {7 }. }) I0 d5 g/ lreturn name;
. `( O! C' _! h4 e}0 `0 W$ j4 N0 k- U+ H9 Q5 o
var result = getdynamicfunction()(mushroom);
; D6 `5 E* A9 A/ D% e# F3.0写惯了是不是看起来很繁琐、落后。 刚学委托时,都把委托理解成函数指针,也来看下用函数指针实现的:
0 t9 w4 ~' ^6 V) U( ^char getname(char p);
9 m2 t6 A+ W' l4 |/ Wtypedef char (*dynamicfunction)(char p);
, U4 k* ~% S; l" o$ |3 n: ?dynamicfunction getdynamicfunction()
: p$ h8 y5 q& f3 n  r{
! F+ j( W" {2 n/ p4 ~/ treturn getname;
, _5 J) P% @" _  I}( _4 @% o/ y: ?
char getname(char p)8 _" ]. c/ @- N. U% Y' `
{
& |1 [* W/ `  oreturn p;) E  o7 E1 G5 V" E" U, ?2 }; w& B
};7 N# M) A: ]+ i+ X/ g# k, \' |" n
char result = getdynamicfunction()('m');! I7 r2 Q2 Z- n/ K) `) l
对比起来和c# 1.0几乎一模一样了(引用/指针差别),毕竟是同一家族的。! t: B; ]! X1 g( r0 O8 q
c# 2.0中,增加匿名函数:  y8 ^- ?$ F8 \& P& F
public delegate string dynamicfunction(string name);
* F& Z7 s% n4 B0 s: Ldynamicfunction result2 = delegate(string name)7 _0 |: i" _; s; V" f+ |  a
{
; p1 d! K# Q, W) P5 P( R' `return name;( \' l2 T( `! {4 D
};; Q( y5 c) W. D# ^) i* ?& R
c# 3.0中,增加lambda表达式,华丽的转身:
( O$ H) X5 O$ ?6 g& Dpublic static func<string, string> getdynamicfunction()
/ ~# W  P0 b7 A: b& W/ O{4 n$ Z% p9 `, B. ~8 ?, J9 r$ b
return name => name;3 \5 w# h7 d+ n' m/ V8 }
}
, T3 y; n6 B2 ]- |. yvar result = getdynamicfunction()(mushroom);
; {+ _5 I, @0 Z" p1 f* L' s匿名函数不足之处( l( C+ B) F' a! x/ h4 I$ A. M9 m
虽然增加lambda表达式,已经极大简化了我们的工作量。但确实有些不足之处:
9 S, t6 o( r% d% Tvar result = name => name;2 C: Y8 x; _" k2 k
这些写编译时是报错的。因为c#本身强类型语言的,提供var语法糖只是为了省去声明确定类型的工作量。 编译器在编译时必须能够完全推断出各参数的类型才行。代码中的name参数类
' j5 H/ W, d" x2 E型,显然在编译时无法推断出来的。
( B( H& G2 Z, [var result = (string name) => name;
* c: |4 y4 S) B( ]* {func<string, string> result2 = (string name) => name;' S4 w% ?6 x4 G" }) J
expression<func<string, string>> result3 = (string name) => name;
6 h0 y# W( s8 r2 E  i% H7 V9 s9 E5 v上面直接声明name类型呢,很遗憾这样也是报错的。代码中已经给出答案了,编译器推断不出右边表达式是属于func<string, string>类型还是expression<func<string, string>>类型
/ ~* e5 h0 u, x! x% B! s! i/ V- W- L  @2 C
dynamic result = name => name;; V( h$ }4 m# s4 h
dynamic result1 = (func<string,string>)(name => name);
2 g% ]. V# l; Z8 g8 G7 U+ s3 G用dynamic呢,同样编译器也分不出右边是个委托,我们显示转换下就可以了。/ O* a, `& U* R: N3 i6 @
func<string, string> function = name => name;4 h. [3 u1 n( w
dynamicfunction df = function;
& q- E# J! T0 g& i$ S% R这里定义个func委托,虽然参数和返回值类型都和dynamicfunction委托一样,但编译时还是会报错:不能隐式转换func<string, string>到dynamicfunction,2个类型是不兼容的。4 z- R/ w% J* i4 D0 e" \1 I
理解c#中的闭包7 e% X/ y( U8 [! D6 [3 o
谈论到动态创建函数,都要牵扯到闭包。闭包这个概念资料很多了,理论部分这里就不重复了。 来看看c#代码中闭包:* l2 V6 i7 Z. O$ g/ x" e" X6 C
func<func<int>> a = () =>* \7 l' a( K# s& D( m- V' @
{+ z' N& b# S0 @
var age = 18;
1 p3 Z3 b, Q* Hreturn () =>  //b函数% ?6 E2 ]" R& m
{6 C3 R& ]: ^9 ]5 j$ b
return age;
4 y7 q9 s+ Z3 K0 U+ w( R};+ P! d9 _1 f. z6 A0 x3 S
};5 b/ [* [) D: {9 C
var result = a()();
4 m2 B$ Y* d/ X4 r! F" u, [5 X上面就是闭包,可理解为就是: 跨作用域访问函数内变量,也有说带着数据的行为。$ G! Q% {1 v; K- |, N* q: D
c#变量作用域一共有三种,即:类变量,实例变量,函数内变量。子作用域访问父作用域的变量(即函数内访问实例/类变量)在我们看来理所当然的,也符合我们一直的编程习惯。
) l6 c5 s8 h" X0 Y, ?例子中匿名函数b是可以访问上层函数a的变量age。对于编译器而言,a函数是b函数的父作用域,所以b函数访问父作用域的age变量是符合规范的。" L4 ?" o2 e# A/ `/ m) h" v
int age = 16;
2 `- m$ [. C6 P$ \4 Y1 h9 _6 q1 z1 wvoid display()
2 s, p* T& D0 R* L: J{' e4 u) U+ ]; f( l$ [' i" a
console.writeline(age);5 H: m  D3 u5 n8 ?# [! T+ H+ N- Q
int age = 18;# I" ?8 W  }* n. M6 r0 J5 R" [/ f
console.writeline(age);
1 R( x  K( |& f" J) f4 y}
7 _  ^- _, _& A& {; j上面编译会报错未声明使用,编译器检查到函数内声明age后,作用域就会覆盖父作用域的age,(像js就undefined了)。7 x$ K% E! J( q- o
func<int> c = () =>0 l5 Z5 b4 i' w' y* c
{
3 M- G; V+ m2 C1 H& q& K; @var age = 19;2 d9 L2 B; w- Z, A; c# @
return age;- p: f  N" w5 P; M
};8 Y! b7 G/ Q) l' [6 Y
上面声明个同级函数c,那么a函数是无法访c函数中的age变量的。 简单来说就是不可跨作用域访问其他函数内的变量。 那编译器是怎么实现闭包机制的呢?4 U/ ^8 g' |! I" [
如上图,答案是升级作用域,把a函数升级为一个实例类作用域。 在编译代码期间,编译器检查到b函数使用a函数内变量时,会自动生成一个匿名类x,把原a函数内变量age提升为x类的
4 e5 V+ K/ B. c6 @6 g; H% M0 B字段(即实例变量),a函数提升为匿名类x的实例函数。下面是编译器生成的代码(精简过):6 h6 F. O$ _/ `: D" k3 c4 L* G
class program1
2 T$ Y8 [( p1 C1 ?  _{: x. w  e8 V; s2 q6 y
static func<func<int>> cachedanonymousmethoddelegate2;0 x& Z2 J! d2 h8 }8 u7 l
static void main(string[] args), G* x* l4 o8 |2 I! \: U
{
6 w; q/ c" k6 k4 v4 }/ ]2 T! l, H) G) Ifunc<func<int>> func = new func<func<int>>(program1.b);8 X0 p( p1 i- x, ?
int num = func()();2 _' X. u# M5 D. ]. q
}
9 S. S; \, m. X$ y" ?static func<int> b()
( r: P. D3 s) h, \/ E{
3 Z- r: o9 p$ w% Z1 xdisplayclass cl = new displayclass();
$ B& X  a( p# A9 {cl.age = 18;6 p% t4 g; y  ^! p/ i5 H8 [
return new func<int>(cl.a);
& d" w9 _, g8 N+ @4 Y: r}/ s) L8 H/ D. t% B4 L7 k8 L, v
}
3 b; C7 z  j5 E* j) X& h7 V8 qsealed class displayclass
8 v9 b+ J" j7 J& ^# e: h# g{% v7 \$ v1 \4 k; w% Y1 ]+ w
public int age;
' O% f  r8 w5 ?/ K( H, S3 bpublic int a()0 r  Z. d! \/ Q9 z1 [( H3 f/ R
{
' q" w% }+ D6 m! ^+ S3 V0 C" m2 Creturn this.age;# D+ ~4 Q: ^+ h3 y
}. d0 f! x8 D2 e2 u4 J; }, b' E3 Q
}
7 w2 I9 W# x9 V3 [, N我们再来看个复杂点的例子:
4 q+ `' j/ v4 Z8 Vstatic func<int, int> getclosurefunction()/ t( f7 v) Y' M' J1 W
{" i$ H& q4 a) Y2 C: Y
int val = 10;
6 x2 ]( c: x$ [* I/ b7 h& [4 ^0 yfunc<int, int> interadd = x => x + val;
' V% L4 R2 ^3 d6 h8 mconsole.writeline(interadd(10));
$ _4 {2 C8 r, E& \val = 30;
, ?' z1 }/ j( h/ kconsole.writeline(interadd(10));
/ t$ F& d$ y8 s7 o8 j' J! r  \return interadd;
1 t  ~! d, \/ R* U}: R) a% O+ m9 o6 S
console.writeline(getclosurefunction()(30));$ G7 j9 Z- v7 ]4 B* ^
输出结果是20、40、60。 当看到这个函数内变量val通过闭包被传递的时候,我们就知道val不仅仅是个函数内变量了。之前我们分析过编译器怎么生成的代码,知道val此时是一个匿名7 p5 X* Z+ F( ^8 b& T
类的实例变量,interadd是匿名类的实例函数。所以无论val传递多少层,它的值始终保持着,直到离开这个(链式)作用域。
7 n8 n; V. r5 V关于闭包,在js当中谈论的比较多,同理,可以对比理解下:6 z: g. T, {9 Y9 W, p
function a() {4 l. k& u- j, R5 h' N
var age = 18;; E, K8 Y# n; t6 G) Z' t
return function () {) s7 v/ |! j  i
return age;
7 v$ F+ M! W( Q. L( H6 F}
1 @0 k* f1 u1 P( A6 U}
0 ]% q7 W( b5 F+ K, m0 M6 F) J4 j6 Ja()();
6 m* k8 [" s1 T7 W, c; y闭包的优点: |4 j7 x" E* H  r
对变量的保护。想暴露一个变量值,但又怕声明类或实例变量会被其他函数污染,这时就可以设计个闭包,只能通过函数调用来使用它。
; [  J( O% I/ }逻辑连续性和变量保持。 a()是执行一部分逻辑,a()()仅接着a()逻辑继续走下去,在这个逻辑上下文期间,变量始终都被保持着,可以随意使用。
9 g, Z, f) R% s( ]0 n& R) u$ V! S9 {- S5 {! `
更多网页制作信息请查看: 网页制作

回复|共 9 个

tohme 发表于 2016-1-26 17:43:08|欧洲 | 查看全部
原来...发神经是这样的啊...
Acropozelan 发表于 2016-1-26 17:43:15|日本 | 查看全部
晚安 别让小嫁再郁闷了 -
wwdu926a 发表于 2016-1-26 17:43:27|马来西亚 | 查看全部
机会就像水中的鱼,耐心等待就能上钩。
bqtklouu 发表于 2016-1-26 17:43:56|西班牙 | 查看全部
什么啊
wwzcdenleclv 发表于 2016-1-26 17:44:13|美国 | 查看全部
baidu是相当能折腾我了
Acropozelan 发表于 2016-1-28 16:21:54|美国 | 查看全部
哎 天理何在啊??
mwxny 发表于 2016-1-28 16:22:12|美国 | 查看全部
这个得知互动技术交流论坛很不错
Mqokjdvq 发表于 2016-1-28 16:22:26|德国 | 查看全部
哈哈 小心被她看见哦~~~~
bqtklouu 发表于 2016-1-28 16:22:34|德国 | 查看全部
楼主说的是什么?我是小白,好像还没看明白!

回复

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

得知互动是一个融创意、设计、开发、营销、生活、互联网于一体的专业交流分享平台。
Copyright © 2026 站长技术交流论坛|互联网技术交流平台 版权所有 All Rights Reserved. Powered by Discuz! X5.0 鄂ICP备15006301号-5|鄂公网安备 42018502006730号
关灯 在本版发帖 扫一扫添加QQ客服 返回顶部
快速回复 返回顶部 返回列表