在使用函数之前必须先定义函数。定义函数的语法格式如下:
fun 函数名(形参列表)[ : 返回值类型] {
}
使用":Unit"指定返回Unit代表没有返回值。相当于java的void(:Unit 可以忽略,默认的是Unit的类型)
![](https://i-blog.csdnimg.cn/blog_migrate/92f98b22bc4678682be9363c1cbb4ffe.webp?x-image-process=image/format,png)
函数的形参的定义:
![](https://i-blog.csdnimg.cn/blog_migrate/a64d5c1d1a91e29b4cc10454a4913648.webp?x-image-process=image/format,png)
当然可以定义返回某个类型的函数:
![](https://i-blog.csdnimg.cn/blog_migrate/82b50cf00019e9a9d1f47c1cde0c86f7.webp?x-image-process=image/format,png)
Kotlin函数也支持但表达式函数:对于但表达式函数而言,编译器可以推断出函数的返回值类型,一次kotlin允许省略点返回值的类型。
![](https://i-blog.csdnimg.cn/blog_migrate/51f534c26f19ae1fcca30487e81d8432.webp?x-image-process=image/format,png)
Kotlin支持函数的形参有默认值:如果第二个参数不传,就默认传的是2。如果第二个参数重新传值就相当于将传来的3把2替换掉。(切记:带有默认值的形参一定要放在不带有默认值的形参的后边)。同样你也可以带参数名来调用,这样在调用的时候就可以打乱顺序。
![](https://i-blog.csdnimg.cn/blog_migrate/92f60d6031e11b6e2c50f9fd49f88d73.webp?x-image-process=image/format,png)
![](https://i-blog.csdnimg.cn/blog_migrate/aea5e80fb4ebdd8fc16b0d4332752f4e.webp?x-image-process=image/format,png)
Kotlin允许定义个数可变的参数,从而允许为函数指定数量不确定的形参。在形参的类型前添加vararg修饰,怎表明该形参可以接受多个参数值。
![](https://i-blog.csdnimg.cn/blog_migrate/31a7a7c1110ae798164c101a68d402c5.webp?x-image-process=image/format,png)
Kotlin支持在函数体内定义函数,这种被放在函数体内定义函数称为局部函数。
![](https://i-blog.csdnimg.cn/blog_migrate/20410ac30e70d745331d90b531f634c9.webp?x-image-process=image/format,png)
Kotlin的高阶函数
首先Kotlin不是纯粹的面向对象语言,Kotlin的函数也是一等公民,因此函数本身也具有自己的类型。(看到这里只熟悉java的朋友会有点蒙圈,怎么函数自己也有自己的类型。那么请听我细细的讲解一下)
![](https://i-blog.csdnimg.cn/blog_migrate/cc9c03f7a4009d31acadff2bbaa8ada9.webp?x-image-process=image/format,png)
首先定义一个2个参数,每个参数都是Int类型和返回值是String类型的变量,之后创建一个pow函数也是同类型,并将pow函数赋值给了myFun。(当直接访问一个函数的函数引用, 而不是调用函数时,需要函数名钱添加两个冒号,而且不能再函数后面添加圆括号-----一旦添加圆括号就变成了调用函数,而不是访问函数的引用)
Kotlin可以使用函数类型作为形参类型:
![](https://i-blog.csdnimg.cn/blog_migrate/93457d82158b4e07411b9af9a2824537.webp?x-image-process=image/format,png)
Kotlin:使用函数类型作为返回值类型:
![](https://i-blog.csdnimg.cn/blog_migrate/72c09a5b9544f6473880f00f05cbbeb2.webp?x-image-process=image/format,png)
Kotlin局部函数与Lambda表达式:
回顾上,用lambda表达式代替局部函数:
定义Lambda表达式总是被大括号括着, 形参列表在->之前声明,参数类型可以省略,函数体放在->之后,函数的最后一个表达式自定被作为Lambda表达式的返回值,无需使用return关键字。
![](https://i-blog.csdnimg.cn/blog_migrate/da75a274e27f9a8641a29079da80c96b.webp?x-image-process=image/format,png)
将Lambda单做一个表达式出来
![](https://i-blog.csdnimg.cn/blog_migrate/faecda5b72f5dbe560e40952583df44c.webp?x-image-process=image/format,png)
![](https://i-blog.csdnimg.cn/blog_migrate/6c25dbdfed2dd8f49a70b205137357ab.webp?x-image-process=image/format,png)
调用Lambda表达式的约定
Kotlin语言有一个约定,如果函数的最后一个参数是函数类型,而且你打算传入一个Lambda表达式作为相应的参数,那么就允许在圆括号之外指定Lambda表达式,(这种用法也不是kotlin独有的,在其他语言中这种用法被称为“尾随闭包”)根据介绍建议将函数类型的形参放在形参列表的最后,这样方便以后传入Lambda表达式作为参数。
![](https://i-blog.csdnimg.cn/blog_migrate/f23e8c2db4bd053fd033eb94507e9586.webp?x-image-process=image/format,png)
Lambda表达式虽然简洁,方便,但是它有一个严重的缺陷就是它表达式不能指定返回值类型。因此如果kotlin无法推断Lambda表达式的返回值类型。此时就需要显示指定的返回值类型,匿名函数可以代替Lambda表达式。如下:
![](https://i-blog.csdnimg.cn/blog_migrate/98ca312a2eeb6f0ba512757cee2af546.webp?x-image-process=image/format,png)
与不普通函数不同的是,如果系统可以推断出函数的形参类型,那么匿名函数允许省略形参类型
![](https://i-blog.csdnimg.cn/blog_migrate/3b3d2967e7a00409e1d07866db9cbdbc.webp?x-image-process=image/format,png)
匿名函数和Lambda表达式的return
匿名函数的本质依然是函数,因此匿名函数的return则返回函数的本身,而Lambda表达式的return返回的是所在的函数。例如:
![](https://i-blog.csdnimg.cn/blog_migrate/9022e39cc7283e248a96171f98c4dd6d.webp?x-image-process=image/format,png)
内联函数
简单介绍一下内联函数的产生条件(当为函数传入函数或者lambda表达式作为参数的调用过程中,程序要将执行顺序转移到被调用表达式或者函数所在的内存地址当被调用表达式或者函数执行完后,再返回原来函数执行的地方)在上边的转移过程中,系统处理如下的事情:
1.为被调用的表达式或者函数创建一个对象。
2.为被调用的表达式或者函数所捕获变量创建一个副本。
3.在跳转被调用的表达式或者函数所在的地址之前,要先保护现场并记录执行地址;从被调用的表达式或者函数地址返回时,要先回复现场,并按原来保存的地址继续执行。也就是通常所说的压栈和出栈。 从上边介绍不难看出,函数调用会产生一定的时间和空间的花销。如果别调用的表达式或者函数的代码量的本身不大,而且该表达式或者函数经常被调用,那么这个时间和空间开销的损耗就不是很划算了。为了避免产生函数的调用过程,我们可以考虑直接把被调用的表达式或者函数的代码嵌入原来的执行流中,简单的说就是将代码复制粘贴过去。为了让编译器帮我们干这个复制和粘贴的活,可以通过内联函数来实现。
使用内联函数非常的简单,只要使用inline关键字修饰带函数的形参的函数即可。例如:
![](https://i-blog.csdnimg.cn/blog_migrate/5242879ed95a7efcc89892482357f1da.webp?x-image-process=image/format,png)
内联函数并不是在什么情况下都可以提高性能的,我们了解到内联函数是可以目标代码的增加为代价来节省时间开销的,因此是否可以使用内联函数的答案是;如果被调用的Lambda表达式或者函数包含大量的执行代码,那么就不应该使用内联函数;如果被调用的Lambda表达式或者函数只包含简单的执行代码(尤其是单表达式),那么就应该使用内联函数。
使用inline修饰函数之后,所有传入该函数的Lambda表达式或函数都会被内联化;如果我们又希望该函数中一个或者几个函数类型的形参不会被内联化,怎可以考虑使用noinline修饰它们。
![](https://i-blog.csdnimg.cn/blog_migrate/7813ec3b8498352fdfa139756aa675a3.webp?x-image-process=image/format,png)
前面已经提过了,在Lambda表达式中直接使用return不是用于返回该表达式,而是用于返回该表达式所在的函数(当然可以用return@ + name指定返回到哪里)但是记住:在默认情况下,在Lambda表达式中并不允许直接使用return,这是因为非内联的Lambda表达式,该Lambda表达式会额外生成一个函数对象,因此这种表达式中的return不可能用于返回它所在的函数。由于内联的Lambda表达式会被直接复制,粘贴到调用他的函数中,故可以在该Lambda表达式中可以使用return。
最后补充一下,目前内联Lambda表达式暂不支持break和continue流程控制,但是未来kotlin计划支持该功能。