JavaWeb - 文件的上传,核心API的使用,文件下载,注解(jdk的注解、自定义注解)

时间:2023-01-27 09:25:05

第一部分:文件的上传

1、什么是文件的上传

1.1 把本地的文件存储到服务器上,这个过程称为文件的上传。比如,网盘、qq空间

1.2 实现文件的上传
(1)jspSmartUpload:应用在jsp的模型一
= 适于嵌入执行上传下载操作的JSP文件中

(2)fileUpload:应用在jsp的模型二(mvc)
= FileUpload 是 Apache commons下面的一个子项目
= 组件FileUpload依赖于Commons IO组件
= 导入jar包(有两个jar包)

(3)要想使用第三方的技术实现文件的上传,首先导入jar包

1.3 要想使用文件的上传,满足三个要求
第一个要求:表单,提交方式是post
= get提交方式:地址栏会携带数据,get大小有限制

第二个要求:在表单里面有文件上传项,必须有name属性 <input type="file" name="filename"/>

第三个要求:在form里面,设置一个属性值 enctype,设置为multipart/form-data

2、使用代码实现文件的上传

2.1 文件上传代码实现的步骤(固定)
第一步:创建磁盘文件项工厂
= new DiskFileItemFactory();
第二步:创建核心上传类
= new ServletFileUpload(FileItemFactory fileItemFactory);
第三步:使用核心上传类解析request对象
= parseRequest(javax.servlet.http.HttpServletRequest request)
= 返回的List集合,集合里面有多个FileItem,List<FileItem>
第四步:遍历list集合,得到每个FileItem
第五步:判断是普通输入项还是文件上传项
= boolean isFormField()
第六步:如果是普通输入项得到值;如果是文件上传项编写上传的代码
= 普输入项
== getFieldName():得到普通输入项name的属性的值
== getString():得到普通输入项里面输入的值
= 文件上传项
== 得到通过表单提交过了的文件的输入流,getInputStream()
== 创建输出流,把文件的输入流写到服务器的一个文件中

3、核心api的使用

3.1 DiskFileItemFactory:磁盘文件项工厂
(1)构造方法
= DiskFileItemFactory(int sizeThreshold, java.io.File repository)
== 有两个参数:
== 第一个参数设置上传文件的缓冲区的大小;
== 第二个参数是如果上传文件超出了缓冲区,产生临时文件,设置临时文件路径

= DiskFileItemFactory()
== setSizeThreshold(int sizeThreshold):设置上传文件的缓冲区的大小
== setRepository(java.io.File repository):设置临时文件路径


3.2 ServletFileUpload:核心上传类
(1)构造方法
= ServletFileUpload(FileItemFactory fileItemFactory)

(2)普通方法
= parseRequest(javax.servlet.http.HttpServletRequest request):解析request对象
返回List集合,集合里面每个部分是FileItem

= setHeaderEncoding(java.lang.String encoding):设置上传文件名称的编码

= setFileSizeMax(long fileSizeMax):设置单个上传文件的大小

= setSizeMax(long sizeMax):设置上传文件的总大小


3.3 FileItem:文件项
(1)普通方法
= boolean isFormField():判断是否是普通输入项,返回boolean,如果返回true是普通输入项

= getFieldName():得到普通输入项name的属性值

= getString():得到普通输入项输入的值
= getString(java.lang.String encoding):输入的值有中文,设置编码

= getName():得到上传文件的名称。在某些浏览器里面得到带路径的名称,截取操作

= getInputStream():得到表单提交的文件的输入流

= delete():删除临时文件

4、练习:js控制多文件的上传

4.1 需求描述
(1)在上传表单里面,有两个按钮,一个是上传,一个是增加
(2)点击增加按钮时候,增加一个文件上传项和删除按钮
(3)点击删除按钮时候,删除一行内容
(4)点击上传按钮时候,把当前文件上传到服务器

4.2 使用js增加一行和删除一行
(1)代码
//增加一行
function addFile() {
//得到div
var div1 = document.getElementById("div1");
div1.innerHTML += "<div><input type='file' name='filename'/><input type='button' value='删除' onclick='del1(this);'/></div>";
}

//删除一行
function del1(who) {
//因为要删除文件上传项和删除按钮,这两个内容都在div里面,删除div就可以了
//得到当前点击的删除按钮所在的div
var div = who.parentNode;
//删除div,使用dom里面的方法 removeChild(),不能自己删除自己,通过父标签删除
//得到要删除div的父标签
var div1 = div.parentNode;
div1.removeChild(div);
}

4.3 使用button提交表单
(1)代码
//提交表单,实现上传(使用button提交的表单)
function uploadfile() {
//得到form标签
var form1 = document.getElementById("form1");
form1.submit();
}

5、文件上传问题的解决

5.1 文件重名的问题
(1)如果上传了多个项目名称的文件,最后一次上传的文件,把之后的文件给覆盖了。

5.2 解决方法
(1)在上传的文件名称里面,添加一个随机的唯一的字符串,保证每个文件名称都是唯一的一个值
(2)生成一个随机的唯一的值
第一种方式:使用毫秒数实现
第二种方式:使用UUID工具类实现
(3)代码
//在文件名称里面添加随机的唯一的值
String uuid = UUID.randomUUID().toString();
//uuid_filename
filename = uuid+"_"+filename;

第二部分:文件的下载

1、什么是文件的下载

1.1 把服务器上文件保存到本地硬盘,这个过程称为文件的下载

1.2 文件下载的实现方式:有两种实现方式
第一种方式:使用超链接实现文件的下载
(1)如果要下载的文件是图片格式,直接在浏览器里面打开;如果下载的是zip格式的文件,才会
提示下载。最终:无论什么格式都是一下载方式打开。

第二种方式:通过代码实现文件的下载
(1)实现的步骤
第一步:设置要下载文件的MIME类型,设置头信息 Content-Disposition,无论是什么格式,都是以下载方式打开
第二步:从服务器上得到要下载的文件的输入流
第三步:创建输出流,通过输出流把文件写到浏览器
第四步:流对接,关闭流

2、代码实现文件的下载

2.1 实现的步骤
第一步:设置要下载文件的MIME类型,设置头信息 Content-Disposition,无论是什么格式,都是以下载方式打开
= 设置mime类型
//设置要下载文件的mime类型
String type = getServletContext().getMimeType(filename);
//设置mime类型
response.setContentType(type);
= 设置头信息
response.setHeader("Content-Disposition", "attachment;filename="+filename);

第二步:从服务器上得到要下载的文件的输入流
//得到要下载文件的输入流
InputStream in = new FileInputStream(path);

第三步:创建输出流,通过输出流把文件写到浏览器
OutputStream out = response.getOutputStream();

第四步:流对接,关闭流


2.2 下载文件名称包含中文的乱码问题解决
(1)不同的浏览器有不同的编码,ie浏览器采用的url编码,火狐采用base64编码
(2)区分不同的浏览器
= 在请求头 User-Agent,得到当前请求的浏览器的类型
(3)代码
//得到当前请求的浏览器类型 ,使用头 User-Agent
String agent = request.getHeader("User-Agent");
if(agent.contains("Firefox")) {//如果是火狐浏览器
//base64编码
filename = "=?UTF-8?B?"+
new BASE64Encoder().encode(filename.getBytes("utf-8"))+"?=";
} else {
//url编码
filename = URLEncoder.encode(filename, "UTF-8");
}

第三部分:注解

1、什么是注解

1.1 是jdk5.0的新特性
(1)jdk5.0很多新特性:泛型、自动拆装箱、增强for循环、可变参数、枚举..

1.2 注解不是注释
(1)注释:给程序员看的
(2)注解:给程序看的
(3)注解就是代码里面的特殊标记,主要功能替代配置文件
其实就是代码里的特殊标记, 它用于替代配置文件,
也就是说,传统方式通过配置文件告诉类如何运行,
有了注解技术后,开发人员可以通过注解告诉类如何运行。
在Java技术里注解的典型应用是:可以通过反射技术去得到类里面的注解,以决定怎么去运行类。

1.3 写法
(1) @注解的名称
(2)可以使用在方法上面,也可以是类上面

2、jdk里面的三个注解

(1) @Override: 限定重写父类方法, 该注解只能用于方法

(2) @SuppressWarnings: 抑制编译器警告.
= @SuppressWarnings("all")
= 不影响程序的运行

(3) @Deprecated: 用于表示某个程序元素(类, 方法等)已过时
= 不影响程序的运行

3、自定义注解

3.1 定义的方式:使用关键字 @interface 注解名称
(1) @interface MyAnno { }

3.2 声明注解的属性
(1)类型 属性的名称();
@interface MyAnno {
//定义属性
String aa();
int bb();
}

3.3 如果定义属性类型是Date类型,报错
Invalid type Date for the annotation attribute MyAnno.date;
only primitive type, String, Class, annotation, enumeration are permitted
or 1-dimensional arrays thereof

(1)属性允许的类型:基本数据类型、string、Class、注解、枚举,一维数组
(2)代码
//自定义注解
@interface MyAnno {
//定义属性
String aa();
int bb();

//定义Date类型
// Date date();

//定义Class类型
Class clazz();

//定义注解
anno ann();

//定义枚举类型
Color c();

//定义一维数组
String[] arr();
}

(3)使用自定义注解
@MyAnno(aa="qq",bb=1,clazz=TestAnno01.class,ann=@anno,c=Color.RED,arr={"aa","bb"})

3.4 特殊属性value
(1)如果把属性的名称写成value,在使用这个注解的时候value可以省略不写

3.5 元注解
(1)修饰注解的注解
@Retention:修饰注解使用范围,自定义一个注解,在默认的情况下是源代码阶段
RetentionPolicy.CLASS: 编译器将把注解记录在 class 文件中. 当运行 Java 程序时, JVM 不会保留注解. 这是默认值
RetentionPolicy.RUNTIME:编译器将把注释记录在 class 文件中. 当运行 Java 程序时, JVM 会保留注解. 程序可以通过反射获取该注释
RetentionPolicy.SOURCE: 编译器直接丢弃这种策略的注释

@Retention(RetentionPolicy.RUNTIME)

@Target:指定注解用于修饰类的哪个成员.
@Target 包含了一个名为 value,类型为ElementType的成员变量。
@Target({ElementType.METHOD,ElementType.PARAMETER})

@Documented: 用于指定被该元 Annotation 修饰的 Annotation 类将被 javadoc 工具提取成文档.

@Inherited: 被它修饰的 Annotation 将具有继承性.
如果某个类使用了被 @Inherited 修饰的 Annotation, 则其子类将自动具有该注解

4、使用反射+注解完成两个练习

4.1 使用反射+注解实现模拟单元测试的效果
(1)什么是单元测试 junit
= 测试一个类中的方法,使用注解方式进行操作 @Test
= 在要测试方法上面添加注解 @Test,选择要测试的方法,点击右键 run as ---> junit test
= 运行之后,在工具里面出现一个JUNIT,有一个绿色的条,表示测试通过了
= 运行之后,在工具里面出现一个JUNIT,如果有红棕色条,表示测试失败
= 在方法上面有 @Test注解,这个方法就可以运行
= @Ignore :当前的方法不参与单元测试

(2)实现的步骤
* 1、得到要测试方法所在类的Class
* = 有三种方式
* 第一种:类名.class
* 第二种:对象.getClass()
* 第三种:Class.forName("包类路径")
* 2、得到类中的所有的方法
* Method[] getMethods() ,返回方法的数组
* 3、遍历数组,得到类中的每个方法
* 4、判断方法上面是否有注解
* boolean isAnnotationPresent(Class<? extends Annotation> annotationClass)
* = 参数:注解的Class
* 5、如果有注解,让方法执行
* invoke(Object obj, Object... args)
* = 第一个参数是类实例;第二个参数执行方法里面的参数

(3)自定义的注解,在默认情况下作用范围是源代码阶段,使用元注解设置的作用范围

(4)代码
//得到要测试方法所在类的Class
Class clazz = TestJunit02.class;
//得到类中的所有的方法
Method[] methos = clazz.getMethods();
//遍历数组,得到每个方法
for (Method m : methos) {
//判断方法上面是否有注解
//自定义注解,在默认情况下作用在源代码阶段
//使用元注解修饰注解的作用范围 @Retention(RetentionPolicy.RUNTIME)
boolean flag = m.isAnnotationPresent(MyTest.class);
//如果有注解,让方法执行
if(flag) {
//方法执行 invoke()
try {
m.invoke(clazz.newInstance());
} catch (Exception e) {
e.printStackTrace();
}
}
}


4.2 使用反射获取注解里面的属性值
(1)使用传统方式实现jdbc操作

(2)使用反射实现jdbc的操作

(3)实现的步骤
第一步:创建注解,注解定义数据库操作信息的属性
第二步:得到类的Class,得到带注解的方法
= Method getMethod(String name, Class<?>... parameterTypes)
== 第一个参数方法名称,第二个参数方法里面参数列表
第三步:得到方法上面的注解
= <T extends Annotation> T getAnnotation(Class<T> annotationClass)
第四步:得到注解里面的属性的值

= 自定义的注解,在默认情况下作用范围是源代码阶段,使用元注解设置的作用范围

(4)核心的代码
//得到Class
Class clazz = TestJDBC02.class;
//得到方法
Method method = clazz.getMethod("testSelect");

//得到方法上面的注解
JdbcInfo info = method.getAnnotation(JdbcInfo.class);
//得到注解里面的属性的值
String drivername = info.drivername();
String url = info.url();
String username = info.username();
String password = info.password();