Android里的匿名内部类

2018-08-10  本文已影响0人  睿尧

(转)https://blog.csdn.net/qq_25827845/article/details/52598319

匿名内部类也就是没有名字的内部类,正因为没有名字,所以匿名内部类只能使用一次,它通常用来简化代码编写。

但使用匿名内部类还有个前提条件:必须继承一个父类或实现一个接口

1、继承一个父类的匿名内部类实现:

abstractclassPeople{

publicabstractvoideat();

}

publicclassDemo{

publicstaticvoidmain(String[] args){

People p =newPeople() {

publicvoideat(){

System.out.println("I can eat ");

            }

        };

        p.eat();

    }

}

2、在接口上使用匿名内部类:

interfacePeople{

publicvoideat();

}

publicclassDemo{

publicstaticvoidmain(String[] args){

People p =newPeople() {

publicvoideat(){

System.out.println("I can eat ");

            }

        };

        p.eat();

    }

}

 此处 new People( )看似实例化了一个接口,事实并非如此,接口式的匿名内部类是实现了一个接口的匿名类。而且只能实现一个接口。

ps再来说一下线程创建的两种方式:

(1)继承Thread类的方式因为耦合性太强,所以一般不用。

(2)常用实现Runnable接口的创建线程方式。

但是我们更喜欢用匿名内部类的方式来创建一个线程。代码如下:

newThread(newRunnable() {

@Override

publicvoidrun(){

inti=0;

while(true){

i++;

System.out.println("this is 线程"+i);

}

}

}).start();

就这一句话就可以创建并且启动一个线程,相对来说比较方便。而且特别直观易懂。

此处的new Runnable( )并没有实例化了一个接口,切记切记!!!!

匿名内部类(Anonymous Inner Class),在创建实例的同时给出类的定义,所有这些在一个表达式中完成。

Runnable rn = newRunnable() {

    public void run() {

    }

};

相当于:

classAnomymous implementsRunnable {

    public void run() {

    }

}

Runnable rn = newAnomymous();

可以看到前者更简洁。(注意前者最后的分号不能省略,编译器把整个看作一条语句)

不过,匿名内部类仅限于只实例化一次的内部类,如果内部类需要多次实例化,通常用后者。

另外,匿名内部类要么继承一个父类,要么实现一个接口,不能两者兼有,实现接口时也不能实现多个接口。

参考示例(转)

匿名内部类也就是没有名字的内部类

正因为没有名字,所以匿名内部类只能使用一次,它通常用来简化代码编写

但使用匿名内部类还有个前提条件:必须继承一个父类或实现一个接口

实例1:不使用匿名内部类来实现抽象方法

abstract class Person {

    public abstract void eat();

}

class Child extends Person {

    public void eat() {

        System.out.println("eat something");

    }

}

public class Demo {

    public static void main(String[] args) {

        Person p = new Child();

        p.eat();

    }

}

运行结果:eat something

可以看到,我们用Child继承了Person类,然后实现了Child的一个实例,将其向上转型为Person类的引用

但是,如果此处的Child类只使用一次,那么将其编写为独立的一个类岂不是很麻烦?

这个时候就引入了匿名内部类

实例2:匿名内部类的基本实现

abstract class Person {

    public abstract void eat();

}

public class Demo {

    public static void main(String[] args) {

        Person p = new Person() {

            public void eat() {

                System.out.println("eat something");

            }

        };

        p.eat();

    }

}

运行结果:eat something

可以看到,我们直接将抽象类Person中的方法在大括号中实现了

这样便可以省略一个类的书写

并且,匿名内部类还能用于接口上

实例3:在接口上使用匿名内部类

interfacePerson {

    publicvoideat();

}

publicclassDemo {

    publicstaticvoidmain(String[] args) {

        Person p = newPerson() {

            publicvoideat() {

                System.out.println("eat something");

            }

        };

        p.eat();

    }

}

运行结果:eat something

由上面的例子可以看出,只要一个类是抽象的或是一个接口,那么其子类中的方法都可以使用匿名内部类来实现

最常用的情况就是在多线程的实现上,因为要实现多线程必须继承Thread类或是继承Runnable接口

实例4:Thread类的匿名内部类实现

publicclassDemo{

publicstaticvoidthreadMethod(String name){

Thread t =newThread(name) {

publicvoidrun(){

for(inti =1; i <=5; i++) {

System.out.print(Thread.currentThread().getName()+":"+i +"  ;  ");

                }

            }

        };

        t.start();

}

publicstaticvoidmain(String[] args){

threadMethod("线程1");

threadMethod("线程2");

    }

}

运行结果(不确定):线程1:1   ;   线程1:2   ;   线程1:3   ;   线程2:1   ;   线程1:4   ;   线程2:2   ;   线程1:5   ;   线程2:3   ;   线程2:4   ;   线程2:5   ;   

实例5:Runnable接口的匿名内部类实现

publicclassDemo{

publicstaticvoidthreadMethod(){

Runnable r =newRunnable() {

publicvoidrun(){

for(inti =1; i <=5; i++) {

System.out.print(Thread.currentThread().getName()+":"+i +"   ;   ");

                }

            }

        };

Thread t1 =newThread(r,"线程1");

Thread t2 =newThread(r,"线程2");

        t1.start();

        t2.start();

    }

publicstaticvoidmain(String[] args){

    threadMethod();

    }

}

运行结果(不确定):线程1:1   ;   线程2:1   ;   线程1:2   ;   线程2:2   ;   线程1:3   ;   线程2:3   ;   线程1:4   ;   线程2:4   ;   线程2:5   ;   线程1:5   ;   

---------------------------------------------------------------

ps:上面的继承Thread和实现Runnable是常用的

之前没写过多线程,也记一下下面的吧

packagecom;

publicclassTestRunnableimplementsRunnable{

privateinti =10;

@Override

publicvoidrun(){

System.out.println("TestRunnable class "+Thread.currentThread().getName()+":"+(--i));

}

publicstaticvoidmain(String[] args){

TestRunnable t1 =newTestRunnable();

TestRunnable t2 =newTestRunnable();

//执行出的结果的顺序不确定  变量i copy

newThread(t1).start();

newThread(t2).start();

//执行出的结果的顺序不确定  变量i 共享

/*new Thread(t1,"thread1").start();

new Thread(t1,"thread2").start();*/

}

}

上一篇下一篇

猜你喜欢

热点阅读