日期:2014-05-20  浏览次数:20756 次

一篇好文,写得比较幽默,看者有分!
java的try-finally给我们提供了一个“保证某个动作必然执行”的机会。 

一个try-finally结构,只要try块开始执行了,finally块里面的代码保证执行一次并且只有一次。 
打个比方,就象你上厕所,只要你一旦开始拉了,我们保证无论如何,是拉稀了也好,放屁了也罢,最终你肯定是擦了屁股走出卫生间。 

应用try-finally,我们可以在异常满天飞的程序里保证我们的关键资源被按时正确清理。一个最常见的应用就是jdbc的Connection, Statement, ResultSet等。 

但是,我最近惊奇地发现,不知道怎么正确清理资源的人大有人在,即使是一些java老手。 

看一个例子先: 



代码 
void f(){ 
Connection conn = ...; 
Statement stmt = conn.createStatement(); 
ResultSet rset = ...; 
... 


典型的jdbc程序。但是也是典型的光着屁股,其臭如兰地走出厕所的典范。哎,你擦屁股了吗? 
有的哥们振振有辞:我不用管,我的jdbc driver/我的应用服务器/garbage collector会处理的。 
这是典型的糊涂蛋逻辑。没有close(),jdbc driver, 应用服务器怎么知道你是拉完了,还是光着屁股出去接个电话先?难不成这driver都智能地会算命了? 
garbage collector倒确实管得了。不过,garbage collector不一定运行啊。你要是有10G得内存,要是你的程序就用了10M,garbage collector说不定就一直睡大觉。而且,就算它管,也许等你光着屁股上班被警察抓起来之后才匆匆赶到,你等的起吗? 


好,有人说,那我擦,我擦,我擦擦擦。行了吧? 



代码 
void f(){ 
Connection conn = ...; 
Statement stmt = conn.createStatement(); 
ResultSet rset = ...; 
rset.close(); 
conn.close(); 
... 



呵呵。我的傻哥们,你只擦了靠近后背的那三公分,剩下的嘛,别人看不见你就乐得省土块儿了是么? 

按jdbc标准,ResultSet, Statement, Connection都要close(),也许有的driver会在Connection关闭的时候同时正确清理ResultSet, Statement,但是,并没有一条规定让所有的driver都这么做。 
另外,也许你的Connection是从一个池里面来的,它只是回到池中去,如果你不关闭Statement, ResultSet,下一个拿到这个Connection的人也许就倒霉了! 
做事要有始有终,既然开始擦了,就擦干净点儿,行不?(那个,谁谁谁,借我个防毒面具先!) 

ok,有个讲卫生的小傻子这样擦: 



代码 
void f(){ 
Connection conn = ...; 
Statement stmt = conn.createStatement(); 
ResultSet rset = ...; 
rset.close(); 
stmt.close(); 
conn.close(); 
... 



然后洋洋得意地说:我是好孩子,我天天擦屁屁。 

是啊,多听话的孩子呀。可惜,某天,这孩子正坐在马桶上美着呢,妈妈喊了嗓子:二傻子,吃饭啦。 
哦!吃饭。二傻子裤子都没提就窜出来了,熏得妈妈一个跟头。 

什么问题,傻子做事一根筋,不能打扰,一旦有异常情况出现,屁股就忘了擦了。 

所以,我这里郑重提醒大家,请用"try-finally"!它独有凹槽,防止侧漏...(糟了,串台了) 

是啊,java老手们都不是傻子,都知道用try-finally的,可是,别美,你现在就保不齐擦没擦屁股呢! 

常见擦法: 


代码 
void f(){ 
Connection conn = null; 
Statement stmt = null; 
ResultSet rset = null; 
try{ 
conn = ...; 
stmt = ...; 
rset = ...; 
... 

finally{ 
if(rset!=null)rset.close(); 
if(stmt!=null)stmt.close(); 
if(conn!=null)conn.close(); 





嗯。怎么说呢。挺聪明的。都学会if(xxx!=null)这种传说中条件判断的上古绝学了。 
可惜,你屁股大,一张纸不够,你用了第一张纸,满意地看着它圆满地完成了金灿灿的任务,再用第二张,靠,只太薄,破了,一手金灿灿地,象带了个金戒指。你大怒,起,绝尘而去。于是也忘了第三张纸, 
哥们儿,close()是可以出异常的,你rset关了,stmt.close()出现了异常,但是conn就不管了? 

近日有位室外高人,据说是鬼谷子高徒,鉴于怜我世人,不擦屁股的实多的高尚情操,亲手赚写一本绝世擦功秘籍,其文美,其意高,除了擦不干净之外,真可以说是称霸擦林。 



代码 
void close(Connection conn){ 
try{ 
if(conn!=null) conn.close(); 

catch(Exception e){ 
e.printStackTrace(); 


void close(ResultSet rset){ 
... 

void close(Statement rset){ 
... 

void f(){ 
Connection conn = null; 
Statement stmt = null; 
ResultSet rset = null; 
try{ 
conn = ...; 
stmt = ...; 
rset = ...; 
... 

finally{ 
close(rset); 
close(stmt); 
close(conn); 





哈,你们不能纸擦破了就不接着擦啊,甚至大而化之,不能擦股用具有了问题就半途而废呀! 

具信,该高人以此法擦遍天下凡十数载,未有擦而无功者。 

可惜,高人却忽视了,除了纸会出故障,甚至大而化之,一切擦具(如土块儿,木条儿,手指)都可能出现故障,还有别的地方也会出故障地!