目录
1.Clonable接口和深拷贝
2.抽象类和接口的区别
3.Object类
4.获取对象的信息
5.对象比较方法equals
6.内部类
1.Clonable接口和深拷贝
Java 中内置了一些很有用的接口, Clonable 就是其中之一,Object 类中存在一个 clone 方法, 调用这个方法可以创建一个对象的 "拷贝". 但是要想合法调用 clone 方法, 必须要先实现 Clonable 接口, 否则就会抛出 CloneNotSupportedException 异常。


 
class Students implements Cloneable { public String name; public int age; @Override protected Object clone() throws CloneNotSupportedException { return super.clone(); } public Students(String name, int age) { this.name = name; this.age = age; } @Override public String toString() { return "{" + "name=" + name + ", age=" + age + "}"; } }public class text { public static void main(String[] args) throws CloneNotSupportedException { Students student1=new Students("张三",10); Students student2=(Students) student1.clone();//强制类型转化 System.out.println(student1); System.out.println(student2); } }

 
 class Money{
    public double m=25.0;
}
class Students  implements Cloneable {
    public String name;
    public int age;
    public Money money=new Money();
    @Override              浅拷贝
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }
    public Students(String name, int age) {
        this.name = name;
        this.age = age;
    }
    @Override
    public String toString() {
        return "{" +
                "name=" + name + " "+
                ", age=" + age +" "+
                ", money=" + money.m +
                '}';
    }
}
public class text {
    public static void main(String[] args) throws CloneNotSupportedException {
        Students student1=new Students("张三",10);
        Students student2=(Students) student1.clone();
        System.out.println(student1);
        System.out.println(student2);
        System.out.println("====");
        student1.money.m=15.0;
        System.out.println(student1);
        System.out.println(student2);
    }
} 
 

 
通过clone,我们只是拷贝了Students对象,但是Students对象中的Money对象,并没有拷贝,通过student1这个引用修改了m的值后,student2这个引用访问m的时候,值也发生了改变。
 
 class Money implements Cloneable{
    public double m=25.0;
    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }
}
class Students  implements Cloneable {
    public String name;
    public int age;
    public Money money=new Money();
    @Override  //深拷贝,将引用的对象(student1)中的对象也拷贝到目标对象中
    protected Object clone() throws CloneNotSupportedException {
        Students temp=(Students) super.clone();
        temp.money=(Money)this.money.clone();
        return temp;
    }
    public Students(String name, int age) {
        this.name = name;
        this.age = age;
    }
    @Override
    public String toString() {
        return "{" +
                "name=" + name + " "+
                ", age=" + age +" "+
                ", money=" + money.m +
                '}';
    }
}
public class text {
    public static void main(String[] args) throws CloneNotSupportedException {
        Students student1=new Students("张三",10);
        Students student2=(Students) student1.clone();
        System.out.println(student1);
        System.out.println(student2);
        System.out.println("====");
        student1.money.m=15.0;
        System.out.println(student1);
        System.out.println(student2);
    }
} 
 

 
通过clone,我们拷贝Students对象,同时将Students对象中的Money对象也进行拷贝。通过student1这个引用修改了m的值后,student2这个引用访问m的时候,值不会发生改变。
2.抽象类和接口的区别
        抽象类和接口都是 Java 中多态的常见使用方式. 都需要重点掌握. 同时又要认清两者的区别,
         核心区别: 抽象类中可以包含普通方法和普通字段, 这样的普通方法和字段可以被子类直接使用(不必重写), 而接口中不能包含普通方法, 子类必须重写所有的抽象方法。
         抽象类存在的意义是为了让编译器更好的校验, 像 Animal 这样的类我们并不会直接使用, 而是使用它的子类.万一不小心创建了 Animal 的实例, 编译器会及时提醒我们。

3.Object类
Object是Java默认提供的一个类。Java里面除了Object类,所有的类都是存在继承关系的。默认会继承Object父类。即所有类的对象都可以使用Object的引用进行接收。
例如:使用Object接收所有类的对象。。
class Person{
}
class Student{}
public class Test {
public static void main(String[] args) {
function(new Person());
function(new Student());
}
public static void function(Object object) {
System.out.println(object);
}}

Object类中有以下方法

4.获取对象的信息
如果要打印对象中的内容,可以直接重写Object类中的toString()方法。
// Object类中的toString()方法实现
public String toString() {
return getClass().getName() + "@" + Integer.toHexString(hashCode());

5.对象比较方法equals
        在Java中,==进行比较时:如果==左右两侧是基本类型变量,比较的是变量中值是否相同;
 如果==左右两侧是引用类型变量,比较的是引用变量地址是否相同;如果要比较对象中内容,必须重写Object中的equals方法,因为equals方法默认也是按照地址比较的。
/ /Object类中的equals方法
public boolean equals(Object obj) {
return (this == obj); // 使用引用中的地址直接来进行比较
}class Person{
private String name ;
private int age ;
public Person(String name, int age) {this.age = age ;
this.name = name ;
}
}
public class Test {
public static void main(String[] args) {
Person p1 = new Person("zhangsan", 20) ;
Person p2 = new Person("lisi", 20) ;
int a = 10;
int b = 10;System.out.println(a == b);
System.out.println(p1 == p2);
System.out.println(p1.equals(p2));}
}

Person类重写equals方法后,然后比较:
class Person{
...
@Override
public boolean equals(Object obj) {
if (obj == null) {
return false ;
}if(this == obj) {
return true ;
}// 不是Person类对象
if (!(obj instanceof Person)) {
return false ;
}Person person = (Person) obj ; // 向下转型,比较属性值
return this.name.equals(person.name) && this.age==person.age ;
}
}
结论:比较对象中内容是否相同的时候,一定要重写equals方法。
6.内部类
// OutClass是外部类public class OutClass {// InnerClass是内部类class InnerClass {}}
public class A {...}class B {...}//A、B是两个独立的类
根据内部类在一个类的位置可以分为实例内部类、静态内部类和局部内部类。
public class OutClass {//成员位置: 实例内部类(未被static修饰)public class InnerClass1{}// 成员位置:静态内部类static class InnerClass2{}public void method(){// 方法中定义内部类: 局部内部类:几乎不用class InnerClass5{}}}
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();
        // 创建实例内部类对象
        OutClass.InnerClass innerClass1 = new OutClass().new InnerClass();
        // 也可以先将外部类对象先创建出来,然后再创建实例内部类对象(常用)
        OutClass.InnerClass innerClass2 = outClass.new InnerClass();
        innerClass2.methodInner();
        }
}注意:外部类中的任何成员都可以在实例内部类方法中直接访问;实例内部类所处的位置与外部类成员位置相同,因此也受public、private等访问限定符的约束;实例内部类对象必须在先有外部类对象前提下才能创建 ;实例内部类的非静态方法中包含了一个指向外部类对象的引用;外部类中,不能直接访问实例内部类中的成员,如果要访问必须先要创建内部类的对象。
2.静态内部类
public class OutClass {
        private int a;
        static int b;
        public void methodA(){
                a = 10;
                System.out.println(a);
        }
        public static void methodB(){
                System.out.println(b);
        }
        // 静态内部类:被static修饰的成员内部类
        static class InnerClass{
                public void methodInner(){
                        // 在内部类中只能访问外部类的静态成员
                        // a = 100; // 编译报错,a不是静态成员变量
                        b =200;
                        // methodA(); // 编译报错,methodB()不是静态成员方法
                        methodB();
                }
        }
        public static void main(String[] args) {
                // 静态内部类对象创建 、成员访问
                OutClass.InnerClass innerClass = new OutClass.InnerClass();
                innerClass.methodInner();
         }
}注意:在静态内部类中只能访问外部类中的静态成员;创建静态内部类对象时,不需要先创建外部类对象。
3.匿名内部类
 
     
    interface IA {
    void test();
}
public class text {
    public static void main(String[] args) {
        //匿名内部类
        IA a = new IA() {
            @Override
            public void test() {
                System.out.println("这是重写了接口的方法!");
            }
        };
        a.test();
        new IA() {
            @Override
            public void test() {
                System.out.println("这是重写了接口的方法!");
            }
        }.test();
    }
} 
    
   4.局部内部类
定义在外部类的方法体或者{}中,该种内部类只能在其定义的位置使用,一般使用的非常少,此处简单了解下语法格式。
public class OutClass {int a = 10;public void method(){int b = 10;// 局部内部类:定义在方法体内部// 不能被public、static等访问限定符修饰class InnerClass{public void methodInnerClass(){System.out.println(a);System.out.println(b);}}// 只能在该方法体内部使用,其他位置都不能用InnerClass innerClass = new InnerClass();innerClass.methodInnerClass();}public static void main(String[] args) {// OutClass.InnerClass innerClass = null; 编译报错}}



















