自定义注解,使用动态代理解决网站的字符集编码问题

第1章设置环境

安装操作系统,安装备份(镜像):

JDK: 设置环境变量
Eclipse:解压即可 
   Eclipse自身解压目录不包括中文
   代码工作空间目录不包括中文
Tomcat:解压不要包含中文目录
MySql:安装: 选择编码utf8

设置Eclipse

1_关联TOMCAT
  开发过程中,环境发生变化(重新关联tomcat),
  *_删除servers窗口下的tomcat
  *_Project Explorer  窗口下的server
  *_重新关联window--->preferences-->servers--->runtime-->tomcat

第2章自定义注解

2.1什么是注解?

——注解和接口,类一样,都是属于数据类型.

开发中地位:类似dom4j解析XML文件. XML文件的解析程序员不会去解析,配置XML文件 后期的开发中不会自定义注解,反射读取注解信息.

2.2注解作用

  1_编译检查
  2_配置 (后期使用最多*)
  3_生成帮助文档

2.3注解的特点

*_注解可以在变量,方法,类之上加载

*_注解可以有属性也可以没有属性 @Override  @Test(timeout=1000)

*_注解有作用范围(源码,编译期间,运行期间)

源码期间有效: String类之上@Author,@Since,@See ----作用:使用命令javadoc命令将当前的源码生成帮助文件, 可以识别String类上的相关的注解

编译期间有效: @Override @Deprecated @Suppresswarning ----作用:告诉编译器部分信息

运行期间有效: @Test ----作用:当我们再当前代码上以Junit方式运行时,Junit会运行方法上包含@Test注解的方法

2.4回顾JDK中出现的3种注解

@Override

@Override 的作用是:如果想重写父类的方法,比如toString()方法的话,在方法前面加上@Override 系统可以帮你检查方法的正确性。

Override的用法:Override:java.lang.Override是一个marker annotation类型,它被用作标注方法。它说明了被标注的方法重载了父类的方法,起到了断言的作用。

@Deprecated

注解 @Deprecated 可以标记 Java API 状态,可以是以下几种:

  • 使用它存在风险,可能导致错误
  • 可能在未来版本中不兼容
  • 可能在未来版本中删除
  • 一个更好和更高效的方案已经取代它。

Java 9 中注解增加了两个新元素:sinceforRemoval

  • since: 元素指定已注解的API元素已被弃用的版本。
  • forRemoval: 元素表示注解的 API 元素在将来的版本中被删除,应该迁移 API。

@Suppresswarning

Java.lang.SuppressWarnings 是 J2SE5.0中标准的Annotation 之一。

可以标注在类,字段,方法,参数,构造方法,以及局部变量上。

作用

告诉编辑器忽略指定的警告,不用再编译完成后出现警告信息。

使用

@ SuppressWarnings(“ ”)

@ SuppressWarnings({ })

@ SuppressWarnings(value={})

注意:

@SuppressWarnings(value={"unchecked", "deprecation"}) 等同于@SuppressWarnings("unchecked", "deprecation")

若需要多个抑制警告类型,我们可以使用逗号隔开 @ SuppressWarnings(value={“unchecked” , “boxing”})

2.5自定义注解

格式:

public @interface 注解名称{ 
  public 属性类型 属性名称1();
  public 属性类型 属性名称2() default 默认值;
 }

自定义注解属性支持的类型:

基本数据类型(4类8种),String,Class,Annotation(注解类型),枚举类型, 以及以上类型的一维数组类型

注解作用:

配置作用 配置:开发的时候部分信息不希望写死在程序中,例如数据库的用户名和密码,可以将用户名和密码存放在.txt , .properties , .xml文件中,利用程序来读取文件中的内容

框架:一大堆工具类组合,目的:加速项目开发 后期的学习中,框架部分hibernate,spring,struts2很多信息需要配置,提供了2种形式配置 (xml,注解)

什么时候用注解来做配置? 如果配置信息不会发生的修改,例如servlet路径,建议使用注解的形式 如果配置信息需要发生频繁的修改,例如数据库的用户名和密码信息, 建议采用传统方法 (.txt , .properties , .xml)


 
  s002
  
   124324 
   12342143 
  
  

测试使用自定义注解

 @MyAnno01(timeout=100,c=java.util.Date.class,strs={"aaa","bbb"})
 public void test01(){
 }

通过反射读取字节码上的注解信息

md.isAnnotationPresent(MyTest.class)

2.6案例:模拟Junit

1_自定义注解@MyTest

通过元注解@Rentention @Target声明当前注解作用域以及目标对象,如果没有声明,在运行期间是无法获取到注解的信息

2_定义UserDao

创建4个方法addUser delUser uptUser getUser ,在前三个方法上加载注解

3_定义类MyJunit

模拟JUnit 将UserDao.class文件加载到内存, 获取到字节码文件上所有的方法 遍历方法,判断每个方法上是否加载了@MyTest注解 如果当前方法上设置@MyTest,执行当前的方法

第3章使用动态代理解决网站的字符集编码问题

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流技术,获取到文件中的数据加载到内存

类创建的方式
隐式加载:不通过在代码里调用ClassLoader来加载需要的类,而是通过JVM来自动加载需要的类到内存,例如:当类中继承或者引用某个类时,JVM在解析当前这个类不在内存中时,就会自动将这些类加载到内存中。
显示加载:在代码中通过ClassLoader类来加载一个类,例如调用this.getClass.getClassLoader().loadClass()或者Class.forName()。

JDK 默认提供了如下几种ClassLoader
Bootstrp loader
Bootstrp加载器是用C++语言写的,它是在Java虚拟机启动后初始化的,它主要负责加载%JAVA_HOME%/jre/lib,-Xbootclasspath参数指定的路径以及%JAVA_HOME%/jre/classes中的类。

ExtClassLoader  
Bootstrp loader加载ExtClassLoader,并且将ExtClassLoader的父加载器设置为Bootstrp loader.ExtClassLoader是用Java写的,具体来说就是 sun.misc.Launcher$ExtClassLoader,ExtClassLoader主要加载%JAVA_HOME%/jre/lib/ext,此路径下的所有classes目录以及java.ext.dirs系统变量指定的路径中类库。

AppClassLoader 
Bootstrp loader加载完ExtClassLoader后,就会加载AppClassLoader,并且将AppClassLoader的父加载器指定为 ExtClassLoader。AppClassLoader也是用Java写成的,它的实现类是 sun.misc.Launcher$AppClassLoader,另外我们知道ClassLoader中有个getSystemClassLoader方法,此方法返回的正是AppclassLoader.AppClassLoader主要负责加载classpath所指定的位置的类或者是jar文档,它也是Java程序默认的类加载器。

ClassLoader加载类的过程
找到.class文件并把这个文件加载到内存中
字节码验证,Class类数据结构分析,内存分配和符号表的链接
类中静态属性和初始化赋值以及静态代码块的执行
为什么要使用双亲委托这种模型呢?
因为这样可以避免重复加载,当父亲已经加载了该类的时候,就没有必要子ClassLoader再加载一次。考虑到安全因素,我们试想一下,如果不使用这种委托模式,那我们就可以随时使用自定义的String来动态替代java核心api中定义的类型,这样会存在非常大的安全隐患,而双亲委托的方式,就可以避免这种情况,因为String已经在启动时就被引导类加载器(Bootstrcp ClassLoader)加载,所以用户自定义的ClassLoader永远也无法加载一个自己写的String,除非你改变JDK中ClassLoader搜索类的默认算法。

自定义ClassLoader
package com.asiainfo.crm.engine.multi;

import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;

/**
 * 
 * 
 * @ClassName: PathClassLoader
 * @Description: java类作用描述
 * @Author: 孤独的main()函数
 * @CreateDate: 2019/2/26 18:46
 * @UpdateUser: 更新者
 * @UpdateDate: 2019/2/26 18:46
 * @UpdateRemark: 更新说明
 * @Version: 1.0
 */

public class PathClassLoader extends ClassLoader{
    public static final String drive = "d://";
    public static final String fileType = ".class";

    public static void main(String[] args) throws Exception
    {
        PathClassLoader loader = new PathClassLoader();
        Class objClass = loader.loadClass("CommonServiceMultiController", true);
        Object obj = objClass.newInstance();
        System.out.println(objClass.getName());
        System.out.println(objClass.getClassLoader());
        System.out.println(obj.getClass().toString());
    }
    @Override
    public Class findClass(String name)
    {
        byte[] data = loadClassData(name);
        return defineClass(name, data, 0, data.length);// 将一个 byte 数组转换为 Class// 类的实例
    }
    public byte[] loadClassData(String name)
    {
        FileInputStream fis = null;
        byte[] data = null;
        try
        {
            fis = new FileInputStream(new File(drive + name + fileType));
            ByteArrayOutputStream baos = new ByteArrayOutputStream();
            int ch = 0;
            while ((ch = fis.read()) != -1)
            {
                baos.write(ch);
            }
            data = baos.toByteArray();
        } catch (IOException e)
        {
            e.printStackTrace();
        }
        return data;
    }

}
————————————————
版权声明:本文为CSDN博主「min函数」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/weixin_40403930/article/details/87941340

字节码加载器:3种
系统引导加载器:

3.3案例:动态代理解决全站乱码问题

步骤

1_new DynamicWeb Project ___>Index.html

post方式提交中文

User:

get方式提交中文

User:

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进行转码,放行

你可能感兴趣的:(自定义注解,使用动态代理解决网站的字符集编码问题)