第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 中注解增加了两个新元素:since 和 forRemoval。
- 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方式提交中文
get方式提交中文
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进行转码,放行