Java中的匿名内部类:由setOnClickListener说起

在初学Android的时候,总是看到这样一段代码:

 Button button = (Button) findViewById(R.id.button);
 button.setOnClickListener(new View.OnClickListener() {

            @Override
            public void onClick(View v) {
                Toast.makeText(MainActivity.this, "onClick", Toast.LENGTH_SHORT).show();
            }
        });

相信大家对上面这段代码都不陌生,确切的说是再熟悉不过了,咱从学习Android的第一天起就已经开始写这种代码了。没错,确实我们在初学Android的时候就已经开始使用匿名内部类了,但是随着使用的深入,会发现越来越多的问题,例如,为什么这个类里面可以随意使用MainActivity(外部类)里面的所有成员变量,即使是private的变量?为什么当需要一个context的时候却要写MainActivity.this这样的形式?为什么在申明这个内部类的方法里定义的局部变量需要定义为final才能在内部类里面使用?别急,在下为大家逐个解释。
要解释这些问题,首先需要从java的内部类说起(本文不打算详细介绍这个语法,想要详细了解的同学可以求助度娘)
java中的内部类一言以蔽之就是,申明在一个类或者方法里面的类(new onClickListener()就是申明在一个类里面的内部类,同时还是匿名的),最典型的内部类申明如下:

class Outer {
    private String name = "Outer";
    class Inner {
        public void print() {
            System.out.println("Class Name is: " + name);
        }
    }
}

上述代码中,类Inner申明在类Outer内部,我们称Inner为Outer的内部类,使用内部类好处多多,例如内部类可以直接使用外部类中的所有成员变量和方法,便于代码管理维护,内部类可以用于隐藏实现细节,可以用于实现类的多重继承…(多了就不写了,当时我在学习的时候看到这么多内部类的使用方法感觉云里雾里,直到后来代码见多了才逐渐理解,所以建议大家刚开始不要想着把所有知识点全部搞清楚,先在自己脑中有点印象,以后在编码或读源码的过程中遇到后再逐步去理解,这样跟着实例去理解不仅具体,而且深刻)。
现在大家清楚了吧,为什么onClickListener这个类里面可以使用MainActivity这个类中的所有成员变量和方法,等等,刚刚不是说可以使用所有成员变量么,那为什么会有MainActivity.this这种奇葩的写法。好吧,刚刚没说清楚,大家有没有想过如果内部类中有同名的变量或者方法的时候怎么处理,很简单就是加上外部类的类名+.然后再写上变量名,例如刚刚那个例子,如果内部类中也有一个同名的变量name,则可以这么写:

class Outer {
    private String name = "Outer";
    class Inner {
        private String name = "Inner";
        public void print() {
            System.out.println("InnerClass Name is: " + name);
            System.out.println("OuterClass Name is: " + Outer.name);
        }
    }
}

这也就解释了为什么要写成MainActivity.this这种形式了,因为内部类onClickListener这个类里面已经有一个同名变量this了,如果想要使用外部类MainActivity的this变量,则需要使用MainActivity.this。
好了,内部类的知识先讲到这里(更多详细用法如果大家真的等不及可以自行百度,不过我真的建议大家在使用过程中逐步去发现内部类的使用实例,自己去理解它的用法),下面再说说匿名内部类。
如果一个内部类在并不是显式得使用class关键字来申明,即没有给该内部类一个名字,我们把这种类叫做匿名的内部类,匿名内部类可以用来实例化一个类或者接口,最常见的匿名内部类当属多线程中的Runnable接口:

final int count = 100;
        new Thread(new Runnable() {
            @Override
            public void run() {
                for(int i = 0; i < count; i++) {
                    System.out.println("current num is:" + i);
                }
            }
        }).start();

相信大家对上面这段代码都不会陌生,简直是太熟悉了啊,基本在使用多线程的时候就能见到(当然不排除你是继承Thread类来做的,那我就没办法了)。这里面的Thread以及Runnable其实都是一种匿名内部类。细心的朋友会注意到,这里count被申明为了final,这是怎么回事,我不申明为final难道不行吗,成,行不行咱先试试,把final去掉,我去,编译器怎么报错了!可以看到,编译器不允许在匿名内部类中使用非final类型的变量,这是什么原因呢?大家有没有想过,在申明内部类的时候,例如上述代码,在执行start之后,该局部方法就返回了,那么作为局部变量的count生命周期结束后会被回收,这个时候thread里面的for循环还在执行,此时再引用局部变量count就会变得很奇怪了吧,明明都回收了还怎么使用。好吧,在java中,编译器是这样处理内部类的:如果这个外部局部变量是常量,则在内部类代码中直接用这个常量。如果是类的实例,则编译器将产生一个内部类的构造参数,将这个final变量传到内部类里,这样即使外部局部变量无效了,还可以使用。
好了,那么关于匿名内部类,就先说到这里,我们在编写代码的时候不光要做到知其然,还要做到知其所以然,希望这篇文章能对大家有所帮助。

你可能感兴趣的:(android,匿名内部类)