>  기사  >  Java  >  10.자바 기초 - 프록시

10.자바 기초 - 프록시

黄舟
黄舟원래의
2017-02-27 10:41:021111검색

기본 개념

Java에서 프록시란 객체에 프록시 객체를 제공하고 프록시는 객체에 대한 액세스를 제어하는 ​​것을 의미합니다. 즉, 클라이언트는 원본 객체를 직접 제어하지 않고 프록시 객체를 통해 원본 객체를 간접적으로 제어합니다.

말로 말하면 기차표를 사는 것과 같습니다. 원래는 기차표를 사려면 기차역이 필요했는데, 지금은 티켓 판매소를 마련했고, 티켓 판매소를 통해 직접 티켓을 살 수 있습니다.

에이전트 유형은 다음과 같습니다.

  • 에이전트 구현은 정적 에이전트와 동적 에이전트로 나눌 수 있습니다.

  • 동적 에이전트도 JDK 동적 프록시와 CLGIB 동적 프록시로 구분됩니다.


정적 프록시

정적 프록시에서 각 프록시 클래스는 하나의 인터페이스만 제공할 수 있습니다.

이 접근 방식의 단점은 분명합니다.

  • 너무 많은 프록시 생성입니다. 여러 인터페이스에는 여러 프록시 클래스가 필요하기 때문입니다.

  • 코드 중복. 모든 프록시 작업은 호출하는 메서드를 제외하고 동일합니다.

정적 프록시의 예를 살펴보겠습니다.

// 定义一个接口,代表书interface Book {    void read(); 
}// 委托类(实现接口,包含了具体的业务逻辑)class Bible implements Book {    @Override
    public void read() {
        System.out.println("Reading...");
    }
}// 静态代理类(其实是对 Bible 类的增强)class BookProxy implements Book {    //委托对象,作为构造函数的参数
    private Book book;  

    BookProxy(Book book) {        this.book = book;
    }    @Override
    public void read() {
        System.out.println("Reading 调用之前");
        book.read();
        System.out.println("Reading 调用之后");
    }
}public class Test {
    public static void main(String[] args) {        //静态代理调用
        Book book = new Bible()
        BookProxy bookProxy = new BookProxy(book);
        bookProxy.read();
    }
}

동적 프록시

동적 프록시로 해결할 수 있습니다. 정적 프록시의 문제점 단점은 프록시 클래스를 통해 모든 프록시 기능을 완료할 수 있다는 것입니다.

동적 프록시는 JDK 동적 프록시와 CLGIB 동적 프록시로 구분됩니다.

DK의 동적 프록시는 인터페이스에 의존하는데 CLGIB 동적 프록시가 이러한 단점을 보완해줍니다.


1. JDK 동적 프록시

JDK의 동적 프록시는 인터페이스 구현에 의존하는 클래스가 있을 경우 JDK 프록시를 사용할 수 없습니다.

예를 살펴보겠습니다.

// 定义一个接口,代表书...// 委托类(实现接口,包含了具体的业务逻辑)...
// 动态代理类
class BookHandler implements InvocationHandler{    
private Object target;    
//绑定委托对象(JDK 动态代理的缺陷,只能绑定接口)并返回一个代理类
    Object bind(Object target){        
    this.target = target;        
    // 取得代理对象
        return Proxy.newProxyInstance(
            target.getClass().getClassLoader(), // 类加载器
            target.getClass().getInterfaces(),  // 类的所有接口
            this); // InvocationHandler
    }    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("Reading 调用之前");
        method.invoke(target, args);
        System.out.println("Reading 调用之后");        return null;
    }
}public class Test {
    public static void main(String[] args) {        // 动态代理调用
        Book book = new Bible();
        BookHandler bookHandler = new BookHandler();
        Book bookProxy = (Book) bookHandler.bind(book);
        bookProxy.read();
    }
}

2.CLGIB 동적 프록시

clgib은 클래스에 대한 프록시를 구현하며 원칙은 하위 클래스를 생성합니다. 지정된 대상 클래스에 대해 해당 메서드를 재정의하여 향상을 달성합니다.

상속을 사용하기 때문에 최종 수정된 클래스는 프록시할 수 없습니다.

예제를 살펴보겠습니다.

// 实现类(没有了接口,直接实现业务逻辑)class Book {    public void read() {
        System.out.println("Reading...");
    }
}// 动态代理类class BookCglib implements MethodInterceptor{    private Object target;

    Object getInstance(Object target){        this.target = target;
        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(this.target.getClass());        //回调方法
        enhancer.setCallback(this);        //创建代理对象
        return enhancer.create();
    }    @Override
    public Object intercept(Object object, Method method, Object[] args, MethodProxy proxy) throws Throwable {
        System.out.println("Reading 调用之前");
        proxy.invoke(target, args);
        System.out.println("Reading 调用之后");        return null;
    }

}public class Test {
    public static void main(String[] args) {
        BookCglib bookCglib =  new BookCglib();
        Book book = (Book) bookCglib.getInstance(new Book());
        book.read();
    }
}

위는 10.Java Basics - Proxy의 내용입니다. 더 많은 관련 내용은 PHP 중국어 홈페이지(www. php.cn)!


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