详解Java内部类——匿名内部类

  今天来看看另一个更加神奇的类——匿名内部类。

  就像它的名字表示的那样,这个类是匿名的,用完之后,深藏功与名,就像扫地僧那样默默潜藏于深山之中。匿名内部类不仅没有名字,连class关键字都省掉了,而且匿名内部类必须继承于某个类或者实现某个接口,长的就像这样:

new 父类(参数列表)|实现接口() {
     //匿名内部类的内部定义
    } 

  来看一个栗子:

public abstract class Human {
  public abstract void walk();
}

  这是一个抽象类,如果使用匿名内部类来继承的话是这样的:

public class AnonymousTest {
  public static void main(String[] args) {
    Human human = new Human(){
      public void walk(){
        System.out.println("AnonymousHuman can walk.");
      };
    };
    human.walk();
  }
}

  简单粗暴,看起来就像局部内部类的简化版。如果不使用匿名内部类,会是怎样呢?

  我们需要先创建一个类来继承这抽象类:

public class Man extends Human {
  @Override
  public void walk() {
    System.out.println("Man can walk.");
  }
}

  然后再来使用这个类:

public class AnonymousTest {
  public static void main(String[] args) {
    Human human = new Man();
    human.walk();
  }
}

  因为一个单独的类往往放在一个单独的文件中,如果这个类只需要创建一个对象,那未免有些大材小用了,从上面的栗子可以比较出匿名内部类的一个优势:在类只需要创建一个对象的情况下更加简单方便。

  再举一个实际一点的栗子:

public class AnonymousTest {
  public static void main(String[] args) {
    Thread t = new Thread() {
      public void run() {
        for (int i = 0; i < 10; i++) {
          try {
            sleep(1000);
          } catch (InterruptedException e) {
            e.printStackTrace();
          }
          System.out.println(i);
        }
      }
    };
    t.start();
  }
}

  这里创建了一个继承于Thread的匿名内部类,覆盖了其中的 run方法,并创建了一个实例返回给了t,然后再调用run方法,可以看到,匿名内部类只能存在一个实例对象,因为new过一次就无法再创建了,也许会觉得局部内部类已经很局限了,为什么要出现比局部内部类适用范围更小的匿名内部类?、

  这你就不懂了吧,在Java的实际使用中,匿名内部类大有用处,为什么要使用匿名内部类呢?

  有时候,我们创建的类只需要一个实例,比如说在多线程中,要使用多线程,一般先继承Thread类或者实现Runnable接口,然后再去调用它的方法,而每个任务一般都不一样,每次都新建一个类显然会很难管理,因为每个类只用一次就丢掉了,这个时候使用匿名内部类就很方便了,不仅不需要管理一堆一次性类,而且创建起来简单粗暴。就像上述栗子,还能简化成这样:

public class AnonymousTest {
  public static void main(String[] args) {
    new Thread() {
      public void run() {
        for (int i = 0; i < 10; i++) {
          try {
            sleep(1000);
          } catch (InterruptedException e) {
            e.printStackTrace();
          }
          System.out.println(i);
        }
      }
    }.start();
  }
}

  创建实例后直接调用run方法,简单粗暴。

  匿名内部类不仅可以继承于类,也可以实现于接口,比如说这样:

public class AnonymousTest {
  public static void main(String[] args) {
    new Thread(new Runnable() {
      public void run() {
        for (int i = 0; i < 10; i++) {
          try {
            sleep(1000);
          } catch (InterruptedException e) {
            e.printStackTrace();
          }
          System.out.println(i);
        }
      }
    }).start();
  }
}

  当然,还有些不得不用内部类的情况,类只能继承于一个类,如果一个类需要使用到另一个包中的另一个类的一个protected方法,却已经继承于另一个类,那么这个时候就不得不用内部类来解决了。

  比如说,还有一个Woman(女人)类:

public class Woman {
  protected void dance(){
    System.out.println("Woman can dance.");
  }
}

  这个时候,如果Man(男人)也难不住寂寞,想要dance(跳舞)一下,那该怎么办呢?继承Woman类?显然不合乎逻辑,而且也无法实现,因为已经继承于Human类了,但就是想要dance,该怎么办?

  内部类的出现让这个问题变得很简单:

public class Man extends Human {
  @Override
  public void walk() {
    System.out.println("Man can walk.");
  }

  public void dance(){
    new Woman(){
      public void manDance(){
        super.dance();
      }
    }.manDance();
  }
}

  因为在不同的包下,不能直接使用Woman的dance方法,但是可以用内部类来继承,从而调用protected方法,然后再放入Man的方法中,这样,Man也能像Woman一样dance了:

public class AnonymousTest {
  public static void main(String[] args) {
    Man human = new Man();
    human.walk();
    human.dance();
  }
}

  当然,使用匿名内部类还是有很多限制的:

  1、匿名内部类必须是继承一个类或者实现一个接口,但是两者不可兼得,同时也只能继承一个类或者实现一个接口。

2、匿名内部类不能定义构造函数。

3、匿名内部类中不能存在任何的静态成员变量和静态方法。

4、匿名内部类是特殊的局部内部类,所以局部内部类的所有限制同样对匿名内部类生效。

5、匿名内部类不能是抽象的,它必须要实现继承的类或者实现的接口的所有抽象方法。

  那么问题来了,怎样初始化一个匿名内部类呢?毕竟匿名内部类是不能有构造器的。

  当然,首先,还是可以使用初始化块来实现的,就像这样:

public class AnonymousTest {
  public static void main(String[] args) {
    Human human = new Human() {
      private String name;
      {
        name = "human";
      }
      @Override
      public void walk() {
        System.out.println(name + " walk.");
      }
    };
    human.walk();
  }
}

  但是这样显然就比较呆板,不够灵活,无法接受外部参数,那么怎样灵活使用呢?不要心急,方法总比问题多,还是有办法解决的:

public class AnonymousTest {
  public static void main(String[] args) {
    Human human = new AnonymousTest().getHumanInstance("Frank");
    human.walk();
  }

  public Human getHumanInstance(final String name){
    return new Human() {
      private String nameA;
      {
        nameA = name;
      }
      @Override
      public void walk() {
        System.out.println(nameA + " walk.");
      }
    };
  }
}

  这里利用初始化块来对匿名内部类进行初始化,注意,如果匿名内部类需要使用外部的参数或者变量,那么必须使用final修饰,因为内部类使用的其实是参数的拷贝,并不是参数本身,为了更明显的表明参数不可变,编译器会要求使用final关键字来修饰需要使用的变量。

  至此,匿名内部类讲解完毕,欢迎大家继续关注!

以上就是详解Java内部类——匿名内部类的详细内容,更多关于Java 匿名内部类的资料请关注我们其它相关文章!

(0)

相关推荐

  • Java内部类和匿名内部类的用法说明

    一.内部类: (1)内部类的同名方法 内部类可以调用外部类的方法,如果内部类有同名方法必须使用"OuterClass.this.MethodName()"格式调用(其中OuterClass与MethodName换成实际外部类名及其方法:this为关键字,表示对外部类的引用):若内部类无同名方法可以直接调用外部类的方法. 但外围类无法直接调用内部类的private方法,外部类同样无法直接调用其它类的private方法.注意:内部类直接使用外部类的方法与该方法的权限与是否static无关,

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

    先看看下面的代码能不能编译通过: public static void main(String[] args) { List l1 = new ArrayList(); List l2 = new ArrayList(){}; List l3 = new ArrayList(){{}}; System.out.println(l1.getClass() == l2.getClass() ); System.out.println(l2.getClass() == l3.getClass() );

  • 老生常谈 java匿名内部类

    匿名内部类: 1.匿名内部类其实就是内部类的简写格式. 2.定义匿名内部类的前提: 内部类必须是继承一个类或者实现接口. 3.匿名内部类的格式:  new 父类或者接口(){定义子类的内容} 4.其实匿名内部类就是一个匿名子类对象.而且这个对象有点胖.    可以理解为带内容的对象. 5.匿名内部类中定义的方法最好不要超过3个. abstract class AbsDemo { abstract void show(); } class Outer { int x = 3; /* class I

  • java中匿名内部类详解

    java匿名内部类: 1:匿名内部类,匿名内部类也就是没有名字的内部类. 2:匿名内部类的作用 正因为没有名字,所以匿名内部类只能使用一次,它通常用来简化代码编写. 3:匿名内部类的实现 匿名内部类的两种实现方式:第一种,继承一个类,重写其方法:第二种,实现一个接口(可以是多个),实现其方法. 4:匿名内部类的创建 匿名类是不能有名称的类,所以没办法引用它们.必须在创建时,作为new语句的一部分来声明它们. package com.mianshi.test; /** * 类名称:Anonymou

  • Java中的匿名内部类小结

    java内部类分为: 成员内部类.静态嵌套类.方法内部类.匿名内部类 . 在java的世界里,提供了匿名内部类语法,用于帮助大家简化代码,本文简要从接口,抽象类以及常规类以代码的形式描述其常用模式. 1. 接口模式 public interface IWriter { void write(); } public static void main(String[] args) { IWriter writer = new IWriter() { @Override public void wri

  • Java Lambda表达式与匿名内部类的联系和区别实例分析

    本文实例讲述了Java Lambda表达式与匿名内部类的联系和区别.分享给大家供大家参考,具体如下: 一 点睛 Lambda表达式与匿名内部类存在如下相同点: Lambda表达式与匿名内部类一样,都可以直接访问"effectively final"的局部变量,以及外部类的成员变量(包括实例变量和类变量). Lambda表达式创建的对象与匿名内部类生成的对象一样, 都可以直接调用从接口继承得到的默认方法. Lambda表达式与匿名内部类主要存在如下区别: 匿名内部类可以为任意接口创建实例

  • java匿名内部类实例代码详解

    这篇文章主要介绍了java匿名内部类实例代码详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 Person.java package insof; public class Person extends Object{ String name; static int age; public Person() { this.name = "tom"; System.out.println("执行的是构造方法");

  • Java匿名内部类原理与用法详解

    本文实例讲述了Java匿名内部类原理与用法.分享给大家供大家参考,具体如下: 一 点睛 匿名内部类适合创建那种只需要一次使用的类,定义匿名内部类的语法格式如下: new 父类构造器(实参列表) | 实现接口() { //匿名内部类的类体部分 } 从上面的定义可以看出,匿名内部类必须继承一个父类,或实现一个接口,但最多只能继承一个父类,或实现一个接口. 两条规则. 匿名内部类不能是抽象类. 匿名内部类不能定义构造器.由于匿名内部类没有类名,所以无法定义构造器,但匿名内部类可以初始化块,可以通过初始

  • Java为什么匿名内部类参数引用需要用final进行修饰?

    事实上,除了匿名内部类参数,方法和作用域内的内部类内部使用的外部变量也必须是final 的.原因大致总结一下: 简单解释就是: 方法中的局部变量的生命周期很短,方法结束后变量就要被销毁,加上final是为了延长变量的生命周期. 进一步解释: 内部类通常都含有回调,引用那个匿名内部类的函数执行完了就没了,所以内部类中引用外面的局部变量需要是final的,这样在回调的时候才能找到那个变量,而如果是外围类的成员变量就不需要是final的,因为内部类本身都会含有一个外围了的引用(外围类.this),所以

  • Java匿名对象与匿名内部类

    匿名对象:没有名字的对象. 非匿名对象: ClassName c=new ClassName(); c.run(); 匿名对象: new ClassName().run(); 注意事项: 1.当对象对方法仅进行一次调用的时候,就可以简化成匿名对象. 2.两个匿名对象不可能是同一个对象. 3.一般不给匿名对象赋予属性值,因为永远无法获取到. 4.运行一次,直接就被回收掉了,节省内存空间. 匿名对象使用的代码例子: public class Anony{ int a=1; int b=2; void

  • JAVA匿名内部类语法分析及实例详解

    1.前言 匿名内部类在我们JAVA程序员的日常工作中经常要用到,但是很多时候也只是照本宣科地用,虽然也在用,但往往忽略了以下几点:为什么能这么用?匿名内部类的语法是怎样的?有哪些限制?因此,最近,我在完成了手头的开发任务后,查阅了一下JAVA官方文档,将匿名内部类的使用进行了一下总结,案例也摘自官方文档.感兴趣的可以查阅官方文档(https://docs.oracle.com/javase/tutorial/java/javaOO/anonymousclasses.html). 2.匿名内部类

  • Java匿名类,匿名内部类实例分析

    本文实例讲述了Java匿名类,匿名内部类.分享给大家供大家参考,具体如下: 本文内容: 内部类 匿名类 首发日期 :2018-03-25 内部类: 在一个类中定义另一个类,这样定义的类称为内部类.[包含内部类的类可以称为内部类的外部类] 如果想要通过一个类来使用另一个类,可以定义为内部类.[比如苹果手机类,苹果手机类中的黄金版的是特别定制的] 内部类的外部类的成员变量在内部类中仍然有效,内部类中的方法也可以调用外部类中的方法.[不论是静态还是非静态的,内部类都可以直接调用外部类中的属性,] 内部

随机推荐