Kotlin-33.异常(Exception)

官方文档: http://kotlinlang.org/docs/reference/exceptions.html

1.异常类(Exception Classes)

与java类似,Kotlin的所有异常类都是Throwable类的子孙类(都继承自Throwable类),
每个异常类成员都有消息(message),堆栈跟踪(stack trace)和可选的起因(optional cause).

1.与Java一样,kotlin使用throw表达式抛出异常(throw exception):
    throw MyException("Hi There!")

2.与Java一样,kotlin使用try表达式捕获异常(catch exceptio):
    try {
        // 一些代码
    } catch (e: SomeException) {
        // 处理程序
    } finally {
        // 可选的finally块
    }
    //可以有零或多个catch块,finally块可省略,但catch和finally块至少应该存在一个!

3.与Java不同,kotlin的try表达式有一个返回值
    try表达式返回值是try块最后一个表达式,或者是catch块最后一个表达式,
    finally块内容不会影响表达式的结果.
    fun main(args: Array<String>) {
        val a: Int? = try { 
                1 //正常运行,返回1               
            } catch (e: NumberFormatException) {
                2
                null
            }  finally {
                3 //finally块不会影响表达式结果
            }

        val b: Int? = try { 
                1
                throw NumberFormatException()
            } catch (e: NumberFormatException) {
                2
                null //捕获异常,返回null
            }  finally {
                3 //finally块不会影响表达式结果
            }

        println(a) //输出1
        println(b) //输出null
    }

2.没有受检异常(Checked Exceptions)

java两种异常类型: 受检异常(checked exception)和非受检异常(unchecked exception)
    1.Error和RuntimeException及其子类都是受检异常(checked exception), 
    2.其余的异常Exception都是非受检异常(unchecked exception);
这两种异常在作用上没有差别,唯一差别在于在编译时编译器会检查受检异常,
所以受检异常需要try catch捕获来避免编译错误,而非受检异常不需要!

可见受检异常(Checked Exceptions)使用比较麻烦,争议非常大,可能会导致java API变得很复杂,
程序跟异常检查代码混杂在一起,这仅仅是为了通过编译器的编译,
许多人批评Java的受检异常,认为受检异常(Checked Exception)是软件工程中一次失败的试验!

kotlin没有受检异常(Checked Exceptions),以下是kotlin不使用受检异常的原因描述:
    JDK的StringBuilder类实现的一个示例接口:
        Appendable append(CharSequence csq) throws IOException;
    这个append函数签名throws IOException,每次追加一个字符串(StringBuilder/某种日志log/控制台console),
    就必须捕获IOException(可能正在执行IO操作Writer也实现了Appendable),所以导致try{}代码随处可见:
        try {
            log.append(message)
        } catch (IOException e) {

        }
    java受检的异常(Checked Exception)很不好用,参见《Effective Java》第65条:不要忽略异常!
    
    Bruce Eckel在《Java是否需要受检的异常?》Does Java need Checked Exceptions?中指出:
        一些小程序测试得出的结论是:受检的异常会提高开发者的生产力和代码质量高,
        但是大型软件项目的经验有不同的结论:生产力降低,代码质量低或没有提高!

    其他相关引证:
        《Java的受检异常是一个错误》Java's checked exceptions were a mistake (Rod Waldhoff)
        《受检异常的烦恼》The Trouble with Checked Exceptions (Anders Hejlsberg)

3.Nothing类型(The Nothing type)

在Kotlin中throw是表达式,所以可以作为Elvis表达式?:的一部分:
    val s = person.name ?: throw IllegalArgumentException("Name required")

throw表达式的类型是Nothing类型,该特殊类型没有值,只用于标记代码位置永远不能到达(never be reached)
所以当person.name为null时, s = 赋值操作永远不会发生(即throw类型Nothing,永远不可到达s = )

可以使用 Nothing 来标记一个函数永远不会返回(never return):
    fun main(args: Array<String>) {
        fun fail(message: String): Nothing {
            throw IllegalArgumentException(message)
        }
        
        //当调用该函数fail()时,编译器会知道执行不会超出该调用(说白了就是程序不会继续执行)
        //程序中断,输出 "java.lang.IllegalArgumentException: Name参数错误,不能为null"
        val name = null  
        val s: String = name ?: fail("Name参数错误,不能为null")
        println(s)
    }

4.kotlin与Java互操作的异常处理(Java Interoperability)

在Kotlin中,所有异常都是非受检的,意味着编译器不会强迫捕获任何异常(try catch)! 
因此,在Kotlin中调用一个受检异常的Java方法,不会强迫你去捕获异常:   
    //kotlin代码,调用java方法,append(CharSequence csq) throws IOException;     
    fun render(list: List<*>, to: Appendable) {
        for (item in list) {
            //在kotlin中不要求捕获异常,但在Java中会强迫捕获异常IOException
            to.append(item.toString()) 
        }
    }

简书:http://www.jianshu.com/p/227f398b929f CSDN博客: http://blog.csdn.net/qq_32115439/article/details/74617358
GitHub博客:http://lioil.win/2017/07/06/Kotlin-exception.html
Coding博客:http://c.lioil.win/2017/07/06/Kotlin-exception.html