try-with-statement语法糖总结

最近处理的sonar问题,近一半都是资源未关闭的问题,sonar提示我们可以使用try-with-statement来解决这类问题,我尝试和总结了如下:

这个是用来替代繁琐的try catch finnally

它会自动close 所有实现了java.lang.AutoCloseable接口的资源

写法是在try后面跟着一个小括号,把资源的声明代码写进去就ok了

try (BufferedReader br =

              new BufferedReader(new FileReader(path))) {

        return br.readLine();

}catch(IOExcepton e){

}

这个里面br就会在使用完毕后自动关闭,比如抛出异常,或是try代码块执行完毕的时候,会自动执行close,妈妈再也不用担心我们的close

这个小括号里面可以声明一个或是多个资源变量

这个写法等价于

BufferedReader br = new BufferedReader(new FileReader(path));

    try {

        return br.readLine();

    } catch(){

    }

    finally {

        // do br.close()  先简单这样理解,编译器对close抛出的异常做了move-exception操作这点无法用代码表述出来,详见后文    }

需要注意一点,它对Exception的抛出做了更友好的改进,通常在

//伪代码

try{

a = ClosableSteam()

} catch(Exception e){

}finnaly{

close()

}

里面如果catch了异常后在执行close的时候又抛出了异常,系统会抛出close时候的异常,这不利于定位问题,所以在使用try-with-statement里面引入了SupressedException的概念,相当于原代码段被处理成了如下方式

TryStudy tryStudy = null;

try{

tryStudy = new TryStudy();

System.out.println(tryStudy);

}catch(Throwable suppressedException) {

if (tryStudy != null) {

try {

tryStudy.close();

}catch(Throwable e) {

e.addSuppressed(suppressedException);

throw e;

}

}

throw suppressedException;

}

这样抛出的异常其实就是原异常,有助于我们定位真实的Exception点,这个原异常会在

try(){ //try-with-statement

}catch(Exception e){

}

这里的e里面拿到,如果一定要拿close时候抛的异常,那么在Exception中getSupressedException就可以了

抛出来的异常大概会长这样

java.io.IOException: doSomething IOException

at sg.bigo.lijiangyan.testtrywithstatement.MyClosableClass.doSomething(MyClosableClass.java:11)

at sg.bigo.lijiangyan.testtrywithstatement.MainActivity.onCreate(MainActivity.java:19)

at android.app.Activity.performCreate(Activity.java:7149)

at android.app.Activity.performCreate(Activity.java:7140)

at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1288)

at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:3017)

at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:3172)

at android.app.servertransaction.LaunchActivityItem.execute(LaunchActivityItem.java:78)

at android.app.servertransaction.TransactionExecutor.executeCallbacks(TransactionExecutor.java:108)

at android.app.servertransaction.TransactionExecutor.execute(TransactionExecutor.java:68)

at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1906)

at android.os.Handler.dispatchMessage(Handler.java:106)

at android.os.Looper.loop(Looper.java:193)

at android.app.ActivityThread.main(ActivityThread.java:6863)

at java.lang.reflect.Method.invoke(Native Method)

at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:537)

at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:858)

Suppressed: java.io.IOException: close IOException

at sg.bigo.lijiangyan.testtrywithstatement.MyClosableClass.close(MyClosableClass.java:16)

at sg.bigo.lijiangyan.testtrywithstatement.MainActivity.onCreate(MainActivity.java:20)

... 15 more

可以看到这里跟了一个Suppressed部分

当然这里也要注意有坑,由于这个语法糖是在java7引入的,如果androidminsdk低于19,就不能用,但是android在支持java8后使用desugar解决了这个问题,所以我们的项目中可以愉快的使用

当然,还有更坑,要注意这是编译器帮我们加的,假如包装类在close中还做了其余可能抛出异常的代码,而这些代码在close之前,依旧可能发生泄露,例如,GZIPOutputStream https://juejin.im/entry/57f73e81bf22ec00647dacd0

try (FileInputStream fin = new FileInputStream(new File("input.txt"));

                GZIPOutputStream out = new GZIPOutputStream(new FileOutputStream(new File("out.txt")))) {

}

GZIPOutputStream里面的close会调用flush之后再调用close,但是flush会抛出异常,而编译器帮我们加的代码是fin和out的close,所以这种情况发生的时候out里面被包装的那个FileOutputStream发生了泄露,正确的做法应该是单独为每个流声明

try (FileInputStream fin = new FileInputStream(new File("input.txt"));

                FileOutputStream fout = new FileOutputStream(new File("out.txt"));

                GZIPOutputStream out = new GZIPOutputStream(fout)) {}

编译器会帮我们close fin,fout,以及out,这样所有资源就可以正确的关闭了

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,384评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,845评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,148评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,640评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,731评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,712评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,703评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,473评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,915评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,227评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,384评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,063评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,706评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,302评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,531评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,321评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,248评论 2 352

推荐阅读更多精彩内容