JAVA匿名內部類(Anonymous Classes)

来源:https://www.cnblogs.com/wuhenzhidu/archive/2019/04/10/anonymous.html
-Advertisement-
Play Games

1.前言 匿名內部類在我們JAVA程式員的日常工作中經常要用到,但是很多時候也只是照本宣科地用,雖然也在用,但往往忽略了以下幾點:為什麼能這麼用?匿名內部類的語法是怎樣的?有哪些限制?因此,最近,我在完成了手頭的開發任務後,查閱了一下JAVA官方文檔,將匿名內部類的使用進行了一下總結,案例也摘自官方 ...


1.前言

匿名內部類在我們JAVA程式員的日常工作中經常要用到,但是很多時候也只是照本宣科地用,雖然也在用,但往往忽略了以下幾點:為什麼能這麼用?匿名內部類的語法是怎樣的?有哪些限制?因此,最近,我在完成了手頭的開發任務後,查閱了一下JAVA官方文檔,將匿名內部類的使用進行了一下總結,案例也摘自官方文檔。感興趣的可以查閱官方文檔(https://docs.oracle.com/javase/tutorial/java/javaOO/anonymousclasses.html)。

2.匿名內部類

匿名內部類可以使你的代碼更加簡潔,你可以在定義一個類的同時對其進行實例化。它與局部類很相似,不同的是它沒有類名,如果某個局部類你只需要用一次,那麼你就可以使用匿名內部類(Anonymous classes enable you to make your code more concise. They enable you to declare and instantiate a class at the same time. They are like local classes except that they do not have a name. Use them if you need to use a local class only once.)

本節包括以下幾個方面:

  1. 定義匿名內部類
  2. 匿名內部類的語法
  3. 訪問作用域的局部變數、定義和訪問匿名內部類成員
  4. 匿名內部類實例

2.1 定義匿名內部類

首先看下官方文檔中給的例子:

 1 public class HelloWorldAnonymousClasses {
 2 
 3     /**
 4      * 包含兩個方法的HelloWorld介面
 5      */
 6     interface HelloWorld {
 7         public void greet();
 8         public void greetSomeone(String someone);
 9     }
10 
11     public void sayHello() {
12 
13         // 1、局部類EnglishGreeting實現了HelloWorld介面
14         class EnglishGreeting implements HelloWorld {
15             String name = "world";
16             public void greet() {
17                 greetSomeone("world");
18             }
19             public void greetSomeone(String someone) {
20                 name = someone;
21                 System.out.println("Hello " + name);
22             }
23         }
24 
25         HelloWorld englishGreeting = new EnglishGreeting();
26 
27         // 2、匿名類實現HelloWorld介面
28         HelloWorld frenchGreeting = new HelloWorld() {
29             String name = "tout le monde";
30             public void greet() {
31                 greetSomeone("tout le monde");
32             }
33             public void greetSomeone(String someone) {
34                 name = someone;
35                 System.out.println("Salut " + name);
36             }
37         };
38 
39         // 3、匿名類實現HelloWorld介面
40         HelloWorld spanishGreeting = new HelloWorld() {
41             String name = "mundo";
42             public void greet() {
43                 greetSomeone("mundo");
44             }
45             public void greetSomeone(String someone) {
46                 name = someone;
47                 System.out.println("Hola, " + name);
48             }
49         };
50         
51         englishGreeting.greet();
52         frenchGreeting.greetSomeone("Fred");
53         spanishGreeting.greet();
54     }
55 
56     public static void main(String... args) {
57         HelloWorldAnonymousClasses myApp = new HelloWorldAnonymousClasses();
58         myApp.sayHello();
59     }
60 }

運行結果為:

1 Hello world
2 Salut Fred
3 Hola, mundo

該例中用局部類來初始化變數englishGreeting,用匿類來初始化變數frenchGreeting和spanishGreeting,兩種實現之間有明顯的區別:

1)局部類EnglishGreetin繼承HelloWorld介面,有自己的類名,定義完成之後需要再用new關鍵字實例化才可以使用;

2)frenchGreeting、spanishGreeting在定義的時候就實例化了,定義完了就可以直接使用;

3)匿名類是一個表達式,因此在定義的最後用分號";"結束。

2.2 匿名內部類的語法

如上文所述,匿名類是一個表達式,匿名類的語法就類似於調用一個類的構建函數(new  HelloWorld()),除些之外,還包含了一個代碼塊,在代碼塊中完成類的定義,見以下兩個實例:

案例一,實現介面的匿名類:

 1  HelloWorld frenchGreeting = new HelloWorld() {
 2    String name = "tout le monde";
 3    public void greet() {
 4          greetSomeone("tout le monde");
 5    }
 6    public void greetSomeone(String someone) {
 7         name = someone;
 8         System.out.println("Salut " + name);
 9    }
10  };

 案例二,匿名子類(繼承父類):

 1 public class AnimalTest {
 2 
 3     private final String ANIMAL = "動物";
 4 
 5     public void accessTest() {
 6         System.out.println("匿名內部類訪問其外部類方法");
 7     }
 8 
 9     class Animal {
10         private String name;
11 
12         public Animal(String name) {
13             this.name = name;
14         }
15 
16         public void printAnimalName() {
17             System.out.println(bird.name);
18         }
19     }
20 
21     // 鳥類,匿名子類,繼承自Animal類,可以覆寫父類方法
22     Animal bird = new Animal("布穀鳥") {
23 
24         @Override
25         public void printAnimalName() {
26             accessTest();           // 訪問外部類成員
27             System.out.println(ANIMAL);  // 訪問外部類final修飾的變數
28             super.printAnimalName();
29         }
30     };
31 
32     public void print() {
33         bird.printAnimalName();
34     }
35 
36     public static void main(String[] args) {
37 
38         AnimalTest animalTest = new AnimalTest();
39         animalTest.print();
40     }
41 }

運行結果:

運行結果:
匿名內部類訪問其外部類方法
動物
布穀鳥

從以上兩個實例中可知,匿名類表達式包含以下內部分:

  1. 操作符:new;
  2. 一個要實現的介面或要繼承的類,案例一中的匿名類實現了HellowWorld介面,案例二中的匿名內部類繼承了Animal父類;
  3. 一對括弧,如果是匿名子類,與實例化普通類的語法類似,如果有構造參數,要帶上構造參數;如果是實現一個介面,只需要一對空括弧即可;
  4. 一段被"{}"括起來類聲明主體;
  5. 末尾的";"號(因為匿名類的聲明是一個表達式,是語句的一部分,因此要以分號結尾)。

3.訪問作用域內的局部變數、定義和訪問匿名內部類成員

 匿名內部類與局部類對作用域內的變數擁有相同的的訪問許可權。

(1)、匿名內部類可以訪問外部內的所有成員;

(2)、匿名內部類不能訪問外部類未加final修飾的變數(註意:JDK1.8即使沒有用final修飾也可以訪問);

(3)、屬性屏蔽,與內嵌類相同,匿名內部類定義的類型(如變數)會屏蔽其作用域範圍內的其他同名類型(變數):

 案例一,內嵌類的屬性屏蔽:

 1 public class ShadowTest {
 2 
 3     public int x = 0;
 4 
 5     class FirstLevel {
 6 
 7         public int x = 1;
 8 
 9         void methodInFirstLevel(int x) {
10             System.out.println("x = " + x);
11             System.out.println("this.x = " + this.x);
12             System.out.println("ShadowTest.this.x = " + ShadowTest.this.x);
13         }
14     }
15 
16     public static void main(String... args) {
17         ShadowTest st = new ShadowTest();
18         ShadowTest.FirstLevel fl = st.new FirstLevel();
19         fl.methodInFirstLevel(23);
20     }
21 }

輸出結果為:

x = 23
this.x = 1
ShadowTest.this.x = 0

這個實例中有三個變數x:1、ShadowTest類的成員變數;2、內部類FirstLevel的成員變數;3、內部類方法methodInFirstLevel的參數。

methodInFirstLevel的參數x屏蔽了內部類FirstLevel的成員變數,因此,在該方法內部使用x時實際上是使用的是參數x,可以使用this關鍵字來指定引用是成員變數x:

 1 System.out.println("this.x = " + this.x); 

利用類名來引用其成員變數擁有最高的優先順序,不會被其他同名變數屏蔽,如:

 1 System.out.println("ShadowTest.this.x = " + ShadowTest.this.x); 

 案例二,匿名內部類的屬性屏蔽

 1 public class ShadowTest {
 2     public int x = 0;
 3 
 4     interface FirstLevel {
 5      void methodInFirstLevel(int x);
 6     }
 7 
 8     FirstLevel firstLevel =  new FirstLevel() {
 9 
10         public int x = 1;
11 
12         @Override
13         public void methodInFirstLevel(int x) {
14             System.out.println("x = " + x);
15             System.out.println("this.x = " + this.x);
16             System.out.println("ShadowTest.this.x = " + ShadowTest.this.x);
17         }
18     };
19 
20     public static void main(String... args) {
21         ShadowTest st = new ShadowTest();
22         ShadowTest.FirstLevel fl = st.firstLevel;
23         fl.methodInFirstLevel(23);
24     }
25 }

輸出結果為:

x = 23
this.x = 1
ShadowTest.this.x = 0

(4)、匿名內部類中不能定義靜態屬性、方法;  

 1 public class ShadowTest {
 2     public int x = 0;
 3 
 4     interface FirstLevel {
 5      void methodInFirstLevel(int x);
 6     }
 7 
 8     FirstLevel firstLevel =  new FirstLevel() {
 9 
10         public int x = 1;
11 
12         public static String str = "Hello World";   // 編譯報錯
13 
14         public static void aa() {        // 編譯報錯
15         }
16 
17         public static final String finalStr = "Hello World";  // 正常
18 
19         public void extraMethod() {  // 正常
20             // do something
21         }
22     };
23 }

(5)、匿名內部類可以有常量屬性(final修飾的屬性);

(6)、匿名內部內中可以定義屬性,如上面代碼中的代碼:private int x = 1;

(7)、匿名內部內中可以可以有額外的方法(父介面、類中沒有的方法);

(8)、匿名內部內中可以定義內部類;

(9)、匿名內部內中可以對其他類進行實例化。

4.匿名內部類實例

官方提供的兩個實例供大家參考:

實例一:

 1 import javafx.event.ActionEvent;
 2 import javafx.event.EventHandler;
 3 import javafx.scene.Scene;
 4 import javafx.scene.control.Button;
 5 import javafx.scene.layout.StackPane;
 6 import javafx.stage.Stage;
 7  
 8 public class HelloWorld extends Application {
 9     public static void main(String[] args) {
10         launch(args);
11     }
12     
13     @Override
14     public void start(Stage primaryStage) {
15         primaryStage.setTitle("Hello World!");
16         Button btn = new Button();
17         btn.setText("Say 'Hello World'");
18         btn.setOnAction(new EventHandler<ActionEvent>() {
19  
20             @Override
21             public void handle(ActionEvent event) {
22                 System.out.println("Hello World!");
23             }
24         });
25         
26         StackPane root = new StackPane();
27         root.getChildren().add(btn);
28         primaryStage.setScene(new Scene(root, 300, 250));
29         primaryStage.show();
30     }
31 }

實例二:

 1 import javafx.application.Application;
 2 import javafx.event.ActionEvent;
 3 import javafx.event.EventHandler;
 4 import javafx.geometry.Insets;
 5 import javafx.scene.Group;
 6 import javafx.scene.Scene;
 7 import javafx.scene.control.*;
 8 import javafx.scene.layout.GridPane;
 9 import javafx.scene.layout.HBox;
10 import javafx.stage.Stage;
11 
12 public class CustomTextFieldSample extends Application {
13     
14     final static Label label = new Label();
15  
16     @Override
17     public void start(Stage stage) {
18         Group root = new Group();
19         Scene scene = new Scene(root, 300, 150);
20         stage.setScene(scene);
21         stage.setTitle("Text Field Sample");
22  
23         GridPane grid = new GridPane();
24         grid.setPadding(new Insets(10, 10, 10, 10));
25         grid.setVgap(5);
26         grid.setHgap(5);
27  
28         scene.setRoot(grid);
29         final Label dollar = new Label("$");
30         GridPane.setConstraints(dollar, 0, 0);
31         grid.getChildren().add(dollar);
32         
33         final TextField sum = new TextField() {
34             @Override
35             public void replaceText(int start, int end, String text) {
36                 if (!text.matches("[a-z, A-Z]")) {
37                     super.replaceText(start, end, text);                     
38                 }
39                 label.setText("Enter a numeric value");
40             }
41  
42             @Override
43             public void replaceSelection(String text) {
44                 if (!text.matches("[a-z, A-Z]")) {
45                     super.replaceSelection(text);
46                 }
47             }
48         };
49  
50         sum.setPromptText("Enter the total");
51         sum.setPrefColumnCount(10);
52         GridPane.setConstraints(sum, 1, 0);
53         grid.getChildren().add(sum);
54         
55         Button submit = new Button("Submit");
56         GridPane.setConstraints(submit, 2, 0);
57         grid.getChildren().add(submit);
58         
59         submit.setOnAction(new EventHandler<ActionEvent>() {
60             @Override
61             public void handle(ActionEvent e) {
62                 label.setText(null);
63             }
64         });
65         
66         GridPane.setConstraints(label, 0, 1);
67         GridPane.setColumnSpan(label, 3);
68         grid.getChildren().add(label);
69         
70         scene.setRoot(grid);
71         stage.show();
72     }
73  
74     public static void main(String[] args) {
75         launch(args);
76     }
77 }

寫在最後:

這篇文章是我在閱讀官方文檔的同時加以自己的理解整理出來的,可能受英文原版的影響,有些地方表達得不准確或是不清楚還希望讀者能夠指正。另外,體會到了那些翻譯英文技術書的人確實不容易,英文的文章看上去意思都很清楚,但是想要再用中文表述出來卻不那麼容易。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一、什麼是外觀模式 定義:為子系統中的一組介面提供一個一致的界面,用來訪問子系統中的一群介面。 外觀模式組成: Facade:負責子系統的的封裝調用 Subsystem Classes:具體的子系統,實現由外觀模式Facade對象來調用的具體任務 二、外觀模式的使用場景 1、設計初期階段,應該有意識 ...
  • 前言: 相信很多人都聽過一個問題:把大象關進冰箱門,需要幾步? 第一,把冰箱門打開;第二,把大象放進去;第三,把冰箱門關上。我們可以看見,這個問題的答案回答的很有步驟。接下來我們介紹一種設計模式——模板方法模式,你會發現,它與這個問題的答案實際上有很多共同之處。 一、定義 定義一個演算法骨架,允許子類 ...
  • 簡介 CAT 是一個實時和接近全量的監控系統,它側重於對Java應用的監控,基本接入了美團上海所有核心應用。目前在中間件(MVC、RPC、資料庫、緩存等)框架中得到廣泛應用,為美團各業務線提供系統的性能指標、健康狀況、監控告警等。 優勢 實時處理:信息的價值會隨時間銳減,尤其是事故處理過程中。 全量 ...
  • 想著想著結果還是過了時間! 這個月的進度堪比之前幾年! 我自己的游戲用UE4寫的風聲水起! 美術的朋友給我做了好多動作! 我都想著以後弄個個人眾籌什麼的了,可見是進展不錯! 公司的事最近不太忙,反而是結婚的事越來越忙,戒指已經買完了,50分的也算不小了,1克拉的她也嫌貴! 前倆禮拜還順便把MACOS ...
  • 好玩圖像PIL處理 一、PIL庫學習總結 1、PIL中的模塊 Image模塊、ImageChops模塊、ImageCrackCode模塊、ImageDraw模塊、ImageEnhance模塊、ImageFile模塊、ImageFileIO模塊、ImageFilter模塊、ImageFont模塊、Im ...
  • # 從決定學習編程語言到正式做出計劃擠出空餘時間,歷經一年半,因工作原因及生活原因不斷擱淺,從湖北到浙江再回湖北,暫時穩定在一家小公司,從日常加班中壓縮時間學習,於此記錄學習進度、學習問題,在此過程中望前輩們不吝指教,自己也會堅持住,希望能早日做到技術實現,在此領域不斷成長! 至此,人生苦短,我用p ...
  • 死磕 java集合之ConcurrentHashMap源碼分析(三) ConcurrentHashMap查詢是否也加鎖? ConcurrentHashMap有哪些值得我們學習的技術? ConcurrentHashMap有哪些不能解決的問題? ...
  • 一.項目結構 二.計算模塊和界面模塊間的調用 三.頁面演示效果 四.遇到問題 五.知識點記錄 六.新知識新技能 七.未實現但計劃實現功能 八.學習目標與總結 項目地址:https://git.dev.tencent.com/hey_wuqw/webArithmetic.git 一.項目結構 二. 計 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...