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

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

Oct 22, 2018 pm 02:53 PM
java jdk proxy_pass 연기

이 글은 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 중국어 웹사이트의 기타 관련 기사를 참조하세요!

본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover

AI Clothes Remover

사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

Video Face Swap

Video Face Swap

완전히 무료인 AI 얼굴 교환 도구를 사용하여 모든 비디오의 얼굴을 쉽게 바꾸세요!

뜨거운 도구

메모장++7.3.1

메모장++7.3.1

사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전

SublimeText3 중국어 버전

중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

드림위버 CS6

드림위버 CS6

시각적 웹 개발 도구

SublimeText3 Mac 버전

SublimeText3 Mac 버전

신 수준의 코드 편집 소프트웨어(SublimeText3)

자바의 완전수 자바의 완전수 Aug 30, 2024 pm 04:28 PM

Java의 완전수 가이드. 여기서는 정의, Java에서 완전 숫자를 확인하는 방법, 코드 구현 예제에 대해 논의합니다.

자바의 웨카 자바의 웨카 Aug 30, 2024 pm 04:28 PM

Java의 Weka 가이드. 여기에서는 소개, weka java 사용 방법, 플랫폼 유형 및 장점을 예제와 함께 설명합니다.

Java의 스미스 번호 Java의 스미스 번호 Aug 30, 2024 pm 04:28 PM

Java의 Smith Number 가이드. 여기서는 정의, Java에서 스미스 번호를 확인하는 방법에 대해 논의합니다. 코드 구현의 예.

Java Spring 인터뷰 질문 Java Spring 인터뷰 질문 Aug 30, 2024 pm 04:29 PM

이 기사에서는 가장 많이 묻는 Java Spring 면접 질문과 자세한 답변을 보관했습니다. 그래야 면접에 합격할 수 있습니다.

Java 8 Stream foreach에서 나누거나 돌아 오시겠습니까? Java 8 Stream foreach에서 나누거나 돌아 오시겠습니까? Feb 07, 2025 pm 12:09 PM

Java 8은 스트림 API를 소개하여 데이터 컬렉션을 처리하는 강력하고 표현적인 방법을 제공합니다. 그러나 스트림을 사용할 때 일반적인 질문은 다음과 같은 것입니다. 기존 루프는 조기 중단 또는 반환을 허용하지만 스트림의 Foreach 메소드는이 방법을 직접 지원하지 않습니다. 이 기사는 이유를 설명하고 스트림 처리 시스템에서 조기 종료를 구현하기위한 대체 방법을 탐색합니다. 추가 읽기 : Java Stream API 개선 스트림 foreach를 이해하십시오 Foreach 메소드는 스트림의 각 요소에서 하나의 작업을 수행하는 터미널 작동입니다. 디자인 의도입니다

Java의 날짜까지의 타임스탬프 Java의 날짜까지의 타임스탬프 Aug 30, 2024 pm 04:28 PM

Java의 TimeStamp to Date 안내. 여기서는 소개와 예제와 함께 Java에서 타임스탬프를 날짜로 변환하는 방법에 대해서도 설명합니다.

캡슐의 양을 찾기위한 Java 프로그램 캡슐의 양을 찾기위한 Java 프로그램 Feb 07, 2025 am 11:37 AM

캡슐은 3 차원 기하학적 그림이며, 양쪽 끝에 실린더와 반구로 구성됩니다. 캡슐의 부피는 실린더의 부피와 양쪽 끝에 반구의 부피를 첨가하여 계산할 수 있습니다. 이 튜토리얼은 다른 방법을 사용하여 Java에서 주어진 캡슐의 부피를 계산하는 방법에 대해 논의합니다. 캡슐 볼륨 공식 캡슐 볼륨에 대한 공식은 다음과 같습니다. 캡슐 부피 = 원통형 볼륨 2 반구 볼륨 안에, R : 반구의 반경. H : 실린더의 높이 (반구 제외). 예 1 입력하다 반경 = 5 단위 높이 = 10 단위 산출 볼륨 = 1570.8 입방 단위 설명하다 공식을 사용하여 볼륨 계산 : 부피 = π × r2 × h (4

미래를 창조하세요: 완전 초보자를 위한 Java 프로그래밍 미래를 창조하세요: 완전 초보자를 위한 Java 프로그래밍 Oct 13, 2024 pm 01:32 PM

Java는 초보자와 숙련된 개발자 모두가 배울 수 있는 인기 있는 프로그래밍 언어입니다. 이 튜토리얼은 기본 개념부터 시작하여 고급 주제를 통해 진행됩니다. Java Development Kit를 설치한 후 간단한 "Hello, World!" 프로그램을 작성하여 프로그래밍을 연습할 수 있습니다. 코드를 이해한 후 명령 프롬프트를 사용하여 프로그램을 컴파일하고 실행하면 "Hello, World!"가 콘솔에 출력됩니다. Java를 배우면 프로그래밍 여정이 시작되고, 숙달이 깊어짐에 따라 더 복잡한 애플리케이션을 만들 수 있습니다.

See all articles