`
xiaolongfeixiang
  • 浏览: 236419 次
  • 性别: Icon_minigender_1
  • 来自: 西安
社区版块
存档分类
最新评论

Java动态代理的设计是否有缺陷??

阅读更多

写一个简单的动态代理:

接口:

package study.proxy;
public interface RealInterface {
	public void sayHello();
}

 实现类:

package study.proxy;
public class RealObject implements RealInterface {
	@Override
	public void sayHello() {
		System.out.println("-------执行中-------");
	}
}

 InvocationHandler:

package study.proxy;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;

public class MyInterceptor implements InvocationHandler {

	private Object target;
	// 这不是必须的接口方法,但是却是编程必须的
	public void setTarget(Object target) {
		this.target = target;
	}

	@Override
	public Object invoke(Object proxy, Method method, Object[] args)
			throws Throwable {
		// proxy 通过 可以转型,但是转型后的Object依然不能用
		// (RealObject)proxy
		System.out.println("Before");
		Object result = method.invoke(target, args);
		System.out.println("After");
		return result;
	}

}

 测试类:

package study.proxy;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;

public class ProxyBasicDemo {

	public static void main(String[] args) throws Exception {

		// 这里不是面向接口的编程
		MyInterceptor handler = new MyInterceptor();
		handler.setTarget(new RealObject());
		
		Class<?> proxyClass = Proxy.getProxyClass(RealInterface.class
				.getClassLoader(), RealInterface.class);

		RealInterface proxyObject = (RealInterface) proxyClass.getConstructor(
				new Class[] { InvocationHandler.class }).newInstance(
				new Object[] { handler });
		
		proxyObject.sayHello();
	}

}

 

我的疑问:

1、在InvocationHandler中的invoke方法中的那个参数 proxy

 

 不能调用toString、hashCode方法;

 虽然可以转型为 实体类,但是 不能 在 method.invoke(obj,args)中使用。

 

 不知道这个参数有什么用???

 

2、既然proxy不知道怎么用,于是,就自己添加个 setTarget(Object  target)方法,注入具体的对象。

 

可行是可行!但是:

// 这里不是面向接口的编程
MyInterceptor handler = new MyInterceptor();

为什么不在InvocationHandler中加入一个setTarget(Object  target)方法的签名呢?

 

 

刚看动态代理,觉得SUN的这个API设计的不好。有可能是我错了。不知道,大家对动态代理有什么看法!

 

欢迎各位批评!!

分享到:
评论
11 楼 xiaolongfeixiang 2010-04-23  
Agrael 写道

这个proxy实际是提供你创建这个代理类时,你传入的接口类型数组所生成的代理信息。并不是目标对象。
但是你可以用它得到代理对象的类型信息,进行反射。这样就避免在调用时硬编码。


得到代理对象的类型信息??我只找到了这样一个方法:

System.out.println("类型信息:"+proxy.getClass().getName());


输出是:

类型信息:$Proxy0

好像不是代理对象的类型信息。  我是不是又错了??
10 楼 Agrael 2010-04-23  
xiaolongfeixiang 写道
Agrael 写道
xiaolongfeixiang 写道
xyz20003 写道
请问为什么会没有什么用呢?如果希望调用被代理实例的方法,不是就会需要proxy吗?


呵呵,热心人。

你是说proxy有 可以调用的方法?

proxy是java.lang.reflect.Proxy的实例吧?如果是的话,里面似乎没有很有用的方法?

另外:调用proxy.hashCode()这个Object中的方法,都会报错的!!

谢谢您的耐心指教!!


貌似我一来就回复了你这个问题,你都不看我的。
通过类型信息反射调用被代理对象的方法。


不好意思,您的回复我看得有点深奥,没怎么明白。于是就先回复他的了。

呵呵,JavaEye就是好心人多、技术牛人多!!

我不知道如何利用proxy这个对象。因为
public Object invoke(Object proxy, Method method, Object[] args)
			throws Throwable {
		System.out.println("Before");
		Object result = method.invoke(proxy, args);
		System.out.println("After");
		return result;
	}


会报错的!!所以我就对proxy这个参数很晕乎!


这个proxy实际是提供你创建这个代理类时,你传入的接口类型数组所生成的代理信息。并不是目标对象。
但是你可以用它得到代理对象的类型信息,进行反射。这样就避免在调用时硬编码。
9 楼 xiaolongfeixiang 2010-04-23  
Agrael 写道
xiaolongfeixiang 写道
xyz20003 写道
请问为什么会没有什么用呢?如果希望调用被代理实例的方法,不是就会需要proxy吗?


呵呵,热心人。

你是说proxy有 可以调用的方法?

proxy是java.lang.reflect.Proxy的实例吧?如果是的话,里面似乎没有很有用的方法?

另外:调用proxy.hashCode()这个Object中的方法,都会报错的!!

谢谢您的耐心指教!!


貌似我一来就回复了你这个问题,你都不看我的。
通过类型信息反射调用被代理对象的方法。


不好意思,您的回复我看得有点深奥,没怎么明白。于是就先回复他的了。

呵呵,JavaEye就是好心人多、技术牛人多!!

我不知道如何利用proxy这个对象。因为
public Object invoke(Object proxy, Method method, Object[] args)
			throws Throwable {
		System.out.println("Before");
		Object result = method.invoke(proxy, args);
		System.out.println("After");
		return result;
	}


会报错的!!所以我就对proxy这个参数很晕乎!
8 楼 xiaolongfeixiang 2010-04-23  
xyz20003 写道
如果你传了一个target进去,还怎么面向切面?


哦,我现在的问题就是(就如我提供的例子), 我必须提供一个setTarget或类似的方法。

如果我这样改:
public class MyInterceptor implements InvocationHandler {

	@Override
	public Object invoke(Object proxy, Method method, Object[] args)
			throws Throwable {
		System.out.println("Before");
                // 由于没有提供任何方法,所以这个参数似乎只能是proxy
                // 但是这样运行时报错!
		Object result = method.invoke(proxy, args);
		System.out.println("After");
		return result;
	}

}



public class ProxyBasicDemo {

	public static void main(String[] args) throws Exception {

		MyInterceptor handler = new MyInterceptor();
		
		Class<?> proxyClass = Proxy.getProxyClass(RealInterface.class
				.getClassLoader(), RealInterface.class);

		RealInterface proxyObject = (RealInterface) proxyClass.getConstructor(
				new Class[] { InvocationHandler.class }).newInstance(
				new Object[] { handler });
		
		proxyObject.sayHello();
	}

}


错误是这样的:
引用

... 943 more
Caused by: java.lang.reflect.UndeclaredThrowableException
at $Proxy0.sayHello(Unknown Source)
... 947 more
Caused by: java.lang.reflect.InvocationTargetException
at sun.reflect.GeneratedMethodAccessor1.invoke(Unknown Source)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(Unknown Source)
at java.lang.reflect.Method.invoke(Unknown Source)
at study.proxy.MyInterceptor.invoke(MyInterceptor.java:18)
... 948 more
Caused by: java.lang.reflect.UndeclaredThrowableException
at $Proxy0.sayHello(Unknown Source)
... 952 more

7 楼 Agrael 2010-04-23  
xiaolongfeixiang 写道
xyz20003 写道
请问为什么会没有什么用呢?如果希望调用被代理实例的方法,不是就会需要proxy吗?


呵呵,热心人。

你是说proxy有 可以调用的方法?

proxy是java.lang.reflect.Proxy的实例吧?如果是的话,里面似乎没有很有用的方法?

另外:调用proxy.hashCode()这个Object中的方法,都会报错的!!

谢谢您的耐心指教!!


貌似我一来就回复了你这个问题,你都不看我的。
通过类型信息反射调用被代理对象的方法。
6 楼 Agrael 2010-04-23  
xiaolongfeixiang 写道
xyz20003 写道
如果想对实体类做AOP,可以使用cglib。
jdk提供的proxy只能代理接口。

我不清楚你为何要将接口强制转型为实现类?从你的代码中没有看到这个必要。是否因为你们的接口设计失误,造成具体调用时都要强制转换成实现类?

也许你的意图是在Interceptor中调用代理的方法,那么只需要把Object转换成接口就行了,如果转换成实现类是会报错的。


谢谢你的回复!

我不是要强制转成实现类,这样不好,我知道的。

我是为了证明,原始的InvocationHandler的invoke方法中的第一个对象proxy,没有什么用。

由于proxy对象没有用,所以我们必须加一个setTarget类似的方法。


第一个参数的作用就是生成的代理的接口,提供类型信息的。proxy.getClass()
5 楼 xiaolongfeixiang 2010-04-23  
xyz20003 写道
请问为什么会没有什么用呢?如果希望调用被代理实例的方法,不是就会需要proxy吗?


呵呵,热心人。

你是说proxy有 可以调用的方法?

proxy是java.lang.reflect.Proxy的实例吧?如果是的话,里面似乎没有很有用的方法?

另外:调用proxy.hashCode()这个Object中的方法,都会报错的!!

谢谢您的耐心指教!!
4 楼 xyz20003 2010-04-23  
请问为什么会没有什么用呢?如果希望调用被代理实例的方法,不是就会需要proxy吗?

如果你传了一个target进去,还怎么面向切面?本来如果系统里有10个实现了这个接口的类,我只要写一个interceptor就可以拦截10个类的方法调用,现在你这样一搞,我必须写10个拦截器,对应十个类才能实现原来一个拦截器的功能。何苦呢?
3 楼 xiaolongfeixiang 2010-04-23  
xyz20003 写道
如果想对实体类做AOP,可以使用cglib。
jdk提供的proxy只能代理接口。

我不清楚你为何要将接口强制转型为实现类?从你的代码中没有看到这个必要。是否因为你们的接口设计失误,造成具体调用时都要强制转换成实现类?

也许你的意图是在Interceptor中调用代理的方法,那么只需要把Object转换成接口就行了,如果转换成实现类是会报错的。


谢谢你的回复!

我不是要强制转成实现类,这样不好,我知道的。

我是为了证明,原始的InvocationHandler的invoke方法中的第一个对象proxy,没有什么用。

由于proxy对象没有用,所以我们必须加一个setTarget类似的方法。
2 楼 Agrael 2010-04-23  
回答:
⒈那个实际是生成的代理接口,所以不能直接使用,但是它包含了你所需要的类型信息。
⒉因为有可能作为匿名类啊这些,那这个签名就没意义。甚至,有可能你的对象不是一个参数可以解决的等等。

PS:如果一切都面向接口编程,那么又由谁来组装呢?
1 楼 xyz20003 2010-04-23  
如果想对实体类做AOP,可以使用cglib。
jdk提供的proxy只能代理接口。

我不清楚你为何要将接口强制转型为实现类?从你的代码中没有看到这个必要。是否因为你们的接口设计失误,造成具体调用时都要强制转换成实现类?

也许你的意图是在Interceptor中调用代理的方法,那么只需要把Object转换成接口就行了,如果转换成实现类是会报错的。

相关推荐

    某外企java笔试题

    【Java 笔试题详解】 1. 选择题:在Java编程中,如果尝试使用一个`char`类型的变量作为参数调用方法,而该方法没有对应`char`类型的重载版本,编译器会报错。这涉及到Java的类型匹配规则和方法重载。 2. 数据库...

    Thinking_in_java(chapter10).pdf

    5. 动态代理:利用反射创建一个接口的代理实例,该代理可以在运行时拦截接口方法的调用。 知识点四:多态性与RTTI 1. 多态性的概念:通过基类引用调用方法时,实际调用的是派生类中重写的方法,称为动态绑定或多态...

    Java架构体系学习线路图

    通过以上对Java架构体系学习线路图的详细解读,我们不难看出,要想成为一名合格的Java开发者,不仅要掌握Java语言本身,还需要具备广泛的工具使用能力、分布式和微服务架构设计能力以及系统性能优化的能力。...

    阿里巴巴Java开发手册v1.2.0_1595390177.docx

    - **表结构设计**:合理设计数据库表结构和索引,避免由于设计不当带来的架构缺陷或性能风险。 ##### 5. 异常与日志 - **异常处理**:规定了如何正确处理异常,以减少系统不稳定因素。 - **日志记录**:规范日志...

    Java编码规范及实践

    《Java编码规范及实践》是针对Java开发人员的一份详尽指南,旨在提升代码质量、增强可读性和可维护性,以及确保团队间代码风格的一致性。本篇将深入探讨该文档涵盖的关键知识点,包括编码规范的重要性和具体实践。 ...

    基于Java的多层分布式应用模型研究.pdf

    在探讨基于Java的多层分布式应用模型研究前,有必要了解多层分布式系统的发展背景及其体系结构的特点。传统的集中模式和两层模式已不能满足大型信息系统对可扩展性、可维护性和高性能的需求。因此,多层模式,特别是...

    面试简历-2年Java开发工程师.docx

    3. **开发经验**:工程师在两个公司任职期间,负责了软件的设计、开发、测试和维护,显示了他在Java Web开发中的全面能力。他熟练使用Spring MVC、Spring、Mybatis等架构,能够处理Web组件和MVC模式,同时对MySQL和...

    一种Java代码覆盖率工具的应用研究.zip

    2. **Java代码覆盖率工具**:Java平台上有多种代码覆盖率工具,如JaCoCo、Cobertura、Emma等。这些工具通过插入字节码或代理类的方式收集覆盖率数据,然后生成报告展示未被测试覆盖的代码。JaCoCo是目前较为流行的一...

    推荐给程序员的经典电子书.docx

    - 内容包括面向对象程序设计、反射与代理、接口与内部类、事件监听器模型、Swing UI工具箱的图形用户界面设计、异常处理、流输入/输出和对象序列化、泛型程序设计等; - 适用于想要深入了解Java核心概念及其应用...

    经典计算机书籍.pdf

    这本经典的Java参考书详细介绍了Java 2开发平台标准版J2SE5.0的基础知识,包括面向对象编程、反射与代理、接口与内部类、事件监听器模型、Swing GUI设计、异常处理、流I/O和对象序列化、泛型编程等多个核心主题。...

    灸哥讲解结构型设计模式

    1.4 尽管门面模式有其优点,但也可能存在缺陷,如如果门面过于庞大,可能会变得难以管理。1.5 缓解这种问题的方法是将大型门面拆分为多个小门面,或者使用策略模式来动态选择不同的子系统组合。 接着,是组合模式。...

    java程序员面试大纲错过了金三银四你还要错过2018吗.docx

    10. **Spring设计模式**:使用了大量的设计模式,如工厂模式、代理模式、观察者模式等。 11. **Spring MVC的工作原理**:接收HTTP请求,通过控制器处理请求,返回视图和模型数据,最终渲染视图响应客户端。 12. **...

    swing用户届面设计.pdf

    Swing的出现主要是为了解决AWT在GUI设计上的局限性和缺陷,如缺乏剪贴板支持、打印功能、键盘导航能力,以及基础组件如弹出菜单和滚动窗格的缺失。Swing的组件几乎都是轻量级组件,这意味着它们不依赖于特定的操作...

    实用文档之软件开发应知应会-84分.pdf

    - 问题:“面向对象、()、独立于程序设计、容易掌握使用是 UML 的特点。”答案是A.可视化和C.独立于过程。 18. 持续交付:持续交付强调快速频繁地交付产品,以满足客户需求。 - 问题:“持续交付指的是以较()地...

    Mockito-and-Junit:Java Mock deafferents方法的示例存储库

    先编写测试,再编写实现,有助于保证代码质量,减少缺陷,同时也有助于设计更好的 API。 10. **持续集成(CI)**: 将这些测试集成到持续集成流程中,如 Jenkins 或 Travis CI,可以确保每次代码变更后,所有的...

    2014年it技术笔试

    这些方法有助于设计出有效的测试用例来发现软件中的缺陷。 18. 测试工具:测试工具在软件开发和测试过程中非常重要,可以提高测试的效率和有效性。常用的测试工具有JMeter用于性能测试,Selenium用于自动化Web应用...

    Python的计算机软件应用技术分析 (1).pdf

    例如,在浏览器插件的帮助下,可以打开相关设置,并将计算机代理设置为***.*.*.*:8888,然后通过此端口编写网络爬虫程序。这一过程的便捷性为开发者提供了极大的便利。 在自动化平台设计方面,基于Python的计算机...

    【卷一/共两卷】AJAX实战pdf高清版90M

    12.5.4 克服Ajax书签支持的缺陷 12.6 重构 12.6.1 XSLT Heler 12.6.2 动态搜索组件 12.6.3 重构报告 12.7 小结 第13章 使用Ajax创建独立的应用 13.1 从外部读取信息 13.1.1 查找XML提要 13.1.2 RSS结构 13.2 创建...

    Springboot+vue的企业质量管理系统(有报告) Javaee项目,springboot vue前后端分离项目

    商业资料、范文/模板/素材的标签表明,这个项目可能包含了一些可以作为参考或者直接使用的材料,可能是数据库设计、API接口文档、UI设计模板等,对于学习或快速启动类似项目非常有价值。 “446.中小型制造企业质量...

Global site tag (gtag.js) - Google Analytics