Summary:类中的某个字段应该在对象建立时被设值,而后就再也不改变。去掉该字段的全部设值函数。java
动机: 程序员
若是你为某个字段提供了设值函数,这就暗示这个字段值能够被改变。若是你不但愿在对象建立以后此字段还有机会被改变,那就不要为它提供设值函数(同时将该字段设为final)。这样你的意图会更加清晰,而且能够排除其值被修改的可能性—这种可能性每每是很是大的。函数
若是你保留了间接访问变量的方法,就可能常常有程序员盲目使用它们。这些人甚至会在构造函数中使用设值函数!他们或许是为了代码的一致性,但却忽视了设值函数日后可能带来的混淆。测试
作法:spa
1. 检查设值函数被使用的状况,看它是否只被构造函数调用,或者被构造函数所调用的另外一个函数调用。rest
2.修改构造函数,使其直接访问设值函数所针对的那个变量。code
à若是某个子类经过设值函数给超类的某个private字段设了值,那么你就不能这样修改。这种状况下你应该试着在超类中提供一个protected函数(最好是构造函数)来给这些字段设值。不论你怎么作,都不要给超类中的函数起一个与设值函数混淆的名字。对象
3. 编译,测试。rem
4.移除这个设值函数,将它所针对的字段设为finalget
5.编译,测试。
范例:
下面是一个简单的例子:
class Account{ private String _id; Account(String id){ setId(id); } void setId(String arg){ _id = arg; } }
以上代码可修改成:
class Account{ private String _id; Account(String id){ _id = id; } }
问题可能以几种不一样的形式出现。首先,你可能会在设值函数中对传入的参数作运算:
class Account{ private String _id; Account(String id){ setId(id); } void setId(String arg){ _id = "zz" + arg; } }
若是对参数的运算很简单(就像上面这样)并且又只有一个构造函数,咱们能够直接在构造函数中作相同的修改。若是修改很复杂,或者有一个以上的函数调用它,就须要提供一个独立函数。咱们须要为新函数起个好名字,清楚表达改函数的用途:
class Account{ private String _id; Account(String id){ initializeId(id); } void initializeId(String arg){ _id = "zz" + arg; } }
若是须要对超类的private变量赋初值,状况就比较麻烦一些:
class InterestAccount extends Account ... private double _interestRate; InterestAccount(String id, double rate){ setId(id); _interestRate = rate; }
问题是咱们没法在InterestAccount中直接访问id变量。最好的解决办法就是使用超类构造函数:
class InterestAccount ... InterestAccount(String id, doulbe rate){ super(id); _interestRate = rate; }
若是不能那样作,那么使用给一个命名良好的函数就是最好的选择:
class InterestAccount ... InterestAccount(String id, doulbe rate){ initializeId(id); _interestRate = rate; }
另一种须要考虑的状况就是对一个集合设值:
class Person{ Vector getCourses(){ return _courses; } void setCourses(Vector arg){ _courses = arg } private Vector _courses; }
在这里,咱们应该将设值函数替换为add操做和remove操做,Encapsulate Collection中谈到了这一点