java框架前的准备.docx
《java框架前的准备.docx》由会员分享,可在线阅读,更多相关《java框架前的准备.docx(25页珍藏版)》请在冰豆网上搜索。
java框架前的准备
Java基础加强总结——代理(Proxy)
一、代理的概念
动态代理技术是整个java技术中最重要的一个技术,它是学习java框架的基础,不会动态代理技术,那么在学习Spring这些框架时是学不明白的。
动态代理技术就是用来产生一个对象的代理对象的。
在开发中为什么需要为一个对象产生代理对象呢?
举一个现实生活中的例子:
歌星或者明星都有一个自己的经纪人,这个经纪人就是他们的代理人,当我们需要找明星表演时,不能直接找到该明星,只能是找明星的代理人。
比如刘德华在现实生活中非常有名,会唱歌,会跳舞,会拍戏,刘德华在没有出名之前,我们可以直接找他唱歌,跳舞,拍戏,刘德华出名之后,他干的第一件事就是找一个经纪人,这个经纪人就是刘德华的代理人(代理),当我们需要找刘德华表演时,不能直接找到刘德华了(刘德华说,你找我代理人商谈具体事宜吧!
),只能是找刘德华的代理人,因此刘德华这个代理人存在的价值就是拦截我们对刘德华的直接访问!
这个现实中的例子和我们在开发中是一样的,我们在开发中之所以要产生一个对象的代理对象,主要用于拦截对真实业务对象的访问。
那么代理对象应该具有什么方法呢?
代理对象应该具有和目标对象相同的方法
所以在这里明确代理对象的两个概念:
1、代理对象存在的价值主要用于拦截对真实业务对象的访问。
2、代理对象应该具有和目标对象(真实业务对象)相同的方法。
刘德华(真实业务对象)会唱歌,会跳舞,会拍戏,我们现在不能直接找他唱歌,跳舞,拍戏了,只能找他的代理人(代理对象)唱歌,跳舞,拍戏,一个人要想成为刘德华的代理人,那么他必须具有和刘德华一样的行为(会唱歌,会跳舞,会拍戏),刘德华有什么方法,他(代理人)就要有什么方法,我们找刘德华的代理人唱歌,跳舞,拍戏,但是代理人不是真的懂得唱歌,跳舞,拍戏的,真正懂得唱歌,跳舞,拍戏的是刘德华,在现实中的例子就是我们要找刘德华唱歌,跳舞,拍戏,那么只能先找他的经纪人,交钱给他的经纪人,然后经纪人再让刘德华去唱歌,跳舞,拍戏。
二、java中的代理
2.1、"java.lang.reflect.Proxy"类介绍
现在要生成某一个对象的代理对象,这个代理对象通常也要编写一个类来生成,所以首先要编写用于生成代理对象的类。
在java中如何用程序去生成一个对象的代理对象呢,java在JDK1.5之后提供了一个"java.lang.reflect.Proxy"类,通过"Proxy"类提供的一个newProxyInstance方法用来创建一个对象的代理对象,如下所示:
1staticObjectnewProxyInstance(ClassLoaderloader,Class
>[]interfaces,InvocationHandlerh)
newProxyInstance方法用来返回一个代理对象,这个方法总共有3个参数,ClassLoaderloader用来指明生成代理对象使用哪个类装载器,Class
>[]interfaces用来指明生成哪个对象的代理对象,通过接口指定,InvocationHandlerh用来指明产生的这个代理对象要做什么事情。
所以我们只需要调用newProxyInstance方法就可以得到某一个对象的代理对象了。
2.2、编写生成代理对象的类
在java中规定,要想产生一个对象的代理对象,那么这个对象必须要有一个接口,所以我们第一步就是设计这个对象的接口,在接口中定义这个对象所具有的行为(方法)
1、定义对象的行为接口
1packagecn.gacl.proxy;
2
3/**
4*@ClassName:
Person
5*@Description:
定义对象的行为
6*@author:
孤傲苍狼
7*@date:
2014-9-14下午9:
44:
22
8*
9*/
10publicinterfacePerson{
11
12/**
13*@Method:
sing
14*@Description:
唱歌
15*@Anthor:
孤傲苍狼
16*
17*@paramname
18*@return
19*/
20Stringsing(Stringname);
21/**
22*@Method:
sing
23*@Description:
跳舞
24*@Anthor:
孤傲苍狼
25*
26*@paramname
27*@return
28*/
29Stringdance(Stringname);
30}
2、定义目标业务对象类
1packagecn.gacl.proxy;
2
3/**
4*@ClassName:
LiuDeHua
5*@Description:
刘德华实现Person接口,那么刘德华会唱歌和跳舞了
6*@author:
孤傲苍狼
7*@date:
2014-9-14下午9:
22:
24
8*
9*/
10publicclassLiuDeHuaimplementsPerson{
11
12publicStringsing(Stringname){
13System.out.println("刘德华唱"+name+"歌!
!
");
14return"歌唱完了,谢谢大家!
";
15}
16
17publicStringdance(Stringname){
18System.out.println("刘德华跳"+name+"舞!
!
");
19return"舞跳完了,多谢各位观众!
";
20}
21}
3、创建生成代理对象的代理类
1packagecn.gacl.proxy;
2
3importjava.lang.reflect.InvocationHandler;
4importjava.lang.reflect.Method;
5importjava.lang.reflect.Proxy;
6
7/**
8*@ClassName:
LiuDeHuaProxy
9*@Description:
这个代理类负责生成刘德华的代理人
10*@author:
孤傲苍狼
11*@date:
2014-9-14下午9:
50:
02
12*
13*/
14publicclassLiuDeHuaProxy{
15
16//设计一个类变量记住代理类要代理的目标对象
17privatePersonldh=newLiuDeHua();
18
19/**
20*设计一个方法生成代理对象
21*@Method:
getProxy
22*@Description:
这个方法返回刘德华的代理对象:
Personperson=LiuDeHuaProxy.getProxy();//得到一个代理对象
23*@Anthor:
孤傲苍狼
24*
25*@return某个对象的代理对象
26*/
27publicPersongetProxy(){
28//使用Proxy.newProxyInstance(ClassLoaderloader,Class
>[]interfaces,InvocationHandlerh)返回某个对象的代理对象
29return(Person)Proxy.newProxyInstance(LiuDeHuaProxy.class
30.getClassLoader(),ldh.getClass().getInterfaces(),
31newInvocationHandler(){
32/**
33*InvocationHandler接口只定义了一个invoke方法,因此对于这样的接口,我们不用单独去定义一个类来实现该接口,
34*而是直接使用一个匿名内部类来实现该接口,newInvocationHandler(){}就是针对InvocationHandler接口的匿名实现类
35*/
36/**
37*在invoke方法编码指定返回的代理对象干的工作
38*proxy:
把代理对象自己传递进来
39*method:
把代理对象当前调用的方法传递进来
40*args:
把方法参数传递进来
41*
42*当调用代理对象的person.sing("冰雨");或者person.dance("江南style");方法时,
43*实际上执行的都是invoke方法里面的代码,
44*因此我们可以在invoke方法中使用method.getName()就可以知道当前调用的是代理对象的哪个方法
45*/
46@Override
47publicObjectinvoke(Objectproxy,Methodmethod,
48Object[]args)throwsThrowable{
49//如果调用的是代理对象的sing方法
50if(method.getName().equals("sing")){
51System.out.println("我是他的经纪人,要找他唱歌得先给十万块钱!
!
");
52//已经给钱了,经纪人自己不会唱歌,就只能找刘德华去唱歌!
53returnmethod.invoke(ldh,args);//代理对象调用真实目标对象的sing方法去处理用户请求
54}
55//如果调用的是代理对象的dance方法
56if(method.getName().equals("dance")){
57System.out.println("我是他的经纪人,要找他跳舞得先给二十万块钱!
!
");
58//已经给钱了,经纪人自己不会唱歌,就只能找刘德华去跳舞!
59returnmethod.invoke(ldh,args);//代理对象调用真实目标对象的dance方法去处理用户请求
60}
61
62returnnull;
63}
64});
65}
66}
测试代码:
1packagecn.gacl.proxy;
2
3publicclassProxyTest{
4
5publicstaticvoidmain(String[]args){
6
7LiuDeHuaProxyproxy=newLiuDeHuaProxy();
8//获得代理对象
9Personp=proxy.getProxy();
10//调用代理对象的sing方法
11StringretValue=p.sing("冰雨");
12System.out.println(retValue);
13//调用代理对象的dance方法
14Stringvalue=p.dance("江南style");
15System.out.println(value);
16}
17}
运行结果如下:
Proxy类负责创建代理对象时,如果指定了handler(处理器),那么不管用户调用代理对象的什么方法,该方法都是调用处理器的invoke方法。
由于invoke方法被调用需要三个参数:
代理对象、方法、方法的参数,因此不管代理对象哪个方法调用处理器的invoke方法,都必须把自己所在的对象、自己(调用invoke方法的方法)、方法的参数传递进来。
三、动态代理应用
在动态代理技术里,由于不管用户调用代理对象的什么方法,都是调用开发人员编写的处理器的invoke方法(这相当于invoke方法拦截到了代理对象的方法调用)。
并且,开发人员通过invoke方法的参数,还可以在拦截的同时,知道用户调用的是什么方法,因此利用这两个特性,就可以实现一些特殊需求,例如:
拦截用户的访问请求,以检查用户是否有访问权限、动态为某个对象添加额外的功能。
3.1、在字符过滤器中使用动态代理解决中文乱码
在平时的JavaWeb项目开发中,我们一般会写一个CharacterEncodingFilter(字符过滤器)来解决整个JavaWeb应用的中文乱码问题,如下所示:
1packageme.gacl.web.filter;
2
3importjava.io.IOException;
4
5importjavax.servlet.Filter;
6importjavax.servlet.FilterChain;
7importjavax.servlet.FilterConfig;
8importjavax.servlet.ServletException;
9importjavax.servlet.ServletRequest;
10importjavax.servlet.ServletResponse;
11
12/**
13*@ClassName:
CharacterEncodingFilter
14*@Description:
解决中文乱码的字符过滤器
15*@author:
孤傲苍狼
16*@date:
2014-9-14下午10:
38:
12
17*
18*/
19publicclassCharacterEncodingFilterimplementsFilter{
20
21@Override
22publicvoidinit(FilterConfigfilterConfig)throwsServletException{
23
24}
25
26@Override
27publicvoiddoFilter(ServletRequestrequest,ServletResponseresponse,
28FilterChainchain)throwsIOException,ServletException{
29//解决以Post方式提交的中文乱码问题
30request.setCharacterEncoding("UTF-8");
31response.setCharacterEncoding("UTF-8");
32response.setContentType("text/html;charset=UTF-8");
33chain.doFilter(request,response);
34}
35
36@Override
37publicvoiddestroy(){
38
39}
40}
但是这种写法是没有办法解决以get方式提交中文参数时的乱码问题的,我们可以用如下的代码来证明上述的解决中文乱码过滤器只对以post方式提交中文参数时有效,而对于以get方式提交中文参数时无效
jsp测试页面如下:
1<%@pagelanguage="java"pageEncoding="UTF-8"%>
2<%--引入jstl标签库--%>
3<%@tagliburi="prefix="c"%>
4
DOCTYPEHTML>
5
6
7
使用字符过滤器解决解决get、post请求方式下的中文乱码问题8
9
10<%--使用c:
url标签构建url,构建好的url存储在servletDemo1变量中--%>
11urlvalue="/servlet/ServletDemo1"scope="page"var="servletDemo1">
12<%--构建的url的附带的中文参数,参数名是:
username,值是:
孤傲苍狼--%>
13paramname="username"value="孤傲苍狼">
param>
14
url>
15<%--使用get的方式访问--%>
16超链接(get方式请求)
17
18<%--使用post方式提交表单--%>
19
20用户名:
21
22
23
24
25
处理请求的ServletDemo1代码如下:
1packageme.gacl.web.controller;
2
3importjava.io.IOException;
4importjava.io.PrintWriter;
5
6importjavax.servlet.ServletException;
7importjavax.servlet.http.HttpServlet;
8importjavax.servlet.http.HttpServletRequest;
9importjavax.servlet.http.HttpServletResponse;
10
11publicclassServletDemo1extendsHttpServlet{
12
13publicvoiddoGet(HttpServletRequestrequest,HttpServletResponseresponse)
14throwsServletException,IOException{
15//接收参数
16Stringusername=request.getParameter("username");
17//获取请求方式
18Stringmethod=request.getMethod();
19//获取输出流
20PrintWriterout=response.getWriter();
21out.write("请求的方式:
"+method);
22out.write("
");
23out.write("接收到的参数:
"+username);
24}
25
26publicvoiddoPost(HttpServletRequestrequest,HttpServletResponseresponse)
27throwsServletException,IOException{
28doGet(request,response);
29}
30}
在web.xml中注册上述的CharacterEncodingFilter和ServletDemo1
1
2CharacterEncodingFilter
3me.gacl.web.filter.CharacterEncodingFilter
4
5
6
7CharacterEncodingFilter
8/*
9
10
11
12ServletDemo1
13me.gacl.web.controller.ServletDemo1
14
15
16
17ServletDemo1
18/servlet/ServletDemo1
19
测试结果如下所示:
从运行结果可以看出,上述的过滤器的确是不能解决以get方式提交中文参数的乱码问题,下面使用动态代理技术改造上述的过滤器,使之能够解决以get方式提交中文参数的乱码问题,改造后的过滤器代码如下:
1packageme.gacl.web.filter;
2
3importjava.io.IOException;
4importjava.lang.reflect.InvocationHandler;
5importjava.lang.reflect.Method;
6importjava.lang.reflect.Proxy;
7
8importjavax.servlet.Filter;
9importjavax.servlet.FilterChain;
10importjavax.servlet.FilterConfig;
11importjavax.servlet.ServletException;
12importjavax.servlet.ServletRequest;
13importjavax.servlet.ServletResponse;
14importjavax.servlet.http.HttpServletRequest;
15importjavax.servlet.http.HttpServletResponse;
16
17/**
18*@ClassName:
CharacterEncodingFilter
19*@Description:
解决中文乱码的字符过滤器
20*@author:
孤傲苍狼
21*@date:
2014-9-14下午10:
38:
12
22*
23*/
24publicclassCharacterEncodingFilterimplementsFilter{
25
26@Override
27publ