在很多时候,我们需要在类的内部初始化一个静态的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接口时,就会报错。
分享到:
相关推荐
Java中的匿名内部类是一种特殊的类,它没有名称,通常用于创建一次性的、简短的类实现,尤其是在需要扩展已有类或实现接口时。在上述代码示例中,我们看到匿名内部类与构造函数的交互。 首先,让我们了解匿名内部类...
本讲义将深入探讨Java中的匿名内部类与适配器模式,这两种概念在实际开发中都有着广泛的用途。 首先,让我们了解一下匿名内部类。匿名内部类,顾名思义,就是没有名字的类,它可以在需要使用类的地方直接定义,无需...
另外,在Java面试中,Anonymous Inner Class(匿名内部类)也是一个重要的知识点。匿名内部类是没有名字的内部类。不能extends(继承)其它类,但一个内部类可以作为一个接口,由另一个内部类实现。 在Java中,还有...
内部类分为四种类型:成员内部类、局部内部类、匿名内部类和静态内部类。它们可以访问外部类的私有属性和方法,提供了一种强大的封装机制。 - 成员内部类与外部类的成员一样,可以是静态或非静态的。 - 局部内部类...
- **匿名内部类**是在Java中创建一个未命名的内部类实例的方式。它可以直接在表达式或语句中创建而无需显式定义类。匿名内部类可以继承其他类或实现接口。 - **继承**:匿名内部类可以继承自某个特定的类。例如,...
Java中 == 和 equals 和 hashCode 的区别 int、char、long 各占多少字节数 int 和 Integer 的区别 谈谈对Java多态的理解 ...成员内部类、静态内部类、方法内部类(局部内部类)和匿名内部类的理解,以及项目中的应用
#### 二、Anonymous Inner Class (匿名内部类) 是否可以extends(继承)其它类,是否可以implements(实现)interface(接口)? **匿名内部类**是一种特殊的内部类,没有名字,只能一次性声明并实例化。 - **是否可以...
#### 第二问:AnonymousInnerClass(匿名内部类)是否可以extends(继承)其它类,是否可以implements(实现)interface(接口)? - 匿名内部类不能直接继承其他类,但它可以直接实现一个或多个接口。匿名内部类的主要用途...
此外,`this`还可以在匿名内部类和lambda表达式中使用,用来引用外部类的实例变量。这使得内部类能够访问和修改外部类的成员,即使它们具有相同的名称。 总结来说,`this`关键字在Java程序中用于明确地引用当前对象...
2. **方法引用和构造器引用**:这些特性进一步增强了lambda表达式的功能,可以直接引用类的静态方法或实例方法,甚至构造器,无需写出完整的匿名内部类。 3. **流(Stream)**:Java 8引入了流API,用于处理集合数据...
在实际应用中,尤其是在Swing编程中,匿名内部类非常常见,用于简化事件监听器的定义。 #### 第三,Static Nested Class 和 Inner Class 的不同 - **Static Nested Class**: 静态嵌套类(也称为静态内部类)是被...
4. 类型推断:Java SE 7引入了钻石操作符(),简化了匿名内部类和泛型实例化时的类型参数指定。 5. 动态语言支持:JSR 292(invokedynamic指令)的引入,使得JVM能够更好地支持动态语言,提升了Java平台的灵活性。...
在匿名类或内部类中,`this`通常指代内部类自身。如果需要访问外部类的成员,必须通过`外部类名.this`的形式来指定。比如: ```java public class A { int i = 1; public A() { Thread thread = new Thread...
至于内嵌类(Inner Class),Java提供了四种类型的内嵌类:成员内嵌类、局部内嵌类、匿名内嵌类和静态内嵌类。它们主要用于创建更复杂的数据结构,提供更好的封装,以及实现特定的功能。 综上所述,Java的面向对象...
第二,Anonymous Inner Class (匿名内部类) 是否可以extends(继承)其它类,是否可以implements(实现)interface(接口)? 可以继承其他类或完成其他接口,在swing编程中常用此方式。 第三,Static Nested Class ...
第二,Anonymous Inner Class (匿名内部类) 是否可以extends(继承)其它类,是否可以implements(实现)interface(接口)? 第三,Static Nested Class 和 Inner Class的不同,说得越多越好(面试题有的很笼统)。 第四...
这通常通过`ActionListener`接口或匿名内部类来实现。例如,添加联系人的按钮可能有一个`addActionListener`,当被点击时,它会调用预定义的方法,执行读取输入数据、验证、存储到文件等操作。 此外,为了保持数据...
- 匿名内部类可以实现接口。 - 但它不能继承其他类,只能继承抽象类或实现接口。 **49. class.forName的作用?** - `Class.forName`用于动态加载类,返回对应的`Class`对象。 **50. super.getClass()方法调用?** ...