Java之内部类

内部类在Android中有着大量的运用

内部类

内部类提供了更好的封装;内部类可以直接访问外部类的私有数据;匿名内部类适合那些只需要使用一次的类。非静态内部类不能拥有静态成员。内部类比外部类可以多使用三个修饰符:private、protected、static

将一个类定义在另一个类的里面,对里面那个类就称为内部类(内置类,嵌套类)

访问特点:

  • 内部类可以直接访问外部类中的成员,包括私有成员
  • 而外部类要访问内部类中的成员必须要建立内部类的对象

内部类的位置

内部类定义在成员位置上

  • 可以被private static成员修饰符修饰
  • 被static修饰的内部类只能访问外部类中的静态成员

内部类定义在局部位置上

  • 也可以直接访问外部类中的成员
  • 同时可以访问所在局部中的局部变量,但必须是被final修饰

非静态内部类

  • 成员内部类是类的成员而局部内部类和匿名内部类则不是
  • 使用ClassName.this.Xxx来调用外部类的重名变量
  • 内部类可以访问外部类的private变量,反之则不行。如果外部类要访问内部类的成员话,则必须显式创建非静态内部类对象来调用访问其实例成员
  • 不允许在非静态内部内定义静态成员
    外部类名.内部类名 变量名 = 外部类对象.内部类对象;
    Outer.Inner in = new Outer().new Inner();
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    class Outer {

    private int x = 3;

    class Inner {
    int x = 4;
    void function() {
    int x = 5;
    System.out.println("outer x: " + Outer.this.x);
    }
    }
    }

静态内部类

  • 用static修饰的内部类,属于类本身
  • 可以包含静态成员也可以包含非静态成员
  • 静态内部类不能访问外部类的实例成员只能访问外部类的类成员。即使是静态内部类的实例方法也不能够访问外部类的实例成员
  • 外部类访问内部类的类成员可以使用类名调用,而实例成员可以通过实例调用
  • 当内部类中定义了静态成员,该内部类必须是static的
    new Outer.Inner().function();
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    class Outer {

    private static int x = 3;

    static class Inner {//静态内部类
    static void function(){
    System.out.println("innner :"+x);
    }
    }
    }

内部类的使用

  • 接口内定义的内部类都为public static修饰的
  • 内部类的使用:

    1. 在外部类中使用内部类
    2. 在外部类中意外使用非静态内部类

      • 此时内部类不可使用private修饰
      • OuterClass.InnerClass varName
      • OuterInstance.new InnerConstructor()

        1
        2
        3
        4
        5
        6
        7
        8
        9
        10
        11
        12
        class Out{
        class In{
        public In (String s){···}
        }
        }
        ···
        Out.In in = new Out().new In("test");
        //等价于
        Out.in in;
        Out out = new Out();
        in = out.new In("test");
        ···
      • 非静态内部类的构造器必须使用外部对象来调用。

        1
        2
        3
        4
        5
        6
        7
        public class SubClass extends Out.In{
        //显示定义SubClass的构造器
        public SubClass ( Out out ){
        //通过传入的Out对象显式调用In的构造器
        out.super("hello");
        }
        }
    3. 在外部类以外使用静态内部类

      • new OuterClass.InnerConstructor()
        1
        2
        3
        4
        5
        6
        7
        8
        9
        10
        class StaticOut{
        static class StaticIn{
        public StaticIn(){···}
        }
        }
        ···
        StaticOut.StaticIn in = new StaticOut.StaticIn();
        //等价于
        StaticOut.StaticIn in;
        in = new StaticOut.StaticIn();

局部内部类

  • 若一个类存在于方法体中,则这个类称之为局部内部类,仅在该方法内有效。局部内部类不能在外部类的方法以外的地方使用,因此不能够使用static修饰
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    class Outer {

    int x = 3;

    void method(final int a) {
    final int y = 4;
    class Inner {
    void function() {
    System.out.println(y);
    }
    }
    new Inner().function();
    }
    }

    class Test {

    public static void main(String[] args) {
    Outer out = new Outer();
    out.method(7); //当此语句执行完毕,释放内存
    out.method(8);
    }
    }

匿名内部类

就是内部类的简化写法

前提:

  • 内部类可以继承或实现一个外部类或者接口

格式为:

  • new 外部类名或者接口名(){覆盖类或者接口中的代码,(也可以自定义内)}

简单理解:

  • 就是建立一个建立一个带内容的外部类或者接口的子类匿名对象
  1. 匿名内部类其实就是内部类的简写格式
  2. 定义匿名内部类的前提:内部类必须是继承一个类或者实现接口
  3. 匿名内部类的格式:new 父类或者接口(){定义子类的内容}
  4. 其实匿名内部类就是一个匿名子类对象。可以理解为带内容的对象
  5. 匿名内部类中定义的方法最好不要超过3个
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
abstract class AbsDemo {
abstract void show();
}

class Outer {
int x = 3;
public void function() {
new AbsDemo() {
int num = 9;
void show() {
System.out.println("num = " + num);
}
void abc() {
System.out.println("增加的方法");
}
}.show();
}
}

class Test {
public static void main(String[] args) {
new Outer().function();
}
}

匿名内部类的应用

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
interface Inter {
void method();
}

class Test
{
//补足代码。通过匿名内部类。
/*
static class Inner implements Inter {
public void method() {
System.out.println("method run");
}
}
*/

static Inter function() {
return new Inter() {
public void method() {
System.out.println("method run");
}
};
}
}

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

//Test.function():Test类中有一个静态的方法function
//.method():function这个方法运算后的结果是一个对象。而且是一个Inter类型的对象
//因为只有是Inter类型的对象,才可以调用method方法
Test.function().method();
show(new Inter(){
public void method() {
System.out.println("method show run");
}
});
}

public static void show(Inter in) {
in.method();
}
}

没有父类及接口如何使用匿名内部类

1
2
3
4
5
6
7
8
9
class InnerTest {
public static void main(String[] args) {
new Object() {
public void function() {
//方法主体
}
}.function();
}
}

Java8改进的匿名内部类

  • 匿名内部类适合只使用一次的类,其定义格式为:

    1
    2
    3
    new 实现接口() | 父类构造器(实参列表){
    //匿名内部类的类体部分
    }
  • 匿名内部类不能是抽象类,因为在创建匿名内部类的同时会创建对象

  • 匿名内部类不能定义构造器。由于匿名内部类没有类名,但却可以使用初始化块
  • 最常用的创建匿名内部类的方式是需要创建某个接口类型对象
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    interface Product{
    public double getPrice();
    public String getName();
    }
    public class AnonymousTest{
    public void test ( Person p ){
    System.out.println( "" + p.getName() + p.getPrice());
    }
    public static void main(String[] args){
    AnonymousTest ta = new AnonymousTest();
    ta.test(new Product(){
    public double getPrice(){
    return 123.4;
    }
    public String getName(){
    return "书本";
    }
    });
    }
    }
1
2
3
4
5
6
7
8
9
class AnonymousProduct implements Product{
public double getPrice(){
return 123.4;
}
public String getName(){
return "书本";
}
}
ta.test ( new AnonymousProduct());
  • 在Java8以前,Java要求被局部内部类、匿名内部类访问的局部变量必须使用final修饰,从Java8开始这个限制被取消了:如果局部变量被匿名内部类访问,那么该局部变量相当于自动使用了final修饰
Donate comment here