Java 匿名内部类

上一篇:Java 内部类

下面这个这个例子看起来会很奇怪,contents() 方法将返回值的生成与表示这个返回值的类的定义结合在一起。但是这个类是匿名的,这个类没有名字。看似你在这里要创建一个Contents 对象,但是你却在这里插入了一个类的定义。这种语法指的是:“创建一个继承自Contents 的匿名的类的对象”,contents() 方法返回的引用会被向上转型为Contents的引用。

public class Parcel {
    public Contents contents(){
        return new Contents() {
            private int num = 10;
            @Override
            public int value() {
                return num;
            }
        };
    }
    public static void main(String[] args) {

        Parcel parcel = new Parcel();
        Contents contents = parcel.contents();

        System.out.println(contents.value());
    }
}

interface Contents{
    int value();
}

输出
10

    在匿名内部类的末尾的分号 ,并不是用来标记此内部类结束的。它标记的是表达式的结束,只不过这个表达式正巧包含了匿名内部类罢了。

如果定义了一个匿名内部类,并希望它使用一个在其外部定义的对象(该对象是通过外围方法传递进来的),那么编译器会要求你只能使用该对象的引用最初始时的状态,即当你调用外部定义的对象的时候你不允许以任何形式去修改或重新定义它们 [在jdk1.8之前要求的是传入的参数必须是final 的,在jdk1.8 之后即使你不将参数定义为final 也是可以的],如果你违反了这条规定那么你将会得到一个编译时期的错误提醒。通过下面的程序我们可以看出:我们在匿名内部类试图修改i 的值,但是却报了一个错误,告诉我们i 需要是final 的,我们在匿名内部类也修改了外部类的num 这是可以的,并且在内部类修改后,外部类的num 值也发生了变化。

public class Parcel {
    private int num = 1;

    public void printNum(){
        System.out.println("num = " +num);
    }
    public Contents contents(int i){
        return new Contents() {
            @Override
            public void print() {
                // i = 100; Variable 'i' is accessed from within inner class,
                // needs to be final or effectively final
                num = 10;
                System.out.println("i = " + i +",num = " + num);
            }
        };
    }
    public static void main(String[] args) {

        Parcel parcel = new Parcel();
        Contents contents = parcel.contents(100);

        parcel.printNum();
        contents.print();
        parcel.printNum();
    }
}

interface Contents{
    void print();
}

输出
num = 1
i = 100,num = 10
num = 10

有的时候你想要在匿名内部类做类似构造器的行为应该怎么做呢?在匿名类中不可能会有命名的构造器,因为这个类根本没有名字。如果我们想要完成这样的功能,我们可以通过实例初始化的方式,就能为匿名内部类达到创建构造器的效果。就像下面这样:

public class Parcel {
    public Contents contents(int i){
        return new Contents(i) {
            @Override
            public void f() {
                System.out.println("f() run .......");
            }
        };
    }
    public static void main(String[] args) {

        Parcel parcel = new Parcel();
        Contents contents = parcel.contents(100);

        contents.f();
    }
}

abstract class Contents{        //注意这里不可以是接口,接口没有构造函数
    Contents(int i ){
        System.out.println("Constructor run ......" + i);
    }
    public abstract void f();
}

输出
Constructor run ……100
f() run …….

匿名内部类与正规的继承相比比较受限,因为匿名匿名内部类既可以扩展类,也可以实现接口,但是两者不能兼备,如果想要实现接口那也只可以实现一个接口。匿名内部类在有些时候为我们提供了很大的方便:有时候我们只是想创建一个简单的线程但是又不想大费周折创建一个类去实现Runnable 接口,这时我们使用匿名内部类就会使代码变得简单许多。

public class ThreadTest {
    public static void main(String[] args) {

        new Thread(new Runnable() {
            @Override
            public void run() {
                for(int i = 0;i < 10; i++){
                    System.out.print(i + " ");
                }
            }
        }).start();
    }
}


下一篇:Java 闭包与回调

                                                                                                            参考书籍:
                                                                                                                《Java 编程思想》Bruce Eckel 著 陈昊鹏 译

<think>嗯,用户想了解Java匿名内部类的使用场景和示例。我需要先回忆一下相关知识,然后根据提供的引用资料来组织回答。首先,匿名内部类是没有名字的局部内部类,通常用于简化代码,特别是在需要一次性实现接口或继承类的情况下。 根据引用[1],匿名内部类的语法是创建一个新的类的实例并在声明时初始化。引用[2]提到它常用于简化代码,比如事件监听器和线程实现。引用[3]中的示例展示了一个抽象类被匿名子类化的情况。引用[4]指出匿名内部类没有类名,构造参数传给父类。 接下来要分几个使用场景。首先是事件监听器,这在Swing或Android开发中很常见。然后是线程实现,用Runnable接口的例子。接着是抽象类或接口的一次性实现,比如引用[3]中的AnimalParent例子。还有比较器Comparator的使用,比如排序集合时。最后是工厂方法中返回接口的实例。 需要确保每个场景都有示例代码,并且符合用户要求的格式,比如代码块用反引号包裹,语言设为java。同时,要检查语法是否正确,比如匿名内部类的正确写法。 可能用户的问题会涉及如何选择匿名内部类与常规类,或者注意事项,比如访问外部变量需要final等。但用户主要问的是使用场景和示例,所以重点放在这上面。 最后生成相关问题,比如与Lambda表达式对比、变量访问限制、优缺点、多方法实现时的注意事项,以及替代方案。这些问题可以帮助用户进一步学习。</think>### Java匿名内部类的使用场景及示例 Java匿名内部类是一种**没有显式类名**的局部内部类,通常用于简化代码结构,特别是在需要**一次性实现接口或继承类**的场景中。以下是其核心使用场景及示例: --- #### 一、事件监听器(如Swing/AWT) 在图形界面开发中,匿名内部类常用于快速定义事件监听逻辑,无需单独创建类文件。 ```java button.addActionListener(new ActionListener() { @Override public void actionPerformed(ActionEvent e) { System.out.println("按钮被点击"); } }); ``` 此例中,直接通过匿名内部类实现了`ActionListener`接口[^2]。 --- #### 二、线程实现(Runnable接口) 创建线程时,可通过匿名内部类快速定义`Runnable`任务: ```java new Thread(new Runnable() { @Override public void run() { System.out.println("线程运行中"); } }).start(); ``` 此方式避免了单独定义实现类[^2]。 --- #### 三、抽象类/接口的一次性实现 当需要临时实现抽象类或接口时,匿名内部类可简化代码: ```java abstract class Animal { public abstract void sound(); } public class Demo { public static void main(String[] args) { Animal cat = new Animal() { @Override public void sound() { System.out.println("喵喵"); } }; cat.sound(); // 输出:喵喵 } } ``` 此例直接实例化了抽象类`Animal`并实现了其抽象方法[^3]。 --- #### 四、Comparator比较器 在集合排序时,可匿名实现`Comparator`接口: ```java List<String> list = Arrays.asList("Apple", "Banana", "Cherry"); Collections.sort(list, new Comparator<String>() { @Override public int compare(String s1, String s2) { return s1.length() - s2.length(); } }); ``` 此方法按字符串长度排序[^2]。 --- #### 五、工厂方法返回接口实例 工厂方法可通过匿名内部类返回接口的具体实现: ```java interface Logger { void log(String message); } class LoggerFactory { public static Logger createLogger() { return new Logger() { @Override public void log(String message) { System.out.println("日志:" + message); } }; } } ``` 调用`LoggerFactory.createLogger()`即可获得匿名实现的日志对象[^4]。 --- ### 注意事项 1. **访问外部变量限制**:若访问外部方法的局部变量,该变量需声明为`final`或等效不可变。 2. **代码可读性**:匿名内部类代码过长时,建议改用具名内部类。 3. **多方法实现**:若需实现多个方法,必须完整覆盖所有抽象方法。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值