文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

【JAVA】Object类与抽象类

2023-09-10 08:17

关注

作者主页:paper jie_的博客

本文作者:大家好,我是paper jie,感谢你阅读本文,欢迎一建三连哦。

本文录入于《JAVASE语法系列》专栏,本专栏是针对于大学生,编程小白精心打造的。笔者用重金(时间和精力)打造,将javaSE基础知识一网打尽,希望可以帮到读者们哦。

其他专栏:《JAVA》《算法详解》《C语言》等

内容分享:本期将会对JAVA中的Object类和内部类进行讲解

目录

 

Object类

获取对象信息toString()方法

对象比较equals方法

 hashcode方法

内部类

内部类的分类

实例内部类

静态内部类

局部内部类 

匿名内部类 


Object类

Object类是java中默认提供的一个类。java里面除了Object类,其他全部的类都是会默认继承Object类的。所以所有的类的对象都可以用Object类的引用来接收。

举个栗子:

class Person{    }class Student{    }public class Test {    public static void function(Object obj) {    System.out.println(obj);}    public static void main(String[] args) {        function(new Person());        function(new Student());    }    }

在java开发的过程中,Objec类是参数的最高统一类型。但是Object类也存在有定义好的一些方法。

这些都是Object类定义好的方法:

这几个方法对于java学习来说是必须掌握,不可缺少的。

获取对象信息toString()方法

如果要打印对象中的内容,直接重写Object类的toSting方法即可:

// Object类中的toString()方法实现:    public String toString() {        return getClass().getName() + "@" + Integer.toHexString(hashCode());    }

其中的道理就是:object类是所有类的父类,重写了tostring方法后,在使用print方法时就会调用我们重写的toString方法以达到效果。

对象比较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("qeo", 20) ;        Person p2 = new Person("qeo", 20) ;        int a = 10;        int b = 10;        System.out.println(a == b); // 输出true        System.out.println(p1 == p2); // 输出false        System.out.println(p1.equals(p2)); // 输出false    }}

上面比较的是地址,想要比较他们的内容就要重写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方法

 hashcode方法

通过观看toString方法的源码我们发现了里面有一个hashcode方法:

    public String toString() {        return getClass().getName() + "@" + Integer.toHexString(hashCode());    }

它的作用就是帮助我们算一个具体的对象位置,它的结果是一个内存地址然后会调用toHexString方法,将这个地址以16进制来打印输出。

hashcode方法源码:

 public native int hashCode();

它是一个native方法,底层是用c/c++代码写的,在java中我们是看不到的。

在我们的认知中,两个名字相同,年龄相同的对象存储在同一个位置,但是不重写hashcode方法的话,结果是不同的:

class Person {    public String name;    public int age;    public Person(String name, int age) {        this.name = name;        this.age = age;    }}public class TestDemo4 {    public static void main(String[] args) {        Person per1 = new Person("go", 20) ;        Person per2 = new Person("go", 20) ;        System.out.println(per1.hashCode());        System.out.println(per2.hashCode());    }} //执行结果        460141958        1163157884

这时我们再重写hashcode方法:

class Person {    public String name;    public int age;    public Person(String name, int age) {        this.name = name;        this.age = age;    } @Override    public int hashCode() {        return Objects.hash(name, age);    }}public class TestDemo4 {    public static void main(String[] args) {        Person per1 = new Person("go", 20) ;        Person per2 = new Person("go", 20) ;        System.out.println(per1.hashCode());        System.out.println(per2.hashCode());    }} //执行结果        460141958        460141958

这时发现他们的哈希值就是一样的了。

结论:

hashcode方法用来确定对象在内存中储存的位置相不相同。

hashcode在散列表中才有用,在散列表中hashcode的作用是获取对象的散列码,来确定对象在散列表中的位置。

内部类

当一个事物的内部,还有一部分需要一个完整的结构进行描述,而这个内部的完整的结构只要为外部的事物提供服务,那这个内部结构使用内部类。

在Java中,可以将一个类定义在另一个类或者方法的内部,前者称为内部类,后者称为外部类。

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

注意:

定义在class类名{}的外部的,即便在一个文件夹内,都不是内部类

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

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

内部类的分类

根据内部类在类中位置不同,可以分为以下几种:

成员内部类:未被static修饰的实例内部类和被static修饰的静态内部类

局部内部类(不谈修饰符)

匿名内部类

public class OutClass {    // 成员位置定义:未被static修饰 --->实例内部类    public class InnerClass1{    } // 成员位置定义:被static修饰 ---> 静态内部类    static class InnerClass2{    }    public void method(){// 方法中也可以定义内部类 ---> 局部内部类:几乎不用        class InnerClass5{        }    }}

实例内部类

未被static修饰的内部类:

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. 外部类中,不能直接访问实例内部类中的成员,如果要访问必须先要创建内部类的对象。

静态内部类

被static修饰的内部类就称为静态内部类

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(); // 编译失败,因为methodA()不是类成员方法            methodB();        }    }    public static void main(String[] args) {// 静态内部类对象创建 & 成员访问        OutClass.InnerClass innerClass = new OutClass.InnerClass();        innerClass.methodInner();    }}

注意:

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

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

局部内部类 

定义在外部类的方法体或者{}中,这种内部类只能在其定义的位置使用,一般使用的非常少:

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; 编译失败    }}

注意:

1. 局部内部类只能在所定义的方法体内部使用

2. 不能被public、static等修饰符修饰

3. 编译器也有自己独立的字节码文件,命名格式:外部类名字$数字内部类名字.class

4. 几乎不会使用 

匿名内部类 

这个内部类是没有名字的:

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();    }


 

来源地址:https://blog.csdn.net/paperjie/article/details/132684042

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     807人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     351人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     314人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     433人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     221人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯