zoukankan      html  css  js  c++  java
  • Effective Java:Ch4_Class:Item14_在public类中应该使用访问方法而不是public域

            你可能偶尔需要编写退化类,目的只是为了集中实例域:

    // Degenerate classes like this should not be public!
    class Point {
      public double x;
      public double y;
    }


            由于这种类的数据域是可以直接访问的,所以他们不能提供“封装”带来的好处。如果不改变API,就不能改变其数据表示法,不能强加任何约束,当域被访问时不能采取任何辅助行为。强硬派的面向对象程序员会非常厌恶这种类,认为这种类应该被包含private域和public访问方法(getter)的类替代。对于可变类,还应该包含setter。

    // Encapsulation of data by accessor and mutators
    class Point {
      private double x;
      private double y;
    
      public Point(double x, double y) {
        this.x = x;
        this.y = y;
      }
    
      public double getX() { return x; }
      public double getY() { return y; }
      public void setX(double x) { this.x = x; }
      public void setY(double y) { this.y = y; }
    }

            毫无疑问,强硬派的观点在public类上是正确的:如果一个类可以被包外访问,那么就要提供访问方法,以便可以灵活地改变类的内部表示。如果public类暴露了其数据域,则要想在将来改变内部表示是不可能的,因为他的客户端代码可能已经遍布各处了。

            然而,如果类是package-private或是private嵌套类,那么把数据域暴露出去并没有本质的错误——假设这些数据域充分描述了该类提供的抽象。无论是在类定义中,还是在客户端代码中,这种方法相对于访问方法更不会产生视觉混乱。虽然客户端代码与类的内部表示紧密相连,但这些代码被限定在同一个包中。如果需要改变内部数据表示,你不必修改包外的任何代码。如果是private嵌套类,则甚至不需要修改类外的任何代码。


            JDK中的几个类违反了“public类不应该直接暴漏其字段”这个告诫。最显著的例子是java.awt包中的Point类和Dimension类。正如Item55中讲的,决定暴露Dimentsion类的内部实现造成了严重的性能问题,而且这个问题至今还存在。

    ——Component.getSize()返回Dimension,每次调用都要new一个新的对象。


            虽然直接暴露public类的字段永远都不是个好主意,但是如果该域是不可变的(immutable),则危害性就小得多。除非修改其API,否则你不能修改其内部数据表示,而且当别人读取该字段时 你不能采取任何附加行动,但是你可以强加约束条件——因为只有你自己可以set该字段)。例如,下面这个类保证了每个实例都表示一个合法的时间。

    // public class with exposed immutable fields - questionable
    public final class Time {
      private static final int HOUR_PER_DAY = 24;
      private static final int MINUTES_PER_HOUR = 60;
    
      public final int hour;
      public final int minute;
    
      public Time() {
        //强加约束条件
        if (hour < 0 || hour >= HOUR_PER_DAY) {
           throw new IllegalArgumentException();
        if (minute < 0 || minute >= MINUTES_PER_HOUR) {
           throw new IllegalArgumentException();
        
        this.hour = hour;
        this.minute = minute;
      }
    
    }


             总之,public类永远不要暴露其可变的字段。而public类暴露其不可变字段随然是有问题,但危害性要小一些。然而,package-private或者private的嵌套类暴露其字段则是可行的,无论该字段是可变还是非可变。




  • 相关阅读:
    spring security几大核心组件
    什么是CSRF攻击?如何避免?
    MyBatis+Oracle实现主键自增长的几种常用方式
    fastjson在将Map<Integer, String>转换成JSON字符串时,出现中文乱码问题
    Layui我提交表单时,table.reload(),表格会请求2次,是为什么?
    Jquery+ajax模板
    HttpServletRequest和ServletRequest的区别
    java技术基础
    mysql数据库基础
    linux基础03
  • 原文地址:https://www.cnblogs.com/riskyer/p/3217727.html
Copyright © 2011-2022 走看看