为了不修改原 ServiceImpl 的业务代码,而对其进行扩充(统计、打印耗时)和拦截(判断是否登录),可以使用代理。
代码如下
1、让 ServiceImpl 处理原先核心的业务逻辑
package constxiong.cxproxy.chapter3.service;
import java.util.HashMap;
import java.util.Map;
/**
* 服务接口实现
* @author ConstXiong
* @date 2019-05-29 11:02:15
*/
public class ServiceImpl implements Service {
/**
* 登录
*/
@Override
public boolean login(String username, String password) {
simulateDaOperation(100);
System.out.println("用户名:" + username + ", 密码:" + password + " 登录成功");
return true;
}
/**
* 根据用户名获取用户信息
*/
@Override
public Map<String, Object> getUserInfo(String username) {
Map<String, Object> userInfo = new HashMap<String, Object>();
simulateDaOperation(150);
userInfo.put("username", username);
userInfo.put("sex", "男");
userInfo.put("age", 18);
System.out.println("用户名:" + username + ", 获取用户信息:" + userInfo);
return userInfo;
}
/**
* 模拟数据库操作,休眠
* @param millis 毫秒数
*/
private void simulateDaOperation(long millis) {
try {
Thread.sleep(millis);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
2、新增一个代理类,代理 Service 接口。
- 代理类持有 Service 接口的实现
- 代理 login 方法,添加打印耗时的功能代码
- 代理 getUserinfo 方法,添加打印耗时、校验是否登录的功能代码
package constxiong.cxproxy.chapter3.proxy;
import java.util.Map;
import java.util.Random;
import constxiong.cxproxy.chapter3.service.Service;
import constxiong.cxproxy.chapter3.service.ServiceImpl;
/**
* Service 接口的静态代理类
* @author ConstXiong
* @date 2019-06-02 00:10:02
*/
public class StaticServiceProxy implements Service {
//持有Serrvice接口的实现类
private Service service;
public StaticServiceProxy() {
service = new ServiceImpl();
}
@Override
public boolean login(String username, String password) {
long start = System.currentTimeMillis();//计时开始
//调用 ServiceImpl 的登录方法
boolean result = service.login(username, password);
long end = System.currentTimeMillis();//计时结束
System.out.println("耗时:" + (end - start) + "毫秒");//打印耗时
return result;
}
@Override
public Map<String, Object> getUserInfo(String username) {
long start = System.currentTimeMillis();//计时开始
Map<String, Object> result = null;
if (checkIsLogined()) {//校验登录
result = service.getUserInfo(username);//调用 ServiceImpl 的获取用户信息方法
}
long end = System.currentTimeMillis();//计时结束
System.out.println("耗时:" + (end - start) + "毫秒");//打印耗时
return result;
}
/**
* 模拟 当前用户是否登录
*/
private boolean checkIsLogined() {
Random r = new Random();
int i = r.nextInt(10);
if (i % 2 == 0) {
System.out.println("已登录");
return true;
}
System.out.println("未登录");
return false;
}
}
3、 调用测试代码
package constxiong.cxproxy.chapter3;
import constxiong.cxproxy.chapter3.proxy.StaticServiceProxy;
import constxiong.cxproxy.chapter3.service.Service;
/**
* 测试类
* @author ConstXiong
* @date 2019-06-02 00:16:30
*/
public class Test {
public static void main(String[] args) {
Service service = new StaticServiceProxy();
service.login("ConstXiong", "123456");
service.getUserInfo("ConstXiong");
}
}
打印的结果跟在 Chapter 2、不使用代理 中的结果一致。
优点:
- 可以不修改原目标类(指ServiceImpl)的代码
- 可以在代理类中对功能进行拦截和扩充
缺点:
- 代理类需要实现与目标类一样的接口,会导致代理类数量较多,不易维护
- 一旦接口增加方法,目标类和代理类都需要维护
鉴于这两个缺点,JDK 提供了动态代理。
完整源码:https://github.com/ConstXiong/xtools cxproxy项目 chapter3
【Java面试题与答案】整理推荐