实例的常见操作

1. 1实例的常见操作

== 能够比较两个对象的内存地址是否一致(两个对象是否是同一个实例),当需要比较两个实例是否 逻辑意义上“相等”时, == 不能保证结果准确。

例:Employee类表示员工

public class Employee {
    public int empno;
    public String ename;
    public double salary;
    
    public Employee() {}
    public Employee(int empno,String ename,double salary){
        this.empno=empno;
        this.ename=ename;
        this.salary=salary;
    }
}

创建两个内容一样的实例

public class Test1 {
    public static void main(String[] args){
        Employee e1=new Employee(1001,"赵四",5000);
        Employee e2=new Employee(1001,"赵四",5000);

        System.out.println(e1==e2);//false
    }
}
public class Employee {
    public int empno;
    public String ename;
    public double salary;
    public Employee() {}
    public Employee(int empno, String ename, double salary) {
        this.empno = empno;
        this.ename = ename;
        this.salary = salary;
    }
    //此时需要在业务逻辑上判定额、e1与e2是相同的,==不能给出正确答案
    //可以通过重写Object类中的equals方法来实现判断Employee实例相等
    public boolean equals(Object obj){
        if (obj instanceof Employee){
            Employee temp=(Employee) obj;
            return (this.empno==temp.empno&&
                    this.ename.equals(temp.ename)&&this.salary==temp.salary);
        }
        return false;
    }
}
//在比较通过e1.equals(e2)可以进行判定

1.2字符串转换

在调用System.out.println(e1)时,默认打印出的是“类型@hashcode”的结构,这是通过调用从Object类继承来的toString()方法实现的。

可以通过重写toString()方法实现打印格式的实现,同时也提供了类转字符串的方法。

public String toString(){
        return "{empno:"+this.empno+",ename:'"+this.ename+"',salary:"+this.salary+"}";
    }

1.3hashcode值

hashcode(哈希码)值是通过对内存地址进行计算,得到一个数字,通常以十六进制方式显示。

它有如下特点:

(1)两个不同对象引用同一个实例,它们一定有相同的hashcode值

(2)两个对象的hashcode值相同,它们不一定是同一个实例

可以通过重写Object类的hashcode方法来重写hashcode获得(不推荐重写)

public int hashCode(){
        return this.empno;
    }

1.4对象的克隆

某些场景下,我们需要复制一个数据相同但是内存不同的新的实例,此时如果这样做会出现问题

Employee e1=new Employee(1001,"赵四");
Employee e2=e1;

此时对象e1和e2指向同一个实例,某个对象实例属性进行修改后,另一个对象也发生改变,我们需要的是两个不相关的实例。(即e1和e2分别指向内容相同的不同实例)。此时我们需要进行克隆操作。

克隆操作需要完成两个步骤:

类实现java.lang.Cloneable接口,这个接口中没有定义任何方法,仅是表明“类的实例允许克隆”

重写Object类中的clone()方法

    public class Employee implements Cloneable{

public Object clone() throws CloneNotSupportedException{
        return new Employee(this.empno,this.ename,this.salary);
    }

}

当需要获得一个克隆对象时

Employee e1=new Employee(1001,"赵四",5000);
Employee e3=(Employee)e1.clone();

e1与e3是两个不同的实例

深度克隆与浅度克隆

我们目前的克隆实现的是浅度克隆,当类中的某个属性是一个对象时,是否对属性对象进行克隆决定了克隆的深浅。

图示:游戏任务类Player中,存在着一个属性:Weaponed(武器)实例

如果没有对Weapone进行克隆,两个Player实例使用同一件武器实例,一方销毁,另一方的武器也会丢失,这种克隆被称为“浅度克隆”

深度克隆是指在Player克隆时,Weapone实例应该也进行克隆,两个Player实例应该分别持有各自的Weapone(当然,两个Weapone实例应该是具备同样的属性)

1.5比较规则

很多时候,我们需要对数组或集合进行排序操作,这样需要提供数组或集合内容的比较规则。通常来说,数字等常见类型拥有自然的比较规则。但是面向对象的类的比较规则需要我们自行设定。

比如两个Employee实例到底哪个更“大”?

比较规则的实现,需要两个步骤:

        类实现java.lang.Comparable接口,并实现compareTo方法

        在compareTo方法编写比较规则

示例

public int compareTo(Employee o){
        //逻辑上,this比o大,返回1;比o小,返回-1;相等,返回0.
        if(this.empno>o.empno){
            return 1;
        } else if (this.empno<o.empno) {
            return -1;
            
        }
        return 0;
    }

排序应用

public class Test2 {
    public static void main(String[] args) {
        Employee[] es = new Employee[3];
        es[0] = new Employee(1002, "刘能", 7000);
        es[1] = new Employee(1003, "广坤", 6000);
        es[2] = new Employee(1001, "赵四", 5000);
        System.out.println("排序前:"+Arrays.toString(es));
        Arrays.sort(es); //排序: 基于自定义的规则 Comparable
        System.out.println("排序后:"+Arrays.toString(es));
    }
}

1.6综合使用

上述的5个操作可以在类中同时设定

public class Employee implements Cloneable ,Comparable<Employee>{
    public int empno;
    public String ename;
    public double salary;
    public Employee() {}
    public Employee(int empno, String ename, double salary) {
        this.empno = empno;
        this.ename = ename;
        this.salary = salary;
    }
    //此时需要在业务逻辑上判定额、e1与e2是相同的,==不能给出正确答案
    //可以通过重写Object类中的equals方法来实现判断Employee实例相等
    public boolean equals(Object obj){
        if (obj instanceof Employee){
            Employee temp=(Employee) obj;
            return (this.empno==temp.empno&&
                    this.ename.equals(temp.ename)&&this.salary==temp.salary);
        }
        return false;
    }

    public String toString(){
        return "{empno:"+this.empno+",ename:'"+this.ename+"',salary:"+this.salary+"}";
    }

    public int hashCode(){
        return this.empno;
    }

    public Object clone() throws CloneNotSupportedException{
        return new Employee(this.empno,this.ename,this.salary);
    }

    public int compareTo(Employee o){
        //逻辑上,this比o大,返回1;比o小,返回-1;相等,返回0.
        if(this.empno>o.empno){
            return 1;
        } else if (this.empno<o.empno) {
            return -1;

        }
        return 0;
    }


}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值