JavaWeb

来源:互联网 发布:黑龙江药品中标数据网 编辑:程序博客网 时间:2024/05/18 03:03

第一部分:文件的上传

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();