一、原则:
软件实体应当对扩展开放,对修改关闭。当应用的需求改变时,在不修改软件实体的源代码或者二进制代码的前提下,可以扩展模块的功能,使其满足新的需求。
二、软件实体包含哪些?
1.项目中划分出的模块
2.类与接口
3.方法
三、作用:
1.对软件测试的影响。软件遵守开闭原则的话,软件测试时只需要对扩展的代码进行测试就可以了,因为原有的测试代码仍然能够正常运行。
2.提高代码的可复用性。粒度越小,被复用的可能性就越大;在面向对象的程序设计中,根据原子和抽象变成可以提高代码的可复用性。
3.可以提高软件的可维护性。遵守开闭原则的软件,其稳定性高和延续性强,从而易于扩展和维护。
四、实现方法:
1.可以通过“抽象约束,封装变化”来实现开闭原则,即通过接口或者抽象类为软件实体定义一个相对稳定的抽象层,而将相同的可变因素封装在相同的具体实现类中。
2.因为抽象灵活性好,适应性广,只要抽象的合理,可以基本保持软件架构的稳定。而软件中易变的细节可以从抽象派生来的实现类来进行扩展,当软件需要发生变化时,只需要根据需求重新派生一个实现类来扩展就可以了。
五、哪些设计模式用到了这个原则?
抽象工厂模式,简单工厂模式,工厂模式。
六、示例:
比如数据库中存储的用户的身份证和姓名都是加密的,平时使用时需要用到未加密的身份证和姓名信息,因此在用户的实体类中加两个get方法,用于获取未加密的身份证和姓名信息,get方法里面添加解密功能。但是不会重写原来获取加密身份证和姓名的方法:
import lombok.Data;
import javax.persistence.*;
import java.io.Serializable;
import java.util.Date;
@Table(name = "`user`")
@Data
public class User implements Serializable {
@Id
@Column(name = "`sid`")
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long sid;
...
/**
* 加密联系人
*/
@Column(name = "`encryption_contact_name`")
private String encryptionContactName;
/**
* 加密联系电话
*/
@Column(name = "`encryption_contact_mobile`")
private String encryptionContactMobile;
...private static final long serialVersionUID = 1L;
//这里是新增的获取明文信息的方法
public String getContactName() {
if (StringUtil.isNotBlank(this.encryptionContactName)) {
return EncryptUtil.decryptName(this.encryptionContactName);
}
return null;
}
public void setContactName(String contactName) {
this.encryptionContactName = EncryptUtil.encryptName(contactName);
}
public String getContactMobile() {
if (StringUtil.isNotBlank(this.encryptionContactMobile)) {
return EncryptUtil.decryptPhone(this.encryptionContactMobile);
}
return null;
}
public void setContactMobile(String contactMobile) {
this.encryptionContactMobile = EncryptUtil.encryptPhone(contactMobile);
}
}