一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - Java教程 - Java 轉型(向上或向下轉型)詳解及簡單實例

Java 轉型(向上或向下轉型)詳解及簡單實例

2020-08-20 11:22Java之家 Java教程

這篇文章主要介紹了Java 轉型(向上或向下轉型)詳解及簡單實例的相關資料,需要的朋友可以參考下

Java編程中經常碰到類型轉換,對象類型轉換主要包括向上轉型和向下轉型。

向上轉型

我們在現實中常常這樣說:這個人會唱歌。在這里,我們并不關心這個人是黑人還是白人,是成人還是小孩,也就是說我們更傾向于使用抽象概念“人”。再例如,麻雀是鳥類的一種(鳥類的子類),而鳥類則是動物中的一種(動物的子類)。我們現實中也經常這樣說:麻雀是鳥。這兩種說法實際上就是所謂的向上轉型,通俗地說就是子類轉型成父類。這也符合Java提倡的面向抽象編程思想。來看下面的代碼:

?
1
2
3
4
5
6
package a.b;
public class A {
public void a1() {
    System.out.println("Superclass");
}
}

A的子類B:

?
1
2
3
4
5
6
7
package a.b;
public class B extends A {
public void a1() {
    System.out.println("Childrenclass"); //覆蓋父類方法
}
    public void b1(){} //B類定義了自己的新方法
}

C類:

?
1
2
3
4
5
6
7
package a.b;
public class C {
public static void main(String[] args) {
    A a = new B(); //向上轉型
    a.a1();
}
}

如果運行C,輸出的是Superclass 還是Childrenclass?不是你原來預期的Superclass,而是Childrenclass。這是因為a實際上指向的是一個子類對象。當然,你不用擔心,Java虛擬機會自動準確地識別出究竟該調用哪個具體的方法。不過,由于向上轉型,a對象會遺失和父類不同的方法,例如b1()。有人可能會提出疑問:這不是多此一舉嗎?我們完全可以這樣寫:

?
1
2
B a = new B();
a.a1();

確實如此!但這樣就喪失了面向抽象的編程特色,降低了可擴展性。其實,不僅僅如此,向上轉型還可以減輕編程工作量。來看下面的顯示器類Monitor:

?
1
2
3
4
5
package a.b;
public class Monitor{
public void displayText() {}
public void displayGraphics() {}
}

液晶顯示器類LCDMonitor是Monitor的子類:

?
1
2
3
4
5
6
7
8
9
package a.b;
public class LCDMonitor extends Monitor {
public void displayText() {
    System.out.println("LCD display text");
}
public void displayGraphics() {
    System.out.println("LCD display graphics");
}
}

陰極射線管顯示器類CRTMonitor自然也是Monitor的子類:

?
1
2
3
4
5
6
7
8
9
package a.b;
public class CRTMonitor extends Monitor {
public void displayText() {
    System.out.println("CRT display text");
}
public void displayGraphics() {
    System.out.println("CRT display graphics");
}
}

等離子顯示器PlasmaMonitor也是Monitor的子類:

?
1
2
3
4
5
6
7
8
9
package a.b;
public class PlasmaMonitor extends Monitor {
public void displayText() {
    System.out.println("Plasma display text");
}
public void displayGraphics() {
    System.out.println("Plasma display graphics");
}
}

現在有一個MyMonitor類。假設沒有向上轉型,MyMonitor類代碼如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
package a.b;
public class MyMonitor {
public static void main(String[] args) {
    run(new LCDMonitor());
    run(new CRTMonitor());
    run(new PlasmaMonitor());
}
public static void run(LCDMonitor monitor) {
    monitor.displayText();
    monitor.displayGraphics();
}
public static void run(CRTMonitor monitor) {
    monitor.displayText();
    monitor.displayGraphics();
}
public static void run(PlasmaMonitor monitor) {
    monitor.displayText();
    monitor.displayGraphics();
}
}

可能你已經意識到上述代碼有很多重復代碼,而且也不易維護。有了向上轉型,代碼可以更為簡潔:

?
1
2
3
4
5
6
7
8
9
10
11
12
package a.b;
public class MyMonitor {
public static void main(String[] args) {
    run(new LCDMonitor());           //向上轉型
    run(new CRTMonitor());           //向上轉型
    run(new PlasmaMonitor());      //向上轉型
}
public static void run(Monitor monitor) { //父類實例作為參數
    monitor.displayText();
    monitor.displayGraphics();
}
}

我們也可以采用接口的方式,例如:

?
1
2
3
4
5
package a.b;
public interface Monitor {
abstract void displayText();
abstract void displayGraphics();
}

將液晶顯示器類LCDMonitor稍作修改:

?
1
2
3
4
5
6
7
8
9
package a.b;
public class LCDMonitor implements Monitor {
public void displayText() {
    System.out.println("LCD display text");
}
public void displayGraphics() {
    System.out.println("LCD display graphics");
}
}

CRTMonitor、PlasmaMonitor類的修改方法與LCDMonitor類似,而MyMonitor可以不不作任何修改。

可以看出,向上轉型體現了類的多態性,增強了程序的簡潔性。

向下轉型

子類轉型成父類是向上轉型,反過來說,父類轉型成子類就是向下轉型。但是,向下轉型可能會帶來一些問題:我們可以說麻雀是鳥,但不能說鳥就是麻雀。來看下面的例子:

A類:

?
1
2
3
4
5
6
package a.b;
public class A {
void aMthod() {
    System.out.println("A method");
}
}

A的子類B:

?
1
2
3
4
5
6
7
8
9
package a.b;
public class B extends A {
void bMethod1() {
    System.out.println("B method 1");
}
void bMethod2() {
    System.out.println("B method 2");
}
}

C類:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
package a.b;
public class C {
   public static void main(String[] args) {
      A a1 = new B(); // 向上轉型
      a1.aMthod();  // 調用父類aMthod(),a1遺失B類方法bMethod1()、bMethod2()
      B b1 = (B) a1; // 向下轉型,編譯無錯誤,運行時無錯誤
      b1.aMthod();  // 調用父類A方法
      b1.bMethod1(); // 調用B類方法
      b1.bMethod2(); // 調用B類方法
      A a2 = new A();
      B b2 = (B) a2; // 向下轉型,編譯無錯誤,運行時將出錯
      b2.aMthod();
      b2.bMethod1();
      b2.bMethod2();
   }
}

從上面的代碼我們可以得出這樣一個結論:向下轉型需要使用強制轉換。運行C程序,控制臺將輸出:

?
1
2
3
4
5
6
Exception in thread "main" java.lang.ClassCastException: a.b.A cannot be cast to a.b.B at
        a.b.C.main(C.java:14)
A method
A method
B method 1
B method 2

其實黑體部分的向下轉型代碼后的注釋已經提示你將發生運行時錯誤。為什么前一句向下轉型代碼可以,而后一句代碼卻出錯?這是因為a1指向一個子類B的對象,所以子類B的實例對象b1當然也可以指向a1。而a2是一個父類對象,子類對象b2不能指向父類對象a2。那么如何避免在執行向下轉型時發生運行時ClassCastException異常?使用5.7.7節學過的instanceof就可以了。我們修改一下C類的代碼:

?
1
2
3
4
5
6
7
A a2 = new A();
if (a2 instanceof B) {
B b2 = (B) a2;
b2.aMthod();
b2.bMethod1();
b2.bMethod2();
}

這樣處理后,就不用擔心類型轉換時發生ClassCastException異常了。

感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!

原文鏈接:http://blog.csdn.net/qq_35101189/article/details/57417933

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 国产一区二区三区在线看 | 91在线亚洲综合在线 | 高h射尿| 国产三级精品三级男人的天堂 | 二区三区视频 | 国产图片综合区 | 亚洲国产99在线精品一区二区 | 天堂网在线网站成人午夜网站 | 精品国产欧美一区二区 | 亚洲色图影院 | 精品福利一区 | 日韩欧美国产一区二区三区 | 国产资源免费观看 | 日本高清视频网站www | 美女艹b | xxxxxx性受 | 免费激情小视频 | 亚洲成A人片在线观看中文L | 欧美成人v视频免费看 | 成人性色生活片免费网 | 粉嫩国产14xxxxx0000 | 日韩伦理在线观看 | 扒开老师挠尿口到崩溃刑罚 | 成人在线观看视频免费 | 午夜一级 | 日本黄a三级三级三级 | 亚洲国产在线观看免费视频 | 成人免费一区二区三区在线观看 | 久久久无码精品亚洲A片猫咪 | 欧美综合精品一区二区三区 | 国产精品四虎在线观看免费 | 暴露狂婷婷医院暴露tx | 男人和女人上床 | yy111111免费观看 | 夫妇交换小说全文阅读 | 秋葵污视频 | 无人在线视频高清免费观看动漫 | 奇米影视久久777中文字幕 | 99久久国产综合精品网成人影院 | 农村妇女野战bbxxx | 亚洲 欧美 偷自乱 图片 |