• 【Java】Object类及类中方法


    活动地址:CSDN21天学习挑战赛
    在这里插入图片描述博客主页: XIN-XIANG荣
    系列专栏:【Java SE】
    一句短话: 难在坚持,贵在坚持,成在坚持!

    1. 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 obj) {
            System.out.println(obj);
        }
    }
    //执行结果:
    Person@1b6d3586
    Student@4554617c
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14

    所以在开发之中,Object类是参数的最高统一类型。但是Object类也存在有定义好的一些方法。如下:

    img

    这里介绍其中的三个方法: toString()方法,equals()方法,hashcode()方法

    2. 重写toString方法打印对象

    当我们想要打印对象中的内容,可以通过重写Object类当中的toString方法来实现!

    下面解释为什么要重写toString()方法

    下面的代码想要打印一个 Person 对象:

    public class Person {
        String name;
        String gender;
        int age;
        public Person(String name, String gender, int age) {
            this.name = name;
            this.gender = gender;
            this.age = age;
    
        }
        public static void main(String[] args) {
            Person person = new Person("Jim","男", 18);
            System.out.println(person);
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15

    看执行结果打印出来并不是对象的具体内容

    img

    这里观察println的方法源码,在源码中是调用了String类中的valueOf方法,在跳转到valueOf处的源码可以发现方法体中实际上是又调用了toString方法

    img

    此时再看toString当中的具体实现

    img

    getClass().getName() 返回此 Object 的运行时类(Class),并且以 String 的形式返回此 Class 对象所表示的实体(类、接口、数组类、基本类型或 void )名称

    hashCode() 方法返回的是"地址"

    Integer.toHexString(hashCode()) 获取此对象的哈希码值(int类型),并且使用包装类 Integer 类,将此int类型的哈希码值,转换为以十六进制无符号整数,并将转换后十六进制的整数以字符串的形式表示

    所以说输出函数println在底层是调用toString实现的,想要实现打印对象的具体内容,只要按我们自己的想法重写Object类中的toString方法即可

    public class Person {
        String name;
        String gender;
        int age;
    
        @Override
        public String toString() {
            return "Person{" +
                    "name='" + name + '\'' +
                    ", gender='" + gender + '\'' +
                    ", age=" + age +
                    '}';
        }
    
        public Person(String name, String gender, int age) {
            this.name = name;
            this.gender = gender;
            this.age = age;
    
        }
        public static void main(String[] args) {
            Person person = new Person("xin","男", 21);
            System.out.println(person);
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25

    执行结果:

    img

    2. 对象比较equals方法

    在Java中,==进行比较时:

    • 如果==左右两侧是基本类型变量,比较的是变量中值是否相同
    • 如果==左右两侧是引用类型变量,比较的是引用变量地址是否相同

    如果要比较对象中内容,必须重写Object中的equals方法,因为equals方法默认也是按照地址比较的,下面是equals方法的源码:

    img

    对象的比较代码实例:

    class Person{
        private String name ;
        private int age ;
        public Person(String name, int age) {
    
            this.age = age ;
            this.name = name ;
        }
    
        @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 ;
        }
    
    }
    
    public class Test {
        public static void main(String[] args) {
            Person p1 = new Person("xin", 20);
            Person p2 = new Person("xin", 20);
            Person p3 = new Person("rong", 21);
    
            System.out.println(p1.equals(p2));
            System.out.println(p1.equals(p3));
    
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38

    执行结果:

    img

    3. hashCode方法

    • hashcode方法用来确定对象在内存中存储的位置是否相同
    • 在散列表中hashCode() 的作用是获取对象的散列码,进而确定该对象在散列表中的位置。

    hashCode方法源码:

    img

    该方法是一个native方法,底层是由C/C++代码写的;编译器中是无法进行观察的。

    我们认为两个名字相同,年龄相同的对象,是同一个对象,应该存储在同一个位置

    看下面给出的代码,没有重写hashCode()方法,两个对象具有相同的内容,但得出的 的hash值不一样 :

    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("xin", 21) ;
            Person per2 = new Person("xin", 21) ;
            System.out.println(per1.hashCode());
            System.out.println(per2.hashCode());
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16

    执行结果:

    img

    所以,如果我们希望认为两个对象具有相同的内容,其在内存中的位置应该是相等的,就需要重写hashCode()方法

    import java.util.Objects;
    
    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("xin", 21) ;
            Person per2 = new Person("xin", 21) ;
            System.out.println(per1.hashCode());
            System.out.println(per2.hashCode());
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24

    执行结果:

    img

  • 相关阅读:
    运行时候报java.lang.NoClassDefFoundError: Could not initialize class
    法定代表人和股东是什么关系
    Note_First:Hadoop安装部署与测试
    【STM32学习(4)】STM32简述定时器
    备战数学建模47-数模常规算法之图论(攻坚站12)
    asp毕业设计——基于asp+access的在线教育系统设计与实现(毕业论文+程序源码)——在线教育系统
    2022年湖北劳务资质办理需要准备什么资料?办理劳务需要注意哪些呢?甘建二
    buuctf刷题9 (反序列化逃逸&shtml-SSI远程命令执行&idna与utf-8编码漏洞)
    如何判断去噪之后的变压器局部放电是内部还是外部信号
    使用MSYS2和UPX显著缩小发布文件体积
  • 原文地址:https://blog.csdn.net/Trong_/article/details/126415806