使用动态代理解决网站乱码问题
3.1 设计模式
软件开发过程中,遇到相似问题,将问题的解决方式抽取模型(套路)
单例,工厂,适配器,装饰者,动态代理
3.2 谷歌汽车场景
*_ java设计了汽车开发约定
interface ICar{ start run stop}
class GoogleCar implements ICar{}
*_希望在将谷歌Car接入到生态圈平台时,增强汽车启动功能
*_装饰者模式
场景:二次开发的时候,无法获取到源码,无法使用继承前提下,要对已经存在对象上的功能进
行增强.
前提: 可以获取到被装饰的对象GoogleCar实现的所有接口
实现思路: 自定定义装饰类实现ICar接口,为自定义装饰类传递被装饰的对象
弊端:如果被实现的接口中的方法过多,装饰类中的方法过多冗余
*_动态代理模式
原理:通过虚拟机在内存中创建类似MyCar.class文件
要创建MyCar.class文件告诉虚拟机:
1_被创建的字节码文件上应该有多少方法
2_被创建的字节码上的方法如何来实现
字节码加载器:
jdk有一些程序,专业将各种字节码文件加载到内存.这类程序简称为字节码加载器
如何将字节码文件class文件加载到内存?
底层实现过程,利用IO流技术,获取到文件中的数据加载到内存
字节码加载器:3种
系统引导加载器:
3.3 案例:动态代理解决全站乱码问题
步骤
1_new DynamicWeb Project ___>Index.html
<h1>post方式提交中文</h1>
<form action="/day18_v3/ServletDemo" method="post">
User:<input type="" name="username"/><br/>
<input type="submit"/>
</form>
<h1>get方式提交中文</h1>
<form action="/day18_v3/ServletDemo" method="get">
User:<input type="" name="username"/><br/>
<input type="submit"/>
</form>
2_ServletDemo
无论是在post/get方法,执行以下语句不存在中文乱码问题
String um=request.getParameter("username");
System.out.println(um);
3_过滤器中,为request上的getParameter()功能进行增强
思路:
判断当前的请求是get/post request.getMethod();
如果是post, 设置一句话: request.setCharacterEncoding(“utf-8”); ,放行
如果是get,调用原先的String v=request.getParameter(name);
将v进行转码,放行
import java.io.IOException;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServletRequest;
public class EncodingFilter implements Filter {
public EncodingFilter() {
}
public void destroy() {
}
public void init(FilterConfig fConfig) throws ServletException {
}
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
//将request对象转换为HttpServletRequest
final HttpServletRequest req=(HttpServletRequest)request;
//让JDK在内存中生成代理对象:增强了req对象上的getParameter(name);API
HttpServletRequest myReq=(HttpServletRequest)Proxy.newProxyInstance(EncodingFilter.class.getClassLoader(), req.getClass().getInterfaces(), new InvocationHandler() {
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
Object obj=null;
if(method.getName().equalsIgnoreCase("getParameter")){
//获取本次请求方式
String md=req.getMethod();
if("get".equalsIgnoreCase(md)){
//get方式的请求
//调用req对象上的getParameter方法
String v=(String)method.invoke(req, args);
//转码
String vv=new String(v.getBytes("iso-8859-1"),"utf-8");
return vv;
}else{
//post方式的请求
req.setCharacterEncoding("utf-8");
obj=method.invoke(req, args);
}
}else{
obj=method.invoke(req, args);
}
return obj;
}
});
//打印代理对象哈希码
System.out.println(myReq.hashCode());
//将代理对象放行
chain.doFilter(myReq, response);
}
}