java返回对象的引用_Java 返回可变引用对象相关问题

1.问题

/**

* 输出: Mon Apr 26 10:54:10 CST 2010

* Mon Apr 26 10:54:10 CST 2010

*/

public static void main(String[] args){

Example test = new Example(new Date());

Date d = test.getDate();

double tenYearsInMillisSeconds = 10 * 365.25 * 24 * 3600 * 1000;

d.setTime((long) (d.getTime() - tenYearsInMillisSeconds));

System.out.println(d);

System.out.println(test.getDate());

}

public class Example {

private Date date;

public Example(Date date){

this.date = date;

}

public Date getDate() {

return date;

}

}

Date类破坏了Example的封装,导致修改实例 d时影响了 test 的值,原因是Date类生成的对象是可变的。

2.对象与对象变量

Date birthday = new Date();

Date deadline = birthday;

这两个变量引用同一个对象(请参见图 4-4 )。

1d0dfd2f5e7eca7b202da511ab95db34.png

但一个对象变量并没有实际包含一个对象,而仅仅引用一个对象。

在 Java 中,任何对象变量的值都是对存储在另外一个地方的一个对象的引用。new 操作符的返回值也是一个引用。

Date birthday = new Date();可以理解为new Date() 构造了一个 Date 类型的对象, 并且它的值是对新创建对象的引用。这个引用存储在变量 birthday中。

Java 对象变量与 C++ 的引用并不同

可以将 Java 的对象变量看作 C++ 的对象指针。例如,

Date birthday; // Java

实际上,等同于

Date* birthday; // C++

所有的 Java 对象都存储在堆中。 当一个对象包含另一个对象变量时, 这个变量依然

包含着指向另一个堆对象的指针。

3.更改器方法与访问器方法

上文还是没有解释清楚为什么Date类的对象是可变对象,原因在这。

假设在上文中Example类中使用Java中与Date类相近的LocalDate类便不会出现上述情况,测试可以自己去尝试。

原因在于假设使用LocalDate类中的plusDays 方法来修改对象变量,它会生成一个新的LocalDate对象,然后把这个新对象赋值给调用者,原来的对象不做任何改动。

此类只访问对象而不修改对象的方法有时称为访问器方法(accessor method)

而像Date类中的setTime方法会使得原对象的状态发生改变,此类称为更改器方法(mutator method)

4.解决方法

如果需要返回一个可变数据域的拷贝,就应该使用 clone。这样会创建一个当前对象的副本,而不会对当前对象造成影响。

public class Example {

private Date date;

public Example(Date date){

this.date = date;

}

public Date getDate() {

return (Date) date.clone();

}

}

5.不可变类

5.1什么是不可变类

不可变类指当类被实例化后,该类的成员变量均不可被改变。

如JDK内部自带的很多不可变类Interger、Long、 Boolean和String等。

5.2优缺点

优点:1.线程安全 2.易于构造、使用和测试 3.可以被自由地共享

缺点:对于每一个不同的值都需要对应一个单独的对象

5.3如何实现不可变类

Class需要用final修饰,保证类不能被继承

所有成员变量需要private修饰,保证成员变量不能直接被访问

类中不允许提供setter方法,保证成员变量不会被改变

在getter方法中不能返回对象本身,返回对象的拷贝

原文:https://www.cnblogs.com/huaranmeng/p/12776250.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Java中的List是一种数据结构,用于存储一组有序的元素。List有多种实现方式,如ArrayList、LinkedList等。在Java中,每个对象都有一个hashCode方法,返回对象的哈希码。如果两个对象的hashCode不同,那么这两个对象肯定不相同。如果List的返回对象的hashCode不一致,可能是因为List中存储的对象的hashCode不同,或者是List的实现类重写了hashCode方法导致的。 ### 回答2: 在Java中,List是一个用于存储对象的集合接口,它提供了许多方法来操作和访问集合中的元素。List的实现类有很多,比如ArrayList和LinkedList等。 当我们使用List的方法返回一个对象时,实际上返回的是一个引用,这个引用指向集合中的某个元素。每个对象都有一个hashCode值,它是根据对象的内部状态计算出来的一个唯一标识。 当我们对List进行操作时,有时候会发现返回对象的hashCode值与我们期望的不一致。这可能是由于几个原因造成的。 首先,如果我们在操作List之前对List进行了修改,比如插入、删除或重新排序等操作,那么List中的元素的顺序可能会发生变化,从而导致返回对象的hashCode值不一致。 其次,如果我们对集合中的元素的属性进行了修改,那么这些修改可能会影响到元素的hashCode值。例如,如果我们修改了某个元素的属性,那么它的hashCode值可能会改变,从而导致返回对象的hashCode值不一致。 另外,如果我们在List中存储的是可变对象,当我们对这些对象进行修改时,它们的hashCode值可能会发生变化,从而导致返回对象的hashCode值不一致。 为了解决这个问题,我们可以在操作List之前进行深拷贝,这样就可以保证返回对象是一个全新的对象,它的hashCode值不会受到影响。另外,我们还可以重写对象的hashCode方法,以确保在对象的内部状态发生变化时,hashCode值的变化是可控的。 总之,Java中List方法返回对象的hashCode不一致,可能是由于对List进行了修改、对元素属性进行了修改或存储了可变对象等原因导致的。我们可以通过深拷贝和重写hashCode方法等方式来解决这个问题

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值