>  기사  >  Java  >  Java의 정적 프록시와 동적 프록시의 네 가지 구현 방법 소개

Java의 정적 프록시와 동적 프록시의 네 가지 구현 방법 소개

不言
不言앞으로
2018-10-22 14:53:532796검색

이 글은 Java에서 정적 프록시와 동적 프록시의 네 가지 구현 방법을 소개합니다. 필요한 친구들이 참고할 수 있기를 바랍니다.

인터뷰 질문: Java의 프록시 디자인 패턴에 대한 구현 방법은 몇 개입니까? 이 질문은 Kong Yiji의 질문인 "회향콩에 fennel이라는 단어를 쓰는 방법은 무엇입니까?"(하단의 RealSubject)와 매우 유사하지만 프록시(Proxy)를 호출하여 실제 개체를 간접적으로 호출합니다.

클라이언트가 실제 객체에 직접 접근하는 것을 원하지 않거나, 실제 객체에 접근하는 데 기술적인 문제가 있기 때문에 일반적으로 프록시 모드를 사용하므로 프록시 객체를 브리지로 사용하여 간접 완성 입장.

Java의 정적 프록시와 동적 프록시의 네 가지 구현 방법 소개

구현 방법 1: 정적 프록시

코드를 작성하기 위한 writeCode 메소드가 포함된 IDeveloper 인터페이스를 개발합니다.

public interface IDeveloper {

     public void writeCode();

}

이 인터페이스를 구현하려면 개발자 클래스를 생성하세요.

public class Developer implements IDeveloper{
    private String name;
    public Developer(String name){
        this.name = name;
    }
    @Override
    public void writeCode() {
        System.out.println("Developer " + name + " writes code");
    }
}

테스트 코드: Jerry라는 개발자 인스턴스를 만들고 코드를 작성하세요!

public class DeveloperTest {
    public static void main(String[] args) {
        IDeveloper jerry = new Developer("Jerry");
        jerry.writeCode();
    }
}

이제 문제가 발생합니다. Jerry의 프로젝트 관리자는 Jerry가 어떠한 문서도 유지하지 않고 코드만 작성했다는 사실에 매우 불만족했습니다. 어느 날 Jerry가 휴가를 떠났고, 다른 프로그래머들이 Jerry의 작업을 대신하기 위해 와서 물음표가 붙은 낯선 코드를 보았다고 가정해 보겠습니다. 전체 그룹의 논의 끝에 각 개발자가 코드를 작성할 때 문서를 동시에 업데이트해야 한다는 결정이 내려졌습니다.

모든 프로그래머가 코드 작성 작업에 영향을 주지 않고 개발 시 문서 작성을 기억하도록 하기 위해 원래 Developer 클래스를 수정하지 않고 새 클래스도 구현합니다. . 이 새로운 클래스 DeveloperProxy는 원래 IDeveloper 인스턴스를 가리키는 멤버 변수를 내부적으로 유지 관리합니다.

public class DeveloperProxy implements IDeveloper{
    private IDeveloper developer;
    public DeveloperProxy(IDeveloper developer){
        this.developer = developer;
    }
    @Override
    public void writeCode() {
        System.out.println("Write documentation...");
        this.developer.writeCode();
    }
}

이 프록시 클래스에 의해 구현된 writeCode 메서드에서는 실제 프로그래머의 writeCode 메서드를 호출하기 전에 문서가 추가됩니다. 호출, 이렇게 하면 프로그래머가 코드를 작성할 때 문서 업데이트가 수반됩니다.

테스트 코드:

Java의 정적 프록시와 동적 프록시의 네 가지 구현 방법 소개

정적 프록시 방법의 장점#🎜 🎜#

1. 이해하고 구현하기 쉽습니다

2. 동적 프록시와 비교하여 프록시 클래스와 실제 클래스 간의 관계는 컴파일 타임에 정적으로 결정됩니다. 실행 중 추가 오버헤드 없이 곧 도입될 예정입니다.

정적 프록시 방법의 단점

모든 실제 클래스에는 새 프록시 클래스를 만들어야 합니다. 위의 문서 업데이트를 예로 들어, 상사가 테스트 엔지니어에게 새로운 요구 사항을 제시하여 테스트 엔지니어가 버그를 발견할 때마다 적시에 해당 테스트 문서를 업데이트하도록 요청한다고 가정해 보겠습니다. 그런 다음 정적 프록시 방법을 사용하여 테스트 엔지니어의 구현 클래스 ITester는 해당 ITesterProxy 클래스도 생성해야 합니다.

public interface ITester {
    public void doTesting();
}
Original tester implementation class:
public class Tester implements ITester {
    private String name;
    public Tester(String name){
        this.name = name;
    }
    @Override
    public void doTesting() {
        System.out.println("Tester " + name + " is testing code");
    }
}
public class TesterProxy implements ITester{
    private ITester tester;
    public TesterProxy(ITester tester){
        this.tester = tester;
    }
    @Override
    public void doTesting() {
        System.out.println("Tester is preparing test documentation...");
        tester.doTesting();
    }
}
Java의 동적 프록시 구현 방법이 탄생한 것은 바로 이러한 정적 코드 방법의 단점 때문입니다.

Java 동적 프록시 구현 방법 1: InvocationHandler

InvocationHandler의 원리 소개할 특별한 글을 작성했습니다: Java 동적 프록시 InvocationHandler는 가장 간단한 입문 튜토리얼

InvocationHandler를 통해 EnginnerProxy 프록시 클래스를 사용하여 개발자와 테스터 모두의 동작을 프록시할 수 있습니다.

public class EnginnerProxy implements InvocationHandler {
    Object obj;
    public Object bind(Object obj)
    {
        this.obj = obj;
        return Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj
        .getClass().getInterfaces(), this);
    }
    @Override
    public Object invoke(Object proxy, Method method, Object[] args)
    throws Throwable
    {
        System.out.println("Enginner writes document");
        Object res = method.invoke(obj, args);
        return res;
    }
}
실제 클래스의 writeCode 및 doTesting 메소드는 동적 프록시 클래스의 리플렉션을 통해 실행됩니다.

테스트 출력:

Java의 정적 프록시와 동적 프록시의 네 가지 구현 방법 소개InvocationHandler를 통한 동적 프록시 제한#🎜🎜 ## 🎜🎜#인터페이스를 구현하지 않는 제품 관리자 클래스(ProductOwner)가 있다고 가정합니다.

public class ProductOwner {
    private String name;
    public ProductOwner(String name){
        this.name = name;
    }
    public void defineBackLog(){
        System.out.println("PO: " + name + " defines Backlog.");
    }
}
여전히 EnginnerProxy 프록시 클래스를 사용하여 프록시를 사용하고 있으며 컴파일 중에 오류가 발생하지 않습니다. 런타임에는 어떤 일이 발생하나요?

ProductOwner po = new ProductOwner("Ross");

ProductOwner poProxy = (ProductOwner) new EnginnerProxy().bind(po);

poProxy.defineBackLog();
실행시 오류가 발생합니다. 따라서 제한 사항은 다음과 같습니다. 프록시된 클래스가 인터페이스를 구현하지 않으면 InvocationHandler 동적 프록시를 통해 해당 동작을 프록시할 수 없습니다.

Java 동적 프록시 구현 방법 2: CGLIBJava의 정적 프록시와 동적 프록시의 네 가지 구현 방법 소개#🎜 🎜#CGLIB는 Java 바이트코드를 생성하고 수정하기 위해 사용하기 쉬운 API를 제공하는 Java 바이트코드 생성 라이브러리입니다. 이 오픈 소스 라이브러리에 대한 자세한 내용을 보려면 github의 CGLIB 저장소(https://github.com/cglib/cglib

)를 방문하세요. 이제 InvocationHandler를 사용하기 전에 프록시에 CGLIB를 사용하려고 합니다. ProductOwner 클래스를 성공적으로 프록시했습니다(이 클래스는 인터페이스를 구현하지 않습니다). 이제 프록시 클래스를 생성하기 위해 대신 CGLIB API를 사용합니다:

public class EnginnerCGLibProxy {
    Object obj;
    public Object bind(final Object target)
    {
        this.obj = target;
        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(obj.getClass());
        enhancer.setCallback(new MethodInterceptor() {
            @Override
            public Object intercept(Object obj, Method method, Object[] args,
            MethodProxy proxy) throws Throwable
            {
                System.out.println("Enginner 2 writes document");
                Object res = method.invoke(target, args);
                return res;
            }
        }
        );
        return enhancer.create();
    }
}

테스트 코드:

ProductOwner ross = new ProductOwner("Ross");

ProductOwner rossProxy = (ProductOwner) new EnginnerCGLibProxy().bind(ross);

rossProxy.defineBackLog();

尽管ProductOwner未实现任何代码,但它也成功被代理了:

Java의 정적 프록시와 동적 프록시의 네 가지 구현 방법 소개

用CGLIB实现Java动态代理的局限性

如果我们了解了CGLIB创建代理类的原理,那么其局限性也就一目了然。我们现在做个实验,将ProductOwner类加上final修饰符,使其不可被继承:

Java의 정적 프록시와 동적 프록시의 네 가지 구현 방법 소개

再次执行测试代码,这次就报错了: Cannot subclass final class XXXX。

所以通过CGLIB成功创建的动态代理,实际是被代理类的一个子类。那么如果被代理类被标记成final,也就无法通过CGLIB去创建动态代理。

Java动态代理实现方式三:通过编译期提供的API动态创建代理类

假设我们确实需要给一个既是final,又未实现任何接口的ProductOwner类创建动态代码。除了InvocationHandler和CGLIB外,我们还有最后一招:

我直接把一个代理类的源代码用字符串拼出来,然后基于这个字符串调用JDK的Compiler(编译期)API,动态的创建一个新的.java文件,然后动态编译这个.java文件,这样也能得到一个新的代理类。

Java의 정적 프록시와 동적 프록시의 네 가지 구현 방법 소개

测试成功:

Java의 정적 프록시와 동적 프록시의 네 가지 구현 방법 소개

我拼好了代码类的源代码,动态创建了代理类的.java文件,能够在Eclipse里打开这个用代码创建的.java文件,

Java의 정적 프록시와 동적 프록시의 네 가지 구현 방법 소개

Java의 정적 프록시와 동적 프록시의 네 가지 구현 방법 소개

下图是如何动态创建ProductPwnerSCProxy.java文件:

Java의 정적 프록시와 동적 프록시의 네 가지 구현 방법 소개

下图是如何用JavaCompiler API动态编译前一步动态创建出的.java文件,生成.class文件:

Java의 정적 프록시와 동적 프록시의 네 가지 구현 방법 소개

下图是如何用类加载器加载编译好的.class文件到内存:

Java의 정적 프록시와 동적 프록시의 네 가지 구현 방법 소개

如果您想试试这篇文章介绍的这四种代理模式(Proxy Design Pattern), 请参考我的github仓库,全部代码都在上面。感谢阅读。

https://github.com/i042416/Ja...

위 내용은 Java의 정적 프록시와 동적 프록시의 네 가지 구현 방법 소개의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
이 기사는 segmentfault.com에서 복제됩니다. 침해가 있는 경우 admin@php.cn으로 문의하시기 바랍니다. 삭제