概念:不可变类的意思是创建该类的实例后,该实例的属性是不可改变的。java中的8个包装类和String类都是不可变类。所以不可变类并不是指该类是被final修饰的,而是指该类的属性是被final修饰的。

自定义不可变类遵守如下原则:

1、使用private和final修饰符来修饰该类的属性。

2、提供带参数的构造器,用于根据传入的参数来初始化属性。

3、仅为该类属性提供getter方法,不要提供setter方法。

4、如果有必要,重写hashCode和equals方法,同时应保证两个用equals方法判断为相等的对象,其hashCode也应相等。

构造一个不可变类非常容易,下面举一个简单例子:

  1. package com.home;
  2.  
  3. public class Address {
  4. private final String detail;
  5.  
  6. public Address() {
  7. this.detail = "";
  8. }
  9.  
  10. public Address(String detail) {
  11. this.detail = detail;
  12. }
  13.  
  14. public String getDetail() {
  15. return detail;
  16. }
  17.  
  18. @Override
  19. public int hashCode() {
  20. return detail.hashCode();
  21. }
  22.  
  23. @Override
  24. public boolean equals(Object obj) {
  25. if (obj instanceof Address) {
  26. Address address = (Address) obj;
  27. if (this.getDetail().equals(address.getDetail())) {
  28. return true;
  29. }
  30. }
  31. return false;
  32. }
  33.  
  34. }

但是值得注意的是,该类的属性虽然是被final修饰的,但若属性是非String的其他引用类型的话,那么虽然该属性的内容(所指对象的地址)不会改变,但其指向的对象却有可能会改变,这样的类当然并不能成为不可变类。比如下面的Person类中有一个Name类型的属性:

  1. package com.home;
  2.  
  3. public class Person {
  4. private final Name name;
  5.  
  6. public Person(Name name) {
  7. super();
  8. this.name = name;
  9. }
  10.  
  11. public Name getName() {
  12. return name;
  13. }
  14.  
  15. public static void main(String[] args) {
  16. Name n = new Name("三", "张");
  17. Person p = new Person(n);
  18. System.out.println(p.getName().getFirstName());
  19. // 改变Person对象Name属性的firstName属性值
  20. n.setFirstName("无忌");
  21. System.out.println(p.getName().getFirstName());
  22. }
  23. }

Name:

  1. package com.home;
  2.  
  3. public class Name {
  4. private String firstName;
  5. private String lastName;
  6.  
  7. public Name() {
  8. super();
  9. }
  10.  
  11. public Name(String firstName, String lastName) {
  12. super();
  13. this.firstName = firstName;
  14. this.lastName = lastName;
  15. }
  16.  
  17. public String getFirstName() {
  18. return firstName;
  19. }
  20.  
  21. public void setFirstName(String firstName) {
  22. this.firstName = firstName;
  23. }
  24.  
  25. public String getLastName() {
  26. return lastName;
  27. }
  28.  
  29. public void setLastName(String lastName) {
  30. this.lastName = lastName;
  31. }
  32. }

运行上面程序可以看到,Person对象的Name属性的firstName属性已经被改变,这就违背了不可变类设计的初衷。我们可以采取如下办法来解决,修改Person类如下:

  1. package com.home;
  2.  
  3. public class Person {
  4. private final Name name;
  5.  
  6. public Person(Name name) {
  7. super();
  8. // 设置name属性为临时创建的Name对象,该对象的firstName属性和lastName属性
  9. // 与传入的name对象的firstName属性和lastName属性相同
  10. this.name = new Name(name.getFirstName(), name.getLastName());
  11. }
  12.  
  13. public Name getName() {
  14.  
  15. // 返回一个匿名对象,该对象的firstName属性和lastName属性
  16. // 与该对象里的name属性的firstName属性和lastName属性相同
  17. return new Name(name.getFirstName(), name.getLastName());
  18. }
  19.  
  20. public static void main(String[] args) {
  21. Name n = new Name("三", "张");
  22. Person p = new Person(n);
  23. System.out.println(p.getName().getFirstName());
  24. // 改变Person对象Name属性的firstName属性值
  25. n.setFirstName("无忌");
  26. System.out.println(p.getName().getFirstName());
  27. }
  28. }

再次运行程序,发现Person对象的Name属性的firstName属性没有改变了。

另外,由于不可变类的实例的状态不可改变,所以可以很方便地被多个对象所共享,那么如果程序要经常使用相同的不可变类实例,为了减少系统开销,一般要考虑使用缓存机制。下面使用数组作为缓存池来构建一个可以缓存实例的不可变类:

  1. package com.home;
  2.  
  3. public class CacheImmutale {
  4. private final String name;
  5. private static CacheImmutale[] cache = new CacheImmutale[10];
  6. private static int pos = 0;
  7.  
  8. public CacheImmutale(String name) {
  9. super();
  10. this.name = name;
  11. }
  12.  
  13. public String getName() {
  14. return name;
  15. }
  16.  
  17. public static CacheImmutale valueOf(String name) {
  18. // 遍历已缓存的对象
  19. for (int i = 0; i < pos; i++) {
  20. // 如果已有相同实例,直接返回该缓存的实例
  21. if (cache[i] != null && cache[i].getName().equals(name)) {
  22. return cache[i];
  23. }
  24. }
  25. // 如果缓冲池已满
  26. if (pos == 10) {
  27. // 把缓存的第一个对象覆盖
  28. cache[0] = new CacheImmutale(name);
  29. pos = 1;
  30. return cache[0];
  31. } else {
  32. // 把新创建的对象缓存起来,pos加1
  33. cache[pos++] = new CacheImmutale(name);
  34. return cache[pos - 1];
  35. }
  36. }
  37.  
  38. @Override
  39. public int hashCode() {
  40. return name.hashCode();
  41. }
  42.  
  43. @Override
  44. public boolean equals(Object obj) {
  45. if (obj instanceof CacheImmutale) {
  46. CacheImmutale ci = (CacheImmutale) obj;
  47. if (name.equals(ci.getName())) {
  48. return true;
  49. }
  50. }
  51. return false;
  52. }
  53.  
  54. public static void main(String[] args) {
  55. CacheImmutale c1 = CacheImmutale.valueOf("hello");
  56. CacheImmutale c2 = CacheImmutale.valueOf("hello");
  57. System.out.println(c1 == c2);// 输出结果为true
  58. }
  59. }

对于缓存的使用,应根据系统需求而定,简单的说,如果某个对象使用的次数不多,重复使用的概率不大,就没必要使用缓存,毕竟缓存的对象也会占用系统内存。如果某个对象需要频换地重复使用,这时就应该使用缓存了。

另外,上面的示例来源疯狂JAVA讲义一书,个人对上面那个Person类里面的属性是引用类型的解决办法存有疑问,他那种办法虽然保证的Person对象的Name属性所指对象的内容没有改变,但Person对象返回的Name属性已经不是同一个属性了,它的地址已发生改变,赋值和返回都是通过new出来的,我个人做了如下改进,觉得更合理:

  1. package com.home;
  2.  
  3. public class Person {
  4. private final Name name;
  5.  
  6. public Person(Name name) {
  7. super();
  8. // 设置name属性为临时创建的Name对象,该对象的firstName属性和lastName属性
  9. // 与传入的name对象的firstName属性和lastName属性相同
  10. this.name = new Name(name.getFirstName(), name.getLastName());
  11. }
  12.  
  13. public Name getName() {
  14. // 直接返回当前实例的name属性即可
  15. return name;
  16. }
  17.  
  18. public static void main(String[] args) {
  19. Name n = new Name("三", "张");
  20. Person p = new Person(n);
  21. System.out.println(p.getName() + " " + p.getName().getFirstName());
  22. // 改变Person对象Name属性的firstName属性值
  23. n.setFirstName("无忌");
  24. System.out.println(p.getName() + " " + p.getName().getFirstName());
  25. }
  26. }

从打印结果可以看出p的name属性的地址和所指内容都没变。

Java中的不可变类的更多相关文章

  1. Java中的不可变类理解

    一.Java中的不可变类 不可变类(Immutable Objects):当类的实例一经创建,其内容便不可改变,即无法修改其成员变量. 可变类(Mutable Objects):类的实例创建后,可以修 ...

  2. 《Java中的不可变类》

    //不可变类举例: /* 下面程序试图定义一个不可变类Person类,但=因为Person类包含一个引用类型的成员变量, 且这个引用类是可变类,所以导致Person类也变成了可变类. */ class ...

  3. 关于java中的不可变类(转)

    如何在Java中写出Immutable的类? 要写出这样的类,需要遵循以下几个原则: 1)immutable对象的状态在创建之后就不能发生改变,任何对它的改变都应该产生一个新的对象. 2)Immuta ...

  4. 深入理解Java中的不可变对象

    深入理解Java中的不可变对象 不可变对象想必大部分朋友都不陌生,大家在平时写代码的过程中100%会使用到不可变对象,比如最常见的String对象.包装器对象等,那么到底为何Java语言要这么设计,真 ...

  5. 为什么 String 在 Java 中是不可变的(终极答案)

    为什么 String 在 Java 中是不可变的(终极答案) 我们可以从2个角度去看待这个问题: 1.为什么要设计成不可变2.如何保证不可变? 1.为什么设计不可变? 1.String对象缓存在Str ...

  6. 为什么 String 在 Java 中是不可变的?

    我最喜欢的 Java 面试问题,很棘手,但同时也非常有用.一些面试者也常问这个问题,为什么 String 在 Java 中是 final 的.字符串在 Java 中是不可变的,因为 String 对象 ...

  7. JAVA中封装JSONUtils工具类及使用

    在JAVA中用json-lib-2.3-jdk15.jar包中提供了JSONObject和JSONArray基类,用于JSON的序列化和反序列化的操作.但是我们更习惯将其进一步封装,达到更好的重用. ...

  8. Java中直接输出一个类的对象

    例如 package com.atguigu.java.fanshe; public class Person { String name; private int age; public Strin ...

  9. Java中Date和Calender类的使用方法

    查看文章     Java中Date和Calender类的使用方法 2009-10-04 20:49 Date和Calendar是Java类库里提供对时间进行处理的类,由于日期在商业逻辑的应用中占据着 ...

随机推荐

  1. josn 转php

    $data = josn_decode(data,[true]); 加true转化为php数组:不加为对象,使用:$data->'字段'.

  2. js页面加载事件

    <body onload="myfunction()" > </body> <script type="text/javascript&qu ...

  3. js面向对象--类式继承

    //待研究//类式继承 //js中模拟类式继承的3个函数 //简单的辅助函数,让你可以将新函数绑定到对象的 prototype 上 Function.prototype.method = functi ...

  4. application/json IE 兼容问题

    由于IE系列浏览器把application/json响应视为文件,并尝试下载在网上看了一下,大致了解,只要修改返回的内容的类型(ContentType)即可解决问题. 由于ajax请求,返回类型默认就 ...

  5. 【转载】ASP.NET线程安全与静态变量的生命周期浅谈

    ASP.NET线程安全所涉及的是什么呢?让我们先来看看静态变量的生命周期问题,下面是我理解的静态变量的生命周期: void Application_Start开始 void Application_E ...

  6. oracle问题 《经由直接路径由 EXPORT:V10.02.01 创建的导出文件 IMP-00013: 只有 DBA 才能导入由其他 DBA 导出的文件》

    问题:  经由直接路径由 EXPORT:V10.02.01 创建的导出文件 : 只有 DBA 才能导入由其他 DBA 导出的文件 解决方法:用sys 登录,给当前用户授权,授权语句:grant dba ...

  7. Message,MessageQueue,Looper,Handler ——由view.post(runnable想到的)

    近日看到代码有view.post(runable),发现对handler机制又有些模糊,故做些复习. 这里就不再对具体的源码原理做深入复习了,就抄一些基本的结论吧. 1.基本概念 Message:基本 ...

  8. 四大跨平台的APP分析

    转载:http://blog.csdn.net/kenkao/article/details/50678269

  9. python下 help()使用方法

    查看python所有的modules:help("modules") 单看python所有的modules中包含指定字符串的modules: help("modules ...

  10. 【USACO 1.5.4】跳棋的挑战

    [问题描述] 检查一个如下的6 x 6的跳棋棋盘,有六个棋子被放置在棋盘上,使得每行,每列,每条对角线(包括两条主对角线的所有对角线)上都至多有一个棋子,如下例,就是一种正确的布局. 上面的布局可以用 ...