建议61:避免在finally内撰写无效代码

在阐述建议之前,需要先提出一个问题:是否存在一种打破try-finally执行顺序的情况,答案是:不存在(除非应用程序本身因为某些很少出现的特殊情况在try块中退出)。应该始终认为finally内的代码会在方法return之前执行,哪怕return在try块中。

正是这点,可能会让你写出无效的代码,有时候,这样的无效代码会是一个隐藏很深的Bug。

看下面代码:

        private static int TestIntReturnBelowFinally()
{
int i;
try
{
i = ;
}
finally
{
i = ;
Console.WriteLine("\t将int结果改为2,finally执行完毕");
}
return i;
}

返回值是2。

但是:

        private static int TestIntReturnInTry()
{
int i;
try
{
return i = ;
}
finally
{
i = ;
Console.WriteLine("\t将int结果改为2,finally执行完毕");
}
}

返回值是1。

再看下面代码:

        static User TestUserReturnInTry()
{
User user = new User() { Name = "Mike", BirthDay = new DateTime(, , ) };
try
{
return user;
}
finally
{
user.Name = "Rose";
user.BirthDay = new DateTime(, , );
Console.WriteLine("\t将user.Name改为Rose");
}
}

user类:

    class User
{ public string Name { get; set; } public DateTime BirthDay { get; set; }
}

TestUserReturnInTry方法返回的User中,Name的值已经改为Rose了。

现在来解释为什么上面3个函数会有3种结果。查看TestIntReturnBelowFinally的finally部分的IL代码:

  finally
{
IL_0004: ldc.i4.2
IL_0005: stloc.0
IL_0006: ldstr bytearray ( 5C 6E D3 7E 9C // ...\i.n.t..~.g9e
3A 4E 0C FF 6E 6C // :N2...f.i.n.a.l.
6C 4C 8C 5B D5 6B ) // l.y.gbL..[.k
IL_000b: call void [mscorlib]System.Console::WriteLine(string)
IL_0010: endfinally
} // end handler
IL_0011: ldloc.0
IL_0012: ret
}

“IL_0004: ldc.i4.2”首先将2压入栈顶

“IL_0005: stloc.0”将最顶层堆栈的值,也就是2赋值给本地变量,也就是 i (index 0)

“IL_0011: ldloc.0”将本地变量 i (index 0)的值再次压入栈

“IL_0012: ret”结束函数,同时把栈内的返回值压入调用者的栈中。就函数将2赋值给了返回值。

看方法TestIntReturnInTry()的Debug版本的IL代码:

.method private hidebysig static int32  TestIntReturnInTry() cil managed
{
// 代码大小 27 (0x1b)
.maxstack
.locals init ([] int32 i,
[] int32 CS$$)
IL_0000: nop
.try
{
IL_0001: nop
IL_0002: ldc.i4.1
IL_0003: dup
IL_0004: stloc.0
IL_0005: stloc.1
IL_0006: leave.s IL_0018
} // end .try
finally
{
IL_0008: nop
IL_0009: ldc.i4.2
IL_000a: stloc.0
IL_000b: ldstr bytearray ( 5C 6E D3 7E 9C // ...\i.n.t..~.g9e
3A 4E 0C FF 6E 6C // :N2...f.i.n.a.l.
6C 4C 8C 5B D5 6B ) // l.y.gbL..[.k
IL_0010: call void [mscorlib]System.Console::WriteLine(string)
IL_0015: nop
IL_0016: nop
IL_0017: endfinally
} // end handler
IL_0018: nop
IL_0019: ldloc.1
IL_001a: ret
} // end of method Program::TestIntReturnInTry

TestIntReturnInTry在IL中创建了两个本地变量 i 和CS$1$0000 ,i 存储的是1,然后finally中 i 被赋值为2。调用者真正得到的是由IL创建的CS$1$0000所对应的值。用Reflector查看C#代码:

private static int TestIntReturnInTry()
{
int i;
int CS$$;
try
{
CS$$ = i = ;
}
finally
{
i = ;
Console.WriteLine("\t将int结果改为2,finally执行完毕");
}
return CS$$;
}

实际上,finally中i=2没有任何意义,所以在本函数的release版本中,IL中找不到对应的代码:

.method private hidebysig static int32  TestIntReturnInTry() cil managed
{
// 代码大小 17 (0x11)
.maxstack
.locals init ([] int32 CS$$)
.try
{
IL_0000: ldc.i4.1
IL_0001: stloc.0
IL_0002: leave.s IL_000f
} // end .try
finally
{
IL_0004: ldstr bytearray ( 5C 6E D3 7E 9C // ...\i.n.t..~.g9e
3A 4E 0C FF 6E 6C // :N2...f.i.n.a.l.
6C 4C 8C 5B D5 6B ) // l.y.gbL..[.k
IL_0009: call void [mscorlib]System.Console::WriteLine(string)
IL_000e: endfinally
} // end handler
IL_000f: ldloc.0
IL_0010: ret
} // end of method Program::TestIntReturnInTry

用Reflector查看release版本中C#代码:

private static int TestIntReturnInTry()
{
int CS$$;
try
{
CS$$ = ;
}
finally
{
Console.WriteLine("\t将int结果改为2,finally执行完毕");
}
return CS$$;
}

再解释第三个方法TestUserReturnInTry为什么返回的是“Rose”。Reflector查看release版本中C#代码:

private static User TestUserReturnInTry()
{
User CS$$;
User <>g__initLocal0 = new User {
Name = "Mike",
BirthDay = new DateTime(0x7da, , )
};
User user = <>g__initLocal0;
try
{
CS$$ = user;
}
finally
{
user.Name = "Rose";
user.BirthDay = new DateTime(0x7da, , );
Console.WriteLine("\t将user.Name改为Rose");
}
return CS$$;
}

User是引用类型, CS$1$0000 = user;说明CS$1$0000和user指向的是同一个对象,当在finally中 user.Name = "Rose"时CS$1$0000的Name也会变为“Rose”。所以返回的CS$1$0000的Name为“Rose”。

再举一个例子:

        private static User TestUserReturnInTry2()
{
User user = new User() { Name = "Mike", BirthDay = new DateTime(, , ) };
try
{
return user;
}
finally
{
user.Name = "Rose";
user.BirthDay = new DateTime(, , );
user = null;
Console.WriteLine("\t将user置为anull");
}
}

返回的结果不是null,而一个Name=“Rose”,BirthDay = new DateTime(2010, 2, 2)的User对象。Reflector查看release版本中C#代码:

private static User TestUserReturnInTry2()
{
User CS$$;
User <>g__initLocal1 = new User {
Name = "Mike",
BirthDay = new DateTime(0x7da, , )
};
User user = <>g__initLocal1;
try
{
CS$$ = user;
}
finally
{
user.Name = "Rose";
user.BirthDay = new DateTime(0x7da, , );
user = null;
Console.WriteLine("\t将user置为anull");
}
return CS$$;
}

CS$1$0000和user指向的是同一个对象,当在finally中 user=null 时,只是user指向为null了,CS$1$0000指向的对象并没有变。

转自:《编写高质量代码改善C#程序的157个建议》陆敏技

编写高质量代码改善C#程序的157个建议——建议61:避免在finally内撰写无效代码的更多相关文章

  1. 编写高质量代码改善C#程序的157个建议[1-3]

    原文:编写高质量代码改善C#程序的157个建议[1-3] 前言 本文主要来学习记录前三个建议. 建议1.正确操作字符串 建议2.使用默认转型方法 建议3.区别对待强制转换与as和is 其中有很多需要理 ...

  2. 读书--编写高质量代码 改善C#程序的157个建议

    最近读了陆敏技写的一本书<<编写高质量代码  改善C#程序的157个建议>>书写的很好.我还看了他的博客http://www.cnblogs.com/luminji . 前面部 ...

  3. 编写高质量代码改善C#程序的157个建议——建议157:从写第一个界面开始,就进行自动化测试

    建议157:从写第一个界面开始,就进行自动化测试 如果说单元测试是白盒测试,那么自动化测试就是黑盒测试.黑盒测试要求捕捉界面上的控件句柄,并对其进行编码,以达到模拟人工操作的目的.具体的自动化测试请学 ...

  4. 编写高质量代码改善C#程序的157个建议——建议156:利用特性为应用程序提供多个版本

    建议156:利用特性为应用程序提供多个版本 基于如下理由,需要为应用程序提供多个版本: 应用程序有体验版和完整功能版. 应用程序在迭代过程中需要屏蔽一些不成熟的功能. 假设我们的应用程序共有两类功能: ...

  5. 编写高质量代码改善C#程序的157个建议——建议155:随生产代码一起提交单元测试代码

    建议155:随生产代码一起提交单元测试代码 首先提出一个问题:我们害怕修改代码吗?是否曾经无数次面对乱糟糟的代码,下决心进行重构,然后在一个月后的某个周一,却收到来自测试版的报告:新的版本,没有之前的 ...

  6. 编写高质量代码改善C#程序的157个建议——建议154:不要过度设计,在敏捷中体会重构的乐趣

    建议154:不要过度设计,在敏捷中体会重构的乐趣 有时候,我们不得不随时更改软件的设计: 如果项目是针对某个大型机构的,不同级别的软件使用者,会提出不同的需求,或者随着关键岗位人员的更替,需求也会随个 ...

  7. 编写高质量代码改善C#程序的157个建议——建议153:若抛出异常,则必须要注释

    建议153:若抛出异常,则必须要注释 有一种必须加注释的场景,即使异常.如果API抛出异常,则必须给出注释.调用者必须通过注释才能知道如何处理那些专有的异常.通常,即便良好的命名也不可能告诉我们方法会 ...

  8. 编写高质量代码改善C#程序的157个建议——建议152:最少,甚至是不要注释

    建议152:最少,甚至是不要注释 以往,我们在代码中不写上几行注释,就会被认为是钟不负责任的态度.现在,这种观点正在改变.试想,如果我们所有的命名全部采用有意义的单词或词组,注释还有多少存在的价值. ...

  9. 编写高质量代码改善C#程序的157个建议——建议151:使用事件访问器替换公开的事件成员变量

    建议151:使用事件访问器替换公开的事件成员变量 事件访问器包含两部分内容:添加访问器和删除访问器.如果涉及公开的事件字段,应该始终使用事件访问器.代码如下所示: class SampleClass ...

  10. 编写高质量代码改善C#程序的157个建议——建议150:使用匿名方法、Lambda表达式代替方法

    建议150:使用匿名方法.Lambda表达式代替方法 方法体如果过小(如小于3行),专门为此定义一个方法就会显得过于繁琐.比如: static void SampeMethod() { List< ...

随机推荐

  1. selenium - 控制浏览器窗口的大小和浏览器最大化

    1.控制浏览器大小 有些前端的页面需要查看在不同像素下的兼容情况,比如把像素设置为 480*800,然后截图看看页面显示有没有问题 WebDriver 提供了 set_windows_size() 方 ...

  2. mysql编译参数详解

    mysql编译参数详解(./configure)   1.--prefix=PREFIX:指定程序安装路径: 2.--enable-assembler:使用汇编模式:(文档说明:compiling i ...

  3. 蓝桥杯 算法训练 ALGO-34 纪念品分组

    算法训练 纪念品分组   时间限制:1.0s   内存限制:256.0MB 问题描述 元旦快到了,校学生会让乐乐负责新年晚会的纪念品发放工作.为使得参加晚会的同学所获得的纪念品价值 相对均衡,他要把购 ...

  4. (转)Oracle执行字符串

    declare v_out ); begin execute immediate 'select p_guid from c_itcomp where rownum = 1 ' into v_out; ...

  5. java代码反转toCharAT()的用法

    总结:反转注意for循环里面的变化 package clientFrame; //字符串反转 public class we { public static void main(String[] ar ...

  6. 1139 First Contact

    题意:给出n个人,m对朋友关系,其中带负号的表示女孩.然后给出k对查询a,b,要求找出所有a的同性别好友c,以及b的同性别好友d,且c和d又是好友关系.输出所有满足条件的c和d,按c的升序输出,若c编 ...

  7. 实例甜点 Unreal Engine 4迷你教程(2)之用C++改变Image小部件的颜色

    完成本迷你教程之前,请前往完成以下迷你教程: ·实例甜点 Unreal Engine 4迷你教程之如何用C++将纹理绘制在UserWidget的Image小部件上: 目标:实现UMG中的此功能: 在上 ...

  8. mysql索引原理与慢查询优化2

    七 正确使用索引 一 索引未命中 并不是说我们创建了索引就一定会加快查询速度,若想利用索引达到预想的提高查询速度的效果,我们在添加索引时,必须遵循以下问题 1 范围问题,或者说条件不明确,条件中出现这 ...

  9. 阻塞IO(blocking IO)

    在linux中,默认情况下所有的socket都是blocking,一个典型的读操作流程大概是这样: 当用户进程调用了recvfrom这个系统调用,kernel就开始了IO的第一个阶段:准备数据.对于n ...

  10. DVWA平台v1.8-反射型XSS(low级别)

    源代码 <?php if(!array_key_exists ("name", $_GET) || $_GET['name'] == NULL || $_GET['name' ...