Kotlin-21.函数(Functions)

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

1.函数定义和调用

相比java, kotlin函数定义/声明非常方便简单
1.函数声明定义/声明(Function Declarations):
    //Unit类型表示没有返回值
    fun demo(): Unit {
        ...
        // return Unit 或 return 是可选的                     
    }

    //可以省略Unit
    fun demo() {
        ...
        // return Unit 或 return 是可选的                     
    }

2.单表达式函数定义/声明:
    //当函数返回单个表达式时,可省略花括号{},并用等号指定函数代码体:
    fun double(x: Int): Int = x * 2
    
    //当返回类型可由编译器推断出时,可省略返回类型:
    fun double(x: Int) = x * 2

1.传统调用

调用普通函数:
    demo()

调用成员函数:
    class Sample() {
        fun foo() { print("Foo") }
    }
    Sample().foo()

2.中缀调用(Infix notation)

中缀调用就是使用[中缀标记]调用函数,非常类似于加减乘除(+ - * /)
必须满足以下条件:
    函数作为类的成员函数 或 扩展函数
    函数必须且只有一个参数
    函数前缀有 infix 标记

// 扩展函数-中缀表示/标记
infix fun Int.add(i: Int): Int{
    return this + i
} 

// 成员函数-中缀表示/标记
class MyInt(var a: Int){    
    infix fun add(i: Int): Int{
        return a + i
    }
}

fun main(args: Array<String>) {          
    println(2 add 3)  //中缀调用,输出5       
    println(2.add(3)) //传统调用,输出5

    val i = MyInt(2)
    println(i add 3)  //中缀调用,输出5
    println(i.add(3)) //传统调用,输出5
}

2.函数参数

函数参数用Pascal表示法定义(即name: type),参数用逗号分隔,参数必须要有显式类型:
    fun myFun(p1: Int, p2: Int) {
    }

1.默认参数(Default Arguments)

函数参数可以有默认值,当函数调用没有参数时,就使用默认值:
    fun myFun(p1: Int = 1, p2: Int = 2) {
    }
 
子类覆盖方法与基类(父类)方法默认参数值相同,父类有默认参数,子类不能写默认参数:
    open class A {
        open fun foo(i: Int = 10) {            
        }
    }

    class B : A() {
        // 父类已有,子类不能有默认参数值
        override fun foo(i: Int) {      
        }
    }

2.命名参数(Named Arguments)

当函数有大量参数时调用会非常不方便:
    fun myFun(a: String = "lioil", 
              b: Boolean = true, 
              c: Int = 1, 
              d: Char = 'w') {        
    }
    //修改参数c,其它参数保持默认,虽然有默认参数,但是依然麻烦!!!
    myFun("lioil",true,2)        
    //此时可用命名参数(Named Arguments)简化:
    myFun(c = 2)

3.参数个数可变(vararg)

函数参数用vararg标记,可使参数个数可变:
    fun <T> asList(vararg ts: T){        
        for (t in ts) //参数ts相当于数组Array <out T>
            print("$t,")
    }

    fun main(args: Array<String>) {
        asList(1,2,3) //输出1,2,3,

        //传递数组,添加伸展(spread)操作符*
        val a = arrayOf(1, 2, 3)
        asList(0, *a, 4) //输出0,1,2,3,4,
    }

如果vararg标记的参数不是最后一个参数,需要用[命名参数]传递其之后的参数:        
    fun <T> asList(vararg ts: T, name: String){  
        for (t in ts)
            print("$t,")
        print(name)
    }

    fun main(args: Array<String>) {            
        asList(1,2,3,name="lioil.win")//输出1,2,3,lioil.win
    }

3.函数种类

Kotlin函数可在文件顶层定义/声明,无需像Java、C#、Scala那样在一个类中定义一个函数,
所以说函数是kotlin世界的第一公民!
此外,Kotlin函数也可在局部作用域定义,作为函数内部的函数,类的成员函数或扩展函数!

1.本地/局部函数(Local Functions)-函数内部的函数
    Kotlin支持局部函数,即一个函数在另一个函数内部:    
    fun main(args: Array<String>) {
        val name = "lioil.win"
        fun myFun() {
        //局部函数可访问外部函数(即闭包)的局部变量name
            println(name)
        } 
        myFun() //输出 lioil.win
    }

2.成员函数(Member Functions)是在类或对象内部定义的函数
    class Sample() {
        fun foo() { print("Foo") }
    }
    //成员函数调用
    Sample().foo()

3.泛型函数(Generic Functions),在函数名前使用尖括号<>指定泛型
    fun <T> singletonList(item: T): List<T> {
    }

4.尾递归函数(Tail recursive functions)
    Kotlin支持尾递归(tail recursion)函数: 使用尾递归函数替循环代码,不会有堆栈溢出的风险!
    使用tailrec修饰符标记函数,编译器会把尾递归优化成高效循环代码!
    使用tailrec修饰符的条件:
        递归调用必须是函数体的最后一个操作(即保证是尾递归)
        不能用在 try/catch/finally 块中
        目前尾部递归只在 JVM 后端中支持

    //尾递归代码
    tailrec fun findFixPoint(x: Double = 1.0): Double
            = if (x == Math.cos(x)) x else findFixPoint(Math.cos(x))
    
    //最终被编译器优化成循环代码
    private fun findFixPoint(): Double {
        var x = 1.0
        while (true) {
            val y = Math.cos(x)
            if (x == y) return y
            x = y
        }
    }

5.内联函数(Inline Functions)、扩展函数(Extension Functions)、
高阶函数(Higher-Order Functions)和Lambda表达式等在其它章节中...

CSDN博客: http://blog.csdn.net/qq_32115439/article/details/73825015
GitHub博客:http://lioil.win/2017/06/27/Kotlin-functions.html
Coding博客:http://c.lioil.win/2017/06/27/Kotlin-functions.html