kotlin开发者大会部分总结

2019-10-08  本文已影响0人  android老男孩

一.kotlin代码简化

println(1f.dp())

比如扩展post

fun Activity?.Main(todo:() -> unit){
     Handler().post {
          todo()
     }
}

fun Activity?.Worker(todo:() -> unit){
     Thread {
          todo()
     }.start()
}

class MainActivity : AppCompatActivity(){
        ...
        Worker{
              Main{

              }
        }
}
BufferedReader bufferedReader = null;
        try {
            bufferedReader = new BufferedReader(new FileReader(new File("readme.md")));
        } catch (IOException ex){
            
        } finally {
            try {
                if(bufferedReader != null){
                   bufferedReader.close();
                }
            } catch (IOException ignore){

            }
        }

优化为
File("readme.md").readLines().foreach(::println)

reified修饰的泛型与java不兼容,必须用inline修饰

startActivity<MainActivity>()
inline fun <reified T:Activity> Activity?.startActivity(){
    this?.startActivity(Intent(this,T::class.java))
}

泛型

kotlin的协变与逆变

协变只能出现在返回值中,逆变只能出现在方法的参数中,协变是out,逆变是in
比如现在有两个类,子类是苹果,父类是水果

具体分为使用处与声明处

java java示例 kotlin示例 kotlin
上界通配符 <? extends Number> <out Number> 使用处协变
下界通配符 <? super Integer > <in Int> 使用处逆变
interface Collection<out E> 声明处协变
interface Comparable<in T> 声明处逆变
//只要有一个满足即成立
val resultAny = list.any {
   it / 2  == 1
}
//所有条件满足即成立
val resultAll = list.all{
     it > 0
}
var time = System.currentTimeMillis()
val list = (1..65535).toList().map {
      it * 2
}.filter {
      it % 3 == 0
}
list.first()
println(System.currentTimeMillis() - time)//29ms

var time = System.currentTimeMillis()
val sequence = (1..65535).asSequence().map {
      it * 2
}.filter {
      it % 3 == 0
}
sequence.first()
println(System.currentTimeMillis() - time)//7ms

函数

kotlin可以将函数转换成一个值,变量,这个变量的类型就是函数

fun Int.sample(a:float,b:double) :Long = this * (a+b).toLong()
//变量 :接收者 (参数) ->返回值 整个作为这个变量的函数类型 = 真正的函数赋值
val function:Int.(Float,Double) -> Long = Int::sample
//这个函数值一定要满足前边的函数类型
fun main() = sample(function)
//这个变量就可以作为参数传入别的函数中
fun sample(a:Int.(Float,Double) -> Long):Long = 3.a(1f,2.0)
//a是传入函数的变量名,可以直接使用a来调用这个函数

函数类型

open class Fruit
class Apple : Fruit()
open class Dog {
     open fun bark():String = "汪汪"
}

class Jinmao : Dog {
    open fun bark():String = "金毛汪汪"
}

fun parent(apple:Apple):Dog{
      println(apple)
      return Dog()
}

fun child(fruit:Fruit):Dog{
      println(fruit)
      return Jinmao()
}

var functionValue :  (Apple) -> Dog = ::parent
fun main(){
      val apple = Apple()
      var dog = functionValue(apple)
      dog.bark() //输出汪汪
      
      functionValue = ::child
      dog = functionValue(apple)
      dog.bark() //输出金毛汪汪
  
      // 编译报错,因为functionValue函数是(Apple) -> Dog类型
      // 即使被child赋值但是它的类型仍然是 (Apple) -> Dog
      val fruit = Fruit()
      functionValue(fruit)
}

那么child为什么可以对functionValue赋值成功

反编译代码会发现

 @NotNull
   private Function1 functionValue = (Function1)(new Function1((Test4)this) {
      // $FF: synthetic method
      // $FF: bridge method
      public Object invoke(Object var1) {
         return this.invoke((Test4.Apple)var1);
      }

      @NotNull
      public final Test4.Dog invoke(@NotNull Test4.Apple p1) {
         Intrinsics.checkParameterIsNotNull(p1, "p1");
         return ((Test4)this.receiver).parent(p1);
      }
      ...

这里用到了Function1接口,后边会详解讲这个接口,这里的in与out对应了前边的协变与逆变,这就是为什么调用不行,而赋值可以成功,因为函数调用在编译阶段是不会涉及到协变逆变的,只是检查调用类型

这里是in就是apple,out就是dog
所以可以理解为apple是fruit的父类,所以child可以赋值成功

public interface Function1<in P1, out R> : Function<R> {
    /** Invokes the function with the specified argument. */
    public operator fun invoke(p1: P1): R
}

lamda表达式实现原理

屏幕快照 2019-09-04 下午12.22.13.png

两种实现的性能

二.性能

inline关键字

扩展函数,或者一些函数式参数建议用inline,其它函数不建议
因为虽然方法的出栈弹栈有性能消耗,但是很小,不是业务开发者需要考虑的问题,把所有的方法都inline到一个方法中显然是不合理,会造成一定的浪费
但是扩展函数,函数参数有一些lamda的实现是用匿名类实现的,会创建多余的对象,所以建议使用inline函数

//加inline关键字的方法,代码直接被编译到调用处
int times = 1000000
int index = 0
for (int i = times;index < i;index++){
     count  = index;
}
//未加inline关键字的方法,创建了lamda对象,然后调用原来的方法
Function lamda = new MyInlineKt$lamda();
noinlineRepeat(1000000,lamda)

inline的场景

其它的一些性能注意

 dataBean?.data?.title1?.toInt()
//反编译
if (dataBean != null) {
         Data var10000 = dataBean.getData();
         if (var10000 != null) {
            String var4 = var10000.getTitle1();
            if (var4 != null) {
               String var2 = var4;
               boolean var3 = false;
               Integer.parseInt(var2);
            }
         }
      }

如果代码有10行,那么kotlin会对每行都判null
但在java中有时我们会写这样的代码,其实后边就都不会执行了

if (dataBean == null) {
    return;
}

总结就是大部分一些kotlin有,java没有的特性,都会有一些实现成本,但不绝对

参考

https://kotlin.gdgbeijing.org/
https://www.jianshu.com/p/f1405bd19dea

上一篇 下一篇

猜你喜欢

热点阅读