美文网首页
Kotlin学习第2篇:函数的定义与调用

Kotlin学习第2篇:函数的定义与调用

作者: tanbiheng | 来源:发表于2018-03-29 10:38 被阅读0次

创建集合

Set:

val set = hashSetOf(1, 7, 53)
val set = setOf(1, 7, 53)

List Map:

val list = arrayListOf(1, 7, 53)
val map = hashMapOf(1 to "one", 7 to "seven", 53 to "fifty-three")

注:list(set).lastOf()函数可获取集合最后一个元素,如果集合泛型为整形可以通过list(set).max()获取数字集合中最大的值

默认参数值

fun<T> joinToString(collection: Collection<T>,
                        separator: String = ",",
                        prefix: String = "",
                        postfix String = "") : String 

//separator prefix postfix 都设置了默认值  
//现在可以用所有参数来调用这个函数或者省略掉部分参数
joinToString(list, "", "", "");
joinToString(list);
joinToString(list, ";");

注:java普遍存在的一个问题是,一些类的重载函数是在太多了,在kotlin中,可以在声明函数的时候,指定参数的默认值,设置默认值的参数可以传也可省略,这样就可以避免创建重载函数;当使用常规调用语法时,必须按照函数声明中定义的参数顺序来给定参数,可以省略的也只有排在末尾的参数,如果使用命名参数,可以省略中间的一些参数:

joinToString(list, prefix = ";", postfix = "#");

消除静态工具类:顶层函数和属性

顶层函数:

// java
package strings;
public class JoinKt{
    public static String joinToString(){}
}
//使用
import strings.JoinKt
...
JoinKt.joinToString();


// kotlin
@file:JvmName("StringFunctions")  // 注解指定类名
package strings;                        // 包名的声明跟在文件注解后

fun joinToString() : String {}
// 使用
import strings.StringFunctions;
...
StringFunctions.joinToString();

注:java编译时会生成类,JVM只能执行类中的代码,kotlin编译生成类的名称对应于包含函数的文件的名称,因此要改变包含kotlin顶层函数生成的类的名称,需要为这个文件添加@JvmName的注解,将其放到这个文件的开头,位于包名的前面

顶层属性:

var opCount = 0

fun performOperation(){
    opCount++;
    ...
}

fun reportOperationCount(){
    println("Operation performed $opCount times")
}

注:和函数一样,属性也可以放到文件的顶层,默认情况下,顶层属性和其他任意属性一样,是通过访问器暴露给java使用的(val只有getter,var对应一对getter和setter)

给别人的类添加方法:扩展函数和属性

扩展函数:

package strings

fun String.lastChar(): Char = this.get(this.length - 1)
// 简化版
fun String.lastChar(): Char = get(length - 1)

// 使用
println("kotlin".lastChar())

注:在扩展函数中,可以像其他成员函数一样用this,也可像普通成员函数一样省略它;扩展函数可以直接访问被扩展的类的其他方法和属性,但不能访问私有或者受保护的成员

导入和扩展函数

import strings.lastChar
// import strings.*

var c = "kotlin".lastChar()

//可以使用关键字as来修改导入的类或者函数名称
import strings.lastChar as last

var c = "kotlin".last() 

不可重写的扩展函数

// 重写成员函数
open class View{
    open fun click() {
        println("view click")
    }
}

open class Button: View() {
    override fun click() {
        println("button click")
    }
}

val view : View = Button()
view.click()

>>> button click

// 扩展函数 
// 假设给view添加了一个扩展函数showOff()
fun View.showOff() = println("view showOff")
fun Button.showOff() = println("button showOff")

val view: View = Button()
view.showOff()
>>> view showOff

注:如你所见,扩展函数不存在重写,因为kotlin会把它们当作静态函数对待;如果一个类的成员函数和扩展函数有相同的签名,成员函数往往会被优先使用

扩展属性:

val String.lastChar: Char
    get() = get(length - 1)
    
var String.lastChar: Char
    get() = get(length - 1)
    set(value : Char){
        this.setCharAt(length - 1, value)
    }

注:当你需要从java中访问扩展属性的时候,应该显式的调用它的getter函数

处理集合:可变参数

可变参数:

val list = listOf(1, 2, 3)

fun listOf<T>(vararg values: T): List<T>{...}

注:java中可变参数用的是三个点,kotlin则是在参数上使用vararg修饰符;kotlin和java的另一个区别是,当需要传递的参数已经包装在数组中时,在java中可以原样传递数组,而kotlin则要求你显式的解包数组,以便每个数组元素在函数中能做为单独的参数来调用,这个功能被称为展开运算符,使用的时候,是在对应的参数前面放一个*:

fun main(args: Array<String>){
    val list = listOf("args:", *args)
    println(list)
}

局部函数和扩展

class User(val id: Int, val name: String, val address: String)

// 带重复代码的函数
fun savaUser(user: User){
    if(user.name.isEmpty()){
        throw Exception("empty name")
    }
    
    if(user.address.isEmpty()){
        throw Exception("empty address")
    }
    
    // 保存用户
    ...
}

// 提取局部函数避免重复
fun savaUser(user: User){
    fun validate(value: String, fieldName: String)
    if(value.isEmpty()){
        throw Exception("empty $fieldName")
    }
    
    validate(user.name)
    validate(user.address)
    // 保存用户
    ...
}

写在最后

  • 学习参考资料来源于《Kotlin实战》

相关文章

网友评论

      本文标题:Kotlin学习第2篇:函数的定义与调用

      本文链接:https://www.haomeiwen.com/subject/aofmcftx.html