日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

Java的動態分派和靜態分派的實現

瀏覽:118日期:2022-09-04 18:24:19

Java 方法執行時的動態分派和靜態分派是 Java 實現多態的本質

背景

Java 的動態分派和靜態分派也是 Java 方法的執行原理。 Java 源代碼的編譯之后,方法之間的調用是使用符號引用來表示的。當字節碼被 JVM 加載之后,符號引用才會被替換為對應方法在方法區的真實內存地址。那么在替換之前,由于 Java 的方法重寫、重載,就導致符號引用對應的方法可能是一個虛方法,那么方法的真實實現在運行時就可能有多個。

所以在將符號引用替換為真實地址時,還需要做一件事情:那就是確定符號引用要替換的方法的版本。

運行時方法幀

與 C,C++ 一樣,JVM 在運行時也會維護一個運行棧,用于方法的調用和返回。當調用一個方法時,會為方法在棧上分配一塊內存區域作為方法的幀。方法調用幀又分為下面幾個區域:

局部變量表

存儲方法參數和方法體中的局部變量,其容量在編譯期就已確定。容量的最小單位是 variable slot(變量槽)。靜態方法的局部變量數就是方法體中聲明的變量數;實例方法的局部變量數會多一個,多出的一個就是我們平時在實例方法中訪問的this。this 其實是編譯器在編譯時悄悄加到實例方法上的,而且是作為第一個參數。

操作數棧

JVM 的字節碼指令執行機制是基于棧的,所以需要一個棧來存儲字節碼指令的操作數。

Android 的 VM 是基于寄存器的,所以沒有操作棧區域。

Android VM 采用寄存器存儲操作數有兩個主要原因:1. 寄存器乃是 CPU 內部的高速內存, 讀寫寄存器是與 CPU 交互最快的方式。2. 智能手機多使用 ARM 架構的 CPU, ARM 架構的 CPU 有很多通用寄存器可使用。

動態鏈接

方法體中調用其他方法時,會把將要調用的方法在常量池中的符號引用,轉化為將要其在方法區內存中的開始地址信息,并儲存到動態鏈接中。

方法返回地址

一個方法執行完畢之后,線程需要值得回到哪里繼續執行,方法返回地址就是存儲這個信息的。返回地址一般就是當前方法的調用者的程序計數器的值(PC寄存器)。

正常完成出口: 方法正常返回時,如果有返回值,返回值會被壓入調用方法的操作數棧中 異常完成出口: 當方法發生了異常,且在異常表中沒有找到匹配的異常處理流程時,方法將不會有返回值

方法調用

方法調用并不等同于方法執行,方法調用階段唯一的任務就是確定被調用方法的版本(即調用哪一個方法)

調用方法的指令

有以下字節碼指令用于方法的調用:

指令 用途 說明 invokestatic 調用類的靜態方法 invokespecfical 調用對象的構造函數和私有方法 invokevirtual 調用對象的 public/protected 的方法 可能通過繼承復寫的方法稱做 virtual method: 表示要到運行時才能定位到真正的方法實現。通過符號引用確定虛方法直接引用的過程又叫做動態分派 invokeinterface 調用接口的方法 具體的實現類將在調用時確定 invokedynamic JDK1.7 為了讓 JVM 支持動態類型語言引入的指令 讓用戶可以決定如何查找目標方法

符號引用到直接引用

由于 Java 的編譯沒有C C++ 編譯過程中的鏈接階段,所以 Class 文件中儲存的只是符號引用,等到了在運行時才通過符號引用定位到方法區中方法代碼在內存布局中的位置--直接引用。符號引用到直接引用的替換又涉及兩種方式。一種是解析,另一種是分派。解析發生在類加載的解析階段,分派發生在編譯或方法調用階段。

解析

在類加載的解析階段會把滿足「編譯期可知,運行期不可變」的方法的符號引用替換為指向方法區的直接引用,不會延遲到運行時再去完成。滿足編譯期可知,運行期不可變的方法有:構造函數、私有方法、靜態方法、final修飾的方法。不滿足上述條件的方法的符號引用替換發生在方法調用期間。

分派 Dispatch

多態的實現原理

變量類型理解分派之前,需要先看兩個類型概念。比如:Object obj = new String('');

靜態類型

定義變量時,聲明的類型。比如這里 obj 的靜態類型就是 Object。靜態類型在編譯期的編譯器就能知道。

實際類型

變量賦值時的實際類型。比如這里 obj 的實際類型就是 String。實際類型在編譯期的編譯器是不可知的。

靜態分派

根據變量的「靜態類型(外觀類型)」匹配調用方法的過程稱為靜態分派。發生的場景為方法重載。如下代碼:

public class StaticDispatch { static abstract class Human { } static class Man extends Human { } static class Woman extends Human { } static class Child extends Human { } public void say(Human human) { System.out.println('human'); } public void say(Man man) { System.out.println('man'); } public void say(Woman woman) { System.out.println('woman'); } public void say(Child child) { System.out.println('child'); }}

public static void main(String[] args) { Human man = new Man(); Human woman = new Woman(); Human child = new Child(); StaticDispatch dispatch = new StaticDispatch(); dispatch.say(man); dispatch.say(woman); dispatch.say(child);}

main 方法的執行結果:

humanhumanhuman

雖然 StaticDispatch 為每種 Human 的子類都重載了一個 say 方法,但是由于重載采用的是靜態分派,是根據對象的靜態類型做方法匹配的。所以結果全都匹配到了 public void say(Human human) 方法。main 方法編譯之后的字節碼:

public static main([Ljava/lang/String;)V NEW method_invoke/StaticDispatch$Man DUP INVOKESPECIAL method_invoke/StaticDispatch$Man.<init> ()V ASTORE 1 NEW method_invoke/StaticDispatch$Woman DUP INVOKESPECIAL method_invoke/StaticDispatch$Woman.<init> ()V ASTORE 2 NEW method_invoke/StaticDispatch$Child DUP INVOKESPECIAL method_invoke/StaticDispatch$Child.<init> ()V ASTORE 3 NEW method_invoke/StaticDispatch DUP INVOKESPECIAL method_invoke/StaticDispatch.<init> ()V ASTORE 4 // 下面為調用 say ALOAD 4 ALOAD 1 INVOKEVIRTUAL method_invoke/StaticDispatch.say (Lmethod_invoke/StaticDispatch$Human;)V ALOAD 4 ALOAD 2 INVOKEVIRTUAL method_invoke/StaticDispatch.say (Lmethod_invoke/StaticDispatch$Human;)V ALOAD 4 ALOAD 3 INVOKEVIRTUAL method_invoke/StaticDispatch.say (Lmethod_invoke/StaticDispatch$Human;)V RETURN

從字節碼也能看到,編譯器確實是按照靜態分派選擇了匹配靜態類型的 StaticDispatch.say(LStaticDispatch$Human;)V 方法,而沒有按照變量的實際類型去匹配重載的方法。

public class Overload { public static void out(char a) { System.out.println('char ' + a); } public static void out(int a) {System.out.println('int ' + a);} public static void out(long a) { System.out.println('long ' + a); } public static void out(float a) { System.out.println('float ' + a); } public static void out(double a) { System.out.println('double ' + a); } public static void out(Integer a) { System.out.println('integer'); } public static void out(Character a) { System.out.println('character'); } public static void out(Serializable a) { System.out.println('serializable ' + a); } public static void out(Comparable a) { System.out.println('comparable ' + a); } public static void out(Object a) { System.out.println('object ' + a); } public static void out(char... a) { System.out.println('char ... ' + Arrays.toString(a)); } public static void main(String[] args) { out(’c’); }}

這段代碼也是一個靜態分派的例子,編譯器會選擇參數類型做合適的函數去調用。可以注釋掉所有 out 函數,留下 out(Serializable a),你會發現程序也能成功編譯和運行。如果留下Serializeable 和 Comparable 編譯則會失敗,提示對 out 的引用不明確。

動態分派

根據變量的「實際類型」匹配調用方法的過程稱為動態分派。發生的場景為方法重寫。當調用一個可能被子類重寫或繼承的方法時,就會觸發動態分派。

public class DynamicDispatch { static class Human { public void say() { System.out.println('human'); } } static class Man extends Human { @Override public void say() { System.out.println('man'); } } static class Woman extends Human { @Override public void say() { System.out.println('woman'); } }}

public static void main(String[] args) { Human human = new Human(); Human man = new Man(); Human woman = new Woman(); human.say(); man.say(); woman.say();}

main 方法的執行結果:

humanmanwoman

意料之中,所謂的多態就是這樣。那多態是如何實現的?

其實多態的實現過程也就是確定被重寫的方法版本的過程。main 方法編譯之后的字節碼:

public static main([Ljava/lang/String;)V NEW method_invoke/DynamicDispatch$Human DUP INVOKESPECIAL method_invoke/DynamicDispatch$Human.<init> ()V ASTORE 1 NEW method_invoke/DynamicDispatch$Man DUP INVOKESPECIAL method_invoke/DynamicDispatch$Man.<init> ()V ASTORE 2 NEW method_invoke/DynamicDispatch$Woman DUP INVOKESPECIAL method_invoke/DynamicDispatch$Woman.<init> ()V ASTORE 3 // 下面為多態調用 say ALOAD 1 INVOKEVIRTUAL method_invoke/DynamicDispatch$Human.say ()V ALOAD 2 INVOKEVIRTUAL method_invoke/DynamicDispatch$Human.say ()V ALOAD 3 INVOKEVIRTUAL method_invoke/DynamicDispatch$Human.say ()V RETURN

這里通過字節碼感覺都會調用Hunman#say方法的,但是運行之后并不是。

當 JVM 執行這兩行字節碼時:

ALOAD 1 // 由上面 ASTORE 1 可知, 局部變量表的第一個變量是 Woman 的對象INVOKEVIRTUAL method_invoke/DynamicDispatch$Human.say ()V// INVOKEVIRTUAL 指令就會到 Woman 類中去尋找 say 方法

調用 say 方法時,JVM 會先去當前調用的對象的類中查找是否存在和目標方法的描述符、簡單名稱一樣的方法,如果存在則將符號引用替換為找到的方法的直接引用,否則就向父類去查找,向父類的父類去查找..., 直到最后找不到拋出NoSuchMethod異常。

Human 的 say 方法的簽名:

public void say(); descriptor: ()V

Woman 的 say 方法的簽名:

public void say(); descriptor: ()V

可見 Woman 類的 Human 類中的 say 方法的描述符和簡單名稱是一樣的,所以 JVM 會優先匹配 Woman 類中的方法。這也是多態調用的底層邏輯。

到此這篇關于Java的動態分派和靜態分派的實現的文章就介紹到這了,更多相關Java 動態分派和靜態分派內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Java
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
国产精品日韩久久久| 国产精久久久| 久久九九国产| 久久久久久网| 三上悠亚国产精品一区二区三区| 亚洲精品**中文毛片| 97人人精品| 久久久久免费av| 不卡视频在线| 亚洲一区中文| 日韩在线观看一区二区| 亚洲精品国产精品粉嫩| 欧美日韩一区二区三区不卡视频| 国产精品久久久久久久久免费高清| 精品国产乱码久久久久久1区2匹| 色天使综合视频| 天堂√8在线中文| 国产精品99一区二区三| 日韩一区欧美| 亚洲在线成人| 欧美日本三区| 麻豆视频在线观看免费网站黄 | 日韩午夜av在线| 亚洲黄页一区| 亚洲综合图色| 美女久久久久久| 久久中文字幕av| 久久不射中文字幕| 欧美成人精品午夜一区二区| 亚洲伦乱视频| 一二三区精品| 精品国产鲁一鲁****| 激情久久五月| 日本麻豆一区二区三区视频| 成人精品动漫一区二区三区| 伊人影院久久| 国产激情一区| 亚洲手机在线| 欧美亚洲专区| 成人免费网站www网站高清| 免费日韩av| 欧美黄色一区| 午夜欧美精品| 国产精品激情| 国产中文一区| 欧美精品影院| 欧美成人基地| 一区二区三区网站| 国产白浆在线免费观看| 中文字幕一区日韩精品| 久久伊人国产| 亚洲神马久久| 成人台湾亚洲精品一区二区| 免费视频国产一区| 国产欧美日韩在线一区二区 | 香蕉成人av| 亚洲欧美专区| 亚洲性色av| 日韩在线电影| 久久91导航| 国产精品一区二区精品| 婷婷精品进入| 麻豆国产91在线播放| 日av在线不卡| 日韩免费在线| 国产精品国码视频| 久久精品主播| 国产精品分类| 视频一区中文字幕国产| 国产一区日韩| 五月国产精品| 国产99精品一区| 国产欧美日韩一级| 亚洲国产一区二区三区在线播放| 国产欧美成人| 亚洲一区久久| 蜜桃成人精品| 国产精品1区在线| 免费人成网站在线观看欧美高清| 欧美xxxx中国| 91精品一区| 亚洲91久久| 欧美精品导航| 蜜臀91精品一区二区三区| 在线观看精品| 久久婷婷国产| 久久福利一区| 久久久久午夜电影| 精品国产亚洲一区二区三区| 亚洲综合欧美| 欧美aa在线观看| 里番精品3d一二三区| 日本一区福利在线| 美女亚洲一区| 欧美好骚综合网| 国产情侣一区在线| 国产精品日韩久久久| 视频二区不卡| 成人在线免费观看网站| 国产精品地址| 欧美一区二区三区久久精品| 天使萌一区二区三区免费观看| 欧美男人天堂| 免费在线亚洲欧美| 欧美中文高清| 伊人久久亚洲| 天堂av在线一区| 国产午夜精品一区二区三区欧美| 国产网站在线| 精品一区二区三区在线观看视频| 日韩欧美另类中文字幕| 国产精品日本| 伊人成人在线视频| 狠狠久久婷婷| 久久国产亚洲| 日韩av免费| 国产 日韩 欧美一区| 色婷婷亚洲mv天堂mv在影片| 国产一区二区三区网| 狠狠久久伊人| 国产一区二区三区不卡视频网站 | 亚洲精品123区| 欧美不卡视频| 蜜臀av免费一区二区三区| 日韩欧美综合| 色婷婷精品视频| 午夜欧美精品久久久久久久| 国产精品普通话对白| 水野朝阳av一区二区三区| 婷婷激情图片久久| 黑丝一区二区| 丝袜亚洲另类欧美| 久久av一区| 蜜臀国产一区二区三区在线播放| 羞羞答答国产精品www一本 | 深夜日韩欧美| 日韩福利视频一区| 国产日韩精品视频一区二区三区| 国产精品亚洲综合在线观看| 国产精品videossex久久发布| 麻豆精品少妇| 国产一区调教| 美女av在线免费看| 亚洲日本网址| 香蕉人人精品| 老色鬼久久亚洲一区二区| 三级一区在线视频先锋| 亚洲永久精品唐人导航网址| 日韩va亚洲va欧美va久久| 欧美亚洲国产日韩| 麻豆久久久久久| 91亚洲一区| 91精品国产调教在线观看| 蜜桃成人av| 亚洲精品乱码| 你懂的国产精品永久在线| 国产在线观看www| 成人日韩在线| 亚洲一区日本| 欧美日韩网址| 国产欧美一区二区三区精品酒店| 欧美日韩视频网站| 香蕉精品999视频一区二区| 日韩av中文字幕一区| 久久99青青| 久久美女精品| 中文字幕av一区二区三区四区| 国产日产一区| 日韩精品看片| 亚洲精品婷婷| 国产精品99一区二区三区| 婷婷综合在线| 日韩和欧美一区二区| 久久亚洲精品中文字幕| 日本不卡免费高清视频在线| 夜夜精品视频| 美女久久99| 国产视频亚洲| 免费在线欧美黄色| 偷拍欧美精品| 麻豆精品av| 黄色成人在线网址| 日本精品在线播放| 日韩欧美二区| 婷婷精品在线| 麻豆成全视频免费观看在线看| 免费成人在线观看| 97精品视频在线看| 香蕉视频成人在线观看| 精品美女久久| 亚洲欧美在线综合| 最近高清中文在线字幕在线观看1| 激情欧美国产欧美| 国产精品一区免费在线| 欧美成人国产| 国产精品s色| 麻豆9191精品国产| 成人在线视频免费| 日韩精品一区二区三区av| 免费av一区二区三区四区|