来自尚学堂Spring视频教程;作文字总结。
系统中现有接口UserDAO及其实现UseDAOImpl:
package com.bjsxt.dao;
import com.bjsxt.model.User;
public interface UserDAO {
public void save(User user);
}
package com.bjsxt.dao.impl;
import com.bjsxt.dao.UserDAO;
import com.bjsxt.model.User;
public class UserDAOImpl implements UserDAO {
public void save(User user) {
//Hibernate
//or JDBC
System.out.println("user saved!");
}
}
现在需要在save方法前记录日志,如:
package com.bjsxt.dao.impl;
import com.bjsxt.dao.UserDAO;
import com.bjsxt.model.User;
public class UserDAOImpl implements UserDAO {
public void save(User user) {
//Hibernate
//or JDBC
System.out.println("logging......");
System.out.println("user saved!");
}
}
当然,像这样直接在源代码里添加当然是最简单的了。但如果拿不到源代码呢?此时可以新写一个类来继承UserDAOImpl,用delegation的做法:
package com.bjsxt.dao.impl;
import com.bjsxt.model.User;
public class UserDAOImpl2 extends UserDAOImpl {
public void save(User user) {
System.out.println("logging......");
super.save(user);
}
}
用组合的方式也可以,如:
package com.bjsxt.dao.impl;
import com.bjsxt.dao.UserDAO;
import com.bjsxt.model.User;
public class UserDAOImpl3 implements UserDAO {
private UserDAO userDAO = new UserDAOImpl();
public void save(User user) {
System.out.println("logging......");
userDAO.save(user);
}
}
这样,把系统的beans.xml文件中配置的UserDAOImpl改成UserDAOImpl2或者UserDAOImpl3,系统就有了记录日志的的功能了。
这三种添加日志功能的方法中:
(1) 直接在源码中写死是最不灵活的方式,因为万一要修改日志功能,要在UserDAOImpl的代码中去搜索日志功能的代码,耦合度太高
(2) 继承的方式也不够灵活。首先UserDAOImpl1这个类无法再继承其他的类;再次,父类的改动必然引起子类的改动,耦合度太高
(3) 组合没有耦合度太高的问题;成员private UserDAO userDAO可以使用多态;继承只能继承一个Impl,而组合更为灵活,可以组合多个Impl,如同时组合UserDAOImpl和ForumDAOImpl。这3种方法中,组合的方式最好,在设计模式中也经常用组合来替代继承
另:这里的继承和组合都是静态代理,对应于[#0x0042]中的动态代理