Java基础(11)内部类、Object类

1.内部类

1. 内部类的概述:把类定义在其他类的内部,这个类就称为内部类
2. 内部类访问的特点
(1)内部类可以直接访问外部类的成员,包括私有
(2)外部类要访问内部类的成员,必须要先创建对象
3. 内部类的分类:根据内部类在外部类的定义位置不同,分成成员内部类和局部内部类
4. 成员内部类
(1)成员内部类的定义位置:类中方法外
(2)创建成员内部类对象:外部类民.内部类名 对象名 = new 外部类名().new 内部类名

public class TestDemo01 {
    public static void main(String[] args) {
        Outer.Inner inner = new Outer().new Inner();
        inner.inShow();
        System.out.println(inner.num3);

        System.out.println("==================");

        Outer outer = new Outer();
        outer.outTest();
    }
}

class Outer{
    int num = 20;
    private double num2 = 3.14;
    public void outShow(){
        System.out.println("外部类的成员方法outShow");
    }

    //定义成员内部类
    class Inner{
        int num3 = 30;
        public void inShow(){
            System.out.println("内部类的inShow方法");
            System.out.println(num);
            System.out.println(num2);
            System.out.println(num3);
            outShow();
        }
    }

    //外部类想要访问内部类的成员,要创建内部类对象
    public void outTest(){
        Inner inner = new Inner();
        System.out.println(inner.num3);
        inner.inShow();
    }
}

(3)private修饰成员内部类:为保证数据的安全性。private修饰的内部类为私有内部类,外界不能直接创建其对象了。如果要访问内部类的成员,需要通过创建外部类的对象来间接访问

public class TestDemo02 {
    public static void main(String[] args) {
        Wai wai = new Wai();
        wai.waiHaHa();
    }
}

class Wai{
    int a = 20;
    private int b = 10;

    public void waiShow(){
        System.out.println("waiShow方法执行");
    }

    private void waiTest(){
        System.out.println("外部类的私有方法");
    }

    private class Nei{
        int c = 233;
        public void neiShow(){
            System.out.println(a);
            System.out.println(b);
            waiShow();
            waiTest();
        }
    }

    public void waiHaHa(){
        Nei nei = new Nei();
        System.out.println(nei.c);
        nei.neiShow();
    }
}

(4)static修饰成员内部类:为了方便访问数据。需要注意的是:①静态内部类访问的外部类数据必须用静态修饰。②成员方法可以是静态的或者非静态的。③成员内部类被static修饰后的访问方式是:外部类名.内部类名 对象名 = new 外部类名.内部类名();

public class TestDemo01 {
    public static void main(String[] args) {
        Wai.Nei nei = new Wai.Nei();
        System.out.println(nei.c);
        nei.neiShow();
    }
}

class Wai{
    private int num = 100;
    private double d = 3.14;
    public void waiShow(){
        System.out.println("waiShow方法执行了");
    }

    public static void waiShow2(){
        System.out.println("静态的waiShow2执行了");
    }

    public static class Nei{
        int c = 20;
        public void neiShow(){
//            System.out.println(num);  //错误 num没有被static修饰
//            waiShow();   //错误 waiShow()没有被static修饰
            waiShow2();
        }
    }
}

(5)一道面试题

//要求:使用已知的变量,在控制台输出30,20,10。
	class Outer {
		public int num = 10;
		class Inner {
			public int num = 20;
			public void show() {
				int num = 30;
				System.out.println(num);  //30
				System.out.println(this.num);  //20
				System.out.println(Outer.this.num);   //10 【重点】
			}
		}
	}
	class InnerClassTest {
		public static void main(String[] args) {
			Outer.Inner oi = new Outer().new Inner();
			oi.show();
		}	
	}

5. 局部内部类【用得不多】
(1)局部内部类的定义位置:类中方法内
(2)局部内部类在外界不能直接访问内部类的成员,需要通过外部类的对象来间接访问内部类成员

public class TestDemo01 {
    public static void main(String[] args) {
        Outer outer = new Outer();
        outer.test();
    }
}

class Outer{
    int num = 20;
    private double a = 3.14;
    public void test(){
        //定义局部内部类
        class Inner{
            int b = 30;
            public void neiShow(){
                System.out.println(num);
                System.out.println(a);
                waiShow();
            }

            public void neiTest(){
                System.out.println(b);
            }
        }

        //创建内部类对象
        Inner inner = new Inner();
        inner.neiTest();
    }

    private void waiShow(){
        System.out.println("外部类show方法");
        test();
    }
}

(3)局部内部类访问外部类的局部变量必须用final修饰,在JDK1.8之后,final会默认加上。

public class TestDemo02 {
}

class Wai{
    int num = 100;
    public void waiTest(final int i){ //局部变量
        final int a = 200;  //局部变量

        class Nei{
            int j = 20;
            public void neiShow(){
                num = 300;
                System.out.println(num);
            }
        }
    }
}

6. 匿名内部类【重点】
(1)匿名内部类的概述:局部内部类的一种简化写法
(2)前提:存在一个类或者接口(这里的类可以是具体类也可以是抽象类)
(3)匿名内部类的本质:是一个继承了该类或者实现了该接口的子类对象
(4)格式:new 类名或者接口名() { 重写方法 };

//重写一个类
public class TestDemo01 {
    public static void main(String[] args) {

        //想要得到AA的子类对象,使用匿名内部类
        new AA(){
            @Override
            public void aa() {
                System.out.println("aaaaa");
            }

            @Override
            public void bb() {
                System.out.println("bbbbb");
            }
        }.aa;    //aaaaa
        
        new AA(){
            @Override
            public void aa() {
                System.out.println("aaaaa");
            }

            @Override
            public void bb() {
                System.out.println("bbbbb");
            }
        }.bb;    //bbbbb
        
    }
}

class AA{
    public void aa(){

    }

    public void bb(){

    }
}
//重写一个接口
public class TestDemo02 {
    public static void main(String[] args) {
        new LOL() {
            @Override
            public void attack() {
                System.out.println("平A");
            }
        }.attack();    //平A
    }
}

interface LOL{
    void attack();
}
public class TestDemo01 {
    public static void main(String[] args) {
   		//使用同一个对象,多态
        LOL lol = new LOL() {
            @Override
            public void attack() {
                System.out.println("普通攻击");
            }

            @Override
            public void magic() {
                System.out.println("魔法攻击");
            }
        };
        
        lol.attack();
        lol.magic();
    }
}

abstract class LOL{
    public abstract void attack();
    public abstract void magic();
}

(5)匿名内部类的应用:可以作为参数来传递,或作为返回值返回
①匿名内部类作为参数来传递

public class TestDemo01 {
    public static void main(String[] args) {
        //当你以后看到一个方法的形参,要一个接口类型,你就可以传递一个该接口的子类对象。
        test(new LOL() {
            @Override
            public void attack() {
                System.out.println("普通攻击");
            }
        });  //普通攻击

    }

    public static void test(LOL lol){
        lol.attack();
    }
}

interface LOL{
    void attack();
}
public class TestDemo02 {
    public static void main(String[] args) {
        set(new Test() {
            @Override
            public void show() {
                System.out.println("重写show方法");
            }
        });   //重写show方法
    }

    public static void set(Test test){
        test.show();
    }
}

abstract class Test{
    public abstract void show();
}

②匿名内部类作为返回值

public class TestDemo01 {
    public static void main(String[] args) {
        MyInterface myInterface = test();
        myInterface.show();  //重写的show方法
    }

    public static MyInterface test(){
        MyInterface myInterface = new MyInterface() {
            @Override
            public void show() {
                System.out.println("重写的show方法");
            }
        };

        return myInterface;
    }
}

interface MyInterface{
    void show();
}

(6)匿名内部类中this关键字的应用

public class TestDemo01 {
    public static void main(String[] args) {
        new Inter(){
            public void show(){
                //this代表接口的子类对象
                System.out.println(this.a);  //23
                System.out.println(Inter.a);  //23
            }
        }.show();
    }
}

interface Inter{
    public static final int a =23;
}

2.Object类

1. Object类概述:所有类的根类,所有类都直接或者间接继承自Object类
2. 构造方法:public Object() //子类的构造方法默认访问的是父类的无参构造方法
3. Object类的hashCode()方法
(1)public int hashCode()
(2)返回该对象的哈希码值。①默认情况下根据该对象的地址值计算。②不同对象的hashCode值不同,同一个对象的hashCode值肯定相同

public class TestDemo01 {
    public static void main(String[] args) {
        Object obj1 = new Object();
        System.out.println(obj1);   //java.lang.Object@3f99bd52
        int code = obj1.hashCode();
        System.out.println(code);   //1067040082

        Object obj2 = new Object();
        System.out.println(obj2);   //java.lang.Object@4f023edb
        int code2 = obj2.hashCode();
        System.out.println(code2);   //1325547227

        System.out.println(code == code2);    //false
    }
}

4. Object类的getClass()方法
(1)public final Class getClass()
(2)获取这个类的字节码文件对象。因为一个类的字节码文件就一份,你一加载进内存,就为这个字节码文件创建了一个对象。你不管通过该类的哪个对象去获取这个类的字节码文件对象,都是同一个字节码对象

public class TestDemo02 {
    public static void main(String[] args) {
        Object obj1 = new Object();
        Class<?> aClass1 = obj1.getClass();
        System.out.println(aClass1);   //class java.lang.Object

        Object obj2 = new Object();
        Class<?> aClass2 = obj2.getClass();
        System.out.println(aClass2);   //class java.lang.Object

        System.out.println(obj1 == obj2);   //false
        System.out.println(aClass1 == aClass2);  //true
    }
}

5. Object类的toString()方法
(1)public String toString()
(2)返回该对象的地址值。我们在输出一个对象的名称时,默认调用父类的toString()方法,打印该对象的地址值
(3)重写toString()方法:打印成员变量的值

public class TestDemo03 {
    public static void main(String[] args) {
        LOL lol1 = new LOL("VN",23);
        System.out.println(lol1.toString());   //LOL{name='VN', age=23}

        LOL lol2 = new LOL("EZ",15);
        System.out.println(lol2.toString());   //LOL{name='EZ', age=15}
    }
}

class LOL{
    private String name;
    private int age;

    public LOL() {
    }

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

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }


    //重写toString方法
    @Override
    public String toString() {
        return "LOL{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}

6. Object类的equals()方法
(1)public boolean equals(Object obj)
(2)默认情况下是比较地址值是否相同
(3)重写equals()方法:比较成员变量的值是否相同

public class TestDemo01 {
    public static void main(String[] args) {
        LOL lol1 = new LOL("酒桶",48);
        LOL lol2 = new LOL("酒桶",48);
        LOL lol3 = new LOL("EZ",15);

        boolean b1 = lol1.equals(lol2);
        System.out.println(b1);   //true

        boolean b2 = lol1.equals(lol3);
        System.out.println(b2);   //false
    }
}

class LOL{
    private String name;
    private int age;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    public LOL(String name, int age) {

        this.name = name;
        this.age = age;
    }

    public LOL() {

    }


    //重写equals()方法
    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        LOL lol = (LOL) o;
        return age == lol.age &&
                Objects.equals(name, lol.name);
    }

}

7. Object类的clone()方法
(1)clone()的权限修饰符是受保护的,在使用的时候要重写该类方法,把该方法的权限修饰符变成public
(2)clone()方法采用的是浅克隆的方式
(3) 如果一个对象需要调用clone的方法克隆,那么该对象所属的类必须要实现Cloneable接口
(4)Cloneable接口只不过是一个标识接口而已,没有任何方法
(5)对象的浅克隆也不会调用到构造方法的。

public class TestDemo01 {
    public static void main(String[] args) throws CloneNotSupportedException {
        Dog dog1 = new Dog("小白",2);
        Dog dog2 = (Dog) dog1.clone();

        System.out.println(dog2.name);   //小白
        System.out.println(dog2.age);   //2
    }
}

class Dog implements Cloneable{
    String name;
    int age;

    public Dog() {
    }

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

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

(6) 对象的浅克隆就是克隆一个对象的时候,如果被克隆的对象中维护了另外一个类的对象,这时候只是克隆另外一个对象的地址,而没有把另外一个对象也克隆一份
在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值