重构36计(31-36)

[b]第三十一计:置空不用的对象[/b]
在C++中,销毁一个对象后,一定要把指针置为NULL,否则会出现野指针,最好写成下面这样,delete后立马置为NULL,
delete pObject;
pObject = NULL;
在Java中,当不再需要一个对象时,最好能把它置为null,这样有利于垃圾回收。

[b]第三十二计:善于利用接口[/b]
1、 回调型接口
在C语言中,回调函数可以通过函数指针来实现,Java中没有指针的概念,可以利用接口来达到同样的目的,例如:
public interface Callback{

public void onChanged();

}

public void execute(Callback callback){
...
callback.onChanged();
...
}

2、标记型接口
这种类型的接口中不包含函数的声明,即接口是空的,主要用来让实现这个接口的类表明自身具有某种特性,起一个标记的作用,例如下面的接口:
public interface Millionaire{

}

public class Member extends User implements Millionaire{
...
}

public boolean check(User user){
if(user instanceOf Millionaire) // 这是百万富豪,直接让它pass
return true;

...
}

3、依赖注入型接口
这种接口一般用在工厂方法中,有选择性地为要创建的对象注入对象引用或者数据,例如:
public interface DataAware{
public void setData(Data data);
}

public interface DataBaseConnectionAware{
public void setConnection(Connection conn);

}

// 用户服务类
public class UserService extends Service implements DataAware,DataBaseConnectionAware{
public void setData(Data data);
public void setConnection(Connection conn);
}

// 创建服务对象
public Service createService(int type){
Service service = null;
if(type == USER){
service = new UserService();
if(service instanceOf DataAware) // 如果它想要数据,则注入数据对象
((DataAware)service).setData(data);
if(service instanceOf DataBaseConnectionAware) // 如果它想要数据库连接,则注入连接对象
((DataBaseConnectionAware)service).setConnection(conn);
return service;
}

[b]第三十三计:简化类关系[/b]
如果类之间的关系比较复杂,那么很可能是面向对象的设计没有做好。一般来说,两个类之间有双向调用关系则说明它们在职责上的分配上不够清晰,例如,一个类(A)频繁地调用另一个类(B)的函数,而且A还把成员变量作为参数传递给B,那么大部分情况下就应该把那个函数定义在A中,从而切断两者之间的关系。另外,系统中有很多类对某些类的状态变化或者事件感兴趣,例如添加一个新用户、网络状态的变化等,对于这样的需求,最好通过各种消息机制来达到它们之间的解耦,如观察者模式、发布订阅模式等。
[b]第三十四计:用多态替换相似条件式[/b]
当一个类中有些函数的处理逻辑很相似,都是根据某个状态值或者某个类型值而采取不同的行为,那么这个类就需要被拆分成多个类了,例如有一个表示形状的类:
public class Shape{
private int type;

public void draw(){
if(type == 0) // 如果是矩形
drawRect();
else if(type == 1) // 如果是圆形
drawCircle();
else if(type == 2) // 如果是直线
drawLine();
}

public boolean isPointIn(float x, float y){
if(type == 0) // 如果是矩形
return isPointInRect(x,y);
else if(type == 1) // 如果是圆形
return isPointInCircle(x,y);
else if(type == 2) // 如果是直线
return isPointInLine();
return false;
}
}

这两个成员函数内部包含相似的处理逻辑,都是根据type的值做相应的处理,对于这种情况,应该充分利用面向对象中的多态性,下面是拆分成多个类的形式:
public abstract class Shape{
public abstract void draw();
public abstract boolean isPointIn(x,y);
}

public class Circleextends Shape{
public void draw(){
drawCircle();
}

public abstract boolean isPointIn(x,y){
isPointInCircle(x,y);
}
}

public class Rect extends Shape{
public void draw(){
drawRec();
}

public abstract boolean isPointIn(x,y){
isPointInRect(x,y);
}
}

public class Line extends Shape{
public void draw(){
drawLine();
}

public abstract boolean isPointIn(x,y){
isPointInLine(x,y);
}
}

[b]第三十五计:合理分层,分离界面显示和业务处理逻辑[/b]
代码的低耦合是保障软件可维护性的关键因素之一,而分层是实现代码低耦合的常用手段,也可以实现人员的有效分工,例如在企业级的Web开发中,一般都会划分出以下几个层次:
1、数据访问层,完成数据库的增删改查,一般都使用ORM框架来实现,例如Hibernate、LINQ等。
2、业务服务层,完成所有的业务处理逻辑
3、请求控制层,处理客户端请求,将业务交给服务层处理,把处理结果以某种形式(HTML、JSON等)返回给客户端。
在本地应用程序的实现中也会有类似的分层,例如MFC中的文档-视图结构也是界面显示和业务处理逻辑的分离,所以在代码中,应该尽量避免界面对象和业务对象的交叉引用,尤其是对于有多种界面呈现方式的系统中。

[b]第三十六计:判断参数有效性[/b]
函数参数有效性的判断是函数实现中很重要的一部分,尤其是作为类的对外接口的公有成员函数,例如判断参数中的对象是否为null、参数值是否合法等,这些工作应该在函数实现的最开始处完成,发现参数不合法时可以直接返回、抛出异常或者提供参数默认值,一定不要通过非代码的方法(比如调用约定)来保证参数的的有效性,断言也不能完全保证函数执行时用到的参数都是合法的,除非在程序发布时所有的公有函数都能被执行到,因此,参数有效性的判断是函数实现中必不可少的部分。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值