Java의 세 가지 프록시 모드는 무엇입니까?
먼저 에이전시 모델이 무엇인지 간단히 설명드리겠습니다.
Proxy는 대상 객체에 액세스하는 또 다른 방법, 즉 프록시 객체를 통해 대상 객체에 액세스하는 방법을 제공하는 디자인 패턴입니다. 이의 장점은 대상 객체의 구현을 기반으로 추가적인 기능 작업을 향상시킬 수 있다는 것입니다. 즉, 대상 개체의 기능을 확장합니다.
여기에서는 프로그래밍의 아이디어가 사용됩니다. 다른 사람이 작성한 코드나 메서드를 마음대로 수정하지 마십시오. 변경해야 하는 경우 메서드를 확장할 수 있습니다. Proxy
예를 들어 에이전트의 역할을 설명하자면: 스타를 초대하고 싶다면 스타와 직접 연락하지 않고 동일한 목적을 달성하기 위해 스타의 에이전트에게 연락한다고 가정해 보겠습니다. 이벤트에서 프로그램에 대한 책임을 져야 하고, 그 외의 사소한 문제는 에이전트(브로커)에게 맡겨 해결하는 것이 현실에서의 에이전시 사고의 예입니다
다음과 같은 다이어그램으로 표현됩니다.
에이전시 모델의 핵심은 프록시 개체와 대상 개체입니다. 프록시 개체는 대상 개체의 확장이며 대상 개체를 호출합니다.
1.1 정적 프록시
정적 프록시를 사용할 때는 다음을 수행해야 합니다. 인터페이스 또는 상위 클래스를 정의하고 프록시 객체와 프록시 객체는 동일한 인터페이스를 구현하거나 동일한 상위 클래스를 상속합니다.
다음은 설명을 위한 예입니다.
저장 작업을 시뮬레이션하고 저장 작업에 대한 인터페이스를 정의합니다. : IUserDao.java, 그리고 대상 객체는 이 인터페이스의 UserDao.java 메소드를 구현합니다. 이때 정적 프록시 메소드를 사용하는 경우 프록시 객체(UserDaoProxy.java)에 IUserDao 인터페이스를 구현해야 합니다.
프록시 객체와 대상 객체는 동일한 인터페이스를 구현해야 하며, 동일한 메소드를 호출하여 대상 객체의 메소드를 호출해야 합니다.
코드 예:
인터페이스: IUserDao.java
/** * 接口 */public interface IUserDao { void save(); }
대상 개체: UserDao.java
/** * 接口实现 * 目标对象 */public class UserDao implements IUserDao { public void save() { System.out.println("----已经保存数据!----"); } }
프록시 개체: UserDaoProxy.java
/** * 代理对象,静态代理 */public class UserDaoProxy implements IUserDao{ //接收保存目标对象 private IUserDao target; public UserDaoProxy(IUserDao target){ this.target=target; } public void save() { System.out.println("开始事务..."); target.save();//执行目标对象的方法 System.out.println("提交事务..."); } }
(동영상 공유 학습: java 동영상 튜토리얼)
테스트 클래스: App.java
/** * 测试类 */public class App { public static void main(String[] args) { //目标对象 UserDao target = new UserDao(); //代理对象,把目标对象传给代理对象,建立代理关系 UserDaoProxy proxy = new UserDaoProxy(target); proxy.save();//执行的是代理的方法 } }
정적 프록시 요약:
1. 대상 객체의 기능을 수정하지 않고도 대상 기능을 확장할 수 있습니다.
2. 단점:
프록시 객체는 대상 객체와 동일한 인터페이스를 구현해야 하기 때문에 많은 프록시 클래스, 너무 많은 클래스 동시에 인터페이스가 메소드를 추가하면 대상 객체와 프록시 객체가 모두 유지되어야 합니다.
정적 프록시의 단점을 해결하는 방법은 Dynamic을 사용할 수 있다는 것입니다. 프록시 메소드
1.2. 동적 프록시
동적 프록시에는 다음과 같은 특징이 있습니다.
1. 프록시 객체는 인터페이스를 구현할 필요가 없습니다.
2. 프록시 객체 생성은 JDK의 API를 사용하여 프록시 객체를 동적으로 구축하는 것입니다. (구현할 프록시 객체/대상 객체를 생성하려면 인터페이스 유형을 지정해야 합니다)
3. 동적 프록시라고도 합니다: JDK 프록시, 인터페이스 프록시
JDK에서 프록시 객체를 생성하기 위한 API
패키지 프록시 클래스는 java.lang.reflect.Proxy에 있습니다.
JDK는 프록시를 구현하기 위해 newProxyInstance 메소드만 사용해야 하지만 이 메소드는 세 개의 매개변수를 수신해야 합니다.
static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces,InvocationHandler h )
이 메소드는 다음과 같습니다. Proxy 클래스의 정적 메소드와 수신된 세 가지 매개변수는 순서대로입니다.
ClassLoader 로더,: 현재 대상 객체가 클래스 로더를 사용하도록 지정합니다. 로더를 얻는 방법은 고정되어 있습니다
Class>[ ] 인터페이스,: 대상 객체가 구현하는 인터페이스 유형, 제네릭을 사용하여 유형 확인
InvocationHandler h: 이벤트 처리, 대상 객체의 메소드 실행 시 이벤트 핸들러의 메소드가 트리거되고 메소드 현재 실행 중인 대상 객체의 매개변수로 전달됩니다
코드 예:
Interface 클래스 IUserDao.java 및 인터페이스 구현 클래스, 대상 객체 UserDao는 수정 없이 동일합니다. 이를 기반으로 프록시 팩토리 클래스(ProxyFactory.java)를 추가합니다. java), 여기에 프록시 클래스를 작성하고 테스트 클래스에서 대상 객체를 생성합니다(프록시를 사용해야 하는 코드). 프록시 객체에 문의한 후 프록시 객체
Proxy 공장에서 동일한 이름의 메소드를 사용합니다. 클래스: ProxyFactory.java
/** * 创建动态代理对象 * 动态代理不需要实现接口,但是需要指定接口类型 */public class ProxyFactory{ //维护一个目标对象 private Object target; public ProxyFactory(Object target){ this.target=target; } //给目标对象生成代理对象 public Object getProxyInstance(){ return Proxy.newProxyInstance( target.getClass().getClassLoader(), target.getClass().getInterfaces(), new InvocationHandler() { @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("开始事务2"); //执行目标对象方法 Object returnValue = method.invoke(target, args); System.out.println("提交事务2"); return returnValue; } } ); } }
테스트 클래스: App.java
/** * 测试类 */public class App { public static void main(String[] args) { // 目标对象 IUserDao target = new UserDao(); // 【原始的类型 class cn.itcast.b_dynamic.UserDao】 System.out.println(target.getClass()); // 给目标对象,创建代理对象 IUserDao proxy = (IUserDao) new ProxyFactory(target).getProxyInstance(); // class $Proxy0 内存中动态生成的代理对象 System.out.println(proxy.getClass()); // 执行方法 【代理对象】 proxy.save(); } }
요약:
프록시 객체는 인터페이스를 구현할 필요가 없지만 대상 객체는 인터페이스를 구현해야 합니다. 그렇지 않으면 동적 프록시
1.3을 사용할 수 없습니다. Cglib 프록시
위의 정적 프록시 및 동적 프록시 모드에서는 대상 개체가 인터페이스를 구현하는 대상 개체여야 하지만 때로는 대상 개체가 별도의 개체일 뿐이고 인터페이스가 없는 경우도 있습니다. 인터페이스를 구현하려면 다음을 사용할 수 있습니다. 프록시를 구현하기 위한 대상 객체 하위 클래스입니다. 이 메서드는 Cglib 프록시
라고 합니다.Cglib代理,也叫作子类代理,它是在内存中构建一个子类对象从而实现对目标对象功能的扩展.
JDK的动态代理有一个限制,就是使用动态代理的对象必须实现一个或多个接口,如果想代理没有实现接口的类,就可以使用Cglib实现.Cglib是一个强大的高性能的代码生成包,它可以在运行期扩展java类与实现java接口.它广泛的被许多AOP的框架使用,例如Spring AOP和synaop,为他们提供方法的interception(拦截)Cglib包的底层是通过使用一个小而块的字节码处理框架ASM来转换字节码并生成新的类.不鼓励直接使用ASM,因为它要求你必须对JVM内部结构包括class文件的格式和指令集都很熟悉.
Cglib子类代理实现方法:
1.需要引入cglib的jar文件,但是Spring的核心包中已经包括了Cglib功能,所以直接引入pring-core-3.2.5.jar即可.
2.引入功能包后,就可以在内存中动态构建子类
3.代理的类不能为final,否则报错
4.目标对象的方法如果为final/static,那么就不会被拦截,即不会执行目标对象额外的业务方法.
代码示例:
目标对象类:UserDao.java
/** * 目标对象,没有实现任何接口 */public class UserDao { public void save() { System.out.println("----已经保存数据!----"); } }
Cglib代理工厂:ProxyFactory.java
/** * Cglib子类代理工厂 * 对UserDao在内存中动态构建一个子类对象 */public class ProxyFactory implements MethodInterceptor{ //维护目标对象 private Object target; public ProxyFactory(Object target) { this.target = target; } //给目标对象创建一个代理对象 public Object getProxyInstance(){ //1.工具类 Enhancer en = new Enhancer(); //2.设置父类 en.setSuperclass(target.getClass()); //3.设置回调函数 en.setCallback(this); //4.创建子类(代理对象) return en.create(); } @Override public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable { System.out.println("开始事务..."); //执行目标对象的方法 Object returnValue = method.invoke(target, args); System.out.println("提交事务..."); return returnValue; } }
测试类:
/** * 测试类 */public class App { @Test public void test(){ //目标对象 UserDao target = new UserDao(); //代理对象 UserDao proxy = (UserDao)new ProxyFactory(target).getProxyInstance(); //执行代理对象的方法 proxy.save(); } }
在Spring的AOP编程中:
如果加入容器的目标对象有实现接口,用JDK代理
如果目标对象没有实现接口,用Cglib代理
相关推荐:java入门教程
위 내용은 Java의 세 가지 프록시 모드는 무엇입니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

핫 AI 도구

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

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

Undress AI Tool
무료로 이미지를 벗다

Clothoff.io
AI 옷 제거제

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

인기 기사

뜨거운 도구

메모장++7.3.1
사용하기 쉬운 무료 코드 편집기

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

스튜디오 13.0.1 보내기
강력한 PHP 통합 개발 환경

드림위버 CS6
시각적 웹 개발 도구

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

뜨거운 주제











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

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

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

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

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

PHP와 Python은 각각 고유 한 장점이 있으며 선택은 프로젝트 요구 사항을 기반으로해야합니다. 1.PHP는 간단한 구문과 높은 실행 효율로 웹 개발에 적합합니다. 2. Python은 간결한 구문 및 풍부한 라이브러리를 갖춘 데이터 과학 및 기계 학습에 적합합니다.

PHP는 서버 측에서 널리 사용되는 스크립팅 언어이며 특히 웹 개발에 적합합니다. 1.PHP는 HTML을 포함하고 HTTP 요청 및 응답을 처리 할 수 있으며 다양한 데이터베이스를 지원할 수 있습니다. 2.PHP는 강력한 커뮤니티 지원 및 오픈 소스 리소스를 통해 동적 웹 컨텐츠, 프로세스 양식 데이터, 액세스 데이터베이스 등을 생성하는 데 사용됩니다. 3. PHP는 해석 된 언어이며, 실행 프로세스에는 어휘 분석, 문법 분석, 편집 및 실행이 포함됩니다. 4. PHP는 사용자 등록 시스템과 같은 고급 응용 프로그램을 위해 MySQL과 결합 할 수 있습니다. 5. PHP를 디버깅 할 때 error_reporting () 및 var_dump ()와 같은 함수를 사용할 수 있습니다. 6. 캐싱 메커니즘을 사용하여 PHP 코드를 최적화하고 데이터베이스 쿼리를 최적화하며 내장 기능을 사용하십시오. 7

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