函数

函数

函数声明

kotlin 中用关键字 fun 声明函数:


fun double(x: Int): Int {
}
函数用法

通过传统的方法调用函数

//  函数的基本用法

fun double(x:Int):Int
{
    return 2 * x
}
class MyClass
{
    fun add(x:Int, y:Int):Int
    {
        return x + y
    }
    companion object {
        fun sub(x:Int, y:Int):Int
        {
            return x - y
        }
    }
}
fun main(args: Array<String>)
{
    println(double(20))
    println(MyClass().add(20,40))
    println(MyClass.sub(40,1))
}
中缀符号

支持中缀标记法调用的函数必须满足如下3个条件:
1. 是成员函数,或是扩展函数
2. 只能有一个参数
3. 使用infix关键字声明

//给 Int 定义一个扩展方法

infix fun String.div(str:String):String
{
    return this.replace(str,"")
}

fun main(args: Array<String>)
{
    //  使用正常的方式调用div函数
    var str = "hello world"
    println(str.div("l"))

    //  使用中缀表达式调用div函数
    println(str div "l")

    println(str div "l" div "o")
}

输出:
这里写图片描述

参数和返回值

函数参数是用 Pascal 符号定义的 name:type。参数之间用逗号隔开,每个参数必
须指明类型。
fun powerOf(number: Int, exponent: Int) {

}

默认参数

函数参数可以设置默认值,当参数被忽略时会使用默认值。这样相比其他语言可以减
少重载。



fun read(b: Array<Byte>, off: Int = 0, len: Int = b.size() ) {
...
}

默认值可以通过在type类型后使用 = 号进行赋值

命名参数

在调用函数时可以参数可以命名。这对于那种有大量参数的函数是很方便的.下面是一个例子:

fun reformat(str: String, normalizeCase: Boolean = true,upperCas
eFirstLetter: Boolean = true,
divideByCamelHumps: Boolean = false,
wordSeparator: Char = ' ') {
...
}

我们可以使用默认参数

reformat(str)

然而当调用非默认参数是就需要像下面这样:

reformat(str, true, true, false, '_')

使用命名参数我们可以让代码可读性更强:

reformat(str,
    normalizeCase = true,
    uppercaseFirstLetter = true,
    divideByCamelHumps = false,
    wordSeparator = '_'
)

如果不需要全部参数的话可以这样:

reformat(str, wordSeparator = '_')

注意,命名参数语法不能够被用于调用Java函数中,因为Java的字节码不能确保方法参数命名的不变性

不带返回值的参数

如果函数不会返回任何有用值,那么他的返回类型就是 Unit . Unit 是一个只有唯一值 Unit 的类型.这个值并不需要被直接返回:

un printHello(name: String?): Unit {
if (name != null)
println("Hello ${name}")
else
println("Hi there!")
// `return Unit` or `return` is optional
}

Unit 返回值也可以省略,比如下面这样:

fun printHello(name: String?) {
...
}
变长参数

函数的参数(通常是最后一个参数)可以用 vararg 修饰符进行标记,标记后,允许给函数传递可变长度的参数:

//  如果给可变参数类型的函数传递数组,前面需要加*
//  函数的可变参数
fun <T> asList(vararg ts:T):List<T>
{
    val result = ArrayList<T>()
    for(t in ts)
    {
        result.add(t)
    }
    return result
}
fun <T> asList1(vararg ts:T,value1:Int, value2:String):List<T>
{
    val result = ArrayList<T>()
    for(t in ts)
    {
        result.add(t)
    }
    println("value1 = ${value1}  value2 = ${value2}")
    return result
}
fun main(args: Array<String>)
{
    var list = asList(1,2,3,"abc",true,30.2)
    println(list.toString())
    var list1 = asList<Int>(1,2,3)
    println(list1.toString())

    var list2 = asList1(1,2,4,5,value1 = 20, value2="hello")

    val arr = arrayOf(1,2,3,4,5,6,7,8,9,10)
    var list3 = asList(*arr)
    println(list3.toString() )
    println(list3.size )
}

输出:
这里写图片描述

只有一个参数可以被标注为 vararg 。加入 vararg 并不是列表中的最后一个参数,那么后面的参数需要通过命名参数语法进行传值,再或者如果这个参数是函数类型,就需要通过lambda法则.当调用变长参数的函数时,我们可以一个一个的传递参数,比如 asList(1, 2,3) ,或者我们要传递一个 array 的内容给函数,我们就可以使用 * 前缀操作符。

单表达式函数

当函数只返回单个表达式
时,大括号可以省略并在 = 后面定义函数体

fun double(x: Int): Int = x*2

在编译器可以推断出返回值类型的时候,返回值的类型可以省略:

fun double(x: Int) = x * 2
明确返回类型

下面的例子中必须有明确返回类型,除非他是返回 Unit 类型的值,Kotlin 并不会对函数体重的返回类型进行推断,因为函数体中可能有复杂的控制流,他的返回类型未必对读者可见(甚至对编译器而言也有可能是不可见的):

函数范围

Kotlin 中可以在文件顶级声明函数,这就意味者你不用像在Java,C#或是Scala一样创建一个类来持有函数。除了顶级函数,Kotlin 函数可以声明为局部的,作为成员函数或扩展函数。

局部函数

Kotlin 支持局部函数,比如在一个函数包含另一函数。

fun dfs(graph: Graph) {
    fun dfs(current: Vertex, visited: Set<Vertex>) {
    if (!visited.add(current)) return
    for (v in current.neighbors)
    dfs(v, visited)
    }
    dfs(graph.vertices[0], HashSet())
}

局部函数可以访问外部函数的局部变量(比如闭包)

fun dfs(graph: Graph) {
    val visited = HashSet<Vertex>()
    fun dfs(current: Vertex) {
    if (!visited.add(current)) return
    for (v in current.neighbors)
    dfs(v)
    }
    dfs(graph.vertices[0])
}

局部函数甚至可以返回到外部函数 qualified return expressions

fun reachable(from: Vertex, to: Vertex): Boolean {
    val visited = HashSet<Vertex>()
        fun dfs(current: Vertex) {
        if (current == to) return@reachable true
        if (!visited.add(current)) return
        for (v in current.neighbors)
        dfs(v)
    }
    dfs(from)
    return false
    }
成员函数

成员函数是定义在一个类或对象里边的

class Sample() {
    fun foo() { print("Foo") }
}

成员函数可以用 . 的方式调用

Sample.foo()

泛型函数

函数可以有泛型参数,样式是在函数后跟上尖括号。

fun sigletonArray<T>(item: T): Array<T> {
    return Array<T>(1, {item})
}

尾递归函数

Kotlin 支持函数式编程的尾递归。这个允许一些算法可以通过循环而不是递归解决问题,从而避免了栈溢出。当函数被标记为 tailrec 时,编译器会优化递归,并用高效迅速的循环代替它。tailrec fun findFixPoint(x: Double = 1.0): Double
= if (x == Math.cos(x)) x else findFixPoint(Math.cos(x))这段代码计算的是数学上的余弦不动点。Math.cos 从 1.0 开始不断重复,直到值不变为止,结果是 0.7390851332151607 这段代码和下面的是等效的:

private fun findFixPoint(): Double {
    var x = 1.0
    while (true) {
    val y = Math.cos(x)
        if ( x == y ) return y
        x = y
    }
}

使用 tailrec 修饰符必须在最后一个操作中调用自己。在递归调用代码后面是不
允许有其它代码的,并且也不可以在 try/catch/finall 块中进行使用。当前的尾递归
只在 JVM 的后端中可以用

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值