美文网首页
Kotlin—基础第三弹

Kotlin—基础第三弹

作者: Lypop | 来源:发表于2017-10-31 13:13 被阅读0次

    Kotlin是一个基于JVM的新的编程语言,由JetBrains开发,由于是Google推荐使用的,可见其重要性,当你真正深入的时候也会发现kotlin的优雅,让你在写代码的时候省时又省力

    对于Kotlin的学习网上有好多,现在分享一些
    官方网址 http://kotlinlang.org/docs/reference/
    别人的总结 https://github.com/youxin11544/Kotlin-learning

    在这里不会像官网那样一一的去讲解,自己也是从官网去学习然后去看别人的博客,对于知识而言可能并不深入,但对于初学者而言也值的一看

    Start

    kotlin委托机制

    委托模式也是代理模式是软件设计模式中的一项基本技巧,在委托模式中,有两个对象参与处理同一个请求,接受请求的对象将请求委托给另一个对象来处理

    在java和Android用到委托机制在AOP里,在java中使用继承InvocationHandler接口的方法来实现代理,而在Android中则使用AspectJ的方式来实现。在这里我们来讲解一下Kotlin的委托模式

    1. 类委托
      也就是上面讲的委托模式定义,然而在Kotlin中实现起来更加的方便更加的灵活,如:

       class MyList(list: ArrayList<String>) : Collection<String> by list {
      
       }
      

    这种也是替代继承的方式,直接使用by关键字就将Collection的方法委托给list对象

    1. 属性委托
      在Kotlin中Delegates类内置了三种委托的方法,分别是lazy、notNull()、observable("")、vetoable("")、Map形式
    • lazy 用于进行懒加载,即第一次使用的时候才执行初始化操作

           //当使用的时候才进行初始化(线程安全的)
           val list: ArrayList<String> by lazy {
               ArrayList<String>()
           }
      
    • notNull() 使用于那些无法再初始化阶段就确定属性值的场合

           //初始化str,用于为空属性进行初始化的场景
           var str: String by Delegates.notNull<String>()
      
    • observable() 观察者和java的观察者模式差不多

           //oldValue是变化之前的值  newValue最新的值
           var str: String by Delegates.observable(""){//初始值
               property, oldValue, newValue ->
               println("${property.name}  old=$oldValue  new=$newValue")
           }   
      
    • vetoable()带条件的委托机制,通过返回的true和false来确定oldValue是上一个还是最初的

           //返回true和observable是一样的效果
           var str1: String by Delegates.vetoable(""){
               property, oldValue, newValue ->
               println("${property.name}  old=$oldValue  new=$newValue")
               true   
           }
           //返回false则标志oldValue永远为初始值
           var str2: String by Delegates.vetoable(""){
               property, oldValue, newValue ->
               println("${property.name}  old=$oldValue  new=$newValue")
               false
           }
      
    • Map 通过一种全新的赋值方法给类属性进行赋值

           class User(val map: Map<String,Any?>){
               val name: String by map
               val age: Int by map
           }
           
           val user = User(mapOf("a" to 1))
      

    除此之外我们也可以自定义委托,具体定义可仿照notNull()委托

    private class NotNullVar<T: Any>() : ReadWriteProperty<Any?, T> {
        private var value: T? = null
    
        public override fun getValue(thisRef: Any?, property: KProperty<*>): T {
            return value ?: throw IllegalStateException("Property ${property.name} should be initialized before get.")
        }
    
        public override fun setValue(thisRef: Any?, property: KProperty<*>, value: T) {
            this.value = value
        }
    }
    

    你也可以设置可读的委托机制,继承ReadOnlyProperty类即可

    kotlin lambda表达式

    首先先给出一个小例子

    val action = {i: Int, j: Int -> println("i=$i   j=$j")};
    
    action(1,2)
    

    注意的是lambda表达式用{}包裹,箭头前面是参数的参数,箭头后面是函数体;从调用上来说比java简单并且优雅了许多,但是实际用途呢?我们可以为一个函数传一个lambda,如:

    fun lambdaTest(arg1: Int, arg2: Int, oper: (a: Int, b: Int) -> Int): Int {
        var result: Int = Int.MIN_VALUE
        if (arg2 != 0) {
            result = oper(arg1, arg2)
        }
        return result
    }
    
    fun main(args: Array<String>) {
        lambdaTest(1, 2, { a: Int, b: Int -> a / b })
    

    既然是传函数当然也可以传递一个函数进去,常使用::,这里延伸出函数的引用,另外还有属性的引用,这里

    listOf(1, 2, 3).forEach(::println)
    
    var name = "1"
    val p = ::name
    p.set/get
    

    当最后一个参数为lambda的时候,可以将大括号放在外面,如:

    lambdaTest(1, 2){ a: Int, b: Int -> a / b }
    

    用lambda表达式的情况还是挺多的,例如在集合中遍历等操作

    val list = arrayListOf(1, 2, 3)
    list.forEach { value -> }
    

    kotlin 运算符重载表达式

    class Rmb(var num: Int) {
        operator fun plus(rmb: Any?) {
    
        }
    }
    

    这样就可以使用Rmb(1) + Rmb(2)进行计算了,是不是很简单。这种操作和前面讲的中缀表达式一样,只不过中间的是运算符了而已

    kotlin 注解与反射

    Kotlin的注解和java是类似的,先确定注解的类型,然后确定生命周期

    @Target(AnnotationTarget.FIELD,
            AnnotationTarget.CLASS,
            AnnotationTarget.FUNCTION)
    @Retention(AnnotationRetention.SOURCE)
    annotation class Path
    

    使用的话直接在class前 fun前面

    @Path class UsePath(val p: String) {
    
    }
    

    Kotlin反射则通过

    String::class.java
    String.javaClass
    

    这两种方式来得到Class<?>对象,进一步可反射得到相应的属性和方法

    至此基础第三部分就结束了,坚持下去,总会有些回报的。

    相关文章

      网友评论

          本文标题:Kotlin—基础第三弹

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