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.)

本节包括以下几个方面:

定义匿名内部类
匿名内部类的语法
访问作用域的局部变量、定义和访问匿名内部类成员
匿名内部类实例

2.1 定义匿名内部类

首先看下官方文档中给的例子:
复制代码

public class HelloWorldAnonymousClasses {/*** 包含两个方法的HelloWorld接口*/interface HelloWorld {public void greet();public void greetSomeone(String someone);}public void sayHello() {// 1、局部类EnglishGreeting实现了HelloWorld接口class EnglishGreeting implements HelloWorld {String name = "world";public void greet() {greetSomeone("world");}public void greetSomeone(String someone) {name = someone;System.out.println("Hello " + name);}}HelloWorld englishGreeting = new EnglishGreeting();// 2、匿名类实现HelloWorld接口HelloWorld frenchGreeting = new HelloWorld() {String name = "tout le monde";public void greet() {greetSomeone("tout le monde");}public void greetSomeone(String someone) {name = someone;System.out.println("Salut " + name);}};// 3、匿名类实现HelloWorld接口HelloWorld spanishGreeting = new HelloWorld() {String name = "mundo";public void greet() {greetSomeone("mundo");}public void greetSomeone(String someone) {name = someone;System.out.println("Hola, " + name);}};englishGreeting.greet();frenchGreeting.greetSomeone("Fred");spanishGreeting.greet();}public static void main(String... args) {HelloWorldAnonymousClasses myApp = new HelloWorldAnonymousClasses();myApp.sayHello();}

}

运行结果为:
Hello world
Salut Fred
Hola, mundo

该例中用局部类来初始化变量englishGreeting,用匿类来初始化变量frenchGreeting和spanishGreeting,两种实现之间有明显的区别:

1)局部类EnglishGreetin继承HelloWorld接口,有自己的类名,定义完成之后需要再用new关键字实例化才可以使用;

2)frenchGreeting、spanishGreeting在定义的时候就实例化了,定义完了就可以直接使用;

3)匿名类是一个表达式,因此在定义的最后用分号";"结束。
2.2 匿名内部类的语法

如上文所述,匿名类是一个表达式,匿名类的语法就类似于调用一个类的构建函数(new HelloWorld()),除些之外,还包含了一个代码块,在代码块中完成类的定义,见以下两个实例:

案例一,实现接口的匿名类:

HelloWorld frenchGreeting = new HelloWorld() {String name = "tout le monde";public void greet() {greetSomeone("tout le monde");}public void greetSomeone(String someone) {name = someone;System.out.println("Salut " + name);}
};

案例二,匿名子类(继承父类):
复制代码

public class AnimalTest {private final String ANIMAL = "动物";public void accessTest() {System.out.println("匿名内部类访问其外部类方法");}class Animal {private String name;public Animal(String name) {this.name = name;}public void printAnimalName() {System.out.println(bird.name);}}// 鸟类,匿名子类,继承自Animal类,可以覆写父类方法Animal bird = new Animal("布谷鸟") {@Overridepublic void printAnimalName() {accessTest();           // 访问外部类成员System.out.println(ANIMAL);  // 访问外部类final修饰的变量super.printAnimalName();}};public void print() {bird.printAnimalName();}public static void main(String[] args) {AnimalTest animalTest = new AnimalTest();animalTest.print();}
}运行结果:匿名内部类访问其外部类方法动物布谷鸟

从以上两个实例中可知,匿名类表达式包含以下内部分:

操作符:new;
一个要实现的接口或要继承的类,案例一中的匿名类实现了HellowWorld接口,案例二中的匿名内部类继承了Animal父类;
一对括号,如果是匿名子类,与实例化普通类的语法类似,如果有构造参数,要带上构造参数;如果是实现一个接口,只需要一对空括号即可;
一段被"{}"括起来类声明主体;
末尾的";"号(因为匿名类的声明是一个表达式,是语句的一部分,因此要以分号结尾)。

3.访问作用域内的局部变量、定义和访问匿名内部类成员

匿名内部类与局部类对作用域内的变量拥有相同的的访问权限。

(1)、匿名内部类可以访问外部内的所有成员;

(2)、匿名内部类不能访问外部类未加final修饰的变量(注意:JDK1.8即使没有用final修饰也可以访问);

(3)、属性屏蔽,与内嵌类相同,匿名内部类定义的类型(如变量)会屏蔽其作用域范围内的其他同名类型(变量):

案例一,内嵌类的属性屏蔽:

public class ShadowTest {public int x = 0;class FirstLevel {public int x = 1;void methodInFirstLevel(int x) {System.out.println("x = " + x);System.out.println("this.x = " + this.x);System.out.println("ShadowTest.this.x = " + ShadowTest.this.x);}}public static void main(String... args) {ShadowTest st = new ShadowTest();ShadowTest.FirstLevel fl = st.new FirstLevel();fl.methodInFirstLevel(23);}}输出结果为:x = 23this.x = 1ShadowTest.this.x = 0

这个实例中有三个变量x:1、ShadowTest类的成员变量;2、内部类FirstLevel的成员变量;3、内部类方法methodInFirstLevel的参数。

methodInFirstLevel的参数x屏蔽了内部类FirstLevel的成员变量,因此,在该方法内部使用x时实际上是使用的是参数x,可以使用this关键字来指定引用是成员变量x:

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

利用类名来引用其成员变量拥有最高的优先级,不会被其他同名变量屏蔽,如:

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

案例二,匿名内部类的属性屏蔽:

public class ShadowTest {public int x = 0;interface FirstLevel {void methodInFirstLevel(int x);}FirstLevel firstLevel =  new FirstLevel() {public int x = 1;@Overridepublic void methodInFirstLevel(int x) {System.out.println("x = " + x);System.out.println("this.x = " + this.x);System.out.println("ShadowTest.this.x = " + ShadowTest.this.x);}};public static void main(String... args) {ShadowTest st = new ShadowTest();ShadowTest.FirstLevel fl = st.firstLevel;fl.methodInFirstLevel(23);}}输出结果为:
x = 23
this.x = 1
ShadowTest.this.x = 0

(4)、匿名内部类中不能定义静态属性、方法;  
复制代码

public class ShadowTest {public int x = 0;interface FirstLevel {void methodInFirstLevel(int x);}FirstLevel firstLevel =  new FirstLevel() {public int x = 1;public static String str = "Hello World";   // 编译报错public static void aa() {        // 编译报错}public static final String finalStr = "Hello World";  // 正常public void extraMethod() {  // 正常// do something}}; }

(5)、匿名内部类可以有常量属性(final修饰的属性);

(6)、匿名内部内中可以定义属性,如上面代码中的代码:private int x = 1;

(7)、匿名内部内中可以可以有额外的方法(父接口、类中没有的方法);

(8)、匿名内部内中可以定义内部类;

(9)、匿名内部内中可以对其他类进行实例化。
4.匿名内部类实例

官方提供的两个实例供大家参考:

实例一:

import javafx.event.ActionEvent;
import javafx.event.EventHandler;
import javafx.scene.Scene;
import javafx.scene.control.Button;
import javafx.scene.layout.StackPane;
import javafx.stage.Stage;public class HelloWorld extends Application {public static void main(String[] args) {launch(args);}@Overridepublic void start(Stage primaryStage) {primaryStage.setTitle("Hello World!");Button btn = new Button();btn.setText("Say 'Hello World'");btn.setOnAction(new EventHandler<ActionEvent>() {@Overridepublic void handle(ActionEvent event) {System.out.println("Hello World!");}});StackPane root = new StackPane();root.getChildren().add(btn);primaryStage.setScene(new Scene(root, 300, 250));primaryStage.show();} }

实例二:

import javafx.application.Application;
import javafx.event.ActionEvent;
import javafx.event.EventHandler;
import javafx.geometry.Insets;
import javafx.scene.Group;
import javafx.scene.Scene;
import javafx.scene.control.*;
import javafx.scene.layout.GridPane;
import javafx.scene.layout.HBox;
import javafx.stage.Stage;public class CustomTextFieldSample extends Application {final static Label label = new Label();@Overridepublic void start(Stage stage) {Group root = new Group();Scene scene = new Scene(root, 300, 150);stage.setScene(scene);stage.setTitle("Text Field Sample");GridPane grid = new GridPane();grid.setPadding(new Insets(10, 10, 10, 10));grid.setVgap(5);grid.setHgap(5);scene.setRoot(grid);final Label dollar = new Label("$");GridPane.setConstraints(dollar, 0, 0);grid.getChildren().add(dollar);final TextField sum = new TextField() {@Overridepublic void replaceText(int start, int end, String text) {if (!text.matches("[a-z, A-Z]")) {super.replaceText(start, end, text);                     }label.setText("Enter a numeric value");}@Overridepublic void replaceSelection(String text) {if (!text.matches("[a-z, A-Z]")) {super.replaceSelection(text);}}};sum.setPromptText("Enter the total");sum.setPrefColumnCount(10);GridPane.setConstraints(sum, 1, 0);grid.getChildren().add(sum);Button submit = new Button("Submit");GridPane.setConstraints(submit, 2, 0);grid.getChildren().add(submit);submit.setOnAction(new EventHandler<ActionEvent>() {@Overridepublic void handle(ActionEvent e) {label.setText(null);}});GridPane.setConstraints(label, 0, 1);GridPane.setColumnSpan(label, 3);grid.getChildren().add(label);scene.setRoot(grid);stage.show();}public static void main(String[] args) {launch(args);} }

JAVA匿名内部类(Anonymous Classes)相关推荐

  1. Lambda and Anonymous Classes

    Lambda 简写的依据 能够使用 Lambda 的依据是必须有相应的函数接口(函数接口,是指内部只有一个抽象方法的接口),也就是说你并不能在代码的任何地方任性的写Lambda表达式.Lambda表达 ...

  2. 关于JAVA匿名内部类,回调,事件模式的一点讨论

    分享一下我老师大神的人工智能教程!零基础,通俗易懂!http://blog.csdn.net/jiangjunshow 也欢迎大家转载本篇文章.分享知识,造福人民,实现我们中华民族伟大复兴! 关于JA ...

  3. 去除残缺条目java_为什么说Java匿名内部类是残缺的闭包

    为什么说Java匿名内部类是残缺的闭包 https://blog.csdn.net/hzy38324/article/details/77986095 前言 我们先来看一道很简单的小题: public ...

  4. 为什么说Java匿名内部类是残缺的闭包

    为什么说Java匿名内部类是残缺的闭包 https://blog.csdn.net/hzy38324/article/details/77986095 前言 我们先来看一道很简单的小题: public ...

  5. java 匿名内部类 百科_java匿名内部类具体概念是什么,在什么地方用到?

    展开全部 java匿名内部类一定是在new的后面,用其隐含实现一个接口或实现一个类,没有类名,根据多态,我们e69da5e887aa62616964757a686964616f313333376138 ...

  6. Java匿名内部类里为什么能用外部变量

    2019独角兽企业重金招聘Python工程师标准>>> 偶然间想到一个问题,Java匿名内部类里为什么能用外部变量?Java到底在背后做了什么呢: final List<Int ...

  7. java 中报错 ~[classes/:na]

    java 中报错 ~[classes/:na] java.lang.NullPointerException: nullat org.asu.pro.bean.Hr.getAuthorities(Hr ...

  8. java匿名内部类,什么是匿名内部类,如何定义匿名内部类,如何使用匿名内部类?

    java匿名内部类 什么是匿名内部类? 匿名内部类的使用场景? 匿名内部类如何创建? 匿名内部类的实现和使用 例1(实现接口) 例2(继承类) 什么是匿名内部类? 匿名内部类,顾名思义,就是不知道这个 ...

  9. Java匿名内部类的用法(简单教学)

    Java匿名内部类笔记 public class Test{public static void main(String[] args){/*语法*///Coo o1 = new Coo(); //创 ...

最新文章

  1. python 字典 get(),items(),iteritems()方法
  2. 【STM32】DMA相关函数和类型
  3. AT3877-[ARC089C]GraphXY【构造】
  4. Linux编程练习 --进程间通信2--两个管道实现双向通信
  5. 30 PP配置-生产车间控制-定义计划策略
  6. InVEST model中生境质量
  7. ionic获取手机通讯录
  8. form图片上传遇到错误
  9. 130. 被围绕的区域
  10. 工欲善其事必先利其器——网络工程师必备六款软件(收藏下载❤️)
  11. 漏斗模型被营销大师们玩烂了
  12. 2.高速PCB设计规范(一)
  13. 四、RocketMq本地集群搭建:多master-slaver异步
  14. 【本/硕毕业生】开题报告及论文指导思路和撰写事项的整理
  15. iTween基础之Move(移动)
  16. 第2章 HFDS的Shell操作
  17. EasyExcel解析动态表头及导出
  18. 如何让iPad变成Mac的扩展屏幕
  19. 【Redis】RedisCluster 批量删除key
  20. 每日一道Leetcode - 剑指 Offer 14- I. 剪绳子【动态规划】

热门文章

  1. 关河因果:钓鱼城引擎技术概述
  2. 邦纳光电传感器QS18VN6R
  3. mtk android 编译过程,MTK android 快速编译方法.doc
  4. CRM是什么?CRM管理系统主要功能,作用,特点分别有哪些?
  5. 我眼中未来的计算机资料,我心目中未来的计算机
  6. 全加器门级(结构级),数据流,行为级实现
  7. 如何在Windows 10上安装PHP 8
  8. 3.7 命令注入攻击
  9. 究竟是量化模型重要还是人重要?
  10. PMP 考点 第十三章 项目相关方管理