Java学习者论坛

 找回密码
 立即注册

QQ登录

只需一步,快速开始

手机号码,快捷登录

恭喜Java学习者论坛(https://www.javaxxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,购买链接:点击进入购买VIP会员
JAVA高级面试进阶视频教程Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程

Go语言视频零基础入门到精通

Java架构师3期(课件+源码)

Java开发全终端实战租房项目视频教程

SpringBoot2.X入门到高级使用教程

大数据培训第六期全套视频教程

深度学习(CNN RNN GAN)算法原理

Java亿级流量电商系统视频教程

互联网架构师视频教程

年薪50万Spark2.0从入门到精通

年薪50万!人工智能学习路线教程

年薪50万!大数据从入门到精通学习路线年薪50万!机器学习入门到精通视频教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程 MySQL入门到精通教程
查看: 394|回复: 0

[Java基础知识]Java中的可变对象和不可变对象

[复制链接]
  • TA的每日心情
    开心
    2021-3-12 23:18
  • 签到天数: 2 天

    [LV.1]初来乍到

    发表于 2014-9-30 17:39:33 | 显示全部楼层 |阅读模式
    一、简单定义
        不可变对象(Immutable Objects)即对象一旦被创建它的状态(对象的数据,也即对象属性值)就不能改变,反之即为可变对象(Mutable Objects)。

        不可变对象的类即为不可变类(Immutable Class)。java平台类库中包含许多不可变类,如String、基本类型的包装类、BigInteger和BigDecimal等。 二、优缺点
    不可变对象有很多优点:
    (1)构造、测试和使用都很简单
    (2)线程安全且没有同步问题,不需要担心数据会被其它线程修改
    (3)当用作类的属性时不需要保护性拷贝
    (4)可以很好的用作Map键值和Set元素

        不可变对象最大的缺点就是创建对象的开销,因为每一步操作都会产生一个新的对象  
      
       
       

         
       

         
       
      
    三、编写不可变类
    可以遵照以下几点来编写一个不可变类:
    (1)确保类不能被继承 - 将类声明为final, 或者使用静态工厂并声明构造器为private
    (2)声明属性为private 和 final
    (3)不要提供任何可以修改对象状态的方法 - 不仅仅是set方法, 还有任何其它可以改变状态的方法
    (4)如果类有任何可变对象属性, 那么当它们在类和类的调用者间传递的时候必须被保护性拷贝
    代码-1:
    1. import java.util.Date;
    2. /**
    3. * Planet是一个不可变类,因为当它构造完成之后没有办法改变它的状态
    4. */
    5. public final class Planet {
    6.     /**
    7.      * 声明为final的基本类型数据总是不可变的
    8.      */
    9.     private final double fMass;
    10.     /**
    11.      * 不可变的对象属性 (String对象不可变)
    12.      */
    13.     private final String fName;
    14.     /**
    15.      * 可变的对象属性. 在这种情况下, 这个可变属性只能被这个类改变。
    16.      * (在其它情况下, 允许在原生类外部改变一个属性是很有意义的;
    17.      * 这种情况就是当属性作为其它地方创建的一个对象引用)
    18.      */
    19.     private final Date fDateOfDiscovery;
    20.     public Planet(double aMass, String aName, Date aDateOfDiscovery) {
    21.         fMass = aMass;
    22.         fName = aName;
    23.         //创建aDateOfDiscovery的一个私有拷贝
    24.         //这是保持fDateOfDiscovery属性为private的唯一方式, 并且保护这个
    25.         //类不受调用者对于原始aDateOfDiscovery对象所做任何改变的影响
    26.         fDateOfDiscovery = new Date(aDateOfDiscovery.getTime());
    27.     }
    28.     /**
    29.      * 返回一个基本类型值.
    30.      *
    31.      * 调用者可以随意改变返回值,但是不会影响类内部。
    32.      */
    33.     public double getMass() {
    34.         return fMass;
    35.     }
    36.     /**
    37.      * 返回一个不可变对象
    38.      *
    39.      * 调用者得到内部属性的一个直接引用. 由于String是不可变的所以没什么影响
    40.      */
    41.     public String getName() {
    42.         return fName;
    43.     }
    44. // /**
    45. // * 返回一个可变对象 - 不是一个好的方式.
    46. // *
    47. // * 调用者得到内部属性的一个直接引用. 这通常很危险,因为Date对象既可以
    48. // * 被这个类改变也可以被它的调用者改变.即,类不再对fDate拥有绝对的控制。
    49. // */
    50. // public Date getDateOfDiscovery() {
    51. // return fDateOfDiscovery;
    52. // }
    53.     /**
    54.      * 返回一个可变对象 - 好的方式.
    55.      *
    56.      * 返回属性的一个保护性拷贝.调用者可以任意改变返回的Date对象,但是不会
    57.      * 影响类的内部.为什么? 因为它们没有fDate的一个引用. 更准确的说, 它们
    58.      * 使用的是和fDate有着相同数据的另一个Date对象
    59.      */
    60.     public Date getDateOfDiscovery() {
    61.         return new Date(fDateOfDiscovery.getTime());
    62.     }
    63.     /**
    64.      * 测试方法
    65.      * @param args
    66.      */
    67.     public static void main(String[] args) {
    68.         Planet planet = new Planet(1.0D, "earth", new Date());
    69.         Date date = planet.getDateOfDiscovery();
    70.         date.setTime(111111111L);
    71.         System.out.println("the value of fDateOfDiscovery of internal class : " +
    72.                    planet.fDateOfDiscovery.getTime());
    73.         System.out.println("the value of date after change its value : " + date.getTime());
    74.     }
    复制代码
    运行结果如下:
       the value of fDateOfDiscovery of internal class : 1393943752205
      the value of date after change its value : 111111111  由此可见Planet类的属性fDateOfDiscovery在对象构造完成之后就没有再改变。在《Effective Java》一书中, Joshua Bloch提出了一个强制性的建议 :
       "类应该是不可变的,除非有很好的理由让它是可变的....如果一个类不能设计为不可变的,也要尽可能的限制它的可变性."  BigDecimal从技术上讲不是不可变的, 因为它没有声明为final. 四、使用场景
        不可变类最适合表示抽象数据类型(如数字、枚举类型或颜色)的值。Java 类库中的基本数据类型的包装类(如Integer 、 Long 和 Float )都是不可变的,其它数字类型(如 BigInteger 和 BigDecimal )也是不可变的。表示复数或任意精度的有理数的类将比较适合设计为不可变类。甚至包含许多离散值的抽象类型(如向量或矩阵)也很适合设计成不可变类,这取决于你的应用程序。    另一个适合用不可变类实现的好示例就是事件 。事件的生命期较短,而且常常会在创建它们的线程之外的线程中消耗,所以使它们成为不可变的是利大于弊。大多数 AWT 事件类都没有严格的作为不可变类来实现。同样地,在通信系统的组件间进行消息传递,将消息对象设计成不可变的是明智的。



      
      
       
       

         
       

         
       
      
    复制代码
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|Java学习者论坛 ( 声明:本站资料整理自互联网,用于Java学习者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2024-5-20 17:03 , Processed in 0.369156 second(s), 46 queries .

    Powered by Discuz! X3.4

    © 2001-2017 Comsenz Inc.

    快速回复 返回顶部 返回列表