Java基础(七)

132 阅读5分钟

持续创作,加速成长!这是我参与「掘金日新计划 · 6 月更文挑战」的第23天,点击查看活动详情

1.Java内部类细节演示

  1. Java中允许将一个类A声明在另一个类B中,则类A就是内部类,类B称为外部类
  2. 内部类的分类:成员内部类(静态、非静态)和 局部内部类(方法内、代码块内、构造器内)
  3. 成员内部类:
    • 一方面,作为外部类的成员:

      • 调用外部类的结构
      • 可以被static修饰
      • 可以被4种不同的权限修饰
    • 另一方面,作为一个类:

      • 类内可以定义属性、方法、构造器等
      • 可以被final修饰,表示此类不能被继承。言外之意,不使用final,就可以被继承
      • 可以被abstract修饰
/**
 * @Author: YuShiwen
 * @Date: 2020/11/19 10:21 AM
 * @Version: 1.0
 */
public class InnerClass {
    public static void main(String[] args) {
        //创建静态的成员内部类实例:
        Student.Maths maths = new Student.Maths();
        maths.appraise();

        //创建非静态的成员内部类实例:
        //错误示范:
        //Student.Physics physics = new Student.Physics();
        //需要先创建外部类对象
        Student student = new Student();
        Student.Physics physics = student.new Physics();
        physics.appraise();
        physics.namesake(18);
    }

}
class Student{
    String name = "Mr.Yu";
    int age = 21;

    public void drink(){
        System.out.println("口渴了,喝水!");
    }

    //静态成员内部类:
    static class Maths{
        String teacher = "Mr.Wu";
        int score;

        public void appraise(){
            System.out.println("该学生成绩优秀,学习刻苦认真!");
            //静态成员中,不能调用非静态属性或方法
            //drink();
            //System.out.println(name);
        }
    }

    //非静态成员内部类
    public final class Physics{
        String teacher = "Mr.Li";
        int age = 48;
        int score;

        public void appraise(){
            System.out.println("该学生成绩优秀,学习刻苦认真!");
            drink();
            System.out.println(name);
        }

        //内部类方法的形参、内部类的属性、外部类的属性同名情况
        public void namesake(int age){
            //内部类方法的形参
            System.out.println(age);
            //内部类的属性
            System.out.println(this.age);
            //外部类的属性
            System.out.println(Student.this.age);
        }
    }


    //以下都为局部内部类演示
    //方法中声明局部内部类
    public void localInnerClassDisplay(){
        //局部内部类
        class LocalInnerClass{

        }
    }
    //代码块中声明局部内部类
    {
        //局部内部类
        class LocalInnerClass{

        }
    }
    //构造器中声明局部内部类
    public Student(){
        //局部内部类
        class LocalInnerClass{

        }
    }


}

输出结果:

该学生成绩优秀,学习刻苦认真!
该学生成绩优秀,学习刻苦认真!
口渴了,喝水!
Mr.Yu
18
48
21

Process finished with exit code 0

  1. 局部内部类作用演示:
interface Person{
    void eat();
    void drink();
}

public class LocalInnerClass {

    //返回一个实现了Person接口的类的对象
    //方式一:
    public Person getPerson1(){
        //1.先创建一个实现了Comparable接口的类:局部内部类
        class Student implements Person{

            @Override
            public void eat() {
                System.out.println("吃饭!");
            }

            @Override
            public void drink() {
                System.out.println("喝牛奶!");
            }
        }
        //2.返回局部内部类对象
        return new Student();
    }

    //返回一个实现了Person接口的类的对象
    //方式二:
    public Person getPerson2(){
        //直接返回一个匿名局部内部类的匿名对象
        return new Person(){

            @Override
            public void eat() {
                System.out.println("吃牛肉!");
            }

            @Override
            public void drink() {
                System.out.println("喝饮料!");
            }
        };
    }
}

class MainTest{
    public static void main(String[] args) {
        LocalInnerClass localInnerClass = new LocalInnerClass();

        Person person1 = localInnerClass.getPerson1();
        Person person2 = localInnerClass.getPerson2();
        person1.eat();
        person1.drink();
        person2.eat();
        person2.drink();
    }
}

运行结果:

吃饭!
喝牛奶!
吃牛肉!
喝饮料!

Process finished with exit code 0

2.Java中父类和接口或接口和接口中出现同名属性或同名同参数方法时,如何区分

1.父类和接口中出现同名属性

/**
 * @Author: YuShiwen
 * @Date: 2020/11/19 12:35 PM
 * @Version: 1.0
 */
interface InterfaceA{
    int number = 0;
}

class SuperClass{
    int number = 1;
}

public class ClassA extends SuperClass implements InterfaceA{
    public void printNumber(){
    	//如果子类没有定义同名的属性,number是不明确的,编译不通过。
        //System.out.println(number);

        //通过super.调用父类中的属性number
        System.out.println(super.number);
        //通过接口名.调用接口中的属性number
        System.out.println(InterfaceA.number);
    }

    public static void main(String[] args) {
        new ClassA().printNumber();
    }
}

输出结果:

1
0

Process finished with exit code 0

2.父类和接口中出现同名方法

我们这里只讨论接口中的静态方法和默认方法,不讨论接口中的抽象方法, 因为接口中的抽象方法实现类(子类)需要重写,所以调用的是子类重写的方法,这里不讨论接口中的抽象方法

/**
 * @Author: YuShiwen
 * @Date: 2020/11/19 12:49 PM
 * @Version: 1.0
 */
interface InterfaceMethod{
    //静态方法
    public static void hobby(){
        System.out.println("我喜欢健身!");
    }
    //默认方法
    public default void drink(){
        System.out.println("口渴了,喝饮料!");
    }

    public default void eat(){
        System.out.println("吃汉堡!");
    }
}

class ClassMethod{
    public void hobby(){
        System.out.println("我喜欢跑步!");
    }

    public void drink(){
        System.out.println("口渴了,喝水!");
    }

    public void eat(){
        System.out.println("吃饭!");
    }
}
public class ClassB extends ClassMethod implements InterfaceMethod{

    public void eat(){
        System.out.println("吃牛排!");
    }

    public void printMethod(){
        //不存在指定不明确情况,如果子类没有重写覆盖该方法,直接是父类中的方法;
        // 因为接口中定义的静态方法,只能通过接口来调用。
        hobby();//也可以写成super.hobby();

        //接口中定义的静态方法,只能通过接口来调用。
        InterfaceMethod.hobby();

        //如果子类(或实现类)继承的父类和实现的接口中声明了同名同参数的默认方法,
        //那么子类在没有重写此方法的情况下,默认调用的是父类中的同名同参数的方法。
        //类优先原则
        drink();
        //接口中的方法
        InterfaceMethod.super.drink();

        //子类有重写此方法的情况
        //子类
        eat();
        //父类
        super.eat();
        //调用接口中的默认方法
        InterfaceMethod.super.eat();
    }

    public static void main(String[] args) {
        new ClassB().printMethod();
    }
}

输出结果:

我喜欢跑步!
我喜欢健身!
口渴了,喝水!
口渴了,喝饮料!
吃牛排!
吃饭!
吃汉堡!

Process finished with exit code 0

3.接口和接口中出现同名属性

/**
 * @Author: YuShiwen
 * @Date: 2020/11/19 1:19 PM
 * @Version: 1.0
 */
interface InterfacePropertyA{
    public static final int number = 0;
}

interface InterfacePropertyB{
    int number = 1;
}
public class ClassC implements InterfacePropertyA,InterfacePropertyB{
    public static void main(String[] args) {
        ClassC classC = new ClassC();
        //如果子类没有定义同名的属性,classC.number指定不明确
        //System.out.println(classC.number);

        System.out.println(InterfacePropertyA.number);
        System.out.println(InterfacePropertyB.number);

    }
}

输出结果:

0
1

Process finished with exit code 0

4.接口和接口中出现同名方法

首先接口中定义的静态方法,只能通过接口来调用,出现同名同参数静态方法只需要用接口进行调用即可。 其次接口中的抽象方法,没有方法体,需要实现类重写,调用时肯定是实现类中重写的方法。 如果实现类实现了多个接口,而这多个接口中定义了同名同参数的默认方法,那么在实现类没有重写此方法的情况下,报错。接口冲突。这就需要我们必须在实现类中重写此方法。