java 는 AOP(로그 기록)의 인 스 턴 스 코드 를 동적 프 록 시 로 구현 합 니 다.

다음은 AOP 가 실현 하 는 간단 한 예 입 니 다.

우선 업무 방법 을 정의 합 니 다.

/**
 * Created with IntelliJ IDEA.
 * Author: wangjie  email:[email protected]
 * Date: 13-9-23
 * Time: 3:49
 */
public interface BussinessService {
    public String login(String username, String password);
    public String find();
}

public class BussinessServiceImpl implements BussinessService {
    private Logger logger = Logger.getLogger(this.getClass().getSimpleName());

    @Override
    public String login(String username, String password) {
        return "login success";
    }

    @Override
    public String find() {
        return "find success";
    }

}


/**
 * Created with IntelliJ IDEA.
 * Author: wangjie  email:[email protected]
 * Date: 13-9-24
 * Time: 10:27
 */
public interface WorkService {
    public String work();
    public String sleep();
}

public class WorkServiceImpl implements WorkService{
    @Override
    public String work() {
        return "work success";
    }

    @Override
    public String sleep() {
        return "sleep success";
    }
}

은 Invocation Handler 인 터 페 이 스 를 실현 하고 map 를 사용 하여 서로 다른 Invocation Handler 대상 을 저장 하여 너무 많은 생 성 을 피한다.

package com.wangjie.aoptest2.invohandler;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.util.Arrays;
import java.util.HashMap;
import java.util.logging.Logger;

/**
 * Created with IntelliJ IDEA.
 * Author: wangjie  email:[email protected]
 * Date: 13-9-23
 * Time: 3:47
 */
public class LogInvoHandler implements InvocationHandler{
    private Logger logger = Logger.getLogger(this.getClass().getSimpleName());

    private Object target; //
    private Object proxy; //

    private static HashMap<Class<?>, LogInvoHandler> invoHandlers = new HashMap<Class<?>, LogInvoHandler>();

    private LogInvoHandler() {
    }

    /**
     * Class Proxy
     * @param clazz
     * @return
     */
    public synchronized static<T> T getProxyInstance(Class<T> clazz){
        LogInvoHandler invoHandler = invoHandlers.get(clazz);

        if(null == invoHandler){
            invoHandler = new LogInvoHandler();
            try {
                T tar = clazz.newInstance();
                invoHandler.setTarget(tar);
                invoHandler.setProxy(Proxy.newProxyInstance(tar.getClass().getClassLoader(),
                        tar.getClass().getInterfaces(), invoHandler));
            } catch (Exception e) {
                e.printStackTrace();
            }
            invoHandlers.put(clazz, invoHandler);

        }

        return (T)invoHandler.getProxy();
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {

        Object result = method.invoke(target, args); //

        //
        logger.info("____invoke method: " + method.getName()
                    + "; args: " + (null == args ? "null" : Arrays.asList(args).toString())
                    + "; return: " + result);


        return result;
    }

    public Object getTarget() {
        return target;
    }

    public void setTarget(Object target) {
        this.target = target;
    }

    public Object getProxy() {
        return proxy;
    }

    public void setProxy(Object proxy) {
        this.proxy = proxy;
    }
}

그리고 Test 클래스 테스트 를 작성 합 니 다:

/**
 * Created with IntelliJ IDEA.
 * Author: wangjie  email:[email protected]
 * Date: 13-9-24
 * Time: 9:54
 */
public class Test {
    public static Logger logger = Logger.getLogger(Test.class.getSimpleName());
    public static void main(String[] args) {

        BussinessService bs = LogInvoHandler.getProxyInstance(BussinessServiceImpl.class);
        bs.login("zhangsan", "123456");
        bs.find();

        logger.info("--------------------------------------");

        WorkService ws = LogInvoHandler.getProxyInstance(WorkServiceImpl.class);
        ws.work();
        ws.sleep();

        logger.info("--------------------------------------");

        BussinessService bss = LogInvoHandler.getProxyInstance(BussinessServiceImpl.class);
        bss.login("lisi", "654321");
        bss.find();

    }
}

이후 새로운 업무 논리 XXxService 를 추가 해 야 합 니 다.호출 만 하면 됩 니 다.
XXXService xs = LogInvoHandler.getProxyInstance(XXXServiceImpl.class);
됐다.
Spring 등 프레임 워 크 의 설정 을 모방 하여 bean 의 클래스 이름 을 xml 파일 에 설정 할 수 있 습 니 다.예 를 들 어:

그리고 자바 코드 에서 xml 를 분석 하고 Class.forName("com.wangjie.aoptest 2.service.impl.BusinessServiceImpl")을 통 해 처리 합 니 다.Class 대상 획득
그리고 LogInvoHandler.getProxyInstance(Class.forName("com.wangjie.aoptest 2.service.impl.BusinessServiceImpl")를 통 해프 록 시 대상 프 록 시 가 져 오기
대리 대상 을 반사 적 으로 호출 하 는 방법 을 사용한다.
 
실행 결 과 는 다음 과 같 습 니 다.
9 월 24,2013 11:08:03 오전 com.wangjie.aoptest 2.invohandler.LogInvoHandler invokeINFO:invoke method: login; args: [zhangsan, 123456]; return:login success 9 월 24,2013 11:08:03 오전 com.wangjie.aoptest 2.invohhandler.LogInvoHandler invokeINFO:invoke method: find; args: null; return:find success 9 월 24,2013 11:08:03 오전 com.wangjie.aoptest 2.Test main INFO:-------------------------------------------------------------------9 월 24,2013 11:08:03 오전 com.wangjie.aoptest 2.invohandinvoke method: work; args: null; return:work success 9 월 24,2013 11:08:03 오전 com.wangjie.aoptest 2.invohhandler.LogInvoHandler invokeINFO:invoke method: sleep; args: null; return:sleep success 9 월 24,2013 11:08:03 오전 com.wangjie.aoptest 2.Test main INFO:-------------------------------------------------------------------------------------------invoke method: login; args: [lisi, 654321]; return:login success 9 월 24,2013 11:08:03 오전 com.wangjie.aoptest 2.invohhandler.LogInvoHandler invokeINFO:invoke method: find; args: null; return: find success

좋은 웹페이지 즐겨찾기