面向对象之内部类

内部类

1.为什么使用内部类?

使用内部类就是在前一个类的内部再定义一个类。比如,A类中定义了一个B类,则B类相对于A类来说称为内部类,A类相对于B类来说就是一个外部类。
使用内部类的好处在于每个内部类都能独立地继承一个(接口的)实现,所以无论外围类是否已经继承了某个(接口的)实现,对于内部类都没有影响。

1.1.使用内部类最大的优点就在于它能够非常好的解决多重继承的问题,使用内部类还能够为我们带来如下特性:

1.内部类可以用多个实例,每个实例都有自己的状态信息,并且与其他外围对象的信息相互独。
2.在单个外围类中,可以让多个内部类以不同的方式实现同一个接口,或者继承同一个类。
3.创建内部类对象的时刻并不依赖于外围类对象的创建。
4.内部类并没有令人迷惑的“is-a”关系,他就是一个独立的实体。
5.内部类提供了更好的封装,除了该外围类,其他类都不能访问。

内部类主要分为:

  • 成员内部类

  • 静态内部类

  • 局部内部类

  • 匿名内部类

成员内部类

//成员内部类
public class Outer1 {
    private int age = 99;
    String name = "Coco";
    public class Inner{
        String name = "Jayden";
        public void show(){
            System.out.println(Outer1.this.name);
            System.out.println(name);
            System.out.println(age);
        }
    }
    public Inner getInnerClass(){
        return new Inner();
    }
    public static void main(String[] args){
        Outer1 o = new Outer1();
        Inner in = o.new Inner();
        in.show();
    }
}
/*1.Inner 类定义在 Outer1 类的内部,相当于 Outer1 类的一个成员变量的位置,Inner 类可以使用任意访问控制符,如 public 、 protected 、 private 等
2.Inner 类中定义的 show() 方法可以直接访问 Outer1 类中的数据,而不受访问控制符的影响,如直接访问 Outer1 类中的私有属性age
3.定义了成员内部类后,必须使用外部类对象来创建内部类对象,而不能直接去 new 一个内部类对象,即:内部类 对象名 = 外部类对象.new 内部类( );
4.编译上面的程序后,会发现产生了两个 .class 文件: Outer1.class,Outer$Inner.class{}
5.成员内部类中不能存在任何 static 的变量和方法,可以定义常量:
(1).因为非静态内部类是要依赖于外部类的实例,而静态变量和方法是不依赖于对象的,仅与类相关,
简而言之:在加载静态域时,根本没有外部类,所在在非静态内部类中不能定义静态域或方法,编译不通过;非静态内部类的作用域是实例级别
(2).常量是在编译器就确定的,放到所谓的常量池了
★★友情提示:
1.外部类是不能直接使用内部类的成员和方法的,可先创建内部类的对象,然后通过内部类的对象来访问其成员变量和方法;
2.如果外部类和内部类具有相同的成员变量或方法,内部类默认访问自己的成员变量或方法,如果要访问外部类的成员变量,可以使用 this 关键字,如:Outer1.this.name*/

静态内部类

静态内部类: 是 static 修饰的内部类

  1. 静态内部类不能直接访问外部类的非静态成员,但可以通过 new 外部类().成员 的方式访问
  2. 如果外部类的静态成员与内部类的成员名称相同,可通过“类名.静态成员”访问外部类的静态成员;如果外部类的静态成员与内部类的成员名称不相同,则可通过“成员名”直接调用外部类的静态成员
  3. 创建静态内部类的对象时,不需要外部类的对象,可以直接创建 内部类 对象名 = new 内部类();
public class Outer2 {
    private int age = 99;
    static String name = "Coco";
    public static class Inner{
        String name = "Jayden";
        public void show(){
            System.out.println(Outer2.name);
            System.out.println(name);
        }
    }
    public static void main(String[] args){
        Inner i = new Inner();
        i.show();
    }
}

局部(方法)内部类

局部(方法)内部类:其作用域仅限于方法内,方法外部无法访问该内部类

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

  2. 只能访问方法中定义的 final 类型的局部变量,因为:
    当方法被调用运行完毕之后,局部变量就已消亡了。
    但内部类对象可能还存在,直到没有被引用时才会消亡。
    此时就会出现一种情况,就是内部类要访问一个不存在的局部变量;
    使用final修饰符不仅会保持对象的引用不会改变,而且编译器还会持续维护这个对象在回调方法中的生命周期。

    局部内部类并不是直接调用方法传进来的参数,而是内部类将传进来的参数通过自己的构造器备份到了自己的内部,自己内部的方法调用的实际是自己的属性而不是外部类方法的参数;防止被篡改数据,而导致内部类得到的值不一致

  3. 注意:在JDK8版本之中,方法内部类中调用方法中的局部变量,可以不需要修饰为 final,匿名内部类也是一样的,主要是JDK8之后增加了Effectively final 功能 http://docs.oracle.com/javase/tutorial/java/javaOO/localclasses.html
    反编译jdk8编译之后的class文件,发现内部类引用外部的局部变量都是 final 修饰的

public class Outer3 {
    public void show(){
        /**
         使用的形参为何要为 final???
         在内部类中的属性和外部方法的参数两者从外表上看是同一个东西,但实际上却不是,所以他们两者是可以任意变化的,也就是说在内部类中我对属性的改变并不会影响到外部的形参,
         而然这从程序员的角度来看这是不可行的,毕竟站在程序的角度来看这两个根本就是同一个,如果内部类该变了,而外部方法的形参却没有改变这是难以理解和不可接受的,
         所以为了保持参数的一致性,就规定使用 final 来避免形参的不改变*/
        final int a = 25;
        int b = 13;
        class Inner{
            int c = 2;
            public void print(){
                System.out.println("访问外部类:" + a);
                System.out.println("访问内部类:" + c);
            }
        }
        Inner i = new Inner();
        i.print();
    }
    public static void main(String[] args){
        Outer3 o = new Outer3();
        o.show();
    }
}

匿名内部类:

  1. 匿名内部类是直接使用 new 来生成一个对象的引用;
  2. 对于匿名内部类的使用它是存在一个缺陷的,就是它仅能被使用一次,创建匿名内部类时它会立即创建一个该类的实例,该类的定义会立即消失,所以匿名内部类是不能够被重复使用;
  3. 使用匿名内部类时,我们必须是继承一个类或者实现一个接口,但是两者不可兼得,同时也只能继承一个类或者实现一个接口;
  4. 匿名内部类中是不能定义构造函数的,匿名内部类中不能存在任何的静态成员变量和静态方法;
  5. 匿名内部类不能是抽象的,它必须要实现继承的类或者实现的接口的所有抽象方法
  6. 匿名内部类初始化:使用构造代码块!利用构造代码块能够达到为匿名内部类创建一个构造器的效果
public class Outer4 {
    //定义一个外部类成员方法,返回一个InnerClass接口类型的引用,该引用指向一个实现了InnerClass接口的匿名内部类,方法名称为getInnerClass,接收一个final的int参数和一个String型参数
    public InnerClass getInnerClass(final int num, String str2){
        /*
        这里的方法体内只有一条语句,就是return(****);
        return的类型呢是一个匿名类,该匿名类没有名称(因此称为匿名),但该匿名类实现了InnerClass接口
        InnerClass接口中定义了一个getNumber()抽象方法
        该匿名类实现了InnerClass接口,类的内容为:
        {
            int number = num + 3; //即接收外部类方法所接受的num数据,将其加3后赋给匿名内部类的比那辆number
            //匿名内部类中重写了接口中的getNumber抽象方法,返回number值
            public int getNumber(){
                return number;
            }
        }
         */
        //可以说new InnerClass()实例化了一个匿名类的对象,该匿名类实现了InnerClass接口
        return new InnerClass(){
            //该内部类的操作就是将num加3赋给number
            int number = num + 3;
            //该内部类中顶一个一个getNumber放啊,返回number的值
            public int getNumber(){
                return number;
            }
        };        /* 注意:分号不能省 */
    }
    public static void main(String[] args) {
        //实例化一个外部类对象
        Outer4 out = new Outer4();
        //定义一个InnerClass类型的引用,名称为inner,将其指向外部类对象调用getInnerClass所返回的匿名内部类
        InnerClass inner = out.getInnerClass(2, "chenssy");
        //通过这个inner这个引用,访问匿名内部类中的getNumber方法
        System.out.println(inner.getNumber());
    }
}
//声明一个接口,接口名为InnerClass
interface InnerClass {
    int getNumber();
}
已标记关键词 清除标记
相关推荐
©️2020 CSDN 皮肤主题: 1024 设计师:白松林 返回首页