ホームページ >Java >&#&チュートリアル >10.Javaの基本 - エージェント

10.Javaの基本 - エージェント

黄舟
黄舟オリジナル
2017-02-27 10:41:021167ブラウズ

基本概念

Javaでは、プロキシとは、オブジェクトにプロキシオブジェクトを提供することを意味し、プロキシオブジェクトは元のオブジェクトへのアクセスを制御します。つまり、クライアントは元のオブジェクトを直接制御しませんが、プロキシ オブジェクトを通じて元のオブジェクトを間接的に操作します。

平たく言えば、鉄道の切符を購入するようなものです。当初は切符を購入するには駅が必要でしたが、現在は切符販売所が設置されており、切符販売所から直接切符を購入できるようになりました。

エージェントの種類は次のとおりです。

  • エージェントの実装は、静的エージェントと動的エージェントに分類できます。

  • 動的エージェントは、さらに JDK 動的エージェントと CLGIB 動的エージェントに分類されます。


静的プロキシ

静的プロキシでは、各プロキシ クラスは 1 つのインターフェイスのみを提供できます。

このアプローチの欠点は明らかです:

  • 生成されるプロキシが多すぎます。複数のインターフェイスには複数のプロキシ クラスが必要なためです。

  • コードの冗長性。すべてのプロキシ操作は、呼び出すメソッドを除いて同じです。

静的プロキシの例を見てみましょう:

// 定义一个接口,代表书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();
    }
}

動的プロキシ

動的プロキシは、1 つのプロキシ クラスを通じてすべてのプロキシ機能を完了できます。

動的プロキシは、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 中国語 Web サイト (www.php.cn) に注目してください。


声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。