이 기사는 Java 동적 프록시의 InvocationHandler에 대한 간략한 소개를 제공합니다. 이는 특정 참조 값을 가지고 있으므로 도움이 될 수 있습니다.
인터넷에는 Java의 동적 프록시, 프록시 및 InvocationHandler 개념에 대한 매우 심층적인 기사가 있습니다. 사실 이러한 개념은 그다지 복잡하지 않습니다. 이제 가장 간단한 예를 통해 InvocationHandler가 무엇인지 이해합니다. InvocationHandler는 Spring 프레임워크 구현에서 널리 사용된다는 점을 언급할 가치가 있습니다. 이는 InvocationHandler를 철저히 이해하면 향후 Spring 소스 코드 학습을 위한 견고한 기반을 마련할 수 있음을 의미합니다.
지정된 사람들에게 "안녕하세요" 또는 "안녕하세요"라고 인사하는 두 가지 방법이 포함된 인터페이스를 개발하세요.
public interface IHello { void sayHello(String name); void sayGoogBye(String name); }
이 IHello 인터페이스를 구현하는 간단한 클래스를 만듭니다.
public class Helloimplements implements IHello { @Override public void sayHello(String name) { System.out.println("Hello " + name); } @Override public void sayGoogBye(String name) { System.out.println(name+" GoodBye!"); } }
이 구현 클래스를 사용하세요. 지금까지는 특별한 것이 없습니다.
이제 다음 요구 사항을 받았다고 가정해 보겠습니다. 상사는 구현 클래스가 누군가에게 인사할 때마다 인사말의 세부 사항을 로그 파일에 기록해야 한다고 요구합니다. 단순화를 위해 인사말 전에 다음 명령문 줄을 인쇄하여 로깅 작업을 시뮬레이션합니다.
System.out.println("问候之前的日志记录...");
이게 간단하지 않다고 생각하시나요? Helloimplements의 해당 메소드를 직접 수정하고 해당 메소드에 이 로그 줄을 삽입합니다.
하지만 사장님의 요청은 원래 Helloimplements 클래스를 수정할 수 없다는 것입니다. 실제 시나리오에서는 타사 jar 패키지에서 Helloimplements를 제공할 수 있으며 코드를 수정할 방법이 없습니다.
디자인 패턴에서 프록시 패턴을 사용할 수 있다고 말할 수 있습니다. 즉, 새 Java 클래스를 프록시 클래스로 생성하고 IHello 인터페이스도 구현한 다음 Helloimplements 클래스의 인스턴스를 전달할 수 있습니다. 프록시 클래스로. Helloimplements의 코드를 수정하지 않아도 되지만 프록시 클래스에 로깅 코드를 작성할 수 있습니다. 전체 코드는 다음과 같습니다.
public class StaticProxy implements IHello { private IHello iHello; public void setImpl(IHello impl){ this.iHello = impl; } @Override public void sayHello(String name) { System.out.println("问候之前的日志记录..."); iHello.sayHello(name); } @Override public void sayGoogBye(String name) { System.out.println("问候之前的日志记录..."); iHello.sayGoogBye(name); } static public void main(String[] arg) { Helloimplements hello = new Helloimplements(); StaticProxy proxy = new StaticProxy(); proxy.setImpl(hello); proxy.sayHello("Jerry"); } }
이 접근 방식은 요구 사항을 충족할 수 있습니다.
동일한 효과를 얻기 위해 InvocationHandler를 사용하는 방법을 살펴보겠습니다.
InvocationHandler是一个JDK提供的标准接口。看下面的代码: import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; public class DynaProxyHello implements InvocationHandler { private Object delegate; public Object bind(Object delegate) { this.delegate = delegate; return Proxy.newProxyInstance( this.delegate.getClass().getClassLoader(), this.delegate .getClass().getInterfaces(), this); } public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { Object result = null; try { System.out.println("问候之前的日志记录..."); // JVM通过这条语句执行原来的方法(反射机制) result = method.invoke(this.delegate, args); } catch (Exception e) { e.printStackTrace(); } return result; }
위 코드의 바인드 메소드는 이전 프록시 클래스인 StaticProxy의 setImpl 메소드와 매우 유사하지만 이 바인드 메소드의 입력 매개변수 유형이 더 보편적입니다. 로깅 코드는 메소드 호출에 작성됩니다.
사용 방법 보기:
static public void main(String[] arg) { DynaProxyHello helloproxy = new DynaProxyHello(); Helloimplements hello = new Helloimplements(); IHello ihello = (IHello) helloproxy.bind(hello); ihello.sayHello("Jerry"); }
실행 효과는 StaticProxy 솔루션과 완전히 동일합니다.
먼저 디버깅해 보겠습니다. 바인딩 메서드가 실행되면 Proxy.newProxyInstance 메서드가 호출되고 Helloimplements 클래스의 인스턴스가 전달됩니다.
디버거의 IHello ihello = (IHello) helloproxy.bind(hello) 문에서 반환된 ihello 변수를 관찰합니다. 정적 유형은 IHello이지만 디버거에서 실제 유형을 관찰할 때 이는 Helloimplements의 인스턴스가 아니라 호출 메서드에서 직접 작성한 로그 줄을 포함하여 JVM이 처리한 내용입니다. 암호. ihello 유형은 $Proxy0입니다.
JVM에서 처리하는 이 변수의 sayHello 메소드가 호출되면 JVM은 호출을 DynaProxyHello.invoke로 자동 전송합니다.
그래서 Invoke 메소드에서는 직접 작성합니다. 로깅 코드가 실행된 다음 원래 sayHello 코드가 Java 리플렉션을 통해 실행됩니다.
어떤 친구들은 InvocationHandler가 StaticProxy보다 더 복잡해 보이나요?라고 물을 수도 있습니다. 이점은 무엇입니까?
상사의 요구 사항이 다시 바뀌고 인사 및 작별 인사 방법을 호출할 때 다양한 로깅 전략을 사용해야 한다고 가정해 보겠습니다.
InvocationHandler를 사용하여 우아하게 구현하는 방법을 살펴보겠습니다.
이 예제를 통해 모든 사람이 Java의 동적 프록시 InvocationHandler에 대한 가장 기본적인 이해를 얻을 수 있기를 바랍니다.
위 내용은 Java 동적 프록시의 InvocationHandler에 대한 간략한 소개의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!