`
该用户名已经存在
  • 浏览: 306488 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

谈谈Java的匿名内部类

    博客分类:
  • Java
阅读更多
在很多时候,我们需要在类的内部初始化一个静态的Map或者List,然后保存一下常量值提供给类内部方法使用。
我们通常的做法是:
首先初始化一个Map的静态变量。
然后在静态块添加常量值:
private final static Map<String, String> CONSTANT = 
    new HashMap<String, String>();
static {
	CONSTANT.put("1", "one");
	CONSTANT.put("2", "two");
}

其实还可以这么写:
private final static Map<String, String> CONSTANT = 
     new HashMap<String, String>() {
	{
		put("1", "one");
		put("2", "two");
	}
};

如果对于这种方式比较陌生,那先看一个熟悉的:
new Thread() {
    public void run() {
        System.out.println("Thread running!");
    };
}.start(); 

实际上上面这段代码的意思就是,声明一个Thread的子类并重写Thread的run()方法,然后创建一个该子类的实例然后调用其start()方法。由于声明的该Thread的子类没有名字,所以叫匿名类。又由于没有名字的类只能存在于一个类或者一个方法内部,所以又称为匿名内部类。

匿名内部类的语法也可以这么写:
Thread thread = new Thread() {
    public void run() {
        System.out.println("Thread running!");
    };
}; 
thread.start();

唯一的区别就是不是直接创建子类并调用其方法,而是声明一个该子类的父类引用thread,然后通过该父类引用调用子类方法。
创建完匿名类的实例后,没有立即执行start(),创建实例和执行实例的方法分开。

两者的区别相当于:
//1
new User().setName("Boyce Zhang");

//2
User user = new User();
user.setName("Boyce Zhang");

匿名内部类的另一个语法场景:
new Thread() {
    public void run() {
        System.out.println("Thread running!");
    };
    {
        start();
    }
};

实际上这种写法就是在匿名子类的类局部代码块中调用其类方法。
局部代码块内的语句是在创建该类的实例后由类加载器隐式立即执行的。
相当于:
public class MyThread extends Thread {
    {
        start();
    }
    public void run() {
        System.out.println("Thread running!");
    };
}

所以三种方式在执行的时刻上略微的差别之外,效果并没有太大的区别。

这样一来,前面初始化Map的方式就不难理解了:
private final static Map<String, String> CONSTANT = new HashMap<String, String>() {
    {
        put("1", "one");
        put("2", "two");
    }
};

原理就是:
声明并实例化一个HashMap的子类(子类没有重写父类HashMap的任何方法),并且在子类的类局部代码块调用父类HashMap的put()方法。
最后声明一个Map接口引用CONSTANT指向实例化的HashMap子类的实例。
根据前面的例子我们知道,类局部代码块中的put()方法调用将在HashMap的匿名子类被实例化后由类加载器隐式的执行。

其实,对于Java的任何类或接口,都可以声明一个匿名类继承或实现它。如:
//重写父类方法,局部代码块调用自己重写过的父类方法。
List<String> list = new ArrayList<String>() {
    public boolean add(String e) {
        System.out.println("Cannot add anything!");
    }
    
    //代码块的顺序在前后都无所谓,可以出现在类范围的任何位置。
    {
        add("Boyce Zhang");
    }
};

//局部代码块调用父类方法。
dao.add(new User(){
    {
        setName("Boyce Zhang");
        setAge(26);
    }
});

//重写父类方法
ThreadLocal<User> threadLocal = new ThreadLocal<User>() {
    protected String initialValue() {
        return new User("Boyce Zhang", 26);
    }
};

在匿名类的内部我们不但可以实现或重写其父类的方法。
而且也可以在其类的局部代码块中执行自己的方法或者其父类的方法。
这并不是匿名内部类的特殊语法,而是Java的语法,对于任何类都适用。

这种写法常常就是用在实例化一个类后立即执行某些方法做一些类实例的数据初始化什么的。
其作用和先实例化一个类,在使用其引用调用需要立即调用的方法是一样的,如:
Map<String, String> map = new HashMap<String, String>();
map.put("1", "one");
map.put("2", "two");

这种语法的优点就是简单,实例化一个类后立即做一些事情,比较方便。
效果有一点儿像Javascript里的即时函数一样。但是有本质的区别。
因为Javascript没有类的概念,或者说Javascript中function就是类,类就是function,所以即时函数是加载完后执行整个function。而Java的局部代码块是可以选择执行类的任何方法。

当然这种写法也有其缺点:
每一个内部类的实例都会隐性的持有一个指向外部类的引用(静态内部类除外),这样一方面是多余的引用浪费,另一方面当串行化这个子类实例时外部类也会被不知不觉的串行化,如果外部类没有实现serialize接口时,就会报错。
6
0
分享到:
评论
1 楼 yizishou 2016-11-30  
引用
另一方面当串行化这个子类实例时外部类也会被不知不觉的串行化,如果外部类没有实现serialize接口时,就会报错。

这个是怎么回事?博主有例子吗

相关推荐

    简单谈谈java中匿名内部类构造函数

    主要简单给我们介绍了java中匿名内部类构造函数,并附上了简单的示例,有需要的小伙伴可以参考下。

    Java面试题,他会给你很好的Java面试准备,第一,谈谈final, finally, finalize的区别,第二,Anonymous Inner Class (匿名内部类) 是否可以extends(继承)其它类,是否可以implements(实现)interface(接口)?

    很好的Java面试题第二,Anonymous Inner Class (匿名内部类) 是否可以extends(继承)其它类,是否可以implements(实现)interface(接口)?

    Java面试题.docx

    9、成员内部类、静态内部类、局部内部类和匿名内部类的理解,以及项目中的应用 10、string 转换成 integer的方式及原理 11-20题: 11、哪些情况下的对象会被垃圾回收机制处理掉? 12、静态代理和动态代理的区别...

    java编程常见面试题目

    第二,Anonymous Inner Class (匿名内部类) 是否可以extends(继承)其它类,是否可以implements(实现)interface(接口)?  第三,Static Nested Class 和 Inner Class的不同,  第四,&和&&的...

    变态JAVA程序员面试32问

    第二,AnonymousInnerClass(匿名内部类)是否可以 extends(继承)其它类,是否可以implements(实 现)interface(接口)? 第三,StaticNestedClass和 InnerClass 的不同,说得越多越好(面试题有的很笼统) 。 ...

    java面试笔试题大汇总+很全面.txt

     第二,Anonymous Inner Class (匿名内部类) 是否可以extends(继承)其它类,是否可以implements(实现)interface(接口)?   第三,Static Nested Class 和 Inner Class的不同,说得越多越好(面试题有的很笼统)。 ...

    Java面试经典试题及答案.doc

    第二,Anonymous Inner Class (匿名内部类) 是否可以extends(继承)其它类,是否可以implements(实现)interface(接口)? 第三,Static Nested Class 和 Inner Class的不同,说得越多越好(面试题有的很笼统)。 第四,...

    JAVA面试题最全集

    谈谈java多线程 23.谈谈文件加密技术 24.软件开发生命周期 25.路由协议种类及特点 26.java的awt和swing组件的GUI设计的关键 27.对于java流的认识 28.简单描述一下awt与swing区别。 29.简述java编程中事件处理...

    java试题,java面试大全

    关于java的面试题目 1, 谈谈final, finally, finalize的区别。 2,Anonymous Inner Class (匿名内部类) 是否可以extends(继承)其它类,是否可以implements(实现)interface(接口)?

    java程序员必知的

    第二,Anonymous Inner Class (匿名内部类) 是否可以extends(继承)其它类,是否可以implements(实现)interface(接口)?  可以继承其他类或完成其他接口,在swing编程中常用此方式。  第三,Static Nested Class ...

    java面试32题概述

    1谈谈final, finally, finalize 的区别。 2Anonymous Inner Class (匿名内部类) 是否可以extends(继承)其它类,是否可以 implements(实现)interface(接口)? 很多的面试问题都会在这里面体现出来

    java_ms.rar_Math Class_java collection

    第二,Anonymous Inner Class (匿名内部类) 是否可以extends(继承)其它类,是否可以implements(实现)interface(接口)?  第三,Static Nested Class 和 Inner Class的不同,说得越多越好(面试题有的很笼统)。  第四...

    Java面试32问

    第二,Anonymous Inner Class (匿名内部类) 是否可以extends(继承)其它类,是否可以implements(实现)interface(接口)? 第三,static Nested Class 和 Inner Class的不同,说得越多越好(面试题有的很...

    java整理试题

    第二,Anonymous Inner Class (匿名内部类) 是否可以extends(继承)其它类,是否可以implements(实现)interface(接口)? 第三,Static Nested Class 和 Inner Class的不同,说得越多越好(面试题有的很笼统)。 第四,&...

    JAVA程序员面试32问

    JAVA程序员面试32问!第一,谈谈final, finally, finalize的区别。 第二,Anonymous Inner Class (匿名内部类) 是否可以extends(继承)其它类,是否可以implements(实现)interface(接口)?

    Java面试宝典2020修订版V1.0.1.doc

    31、Anonymous Inner Class (匿名内部类) 是否可以extends(继承)其它类,是否可以implements(实现)interface(接口)? 21 33、数字转字符有多少种方式,分别是什么 22 34、Java创建对象有几种方式 22 35、写出验证...

    中软国际Java程序员笔试题及答案

    中软国际Java程序员笔试题及答案 谈谈final, finally, finalize的区别。 Anonymous Inner Class (匿名内部类) 是否可以extends(继承)其它类,是否可以implements(实现)interface(接口)?

    java九阴真经

    程序员面试之九阴真经 谈谈final, finally, finalize的...Anonymous Inner Class (匿名内部类) 是否可以extends(继承)其它类,是否可以implements(实现)interface(接口): &和&&的区别: HashMap和Hashtable的区别:

    java 面试常提到的32个问题

    第一,谈谈final, finally, finalize的...第二,Anonymous Inner Class(匿名内部类) 是否可以extends(继承)其它类,是否可以implements(实现)interface(接口)? …………………………………………………………

    JAVA面试基础试题

    第一,谈谈final, finally, finalize的区别。Anonymous Inner Class(匿名内部类) 是否可以extends(继承)其它类,是否可以implements(实现)interface(接口)?

Global site tag (gtag.js) - Google Analytics