• Java内部类


    内部类的基本概念

    • 当一个类的定义出现在另外一个类的类体中时,那么这个类叫做内部类 (Inner),而这个内部类所在的类叫做外部类(Outer)。

    • 类中的内容:成员变量、成员方法、构造方法、静态成员、构造块和静 态代码块、内部类。

    实际作用

    当一个类存在的价值仅仅是为某一个类单独服务时,那么就可以将这个 类定义为所服务类中的内部类,这样可以隐藏该类的实现细节并且可以 方便的访问外部类的私有成员而不再需要提供公有的get和set方法。
     

    内部类的分类

    • 普通内部类 - 直接将一个类的定义放在另外一个类的类体中。
    • 静态内部类 - 使用static关键字修饰的内部类,隶属于类层级。
    • 局部内部类 - 直接将一个类的定义放在方法体的内部时。
    • 匿名内部类 - 就是指没有名字的内部类。
     
    普通内部类格式
    访问修饰符 class 外部类的类名 {
      访问修饰符 class 内部类的类名 {

        内部类的类体; }

    }

    code

    public class NormalOuter {
        private int cnt = 1;
    
        // 定义普通内部类,隶属于外部类的成员,并且是对象层级
        /*private*/public /*final*/ class NormalInner {
            private int ia = 2;
            private int cnt = 3;
            public NormalInner() {
                System.out.println("普通内部类的构造方法体执行到了!");
            }
    
            public void show() {
                System.out.println("外部类中变量cnt的数值为:" + cnt); // 1
                System.out.println("ia = " + ia); // 2
            }
    
            public void show2(int cnt) {
                System.out.println("形参变量cnt = " + cnt);  // 局部优先原则  4
                System.out.println("内部类中cnt = " + this.cnt); // 3
                System.out.println("外部类中cnt = " + NormalOuter.this.cnt); // 1
            }
        }
    public class NormalOuterTest {
    
        public static void main(String[] args) {
    
            // 1.声明NormalOuter类型的引用指向该类型的对象
            NormalOuter no = new NormalOuter();
            // 2.声明NormalOuter类中内部类的引用指向内部类的对象
            NormalOuter.NormalInner ni = no.new NormalInner();
            // 调用内部类中的show方法
            ni.show();
    
            System.out.println("---------------------------------------------");
            ni.show2(4);
        }
    }
    普通内部类的使用方式
    • 普通内部类和普通类一样可以定义成员变量、成员方法以及构造方法等。

    • 普通内部类和普通类一样可以使用final或者abstract关键字修饰。

    • 普通内部类还可以使用private或protected关键字进行修饰。

    • 普通内部类需要使用外部类对象来创建对象。

    • 如果内部类访问外部类中与本类内部同名的成员变量或方法时,需要使 用this关键字。

    静态内部类格式
    访问修饰符 class 外部类的类名 {
      访问修饰符 static class 内部类的类名 {

        内部类的类体; }

    }

    code

    /**
     * 实现静态内部类的定义和使用
     */
    public class StaticOuter {
        private int cnt = 1;        // 隶属于对象层级
        private static int snt = 2; // 隶属于类层级
    
        public /*static*/ void show() {
            System.out.println("外部类的show方法就是这里!");
        }
    
        /**
         * 定义静态内部类   有static关键字修饰隶属于类层级
         */
        public static class StaticInner {
            private int ia = 3;
            private static int snt = 4;
    
            public StaticInner() {
                System.out.println("静态内部类的构造方法哦!");
            }
    
            public void show() {
                System.out.println("ia = " + ia); // 3
                System.out.println("外部类中的snt = " + snt); // 2
                //System.out.println("外部类的cnt = " + cnt); // Error:静态上下文中不能访问非静态的成员,因此此时可能还没有创建对象
            }
    
            public void show2(int snt) {  // 就近原则
                System.out.println("snt = " + snt); // 5
                System.out.println("内部类中的成员snt = " + StaticInner.snt); // 4
                System.out.println("外部类中的成员snt = " + StaticOuter.snt); // 2
                //StaticOuter.show();
                new StaticOuter().show();
            }
        }
    }
    public class StaticOuterTest {
    
        public static void main(String[] args) {
    
            // 1.声明StaticInner类型的引用指向该类型的对象
            StaticOuter.StaticInner si = new StaticOuter.StaticInner();
            // 2.调用show方法进行测试
            si.show();
    
            System.out.println("---------------------------------------------");
            si.show2(5);
        }
    }

    静态内部类使用方式

    • 静态内部类不能直接访问外部类的非静态成员。

    • 静态内部类可以直接创建对象。

    • 如果静态内部类访问外部类中与本类内同名的成员变量或方法时,需要 使用类名.的方式访问。

    局部(方法)内部类格式

    访问修饰符 class 外部类的类名 {
      访问修饰符 返回值类型 成员方法名(形参列表) {

        class 内部类的类名 {

          内部类的类体;}

      }

    }

    局部内部类使用方式

      局部内部类只能在该方法的内部可以使用。

      局部内部类可以在方法体内部直接创建对象。

      局部内部类不能使用访问控制符和static关键字修饰符。

      局部内部类可以使用外部方法的局部变量,但是必须是final的。由局部内部类和局部变量的声明周期不同所致

    public class AreaOuter {
        private int cnt = 1;
    
        public void show() {
    
            // 定义一个局部变量进行测试,从Java8开始默认理解为final关键字修饰的变量
            // 虽然可以省略final关键字,但建议还是加上
            final int ic = 4;
    
            // 定义局部内部类,只在当前方法体的内部好使
            class AreaInner {
                private int ia = 2;
    
                public AreaInner() {
                    System.out.println("局部内部类的构造方法!");
                }
    
                public void test() {
                    int ib = 3;
                    System.out.println("ia = " + ia); // 2
                    System.out.println("cnt = " + cnt); // 1
                    System.out.println("ic = " + ic); // 4
                }
            }
    
            // 声明局部内部类的引用指向局部内部类的对象
            AreaInner ai = new AreaInner();
            ai.test();
        }
    
    }
    public class AreaOuterTest {
    
        public static void main(String[] args) {
    
            // 1.声明外部类类型的引用指向外部类的对象
            AreaOuter ao = new AreaOuter();
            // 2.通过show方法的调用实现局部内容类的定义和使用
            ao.show();
        }
    }

    回调模式概念

    回调模式是指——如果一个方法的参数是接口类型,则在调用该方法时, 需要创建并传递一个实现此接口类型的对象;而该方法在运行时会调用 到参数对象中所实现的方法(接口中定义的)。
     

    匿名内部类格式

    接口/父类类型 引用变量名 = new 接口/父类类型() { 方法的重写 };

     code

    public interface AnonymousInterface {
        // 自定义抽象方法
        public abstract void show();
    }
    public class AnonymousInterfaceImpl implements AnonymousInterface {
        @Override
        public void show() {
            System.out.println("这里是接口的实现类!");
        }
    }
    public class AnonymousInterfaceTest {
    
        // 假设已有下面的方法,请问如何调用下面的方法?
        // AnonymousInterface ai = new AnonymousInterfaceImpl();
        // 接口类型的引用指向实现类型的对象,形成了多态
        public static void test(AnonymousInterface ai) {
            // 编译阶段调用父类版本,运行调用实现类重写的版本
            ai.show();
        }
    
        public static void main(String[] args) {
    
            //AnonymousInterfaceTest.test(new AnonymousInterface()); // Error:接口不能实例化
            AnonymousInterfaceTest.test(new AnonymousInterfaceImpl());
    
            System.out.println("---------------------------------------------------------------");
            // 使用匿名内部类的语法格式来得到接口类型的引用,格式为:接口/父类类型 引用变量名 = new 接口/父类类型() { 方法的重写 };
            AnonymousInterface ait = new AnonymousInterface() {
                @Override
                public void show() {
                    System.out.println("匿名内部类就是这么玩的,虽然你很抽象!");
                }
            };
    
            // 从Java8开始提出新特性lamda表达式可以简化上述代码,格式为:(参数列表) -> {方法体}
            AnonymousInterface ait2 = () -> System.out.println("lamda表达式原来是如此简单!");
            AnonymousInterfaceTest.test(ait2);
        }
    }
  • 相关阅读:
    [CF724G]Xor-matic Number of the Graph
    [SOJ #537]不包含 [CF102129I]Incomparable Pairs(2019-8-6考试)
    [SOJ #538]好数 [CC]FAVNUM(2019-8-6考试)
    [洛谷P4052][JSOI2007]文本生成器
    [洛谷P3966][TJOI2013]单词
    [洛谷P5158]【模板】多项式快速插值
    [洛谷P3227][HNOI2013]切糕
    【bzoj】3477: [Usaco2014 Mar]Sabotage 01分数规划
    【SPOJ
    【以前的空间】系列
  • 原文地址:https://www.cnblogs.com/goldenwangyi/p/15170313.html
Copyright © 2020-2023  润新知