如何使java中包含日期字段的类不可变?
我创建了一个带有日期字段的不可变类。我如何确保即使是日期字段也是不可变的,因为即使将日期字段设置为final,以后也可以为其分配不同的值?在如何使java中包含日期字段的类不可变?,java,immutability,Java,Immutability,我创建了一个带有日期字段的不可变类。我如何确保即使是日期字段也是不可变的,因为即使将日期字段设置为final,以后也可以为其分配不同的值?在getDate()方法中,返回一个new date()实例,而不是相同的实例 public Date getDate() { // Not correct. return this.date; // This will make your class mutable. // Instead use, return new
getDate()
方法中,返回一个new date()
实例,而不是相同的实例
public Date getDate() {
// Not correct.
return this.date; // This will make your class mutable.
// Instead use,
return new Date(this.date.getTime()); // This will make sure your date field cannot be changed.
}
使用防御性复制。这意味着,如果必须将日期传递给其他类,例如通过方法参数或从方法返回的日期,则应创建副本 制作日期副本很容易:
new Date(myDate.getting())
下面是
Java
public final class Employee{
final String pancardNumber;
public Employee(String pancardNumber){
this.pancardNumber=pancardNumber;
}
public String getPancardNumber(){
return pancardNumber;
}
}
上述类是不可变的,因为:
Date
属性,您可以使用构造函数为每个新对象设置日期,并导入org.joda.time.DateTime
类。这是一个比java.util.Date
更好的版本,因为它是不可变的。使用java.util.Date会很危险,因为它是一个可变类,我们无法控制调用线程(这可能会修改它)。下面是一个例子
public final class Bill {
private final int amount;
private final DateTime dateTime;
public Bill(int amount, DateTime dateTime) {
this.amount = amount;
this.dateTime = dateTime;
}
public int getAmount() {
return amount;
}
public DateTime getDateTime() {
return dateTime;
}
}
这是一个带有不可变HAS-a-Date对象的Bean(类)示例
import java.util.Date;
public class MyBean {
private Date date; // Immutable Date Step 1 Make Private
public MyBean(Date date)
{
// Immutable Date Step 2 If Set through Constructor then get a specialised (sub class) Date.
this.date= getImmutableDate(date); // THIS METHOD RETURNS AN IMMUTABLE DATE
}
public MyBean(){} // Allow Default Constructor
public Date getDate() {
return date;
}
// Immutable Date Step 3- Allow setting of date only once!!
public void setDate(Date date) {
if(this.date==null)
{
this.date= getImmutableDate(date);
}
}
/* Override all Setter methods of Date class. So even if user gets reference of Date Object it is not the original date object
* it would be a modified date object whose all setter methods do nothing*/
private Date getImmutableDate(Date date)
{
/* This is an Anonymous Inner Class that extends java.util.Date class, it overrides all the setter methods
* making the date object IMMUTABLE( i.e setXXX has no effect)
* */
date =new Date(date.getTime()){
private static final long serialVersionUID = 1L;
@Override
public void setYear(int year) {}
@Override
public void setMonth(int month) {}
@Override
public void setDate(int date) {}
@Override
public void setHours(int hours) {}
@Override
public void setMinutes(int minutes) {}
@Override
public void setSeconds(int seconds) {}
@Override
public void setTime(long time) {}
};
return date;
}
}识别可变实例变量(如date或hashmap)返回新对象,并为所有可变对象复制内容。不可变变量可以安全地返回,无需额外努力 请参见以下示例:
import java.util.Date;
public final class ImmutableClass
{
/**
* Integer class is immutable as it does not provide any setter to change its content
* */
private final Integer immutableField1;
/**
* String class is immutable as it also does not provide setter to change its content
* */
private final String immutableField2;
/**
* Date class is mutable as it provide setters to change various date/time parts
* */
private final Date mutableField;
//Default private constructor will ensure no unplanned construction of class
private ImmutableClass(Integer fld1, String fld2, Date date)
{
this.immutableField1 = fld1;
this.immutableField2 = fld2;
this.mutableField = new Date(date.getTime());
}
//Factory method to store object creation logic in single place
public static ImmutableClass createNewInstance(Integer fld1, String fld2, Date date)
{
return new ImmutableClass(fld1, fld2, date);
}
//Provide no setter methods
/**
* Integer class is immutable so we can return the instance variable as it is
* */
public Integer getImmutableField1() {
return immutableField1;
}
/**
* String class is also immutable so we can return the instance variable as it is
* */
public String getImmutableField2() {
return immutableField2;
}
/**
* Date class is mutable so we need a little care here.
* We should not return the reference of original instance variable.
* Instead a new Date object, with content copied to it, should be returned.
* */
public Date getMutableField() {
return new Date(mutableField.getTime());
}
@Override
public String toString() {
return immutableField1 +" - "+ immutableField2 +" - "+ mutableField;
}
}
有关详细信息:
java.time
其他答案是正确的,在展示固定对象中的值的策略时
我还建议您使用现代的java.time类,而不是糟糕的遗留类。使用“日期”代替“日期”。使用“日历”代替“日历”
java.time类被设计为不可变对象。方法,如加上…
,减…
,到…
,以及与
都会生成一个新的对象,保留原始对象不变。这些类不携带setter方法
额外提示:在您自己的不可变类中,您可能会发现遵循java.time类建立的方法命名模式很有用
关于java.time 该框架内置于Java8及更高版本中。这些类取代了麻烦的旧日期时间类,例如,& 该项目现已启动,建议迁移到类 要了解更多信息,请参阅。并搜索堆栈溢出以获得许多示例和解释。规格是 使用兼容的或更高版本,您可以直接与数据库交换java.time对象。不需要字符串,也不需要java.sql.*类 从哪里获得java.time类
- ,及以后
- 内置的李>
- 标准JavaAPI的一部分,带有捆绑实现
- Java9添加了一些次要功能和修复
- 及
- 大部分java.time功能都在中向后移植到Java6和Java7
-
- 更高版本的Android捆绑包实现了java.time类
- 对于早期的Android,该项目采用了ThreeTen Backport(如上所述)。看
创建自定义DateClass并重写set**方法,并且不为此方法提供任何功能 您可以克隆日期对象,使其无法修改
public Date getDate (){
return (Date) this.date.clone ();
}
-Oracle除了建议或暗示对日期使用getter的答案之外,请确保您的日期字段是私有的,以便其他类无法修改它。如果用户为他们从中获取的日期的变量指定了不同的值,则不会更改您的日期字段中的日期值class@MadProgrammer是的,但是getter返回的日期不是不变的。这就是为什么在getter中复制它很重要的原因。@McBrainy是的,这部分是正确的,但这不是你的问题所说的;)最简单的方法是使用字符串。但是如果您想要“java.util.Date”,那么上面的示例将在setter中无声地失败,就像这是一个等待发生的bug@Oliver以毫秒为单位的时间比字符串好,但日期最好。@McBrainy你认为这段代码怎么会失败?请再解释一下。@Oliver,如果你覆盖了所有这些方法,使用你的API的人会非常困惑。在许多情况下,这可能不是一个好的编程实践。@Codebender这很管用。我认为使用JodaAPI是一个更好的选择,因为DateTime对象无论如何都是不可变的。但是,如果你不想使用像Joda这样的完整API,这可能是一个很好的快速修复方法。确切地说,为什么我们会担心“调用线程”修改我们的私有字段?通过getter?你可以像Codebender或Joni那样,在从getter返回日期时,有效地克隆日期。@McBrainy:说得好。我同意你的看法。我们可以这样做。此方法返回的所有实例仍然是可变的。this.date无法更改,因此类将保持不变,尽管您可以更改克隆的对象。除此之外,我们还可以在构造函数中执行此操作:
public constructor(date d){this.date=new date(d.getTime());}
这样,如果对d
对象进行更改,也不会影响原始对象中的日期。