java内部类的详细讲解以及接口的补充

目录

Clonable 接口深浅拷贝

抽象类和接口的区别

equals方法

Object类中的toString()方法

内部类

内部类的分类

实例内部类

静态内部类与局部内部类

匿名内部类(很重要)


Clonable 接口深浅拷贝

简单点说:

浅拷贝:指向的内容是同一个,会随着你克隆的对象的改变而改变

java内部类的详细讲解以及接口的补充_第1张图片

深拷贝:在拷贝的同时把原对象包含的引用所指向的对象也拷贝一份出来,每一个都要拷贝到

java内部类的详细讲解以及接口的补充_第2张图片

详细解释:

浅拷贝(Shallow Copy): 在浅拷贝中,拷贝的是对象的引用,而不是对象的内容。因此,原对象和浅拷贝的对象共享相同的内部对象实例。如果被拷贝的对象包含引用类型的成员变量,那么这些成员变量实际上是指向相同的内部对象。浅拷贝可以通过对象的 clone 方法实现,或者手动复制对象的成员变量。

深拷贝(Deep Copy): 在深拷贝中,拷贝的是对象及其所有引用类型的内容,而不是引用。深拷贝创建了一个新的对象,并递归地将原对象的所有引用类型成员变量复制到新对象中。因此,原对象和深拷贝的对象拥有独立的内部对象实例,对一个对象的修改不会影响到另一个对象。实现深拷贝的方式通常是通过序列化和反序列化、手动递归复制等方法。

用法例子:

import java.util.Objects;
class Money implements Cloneable {
    public double m = 19.9;

    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }
}

class Person implements Cloneable {
    public String name;
    public int age;
    public Money money = new Money();

    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }

    @Override
    protected Object clone() throws CloneNotSupportedException {

        Person tmp = (Person) super.clone();//深拷贝
        tmp.money = (Money) this.money.clone();//深拷贝
        return tmp;
        //return super.clone();
    }

    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        Person person = (Person) o;
        return age == person.age && Objects.equals(name, person.name);
    }

    @Override
    public int hashCode() {
        return Objects.hash(name, age);
    }
}

public class Test {
    public static void main(String[] args) throws CloneNotSupportedException {
        Person person = new Person("he", 11);
        Person person1 = new Person("he", 11);
        System.out.println(person.hashCode());
        System.out.println(person1.hashCode());

        //Person person1 = (Person) person.clone();//这里得转型,因为父类是Object,所以得向下转型

        System.out.println(person.money.m);
        System.out.println(person1.money.m);
        person.money.m = 99.9;
        System.out.println(person.money.m);
        System.out.println(person1.money.m);

    }

    public static void main2(String[] args) throws CloneNotSupportedException {
        Person person = new Person("he", 11);

        Person person1 = (Person) person.clone();//这里得转型,因为父类是Object,所以得向下转型

        Person person2 = (Person) new Person("he", 11).clone();

        System.out.println(person);
        System.out.println(person1);
        System.out.println(person2);
    }
}

抽象类和接口的区别

java内部类的详细讲解以及接口的补充_第3张图片

核心区别: 抽象类中可以包含普通方法和普通字段, 这样的普通方法和字段可以被子类直接使用(不必重写), 而接口中不能包含普通方法, 子类必须重写所有的抽象方法


equals方法

比较当前定义的两个对象一不一样

用法例子:

java内部类的详细讲解以及接口的补充_第4张图片


Object类中的toString()方法

 用法例子:

    public String toString() {

        return getClass().getName() + "@" + Integer.toHexString(hashCode());

    }

默认返回的姓名加@加地址的hascode值


内部类

可以将一个类定义在另一个类或者一个方法的内部,前者称为内部类,后者称为外部类。内部类也是封装的一种体现

例子:

public class OutClass { 
class InnerClass{ 
} 
} 
// OutClass是外部类 
// InnerClass是内部类

注意事项:

1. 定义在class 类名{}花括号外部的,即使是在一个文件里,都不能称为内部类

public class A{ 
} 
class B{ 
} 
// A 和 B是两个独立的类,彼此之前没有关系 

2. 内部类和外部类共用同一个java源文件,但是经过编译之后,内部类会形成单独的字节码文件


内部类的分类

实例内部类、静态内部类、匿名内部类、局部内部类(不怎么用)

根据内部类定义的位置不同,一般可以分为以下几种形式:   

 public class OutClass {

        // 成员位置定义:未被static修饰 --->实例内部类

        public class InnerClass1{

        }

        // 成员位置定义:被static修饰 ---> 静态内部类

        static class InnerClass2{

        }

        public void method(){

// 方法中也可以定义内部类 ---> 局部内部类:几乎不用

            class InnerClass5{

            }

        }

    }

1. 成员内部类(普通内部类:未被static修饰的成员内部类 和 静态内部类:被static修饰的成员内部类)

2. 局部内部类(不谈修饰符)、匿名内部类


实例内部类

public class OutClass { 
private int a; 
static int b; 
int c; 
public void methodA(){ 
a = 10; 
System.out.println(a); 
} 
public static void methodB(){ 
System.out.println(b); 
} 
// 实例内部类:未被static修饰 
class InnerClass{ 
int c; 
public void methodInner(){ 
// 在实例内部类中可以直接访问外部类中:任意访问限定符修饰的成员 
a = 100; 
b =200; 
methodA(); 
methodB(); 
// 如果外部类和实例内部类中具有相同名称成员时,优先访问的是内部类自己的 
c = 300; 
System.out.println(c); 
// 如果要访问外部类同名成员时候,必须:外部类名称.this.同名成员名字 
OutClass.this.c = 400; 
System.out.println(OutClass.this.c); 
} 
} 
public static void main(String[] args) { 
// 外部类:对象创建 以及 成员访问 
OutClass outClass = new OutClass(); 
System.out.println(outClass.a); 
System.out.println(OutClass.b); 
System.out.println(outClass.c); 
outClass.methodA(); 
outClass.methodB(); 
System.out.println("=============实例内部类的访问============="); 
// 要访问实例内部类中成员,必须要创建实例内部类的对象 
// 而普通内部类定义与外部类成员定义位置相同,因此创建实例内部类对象时必须借助外部类 
// 创建实例内部类对象 
OutClass.InnerClass innerClass1 = new OutClass().new InnerClass(); 
// 上述语法比较怪异,也可以先将外部类对象先创建出来,然后再创建实例内部类对象 
OutClass.InnerClass innerClass2 = outClass.new InnerClass(); 
innerClass2.methodInner(); 
} 

注意:

1. 外部类中的任何成员都可以在实例内部类方法中直接访问

2. 实例内部类所处的位置与外部类成员位置相同,因此也受public、private等访问限定符的约束

3. 在实例内部类方法中访问同名的成员时,优先访问自己的,如果要访问外部类同名的成员,必须:外部类名

称.this.同名成员 来访问

4. 实例内部类对象必须在先有外部类对象前提下才能创建

5. 实例内部类的非静态方法中包含了一个指向外部类对象的引用

6. 外部类中,不能直接访问实例内部类中的成员,如果要访问必须先要创建内部类的对象。


静态内部类与局部内部类

用法例子:

class OutClass {
    public int data = 0;
    public static int a = 1;

    static class InClass {//静态内部类
        public static int s = 3;
        public int data1 = 2;

        public void test1() {
            OutClass outClass = new OutClass();//实例化外部类对象

            System.out.println("In");
            System.out.println(s);
            System.out.println(data1);
            System.out.println(a);//这个可以直接访问,因为外部类的那个变量是静态的

            System.out.println(outClass.data);//静态内部类要想访问外部类的变量,先实例化外部类的对象
        }
    }

    public void test() {
        System.out.println("Out");
    }
}

public class Test {

    public void fun() {//局部内部类
        class A {
            public int a = 10;

            public void fun1() {

            }
        }
    }

    public static void main(String[] args) {
        OutClass.InClass inClass = new OutClass.InClass();//静态内部类的实例化
        inClass.test1();
    }

}

注意事项

1. 在静态内部类中只能访问外部类中的静态成员

如果确实想访问,我们该如何做?如下的代码,需要先实例化外部类对象

 public void test1() {
            OutClass outClass = new OutClass();//实例化外部类对象

            System.out.println("In");
            System.out.println(s);
            System.out.println(data1);
            System.out.println(a);//这个可以直接访问,因为外部类的那个变量是静态的

            System.out.println(outClass.data);//静态内部类要想访问外部类的变量,先实例化外部类的对象
        }
    }

2. 创建静态内部类对象时,不需要先创建外部类对象


匿名内部类(很重要)

用法例子:

interface IA {
    void test();
}

public class Test {

    public static void main(String[] args) {
        IA a = new IA() {
            @Override
            public void test() {
                System.out.println("这是重写了接口的方法");
            }
        };//匿名内部类的对象
        a.test();

        //两种方法调用test方法

        new IA() {
            @Override
            public void test() {
                System.out.println("这是重写了接口的方法");
            }
        }.test();
    }
}

欢迎大家一起分享java知识,一起进步!!!

你可能感兴趣的:(java)