基础加强

学习目标

案例-自定义单元测试@MyTest

案例-全局的编码的解决

一、类加载器

1.什么是类加载器,作用是什么?

类加载器就加载字节码文件(.class)

2.类加载器的种类

类加载器有三种,不同类加载器加载不同的

1)BootStrap:引导类加载器:加载都是最基础的文件

2)ExtClassLoader:扩展类加载器:加载都是基础的文件

3)AppClassLoader:应用类加载器:三方jar包和自己编写java文件

怎么获得类加载器?(重点)

ClassLoader 字节码对象.getClassLoader();

  1. public class Demo {
  2.  
  3. public static void main(String[] args) {
  4.  
  5. //获得Demo字节码文件的类加载器
  6.  
  7. Class clazz = Demo.class;//获得Demo的字节码对象
  8.  
  9. ClassLoader classLoader = clazz.getClassLoader();//获得类加载器
  10.  
  11. //getResource的参数路径相对classes(src)
  12.  
  13. //获得classes(src)下的任何的资源
  14.  
  15. String path = classLoader.getResource("com/itheima/classloader/jdbc.properties").getPath();
  16.  
  17. //classLoader.getResourceAsStream("");
  18.  
  19. System.out.println(path);
  20.  
  21. }
  22.  
  23. }

二、注解 @xxx

1.什么是注解,注解作用

注解就是符合一定格式的语法 @xxxx

注解作用:

注释:在阅读程序时清楚----给程序员看的

注解:给jvm看的,给机器看的

注解在目前而言最主流的应用:代替配置文件

关于配置文件与注解开发的优缺点:

注解优点:开发效率高 成本低

注解缺点:耦合性大 并且不利于后期维护

企业的趋势:混合使用,不经常修改的可以用注解的形式。

2.jdk5提供的注解

@Override:告知编译器此方法是覆盖父类的

@Deprecated:标注过时

@SuppressWarnings:压制警告

deprecation,忽略过时

unused,忽略不使用

rawtypes,忽略类型安全

all,忽略所有

…..

发现的问题:

不同的注解只能在不同的位置使用(方法上、字段上、类上)

3.自定义注解(了解)

1)怎样去编写一个自定义的注解

2)怎样去使用注解

3)怎样去解析注解-----使用反射知识

(1)编写一个注解

关键字:@interface

注解的属性:

语法:返回值 名称();

注意:如果属性的名字是value,并且注解的属性值有一个 那么在使用注解时可以省略value

注解属性类型只能是以下几种

1.基本类型

2.String

3.枚举类型

4.注解类型

5.Class类型

6.以上类型的一维数组类型

(2)使用注解

在类/方法/字段 上面是@XXX

(3)解析使用了注解的类

介入一个概念:元注解:代表修饰注解的注解,作用:限制定义的注解的特性

@Retention

SOURCE: 注解在源码级别可见

CLASS:注解在字节码文件级别可见

RUNTIME:注解在整个运行阶段都可见

@Target

代表注解修饰的范围:类上使用,方法上使用,字段上使用

FIELD:字段上可用此注解

METHOD:方法上可以用此注解

TYPE:类/接口上可以使用此注解

注意:要想解析使用了注解的类 , 那么该注解的Retention必须设置成Runtime

关于注解解析的实质:从注解中解析出属性值

字节码对象存在于获得注解相关的方法

isAnnotationPresent(Class<? extends Annotation> annotationClass) 判断该字节码对象身上是否使用该注解了

getAnnotation(Class<A> annotationClass) :获得该字节码对象身上的注解对象

4.案例-自定义单元测试@MyTest

  1. public class MyTestParster {
  2.  
  3. public static void main(String[] args) throws IllegalAccessException, IllegalArgumentException, InvocationTargetException, InstantiationException {
  4.  
  5. //获得TestDemo
  6.  
  7. Class clazz = TestDemo.class;
  8.  
  9. //获得所有的方法
  10.  
  11. Method[] methods = clazz.getMethods();
  12.  
  13. if(methods!=null){
  14.  
  15. //获得注解使用了@MyTest的方法
  16.  
  17. for(Method method:methods){
  18.  
  19. //判断该方法是否使用了@MyTest注解
  20.  
  21. boolean annotationPresent = method.isAnnotationPresent(MyTest.class);
  22.  
  23. if(annotationPresent){
  24.  
  25. //该方法使用MyTest注解了
  26.  
  27. method.invoke(clazz.newInstance(), null);
  28.  
  29. }
  30.  
  31. }
  32.  
  33. }
  34.  
  35. }
  36.  
  37. }

三、动态代理

1.什么是代理(中介)

目标对象/被代理对象 ------ 房主:真正的租房的方法

代理对象 ------- 黑中介:有租房子的方法(调用房主的租房的方法)

执行代理对象方法的对象 ---- 租房的人

流程:我们要租房----->中介(租房的方法)------>房主(租房的方法)

抽象:调用对象----->代理对象------>目标对象

2.动态代理

动态代理:不用手动编写一个代理对象,不需要一一编写与目标对象相同的方法,这个过程,在运行时 的内存中动态生成代理对象。------字节码对象级别的代理对象

动态代理的API:

在jdk的API中存在一个Proxy中存在一个生成动态代理的的方法newProxyInstance

static Object

newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)

返回值:Object就是代理对象

参数:loader:代表与目标对象相同的类加载器-------目标对

象.getClass().getClassLoader()

interfaces:代表与目标对象实现的所有的接口字节码对象数组

h:具体的代理的操作,InvocationHandler接口

注意:JDK的Proxy方式实现的动态代理 目标对象必须有接口 没有接口不能实现jdk版动态代理

例子一:

  1. public class ProxyTest {
  2.  
  3. @Test
  4.  
  5. public void test1(){
  6.  
  7. //获得动态的代理对象----在运行时 在内存中动态的为Target创建一个虚拟的代理对象
  8.  
  9. //objProxy是代理对象 根据参数确定到底是谁的代理对象
  10.  
  11. TargetInterface objProxy = (TargetInterface) Proxy.newProxyInstance(
  12.  
  13. Target.class.getClassLoader(), //与目标对象相同的类加载器
  14.  
  15. new Class[]{TargetInterface.class},
  16.  
  17. new InvocationHandler() {
  18.  
  19. //invoke 代表的是执行代理对象的方法
  20.  
  21. @Override
  22.  
  23. //method:代表目标对象的方法字节码对象
  24.  
  25. //args:代表目标对象的响应的方法的参数
  26.  
  27. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  28.  
  29. System.out.println("目标方法前的逻辑");
  30.  
  31. //执行目标对象的方法
  32.  
  33. Object invoke = method.invoke(new Target(), args);
  34.  
  35. System.out.println("目标方法后的逻辑");
  36.  
  37. return invoke;
  38.  
  39. }
  40.  
  41. }
  42.  
  43. );
  44.  
  45. objProxy.method1();
  46.  
  47. String method2 = objProxy.method2();
  48.  
  49. System.out.println(method2);
  50.  
  51. }
  52.  
  53. }

例子二:

  1. public class ProxyTest2 {
  2.  
  3. public static void main(String[] args) {
  4.  
  5. final Target target = new Target();
  6.  
  7. //动态创建代理对象
  8.  
  9. TargetInterface proxy = (TargetInterface) Proxy.newProxyInstance(
  10.  
  11. target.getClass().getClassLoader(),
  12.  
  13. target.getClass().getInterfaces(),
  14.  
  15. new InvocationHandler() {
  16.  
  17. @Override
  18.  
  19. //被执行几次?------- 看代理对象调用方法几次
  20.  
  21. //代理对象调用接口相应方法 都是调用invoke
  22.  
  23. /*
  24.  
  25. * proxy:是代理对象
  26.  
  27. * method:代表的是目标方法的字节码对象
  28.  
  29. * args:代表是调用目标方法时参数
  30.  
  31. */
  32.  
  33. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  34.  
  35. //反射知识点
  36.  
  37. Object invoke = method.invoke(target, args);//目标对象的相应方法
  38.  
  39. //retrun返回的值给代理对象
  40.  
  41. return invoke;
  42.  
  43. }
  44.  
  45. }
  46.  
  47. );
  48.  
  49. proxy.method1();//调用invoke---Method:目标对象的method1方法 args:null 返回值null
  50.  
  51. String method2 = proxy.method2();//调用invoke---Method:目标对象的method2方法 args:null 返回值method2
  52.  
  53. int method3 = proxy.method3(100);////调用invoke-----Method:目标对象的method3方法 args:Object[]{100} 返回值100
  54.  
  55. System.out.println(method2);
  56.  
  57. System.out.println(method3);
  58. }
  59. }

3.案例-全局的编码的解决

  1. public class EncodingFilter implements Filter{
  2.  
  3. @Override
  4.  
  5. public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)
  6.  
  7. throws IOException, ServletException {
  8.  
  9. final HttpServletRequest req = (HttpServletRequest) request;
  10.  
  11. //使用动态代理完成全局编码
  12.  
  13. HttpServletRequest enhanceRequset = (HttpServletRequest) Proxy.newProxyInstance(
  14.  
  15. req.getClass().getClassLoader(),
  16.  
  17. req.getClass().getInterfaces(),
  18.  
  19. new InvocationHandler() {
  20.  
  21. @Override
  22.  
  23. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  24.  
  25. //对getParameter方法进行增强
  26.  
  27. String name = method.getName();//获得目标对象的方法名称
  28.  
  29. if("getParameter".equals(name)){
  30.  
  31. String invoke = (String) method.invoke(req, args);//乱码
  32.  
  33. //转码
  34.  
  35. invoke = new String(invoke.getBytes("iso8859-1"),"UTF-8");
  36.  
  37. return invoke;
  38. }
  39.  
  40. return method.invoke(req, args);
  41. }
  42. }
  43. );
  44.  
  45. chain.doFilter(enhanceRequset, response);
  46.  
  47. //request.setCharacterEncoding("UTF-8");
  48.  
  49. //在传递request之前对request的getParameter方法进行增强
  50.  
  51. /*
  52.  
  53. * 装饰者模式(包装)
  54.  
  55. *
  56.  
  57. * 1、增强类与被增强的类要实现统一接口
  58.  
  59. * 2、在增强类中传入被增强的类
  60.  
  61. * 3、需要增强的方法重写 不需要增强的方法调用被增强对象的
  62.  
  63. *
  64.  
  65. */
  66.  
  67. //被增强的对象
  68.  
  69. //HttpServletRequest req = (HttpServletRequest) request;
  70.  
  71. //增强对象
  72.  
  73. //EnhanceRequest enhanceRequest = new EnhanceRequest(req);
  74.  
  75. //chain.doFilter(enhanceRequest, response);
  76.  
  77. }
  78.  
  79. @Override
  80.  
  81. public void destroy() {
  82. }
  83. @Override
  84.  
  85. public void init(FilterConfig filterConfig) throws ServletException {
  86. }
  87. }
  88.  
  89. class EnhanceRequest extends HttpServletRequestWrapper{
  90.  
  91. private HttpServletRequest request;
  92.  
  93. public EnhanceRequest(HttpServletRequest request) {
  94.  
  95. super(request);
  96.  
  97. this.request = request;
  98.  
  99. }
  100.  
  101. //对getParaameter增强
  102.  
  103. @Override
  104.  
  105. public String getParameter(String name) {
  106.  
  107. String parameter = request.getParameter(name);//乱码
  108.  
  109. try {
  110.  
  111. parameter = new String(parameter.getBytes("iso8859-1"),"UTF-8");
  112.  
  113. } catch (UnsupportedEncodingException e) {
  114.  
  115. e.printStackTrace();
  116.  
  117. }
  118.  
  119. return parameter;
  120.  
  121. }
  122.  
  123. }

【JAVAWEB学习笔记】25_基础加强:类加载器、注解 @xxx和动态代理的更多相关文章

  1. Javaweb学习笔记——(二十八)——————Servlet3.0、动态代理、类加载器

    web最后一天:完了. Servlet3.0          一.要求         1.MyEclipse10.0或以上版本         2.发布到Tomcat7.0或以上版本 二.步骤   ...

  2. 【JVM学习笔记】线程上下文类加载器

    有许多地方能够看到线程上下文类加载的设置,比如在sun.misc.Launcher类的构造方法中,能够看到如下代码 先写一个例子建立感性认识 public class Test { public st ...

  3. 大数据之路week05--day07(序列化、类加载器、反射、动态代理)

    遇到这个 Java Serializable 序列化这个接口,我们可能会有如下的问题 a,什么叫序列化和反序列化b,作用.为啥要实现这个 Serializable 接口,也就是为啥要序列化c,seri ...

  4. JavaWeb学习笔记——jsp基础语法

    1.JSP注释 显式注释 <!-- 注释内容 --> 隐式注释,隐式注释在客户端无法看见 // /* */ <% 注释内容 %> 2.Scriptlet(小脚本程序) 所有嵌入 ...

  5. JavaWeb学习笔记——JavaEE基础知识

  6. 《Java编程思想》学习笔记(二)——类加载及执行顺序

    <Java编程思想>学习笔记(二)--类加载及执行顺序 (这是很久之前写的,保存在印象笔记上,今天写在博客上.) 今天看Java编程思想,看到这样一道代码 //: OrderOfIniti ...

  7. MyBatis:学习笔记(1)——基础知识

    MyBatis:学习笔记(1)--基础知识 引入MyBatis JDBC编程的问题及解决设想 ☐ 数据库连接使用时创建,不使用时就释放,频繁开启和关闭,造成数据库资源浪费,影响数据库性能. ☐ 使用数 ...

  8. JavaWeb学习笔记总结 目录篇

    JavaWeb学习笔记一: XML解析 JavaWeb学习笔记二 Http协议和Tomcat服务器 JavaWeb学习笔记三 Servlet JavaWeb学习笔记四 request&resp ...

  9. Quartz学习笔记:基础知识

    Quartz学习笔记:基础知识 引入Quartz 关于任务调度 关于任务调度,Java.util.Timer是最简单的一种实现任务调度的方法,简单的使用如下: import java.util.Tim ...

随机推荐

  1. Bar 3D 和Pie 3D的统计图形

    最近在做一个关于图形统计的界面,主要用到的是Dev控件ChartControl(功能很强大,能解决基本和复杂图形统计问题). ChartControl主要有Diagram.Series.Legend三 ...

  2. 网站与域名知识扫盲-DNS

    域名概述 域名的概念 IP地址不易记忆 早期使用Hosts解析域名 主机名称重复 主机维护困难 DNS(Domain Name System 域名系统) 分布式 层次性 域名空间结构 根域 组织域[. ...

  3. linux入门之用户管理

    用户管理 添加用户 以root用户执行 adduser  或 useradd [new_account] -u  UID   -d 指定家目录 -g GID 指定一个基本组ID   -G指定一个附加组 ...

  4. nodejs + nginx + ECS阿里云服务器环境设置

    nodejs + nginx + ECS阿里云服务器环境设置 部署 nodejs ECS 基于 CentOS7.2 详细步骤:click 部署 nginx 安装 添加Nginx软件库: [root@l ...

  5. redis 字符串

    redis 字符串 概述 redis 没有使用 c 语言风格的字符串表示(以 "\0" 作为结尾), 而是使用自定义的 sds 结构 字符串结构 定义位置 (src/sds.h) ...

  6. iOS 2017年, 上传审核被拒绝.到奔溃

    2017年,苹果并没有因为新年的气氛而对CP们"网开一面".频繁锁榜.调整排名规则以及关键词覆盖算法--不断抛出的大动作,让CP们叫苦不迭.且从1月初开始,苹果还进一步加强了对应用 ...

  7. CPP笔记_泛型编程简单总结

    本篇是基于<Essential C++>第三章泛型编程风格的一个简单总结 1 Iterator vector<string>::iterator  表明此iterator是位于 ...

  8. HashMap 构造函数

    HashMap总共提供了三个构造函数 /** * Constructs an empty <tt>HashMap</tt> with the default initial c ...

  9. phpcms基础

    CSM基础(做中小型企业网站) 做一个企业站,三个页面比较重要1.首页2.列表页3.内容页 做企业站的流程:1.由美工出一张,设计效果图2.将设计图静态化3.开始安装CMS4.强模板文件放到CSM里面 ...

  10. Linux基础(4)

    Linux基础(四) 通过前面的知识的学习,来现学现卖咯! 1.题目:集群搭建 1.1.部署nginx反向代理三个web服务,调度算法使用加权轮询: 1.2.所有web服务使用共享存储nfs,保证所有 ...