[Java开发之路](24)内部类

在Java中,可以将一个类定义在另一个类里面或者一个方法里面,这样的类称为内部类。广泛意义上的内部类一般来说包括这四种:成员内部类、局部内部类、匿名内部类和静态内部类。下面就先来了解一下这四种内部类的用法。

1. 成员内部类

成员内部类是定义在另一个类内部的类。

  1. package com.qunar.fresh;
  2. /**
  3. * Created by xiaosi on 16-3-29.
  4. */
  5. public class Circle {
  6.    private double radius;
  7.    public Circle(double radius) {
  8.        this.radius = radius;
  9.    }
  10.    // 内部类
  11.    class Draw{
  12.        public void drawCircle(){
  13.            System.out.println("Circle---->" + radius);
  14.        }
  15.    }
  16.    public Draw getDrawInstance(){
  17.        return new Draw();
  18.    }
  19.    public static void main(String[] args) {
  20.        Circle circle = new Circle(12.5);
  21.        circle.getDrawInstance().drawCircle(); // 12.5
  22.    }
  23. }

Circle称为外部类。成员内部类可以无条件访问外部类的所有成员属性和成员方法(包括private成员和静态成员)。在内部类drawCircle方法中,可以很轻松的访问外部类的私有成员radius。

注意点:

(1)当成员内部类拥有和外部类同名的成员变量或者方法时,会发生隐藏现象,即默认情况下访问的是成员内部类的成员。

  1. public class Circle {
  2.    private double radius;
  3.    public Circle(double radius) {
  4.        this.radius = radius;
  5.    }
  6.    // 内部类
  7.    class Draw {
  8.        private double radius;
  9.        public Draw(double radius) {
  10.            this.radius = radius;
  11.        }
  12.        public void drawCircle() {
  13.            // 成员内部类拥有和外部类同名的成员变量或者方法时,默认情况下访问的是成员内部类的成员
  14.            System.out.println("Circle---->" + radius);
  15.        }
  16.    }
  17.    public Draw getDrawInstance(double radius) {
  18.        return new Draw(radius);
  19.    }
  20.    public static void main(String[] args) {
  21.        Circle circle = new Circle(12.5);
  22.        circle.getDrawInstance(11.6).drawCircle(); // 11.6
  23.    }
  24. }

如果要访问外部类的同名成员,需要以下面的形式进行访问:

  • 外部类.this.成员变量
  • 外部类.this.成员方法
  1. public void drawCircle() {
  2.    System.out.println("Circle---->" + Circle.this.radius); // 12.5
  3. }

(2)成员内部类可以无条件地访问外部类的成员,而外部类访问成员内部类的成员却这么简单。在外部类中访问成员内部类的成员,必须先创建一个成员内部类的对象,再通过指向这个对象的引用来访问:

  1. public class Circle {
  2.    private double radius;
  3.    public Circle(double radius) {
  4.        this.radius = radius;
  5.        // 需要创建成员内部类的对象
  6.        new Draw().drawCircle();
  7.    }
  8.    // 成员内部类
  9.    class Draw {
  10.        public void drawCircle() {
  11.            System.out.println("Circle---->" + radius); // 12.5
  12.        }
  13.    }
  14.    public static void main(String[] args) {
  15.        Circle circle = new Circle(12.5); // Circle---->12.5
  16.    }
  17. }

(3)成员内部类是依附外部类而存在的,也就是说,如果要创建成员内部类的对象,前提是必须存在一个外部类的对象。创建成员内部类对象的一般方式如下:

  1. package com.qunar.fresh;
  2. /**
  3. * Created by xiaosi on 16-3-29.
  4. */
  5. public class Outer {
  6.    private String name;
  7.    public Outer(String name) {
  8.        this.name = name;
  9.    }
  10.    // 成员内部类
  11.    class Inner {
  12.        public void print() {
  13.            System.out.println("Outer---->" + name); // Outer---->OuterClass
  14.        }
  15.    }
  16.    public static void main(String[] args) {
  17.        Outer outer = new Outer("OuterClass");
  18.        Outer.Inner inner = outer.new Inner();
  19.        inner.print();
  20.    }
  21. }

第2种方式:

  1. package com.qunar.fresh;
  2. /**
  3. * Created by xiaosi on 16-3-29.
  4. */
  5. public class Outer {
  6.    private String name;
  7.    private Inner inner = null;
  8.    public Outer(String name) {
  9.        this.name = name;
  10.    }
  11.    public Inner getInstance() {
  12.        if (inner == null) {
  13.            return new Inner();
  14.        }
  15.        return inner;
  16.    }
  17.    // 成员内部类
  18.    class Inner {
  19.        public void print() {
  20.            System.out.println("Outer---->" + name); // Outer---->OuterClass
  21.        }
  22.    }
  23.    public static void main(String[] args) {
  24.        Outer outer = new Outer("OuterClass");
  25.        Outer.Inner inner = outer.getInstance();
  26.        inner.print();
  27.    }
  28. }

(4)内部类可以拥有private、protected、public及包访问权限。比如上面的例子,如果成员内部类Inner用private修饰,则只能在外部类的内部访问,如果用public修饰,则任何地方都能访问;如果用protected修饰,则只能在同一个包下或者继承外部类的情况下访问;如果是默认访问权限,则只能在同一个包下访问。这一点和外部类有一点不一样,外部类只能被public和包访问两种权限修饰。

2. 局部内部类

局部内部类是定义在一个方法或者一个作用域内的类,它和成员内部类的区别在于局部内部类的访问仅限于方法内或者该作用域内。

备注:

局部内部类就像是方法里面的一个局部变量一样,是不能有public、protected、private以及static修饰符的。

3. 匿名内部类

匿名类是不能有名称的类,所以没办法引用它们。必须在创建时,作为new语句的一部分来声明它们。这就要采用另一种形式的new语句,如下所示: new <类或接口> <类的主体> 这种形式的new语句声明一个新的匿名类,它对一个给定的类进行扩展,或者实现一个给定的接口。它还创建那个类的一个新实例,并把它作为语句的结果而返回。

举个例子来说,假设你有一个Map,key是物品,value是对应的价格,单位是人民币。现在有个需求是将里面的价格都转换为美元,传统的做法是遍历整个Map,然后更新每个value值,将价格转换为美元价格,这种方式比较麻烦。我们使用Function可以解决这一问题。

  1. @Test
  2.    public void test1(){
  3.        Map<String,Double> map = Maps.newHashMap();
  4.        map.put("apple",4.5);
  5.        map.put("bear",6.3);
  6.        // {bear=6.3, apple=4.5}
  7.        System.out.println(map.toString());
  8.        Map<String,Double> newMap = Maps.transformValues(map, new Function<Double, Double>() {
  9.            double rate = 0.1544;
  10.            @Override
  11.            // 转换为美元
  12.            public Double apply(Double input) {
  13.                return rate * input;
  14.            }
  15.        });
  16.        // {bear=0.97272, apple=0.6948000000000001}
  17.        System.out.println(newMap.toString());
  18.    }

new Function<Double,Double>(){...}就是创建一个内部类。

匿名内部类是唯一一种没有构造器的类(不能定义构造器)。正因为其没有构造器,所以匿名内部类的使用范围非常有限,大部分匿名内部类用于接口回调。匿名内部类在编译的时候由系统自动起名为Outter$1.class。一般来说,匿名内部类用于继承其他类或是实现接口,并不需要增加额外的方法,只是对继承方法的重写或者对接口方法的实现。

匿名内部类不能是抽象类,因为系统在创建匿名内部类的时候,会立即创建内部类的对象。因此不允许将匿名内部类定义成抽象类。

4. 静态内部类

静态内部类也是定义在另一个类里面的类,只不过在类的前面多了一个关键字static。静态内部类是不需要依赖于外部类的,这点和类的静态成员属性有点类似,并且它不能使用外部类的非static成员变量或者方法,因为在没有外部类的对象的情况下,可以创建静态内部类的对象,如果允许访问外部类的非static成员就会产生矛盾,因为外部类的非static成员必须依附于具体的对象。

  1. public class Outer2 {
  2.    private static String sname = "Outer2";
  3.    private String name;
  4.    public Outer2(String name) {
  5.        this.name = name;
  6.    }
  7.    // 成员内部类
  8.    public static class Inner {
  9.        public void print() {
  10.            // 静态内部类不能访问外部类的非static成员变量或者方法
  11.            // System.out.println("Outer---->" + name);
  12.            // 可以使用静态变量或者方法
  13.            System.out.println("Outer---->" + sname);
  14.        }
  15.    }
  16. }

调用:

  1. Outer2.Inner inner2 = new Outer2.Inner();
  2. inner2.print();
时间: 2024-11-09 01:07:50

[Java开发之路](24)内部类的相关文章

[Java开发之路](7)RandomAccessFile类详解

RandomAccessFile适用于大小已知的记录组成的文件,提供的对文件访问,既可以读文件,也可以写文件,并且支持随机访问文件,可以访问文件的任意位置.文件中记录的大小不一定都相同,只要我们知道记录的大小和位置.但是该类仅限于操作文件. RandomAccessFile不属于InputStream和OutputStream继承层次结构中的一部分.除了实现DataInput和DataOutput接口之外(DataInputStream和DataOutputStream也实现了这两个接口),它和

7.[Java开发之路](5)异常

1. 异常分类 在Java程序设计语言中,异常对象都是派生于Throwable类的一个实例.其是如果Java中的异常类不能满足需求,用户可以创建自己的异常类. 下图是Java异常层次结构的一个简化示意图. 从图上可以看出,所有的异常都是继承于Throwable类,但是在下一层立即分解为两个分支:Error和Exception. (1)Error Error描述了Java运行时系统的内部错误和资源耗尽错误.应用程序不应该抛出这种类型的错误,如果出现了这样的内部错误,除了通告用户,并尽力使程序安全的

[Java开发之路](7)RandomAccessFile类具体解释

RandomAccessFile适用于大小已知的记录组成的文件.提供的对文件訪问.既能够读文件.也能够写文件,而且支持随机訪问文件.能够訪问文件的任何位置. 文件里记录的大小不一定都同样.仅仅要我们知道记录的大小和位置.可是该类仅限于操作文件. RandomAccessFile不属于InputStream和OutputStream继承层次结构中的一部分.除了实现DataInput和DataOutput接口之外(DataInputStream和DataOutputStream也实现了这两个接口),

[Java开发之路](18)关于Class.getResource和ClassLoader.getResource的路径问题

Java中取资源时,经常用到Class.getResource和ClassLoader.getResource.昨天老师讲解题目时候,问我们为什么你们都是在文件前家上"/": String path = Resources.class.getResource("/a.txt").getPath(); 注:在Resources文件下创建了a.txt文件 我想我反正是试出来的,不使用"/"不行.为了正式解答心中的疑惑,我们正式来看看Resources

[Java开发之路](20)try-with-resource 异常声明

Try-with-resources是java7中一个新的异常处理机制,它能够很容易地关闭在try-catch语句块中使用的资源. 在java7以前,程序中使用的资源需要被明确地关闭,过程有点繁琐,如下所示: package com.qunar.lectures.tryResource; import java.io.*; import java.util.ArrayList; import java.util.List; /** * Created by xiaosi on 16-3-4. *

[Java开发之路](8)图说字符串的不变性

我们用下面一组图来说明Java的不变性. 1.声明一个字符串 String s = "abcd"; s存储了字符串对象的引用.下面图片中的箭头就表示这种存储引用. 2. 将一个字符串变量赋值给另外一个字符串变量 String s2 = s; s2变量存储了同样的引用值.所以,两个变量指向同一个字符串对象. 3. 合并字符串 s = s.concat("ef"); s现在存储的是新生成的字符串对象的引用. 4. 总结 一旦一个字符串在内存(堆)上创建,这个字符串就不会

[Java开发之路](19)Long缓存问题

Long中有个小小的陷阱,就是在-128至127范围内,Long.valueOf(long l)返回的Long的实例是相同的,而在此范围之外每次使用valueOf(long l)时,返回的实例都是不同的. 举例: System.out.println(Long.valueOf(-129) == Long.valueOf(-129)); // false System.out.println(Long.valueOf(-128) == Long.valueOf(-128)); // true Sy

[Java开发之路](14)反射机制

1. Class类 普通对象构造方式: // 创建Book实例对象 Book book = new Book(); 对于Class的实例对象如何构造呢? Class的构造函数是私有的,只有JVM才能创建实例对象 // Class的构造函数是私有的,只有JVM才能创建Class实例对象 Class class1 = new Class(); // 错误 public final class Class<T> implements java.io.Serializable, java.lang.r

[Java开发之路](6)File类的使用

1. 构造方法 构造方法 描写叙述 File(String pathname) 通过将给定的路径名字符串转换为抽象路径名来创建一个新的文件实例. File(String parent , String child) 通过给定的父路径名字符串和子路径名字符串来创建一个新的文件实例. File(File parent , String child) 通过给定的父抽象路径对象和子路径名字符串来创建一个新的文件实例. File(URI uri) 通过给定的URI来创建一个新的文件实例 package c