zoukankan      html  css  js  c++  java
  • 【java基础】内部类

    如果一个事物的内部包含另一个事物,那么这就是一个类内部包含另一个类。
    例如:身体和心脏的关系。又如:汽车和发动机的关系。

    分类:
    1. 成员内部类
    2. 局部内部类(包含匿名内部类)

    修饰符 class 外部类名称 {
    修饰符 class 内部类名称 {
    // ...
    }
    // ...
    }

    注意:

    1)内用外,随意访问;外用内,需要内部类对象。

    2)如果出现了重名现象,那么格式是:外部类名称.this.外部类成员变量名

    ==========================
    如何使用成员内部类?有两种方式:
    1. 间接方式:在外部类的方法当中,使用内部类;然后main只是调用外部类的方法。
    2. 直接方式,公式:
    类名称 对象名 = new 类名称();
    【外部类名称.内部类名称 对象名 = new 外部类名称().new 内部类名称();】

    public class Body { // 外部类
    
        public class Heart { // 成员内部类
            int age=10; //重名变量
            // 内部类的方法
            public void beat() {
                System.out.println("心脏跳动:蹦蹦蹦!");
                System.out.println("我叫:" + name); // 正确写法!
                System.out.println("内部类的年龄为"+this.age);
                System.out.println("外部类的年龄为"+Body.this.age);
            }
    
        }
        private String name;
        int age=20;
        public void methodBody() {// 外部类的方法
            System.out.println("外部类的方法");
            new Heart().beat();  //访问内部类的以一种方法,在外部类实例化内部类一个对象
        }
        public String getName() {
            return name;
        }
        public void setName(String name) {
            this.name = name;
        }
    }
    ====================================
    public class Demo01InnerClass {
    
        public static void main(String[] args) {
            Body body = new Body(); // 外部类的对象
            // 访问方法1:通过外部类的对象,调用外部类的方法,里面间接在使用内部类Heart
            body.methodBody();
            System.out.println("=====================");
    
            // 访问方法1:按照公式写
            Body.Heart heart = new Body().new Heart();
            heart.beat();
        }
    

     局部内部类

    如果一个类是定义在一个方法内部的,那么这就是一个局部内部类。
    “局部”:只有当前所属的方法才能使用它,出了这个方法外面就不能用了。

    局部内部类,如果希望访问所在方法的局部变量,那么这个局部变量必须是【有效final的】

    定义格式:
    修饰符 class 外部类名称 {
        修饰符 返回值类型 外部类方法名称(参数列表) {
            class 局部内部类名称 {
                // ...
            }
        }
    }

    定义一个类的时候,权限修饰符规则:
    1. 外部类:public / (default)
    2. 成员内部类:public / protected / (default) / private
    3. 局部内部类:什么都不能写

    class Outer {
        public void methodOuter() {
            int numMethod=20;
            class Inner { // 局部内部类
                int num = 10;
                public void methodInner() {
                    System.out.println(num); // 10
                    System.out.println(numMethod); // 20
                }
            }
            Inner inner = new Inner();
            inner.methodInner();
        }
    }
    ===========================
    public class DemoMain {
        public static void main(String[] args) {
            Outer obj = new Outer();
            obj.methodOuter();
        }
    }
    

      

    如果接口的实现类(或者是父类的子类)只需要使用唯一的一次,
    那么这种情况下就可以省略掉该类的定义,而改为使用【匿名内部类】。

    匿名内部类的定义格式:
    接口名称 对象名 = new 接口名称() {
    // 覆盖重写所有抽象方法
    };

    另外还要注意几点问题:
    1. 匿名内部类,在【创建对象】的时候,只能使用唯一一次。
    如果希望多次创建对象,而且类的内容一样的话,那么就需要使用单独定义的实现类了。
    2. 匿名对象,在【调用方法】的时候,只能调用唯一一次。
    如果希望同一个对象,调用多次方法,那么必须给对象起个名字。
    3. 匿名内部类是省略了【实现类/子类名称】,但是匿名对象是省略了【对象名称】
    强调:匿名内部类和匿名对象不是一回事!!!

    public interface MyInterface {
    
        void method1(); // 抽象方法
    
        void method2();
    
    }
    ===========================
    //传统的类
    public class MyInterfaceImpl implements MyInterface {
        @Override
        public void method1() {
            System.out.println("实现类覆盖重写了方法!111");
        }
    
        @Override
        public void method2() {
            System.out.println("实现类覆盖重写了方法!222");
        }
    }
    =================================
    public class DemoMain {
    
        public static void main(String[] args) {
    //        MyInterface obj = new MyInterfaceImpl();
    //        obj.method();
    
    //        MyInterface some = new MyInterface(); // 错误写法!
    
            // 使用匿名内部类,但不是匿名对象,对象名称就叫objA
            MyInterface objA = new MyInterface() {
                @Override
                public void method1() {
                    System.out.println("匿名内部类实现了方法!111-A");
                }
    
                @Override
                public void method2() {
                    System.out.println("匿名内部类实现了方法!222-A");
                }
            };
            objA.method1();
            objA.method2();
            System.out.println("=================");
    
            // 使用了匿名内部类,而且省略了对象名称,也是匿名对象
            new MyInterface() {
                @Override
                public void method1() {
                    System.out.println("匿名内部类实现了方法!111-B");
                }
    
                @Override
                public void method2() {
                    System.out.println("匿名内部类实现了方法!222-B");
                }
            }.method1();
            // 因为匿名对象无法调用第二次方法,所以需要再创建一个匿名内部类的匿名对象
            new MyInterface() {
                @Override
                public void method1() {
                    System.out.println("匿名内部类实现了方法!111-B");
                }
    
                @Override
                public void method2() {
                    System.out.println("匿名内部类实现了方法!222-B");
                }
            }.method2();
        }
    
    }
    

      

  • 相关阅读:
    网易163邮箱被盗号找回经历
    C++中基类的析构函数为什么要用virtual虚析构函数
    像linux ls命令一样优雅地打印
    【Linux】- 六个超赞的字符画生成器
    linux欢迎界面 /etc/motd
    Linux 的 FIGlet 指令产生 ASCII Art 大型文字教学
    趣玩 Linux:四个生成字符图案(字符画)的命令
    案例参考手册-第四章 Curses字符界面.docx
    读取键盘输入流改为原始模式
    centos 7配置系统调度isolcpus(软中断绑定)
  • 原文地址:https://www.cnblogs.com/paulwinflo/p/13029006.html
Copyright © 2011-2022 走看看