成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Android Studio3支持Java8了,就問你敢用嗎

移動開發 Android
日前,Google 發布了 AS 3.0,以及一系列的 Support 包,有意思的新東西挺多,因為之前一直在看 kotlin的支持,特地翻了一下對 Java8 的支持方式,結果……

日前,Google 發布了 AS 3.0,以及一系列的 Support 包,有意思的新東西挺多,因為之前一直在看 kotlin的支持,特地翻了一下對 Java8 的支持方式,結果……

支持 Java 8

kotlin 相關的東西很早以前我就在講了,這里就不再細說了。AS3里面有一個亮眼的特性就是支持J8。首先說一下為什么以前我們不能用Java8的新特性,最主要的原因就是 lambda 語法。在 JVM 中,Java8 的語法是通過一個叫做 invokedynamic 的字節碼操作命令完成的,但是這東西在 dalvik 中并沒有,因此一直不能用。

現在AS3.0之所以能用,實際上是在新的Android Studio中加入了一個 desugar 的東西,他就類似 JVM 上的 invokedynamic ,把Java8的字節碼翻譯成 dalvik 可識別的。

官網介紹:

Android Studio provides built-in support for using certain Java 8 language features and third-party libraries that use them. As shown in figure 1, the default toolchain implements the new language features by performing bytecode transformations, called desugar, on the output of the javac compiler.

desugar 能干啥

首先看張官方圖:

Android Studio3支持Java8了,就問你敢用嗎

在 javac 執行后,desugar 會對 class 做操作,將內部的lambda相關的語法轉換為 dalvik 可識別的語法。

說的太抽象具體表現我們看代碼。

 

  1. @Override 
  2. protected void onCreate(Bundle savedInstanceState) { 
  3.     super.onCreate(savedInstanceState); 
  4.     setContentView(R.layout.activity_main); 
  5.  
  6.     new Thread(() -> { 
  7.         Log.d("kymjs""========"); 
  8.     }).start(); 

一個這樣的Java8 lambda語法的代碼被編譯以后,反編譯它,可以看到變成了這樣:

 

  1. protected void onCreate(Bundle savedInstanceState) { 
  2.     super.onCreate(savedInstanceState); 
  3.     this.setContentView(2131296283); 
  4.     (new Thread(MainActivity$$Lambda$0.$instance)).start(); 
  5.  
  6. // $FF: synthetic class 
  7. final class MainActivity$$Lambda$0 implements Runnable { 
  8.     static final Runnable $instance = new MainActivity$$Lambda$0(); 
  9.  
  10.     private MainActivity$$Lambda$0() { 
  11.     } 
  12.  
  13.     public void run() { 
  14.         MainActivity.lambda$onCreate$0$MainActivity(); 
  15.     } 

我們看到上面的代碼,在編譯后的 run() 方法內有一句 MainActivity.lambda$onCreate$0$MainActivity(); 其實這一句就是原本lambda body,他被轉換成了 MainActivity 類中的一個 static method。在最終編譯成 dex 后會再次優化,減少一次方法調用直接變成run方法的body(相當于內聯)。具體原理操作請見源碼的 visitInvokeDynamicInsn 方法: GoogleCode請自備梯子

上面的代碼演示了純函數(什么是純函數自己wiki)的操作,下面看一個非純函數的。

編譯前:

 

  1. public class MainActivity extends AppCompatActivity { 
  2.  
  3.     String mString = "hello"
  4.  
  5.     @Override 
  6.     protected void onCreate(Bundle savedInstanceState) { 
  7.         super.onCreate(savedInstanceState); 
  8.         setContentView(R.layout.activity_main); 
  9.  
  10.         new Thread(() -> { 
  11.             Log.d("kymjs""========" + mString); 
  12.         }).start(); 
  13.     } 

編譯后:

 

  1. protected void onCreate(Bundle savedInstanceState) { 
  2.     super.onCreate(savedInstanceState); 
  3.     this.setContentView(2131296283); 
  4.     (new Thread(new MainActivity$$Lambda$0(this))).start(); 
  5.  
  6. // $FF: synthetic class 
  7. final class MainActivity$$Lambda$0 implements Runnable { 
  8.     private final MainActivity arg$1; 
  9.  
  10.     MainActivity$$Lambda$0(MainActivity var1) { 
  11.         this.arg$1 = var1; 
  12.     } 
  13.  
  14.     public void run() { 
  15.         this.arg$1.lambda$onCreate$0$MainActivity(); 
  16.     } 

原本的 lambda 靜態對象不再是靜態的了;lambda 類的構造方法多了一個外部類對象的引用。因此,如果 lambda body 不是一個非純函數,是有可能會造成內存泄漏的(原因跟內部類持有外部類對象是一樣)。

方法引用

這個就實在是讓我哭笑不得了。官網標注,Method References 完全支持了,原本想到kotlin 的高階函數會有性能問題,還想看看Java8會不會有這個問題。但是我用了一下,槽點滿滿。不管是 Supplier 還是 Predicate , Function 所有的方法調用都得要最低 API24,我靠現在普遍都是兼容到14的吧,你這讓我怎么用高階方法。不過我也嘗試不考慮低版本寫了一個,看了一下效果。

首先是Java8編譯前代碼:

 

  1. public String str = "hello"
  2.  
  3. @Override 
  4. protected void onCreate(Bundle savedInstanceState) { 
  5.     super.onCreate(savedInstanceState); 
  6.     setContentView(R.layout.activity_main); 
  7.  
  8.     test(() -> str); 
  9.  
  10. public void test(Supplier<String> block) { 
  11.     System.out.println("=======" + block.get()); 

Java8編譯后,貌似換湯不換藥,只替換lambda部分,方法內依舊是普通對象方法調用:

 

  1. protected void onCreate(Bundle savedInstanceState) { 
  2.     super.onCreate(savedInstanceState); 
  3.     this.setContentView(2131296283); 
  4.     this.test(new MainActivity$$Lambda$0(this)); 
  5.  
  6. public void test(Supplier<String> block) { 
  7.     System.out.println("==========" + (String)block.get()); 
  8.  
  9. // $FF: synthetic class 
  10. final class MainActivity$$Lambda$0 implements Supplier { 
  11.     private final MainActivity arg$1; 
  12.  
  13.     MainActivity$$Lambda$0(MainActivity var1) { 
  14.         this.arg$1 = var1; 
  15.     } 
  16.  
  17.     public Object get() { 
  18.         return this.arg$1.lambda$onCreate$0$MainActivity(); 
  19.     } 

kotlin編譯前代碼:

 

  1. val str: String = "hello" 
  2.  
  3. override fun onCreate(savedInstanceState: Bundle?) { 
  4.     super.onCreate(savedInstanceState) 
  5.     setContentView(R.layout.activity_main) 
  6.     test { 
  7.         str 
  8.     } 
  9.  
  10. fun test(block: () -> String) { 
  11.     println("=========${block.invoke()}"

kotlin編譯后的代碼:

 

  1. protected void onCreate(@Nullable Bundle savedInstanceState) { 
  2.     super.onCreate(savedInstanceState); 
  3.     setContentView((int) R.layout.activity_main); 
  4.     test(new MainActivity$onCreate$1(this)); 
  5.  
  6. public final void test(@NotNull Function0<String> block) { 
  7.     Intrinsics.checkParameterIsNotNull(block, "block"); 
  8.     System.out.println("=========" + ((String) block.invoke())); 
  9.  
  10. /* compiled from: MainActivity.kt */ 
  11. final class MainActivity$onCreate$1 extends Lambda implements Function0<String> { 
  12.     final /* synthetic */ MainActivity this$0; 
  13.  
  14.     MainActivity$onCreate$1(MainActivity mainActivity) { 
  15.         this.this$0 = mainActivity; 
  16.         super(0); 
  17.     } 
  18.  
  19.     @NotNull 
  20.     public final String invoke() { 
  21.         return this.this$0.getStr(); 
  22.     } 

所以說,kotlin在實現上跟Java也依舊是一模一樣,首先生成一個類,把lambda轉換成對象,再調用這個對象的 invoke() 方法。但是別忘了,kotlin有神奇的 inline 關鍵字,就專門用來解決這種莫名其妙的多生成一大堆對象的情況。

假設給上面的 test() 方法加上 inline 關鍵字后,編譯后的代碼就變成了這樣,相當于并沒有調用test()方法,因此也就不存在多生成的 Function0 對象了:

 

  1. protected void onCreate(@Nullable Bundle savedInstanceState) { 
  2.     super.onCreate(savedInstanceState); 
  3.     setContentView((int) R.layout.activity_main); 
  4.     MainActivity this_$iv = this; 
  5.     System.out.println("=========" + this.str); 
  6.  
  7. public final void test(@NotNull Function0<String> block) { 
  8.     Intrinsics.checkParameterIsNotNull(block, "block"); 
  9.     System.out.println("=========" + ((String) block.invoke())); 

原本我是想夸一夸J8的支持,因為我最初以為他是將 lambda body 的純函數轉換成靜態方法,直接將 lambda 改成靜態方法調用來做的,結果沒想到還不如 kotlin。感覺就是 Google 為了 KPI 去加了一個 Java8 支持的噱頭。

責任編輯:未麗燕 來源: KymJS
相關推薦

2011-03-07 12:31:54

Filezilla

2012-05-16 09:27:53

Chrome瀏覽器

2010-05-13 14:08:58

Visual Stud

2012-06-21 09:37:50

Windows Pho存儲擴充

2011-06-20 10:21:29

Chrome 13

2019-02-27 16:00:28

IT資產審計

2009-04-02 08:57:23

IE8Firefox插件

2012-01-10 09:36:24

Windows 8ARM

2013-08-23 14:56:24

Windows 8.1

2021-07-19 22:20:24

微軟Windows 11Windows

2010-01-27 09:17:43

Office 2010GUP加速

2012-03-02 11:37:43

Kubuntu社區項目

2012-09-19 10:57:02

vSphere 5.1VMwareVMware View

2015-07-20 13:24:42

Windows 10SD卡

2011-03-29 14:09:00

Windows Ser藍牙

2022-02-18 18:00:00

數字人人工智能冬奧黑科技

2010-08-19 09:37:35

IE6fixed

2012-12-14 13:31:02

2011-07-19 13:39:20

iOS HTML5

2011-10-11 11:17:24

UbuntuARM
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 中文字幕视频三区 | 日本不卡免费新一二三区 | 99精品国产一区二区三区 | 日韩一区二区福利视频 | 国产99视频精品免视看9 | 久久久久久国产精品mv | 久久久久久免费毛片精品 | 伊人伊人伊人 | 成人免费在线观看 | av片网站 | 欧美国产日韩在线 | 国产精品1区2区3区 中文字幕一区二区三区四区 | 国产亚洲精品精品国产亚洲综合 | 日韩精品激情 | a级免费黄色片 | 毛片一区二区 | 亚洲一区二区视频 | 青青草一区二区三区 | 一级高清视频 | 8x国产精品视频一区二区 | 欧美成人精品在线 | 欧美日韩在线一区二区 | 农村真人裸体丰满少妇毛片 | 综合久久综合久久 | 国产福利视频 | 日日干夜夜操天天操 | 欧美精品一区二区在线观看 | 色接久久 | 久热精品在线观看视频 | 成人免费视频在线观看 | 亚洲天天 | 色av一区二区三区 | 超碰在线国产 | 看av网| 黑人精品欧美一区二区蜜桃 | 免费在线观看黄视频 | 一区欧美 | 欧美日韩国产在线 | 色爱综合网 | 台湾佬伊人 | 欧美一级在线免费观看 |