19-JavaWeb基础-Filter

1. 介绍

当访问资源时,如果不希望此资源被访问,可以使用过滤器进行拦截。
访问一个资源,需要路径,过滤器通过路径匹配进行资源拦截。
过滤器必须实现接口:javax.servlet.Filter

Filter
    init(FilterConfig):void        初始化
    doFilter(ServletRequest,ServletResponse,FilterChain):void    过滤方法
    destroy():void        销毁

默认情况下过滤器会拦截匹配路径和执行的资源
需要手动放行:chain.doFilter(request,response);

2.使用方法

详见统计访问次数的案例

3.生命周期

3.1 init(FilterConfig)

初始化方法
    执行时机:服务器启动时执行
    执行者:tomcat
FilterConfig:当前过滤器的配置对象
    getFilterName():String    过滤名称,相当于<filter-name>标签
    getServletContext():ServletContext    ServletContext对象引用
    getInitParametes(String):String    获得过滤器初始化参数
    getInitParameterNames():Enumeration    获得过滤器初始化参数的所有名字

    <filter>
        <filter-name>CountFilter</filter-name>
        <filter-class>com.lujiahao.cms.filter.CountFilter</filter-class>
        <init-param>
            <param-name>encoding</param-name>
            <param-value>UTF-8</param-value>
    </filter>

3.2 doFilter(ServletRequest,ServletResponse ,FilterChain)

路径匹配一次执行一次。一次请求拦截一次。

3.3 destroy()

执行时机:服务器正常关闭时执行
执行者:tomcat

4.doFilter方法详解

参数1ServletRequest 和参数2 ServletResponse

回顾:Object               List               ArrayList
      ServletRequest        HttpServletRequest        tomcat实现类
结论:可以强转获得Http协议有关对象
    HttpServletRequest  request = (HttpServletRequest) req;
    HttpServletResponse  request = (HttpServletResponse) res;

参数3:FilterChain

FilterChain 过滤器链,当请求资源时,多个过滤器都生效,tomcat将生成一个链,用于存放多个过滤器。
第一个过滤器执行完成,并发行时,将执行第二个过滤器,。。。。。当最后一个过滤器放行时,将执行资源。

aaaaa.png

过滤器链中 过滤器执行顺序 与 映射在web.xml配置顺序一致的。web.xml 先配置先执行

5.<url-pattern>匹配路径

回顾:servlet路径配置

1.完全匹配,必须/开头 ,例如:/a/b/c/oneServlet
2.不完全匹配,以/开头   *结尾,例如:/a/b/*  , /*
3.通配符匹配. *.开头,例如: *.jsp  、 *.do   、*.action 等
4 缺省  /   以上都没有匹配将执行。

过滤器路径编写,与servlet一致的,过滤器过滤的就是指定servlet。

/oneServlet  、 /twoServlet
/*   过滤器 将匹配以上两个servlet

6.dispatcher配置

用于配置过滤器在何处进行拦截。

取值:    REQUEST、 FORWARD、 INCLUDE、ERROR
request : 表示在请求开始时拦截。默认值。
forward:表示在请求转发开始时拦截。及 request.getRequestDistacher(..).forward 被调用时。
include:表示在请求包含开始时拦截。及 request.getRequestDistacher(..).include 被调用时。
error : 表示程序异常,显示友好页面之前拦截。

案例〇 : dispatcher配置友好错误页面

IE会将500的错误以他自己的方式展现出来,这就不符合需求,所以需要手动配置友好错误页面.
如果有多个友好错误界面,不能在每个jsp页面更改状态码,这就需要一个过滤器来统一过滤.

1.创建FriendErrorFilter实现javax.servlet.Filter接口

/**
 * 友好错误界面过滤器
 * Created by lujiahao on 2016/7/27.
 */
public class FriendErrorFilter implements Filter {
    @Override
    public void init(FilterConfig filterConfig) throws ServletException {

    }

    @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
        HttpServletRequest request = (HttpServletRequest) servletRequest;
        HttpServletResponse response = (HttpServletResponse) servletResponse;

        // 修改状态码
        response.setStatus(200);
        // 放行
        filterChain.doFilter(request,response);
    }

    @Override
    public void destroy() {

    }
}

2.配置到tomcat(web.xml)

<!--配置友好错误页面过滤器-->
<error-page>
    <error-code>500</error-code>
    <location>/demo0/frienderror.jsp</location>
</error-page>
<!--友好错误页面过滤器-->
<filter>
    <filter-name>FriendErrorFilter</filter-name>
    <filter-class>com.lujiahao.filter.FriendErrorFilter</filter-class>
</filter>
<filter-mapping>
    <filter-name>FriendErrorFilter</filter-name>
    <url-pattern>/frienderror.jsp</url-pattern>
    <dispatcher>ERROR</dispatcher>
</filter-mapping>

3.编写友好错误界面(error.jsp)

<html>
    <head>
        <title>Title</title>
    </head>
    <body>
        我是友好错误界面<br/>
        服务器繁忙,请稍后重试
    </body>
</html>

4.模拟服务器500错误

<%
    int i = 1/0;
%>

案例一 : 统计访问次数

原理:将数据记录到ServletContext 所有用户共享

1.创建CountFilter实现javax.servlet.Filter接口

/**
 * 统计次数的过滤器
 * Created by lujiahao on 2016/7/27.
 */
public class CountFilter implements Filter {
    @Override
    public void init(FilterConfig filterConfig) throws ServletException {

    }

    @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
        HttpServletRequest request = (HttpServletRequest) servletRequest;
        HttpServletResponse response = (HttpServletResponse) servletResponse;

        // 获得访问路径   /pages/main/main.jsp
        String servletPath = request.getServletPath();
        String filePath = servletPath.substring(servletPath.lastIndexOf("/")+1,servletPath.lastIndexOf("."));

        // 获得servletContext获取统计数据
        ServletContext servletContext = request.getSession().getServletContext();
        Integer num = (Integer) servletContext.getAttribute(filePath);
        if (num == null) {
            num = 1;// 第一次访问
        } else {
            num ++;
        }
        // 存放到ServletContext中
        servletContext.setAttribute(filePath,num);

        // 放行
        filterChain.doFilter(request,response);
    }

    @Override
    public void destroy() {

    }
}

2.配置到tomcat(web.xml)

<!--统计次数的过滤器-->
<filter>
    <filter-name>CountFilter</filter-name>
    <filter-class>com.lujiahao.cms.filter.CountFilter</filter-class>
</filter>
<filter-mapping>
    <filter-name>CountFilter</filter-name>
    <url-pattern>/pages/main/main.jsp</url-pattern>
</filter-mapping>

3.JSP中使用

访问次数:${applicationScope.main}次<br/>

案例三 : 自动登录

原理图:
QQ截图20160727163940.png

1.修改以前的登录逻辑,添加对自动登录的标记

/**
 * 登录操作
 */
private void login(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
    // 1.获取数据并封装
    Customer customer = LBeanUtils.populate(Customer.class, request.getParameterMap());
    // 2.通知service进行登录
    CustomerServeice customerServeice = new CustomerServiceImpl();
    Customer loginCustomer = customerServeice.login(customer);

    // 3.处理
    if (loginCustomer != null) {
        // -------------自动登录start-------------
        // checkbox选中的话获得的值是on
        String autologin = request.getParameter("autologin");
        if (autologin != null) {
            // 勾选了记住密码,  使用cookie将用户的登录名和密码发送到浏览器,并通知浏览器保存(持久化cookie)
            // 1.创建cookie对象
            String username = URLEncoder.encode(loginCustomer.getName(), "UTF-8");
            String password = URLEncoder.encode(loginCustomer.getPwd(), "UTF-8");
            // 存入需要进行URLEndode编码,不然会报错:http://blog.csdn.net/liuxiao723846/article/details/22155393
            Cookie cookie = new Cookie("autoLoginCookie", username + "&" + password);
            // 2.设置有效时间
            cookie.setMaxAge(60 * 60);// 1小时
            // 3.设置路径
            cookie.setPath("/");
            // 4.发送cookie
            response.addCookie(cookie);
        }
        // -------------自动登录end-------------

        // 成功  -- session记录登录状态,重定向到主页面
        // * session作用域保存数据
        request.getSession().setAttribute("loginCustomer", loginCustomer);
        response.sendRedirect(request.getContextPath() + "/pages/main/main.jsp");
    } else {
        // 不成功  --- request记录当次请求提示,请求转发到login.jsp 显示数据
        request.setAttribute("msg", "用户名和密码不匹配");
        request.setAttribute("customer", customer);// 回显数据
        request.getRequestDispatcher("pages/login/login.jsp").forward(request, response);
    }
}

2.编写自动登录的过滤器

/**
 * 自动登录的过滤器
 * Created by lujiahao on 2016/7/27.
 */
public class AutoLoginFilter implements Filter {
    @Override
    public void init(FilterConfig filterConfig) throws ServletException {

    }

    @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
        HttpServletRequest request = (HttpServletRequest) servletRequest;
        HttpServletResponse response = (HttpServletResponse) servletResponse;

        Customer loginCustomer = (Customer) request.getSession().getAttribute("loginCustomer");
        // 1.已经登录
        if (loginCustomer != null) {
            filterChain.doFilter(request,response);
            return;
        }

        // 2.没有登录,获取cookie信息
        Cookie[] allCookie = request.getCookies();
        String cookieValue = null;
        if (allCookie != null) {
            for (Cookie cookie : allCookie) {
                if ("autoLoginCookie".equals(cookie.getName())) {
                    cookieValue = cookie.getValue();
                    break;// 一旦查询到了就跳出循环,提升性能
                }
            }
        }

        // 3.没有cookie信息
        if (cookieValue == null) {
            filterChain.doFilter(request,response);
            return;
        }

        // 4.查询到cookie信息,获得用户数据
        String username = URLDecoder.decode(cookieValue.split("&")[0],"UTF-8");
        String password = URLDecoder.decode(cookieValue.split("&")[1],"UTF-8");
        Customer customer = new Customer(username,password);

        // 5.查询用户信息
        CustomerServeice customerServeice = new CustomerServiceImpl();
        loginCustomer = customerServeice.login(customer);

        // 6.没有查询到用户信息----例如:密码修改
        if (loginCustomer == null) {
            filterChain.doFilter(request,response);
            // 应该删除cookie
            Cookie cookie = new Cookie("autoLoginCookie","");
            cookie.setMaxAge(0);// 删除
            cookie.setPath("/");// 保存的时候写的路径要和这里一样
            response.addCookie(cookie);
            return;
        }

        // 7.查询到用户信息,执行自动登录
        request.getSession().setAttribute("loginCustomer",loginCustomer);

        // 放行
        filterChain.doFilter(request,response);
    }
    @Override
    public void destroy() {

    }
}

案例四 : GET/POST中文乱码统一处理

1.旧的处理方式:

public class GetPostServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        /*
        String username = request.getParameter("username");
        String password = request.getParameter("password");

        //处理编码
        username = new String(username.getBytes("ISO8859-1"),"UTF-8");
        password = new String(password.getBytes("ISO8859-1"),"UTF-8");

        System.out.println("get:");
        System.out.println(username);
        System.out.println(password);
        */
        this.doPost(request, response);
    }

    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //post请求乱码
        //request.setCharacterEncoding("UTF-8");

        String username = request.getParameter("username");
        String password = request.getParameter("password");

        System.out.println("post:");
        System.out.println(username);
        System.out.println(password);
    }
}

2.使用过滤器+装饰着模式的处理方式:

/**
 * 处理Get/Post请求乱码的过滤器
 * Created by lujiahao on 2016/7/27.
 */
public class GetPostEncodingFilter implements Filter {
    @Override
    public void init(FilterConfig filterConfig) throws ServletException {

    }

    @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
        HttpServletRequest request = (HttpServletRequest) servletRequest;
        HttpServletResponse response = (HttpServletResponse) servletResponse;

        //编码设置 另一个filter编写   这个只能解决POST乱码的问题
        request.setCharacterEncoding("UTF-8");

        // 这个里面有处理GET乱码
        // 使用装饰者模式增强的request
        MyRequest myRequest = new MyRequest(request);
        // 放行
        filterChain.doFilter(myRequest,response);
    }

    @Override
    public void destroy() {

    }
}

/**
 * 使用装饰者设计模式
 * HttpServletRequestWrapper是系统为我们准备好的装饰者的父类,已经完成了相应的工作
 * Created by lujiahao on 2016/7/27.
 */
public class MyRequest extends HttpServletRequestWrapper {
    private boolean isEncoding = false;// 默认是没有编码的
    private HttpServletRequest request;
    public MyRequest(HttpServletRequest request) {
        super(request);// 将tomcat的request传入父类,提供给不需要增强的方法使用
        this.request = request;
    }

    // 需要增强的方法直接覆写父类方法就好了

    // 通过名称获得第一个值
    @Override
    public String getParameter(String name) {
        String[] parameterValues = this.getParameterValues(name);
        if (parameterValues == null) {
            return null;
        }
        return parameterValues[0];
    }

    // 通过名称获得所有的值
    @Override
    public String[] getParameterValues(String name) {
        return this.getParameterMap().get(name);
    }

    @Override
    public Map<String, String[]> getParameterMap() {
        // 1.获得tomcat原始数据
        Map<String, String[]> map = request.getParameterMap();
        // 2.处理get请求的乱码
        if ("GET".equals(request.getMethod()) && !isEncoding) {
            // 3.遍历map
            for(Map.Entry<String,String[]> entry : map.entrySet()){
                // 4.获得所有value数据
                String[] value = entry.getValue();
                // 处理所有乱码
                for (int i = 0; i < value.length; i++) {
                    try {
                        value[i] = new String(value[i].getBytes("ISO-8859-1"),"UTF-8");
                    } catch (UnsupportedEncodingException e) {
                        throw new RuntimeException(e);
                    }
                }
            }
            isEncoding = true;// 已经解决完乱码就不要再次解决了
        }
        return map;
    }
}

案例五 : 页面静态化 & 全栈压缩

页面静态化原理图:
QQ截图20160728133730.png
1.装饰者模式增强HttpServletResponse

public class MyResponse extends HttpServletResponseWrapper{

    private HttpServletResponse response;

    //提供自定义缓存
    private ByteArrayOutputStream baos = new ByteArrayOutputStream();

    private PrintWriter pw = null;
    public MyResponse(HttpServletResponse response) {
        super(response);
        this.response = response;
    }

    @Override
    public PrintWriter getWriter() throws IOException {
        if(pw == null){
            pw = new PrintWriter(new OutputStreamWriter(baos , "UTF-8" ));
            System.out.println("执行" + pw);
        }
        return pw;
    }

    /**
     * 获得自定义缓存内容
     * @return
     */
    public byte[] getData(){
        return baos.toByteArray();
    }

}

2.过滤器写法

public class PageStaticFilter implements Filter {

    @Override
    public void init(FilterConfig filterConfig) throws ServletException {
    }

    @Override
    public void doFilter(ServletRequest req, ServletResponse resp,
            FilterChain chain) throws IOException, ServletException {
        HttpServletRequest request = (HttpServletRequest) req;
        HttpServletResponse response = (HttpServletResponse) resp;

        //1 获得页面位置
        //D:\java\tomcat\apache-tomcat-7.0.53\webapps\day19_demo\demo04\1.html
        ServletContext sc = request.getSession().getServletContext();
        String id= request.getParameter("id");
        String path = "/demo04/"+id+".html";// 静态页面存放的位置
        String htmlPath = sc.getRealPath(path);

        //2 文件是否存在
        File htmlFile = new File(htmlPath);
        if(htmlFile.exists()){
            //存在就显示
            request.getRequestDispatcher(path).forward(request, response);
            return;
        }

        // 自定义response 提供缓存,用于存放 发送到浏览器所有内容
        MyResponse myResponse = new MyResponse(response);
        //放行
        chain.doFilter(request, myResponse);

        //内容已经写入到自定义缓存
        byte[] arrayByte = myResponse.getData();

        /***写入到文件 start**/
        //将指定的内容内容到文件
        FileOutputStream out = new FileOutputStream(htmlFile);
        out.write(arrayByte);
        out.close();

        //手动写入到浏览器
        response.getOutputStream().write(arrayByte);
        /***写入到文件 end**/

        /**全站压缩 , 将需要发送的数据,压缩之后再发给浏览器,节省流量
            全站压缩和页面静态化是冲突的,需要一个一个来测试**/
        /*
        // 设置发送的数据位压缩数据
        response.setHeader("content-encoding", "gzip");
        ByteArrayOutputStream baos = new ByteArrayOutputStream(); //存放压缩后的结果
        GZIPOutputStream gzipOut = new GZIPOutputStream(baos); //压缩的位置
        gzipOut.write(arrayByte); //需要压缩的原始数据
        gzipOut.close();

        response.getOutputStream().write(baos.toByteArray());
        */
    }

    @Override
    public void destroy() {
    }
}

扩展:装饰者设计模式

乱码解决的十几分钟的时候讲了这个东西

jar包和war包的区别

java项目 压缩 :jar包
web项目 压缩:war包 。 当war添加到tomcat/webapps下将自动解压
jar包和war包压缩格式都是 zip

升级订制版HttpFilter

这个非常好,用到的思想和Servlet中的service()方法非常类似,好好总结一下
http://www.cnblogs.com/jianjianyang/p/5001471.html

11-JavaWeb基础-EL-JSTL

EL表达式

EL介绍

EL(Expression Language) 目的:为了使JSP写起来更加简单。表达式语言的灵感来自于 ECMAScript 和 XPath 表达式语言,它提供了在 JSP 中简化表达式的方法,让Jsp的代码更加简化。

阅读更多

10-JavaWeb基础-JSP笔记

什么是JSP

  • tomcat安装目录下的work目录就是tomcat处理jsp的工作目录
  • JSP是以Java语言为基础的动态网页开发技术(Servlet也是动态网页开发技术)
  • 二者特点
    • servlet特点:在java代码中嵌入html代码
    • jsp特点:在html代码中嵌入java代码
阅读更多

基础加强--框架加强

新特性

Static Import : 静态导入,当前类中可以直接使用静态资源(字段|方法)

格式:import static java.lang.Math.*;

Varargs :可变参数

public void demo(String... args){ 当成数组使用 }

Autoboxing/Unboxing 自动装箱、自动拆箱(基本类型 <–> 包装类)

装箱:基本类型 --> 包装类。例如:Integer i = 10;  --> Integer m = new Integer(10);
拆箱:包装类 --> 基本类型。例如:int j = i;  --> int n = m.intValue();

Enhanced for Loop 增强for循环

for(类型 变量 : 容器){}   --> 容器:数组、Iterable接口(iterator方法)

Typesafe Enums 枚举

public class Hello {}
public enum Color {}

泛型

特点

1.将运行时的问题提升到编译时
2.方法级别的泛型定义主要是用在工具类中的
    public class TestGeneric2 {
        public void demo(){
            int m = 10;
            String s = init("abc");
            Integer i = init(123);
        }
        //方法级别的定义  MyBeanUtils
        public <T> T init(T obj){
            //算法
            System.out.println(obj);
            return obj;
        }
    }

泛型擦除

以下两个方法不能同时存在
public void init(List<String> list){}
public void init(List<Integer> list){}
类型擦除之后,两个方法是完全相同

反射中Class对象的三种获取方式:

Class.forName("全限定类名");
String.class
obj.getClass

反射和泛型混合使用的例子:

public class Dao<T> {
    private Session session;
    private Class beanClass;
    public Dao() {
        //1 在运行时,获得当前运行类 父类中泛型的实际参数
        //this.getClass().getSuperclass(); //jdk1.4之前,获得父类,不含泛型信息
        // * 获得被参数化的类型  ArrayList<Integer> ,含有泛型信息
        ParameterizedType type = (ParameterizedType) this.getClass().getGenericSuperclass();
        //2 获得所有的实际参数类型。类<A,B,C,D> ,返回数组,获得第一个实际参数
        Type firstType = type.getActualTypeArguments()[0];

        this.beanClass = (Class) firstType;

        /*
            //回顾 反射 api
            beanClass.getName(); //类名
            beanClass.getMethod(name, parameterTypes)  //方法
            beanClass.getConstructor(parameterTypes) //构造
            beanClass.getField(name)  //字段
        */
        System.out.println(beanClass);
    }

    /* 泛型 ,将运行时问题 提升 编译时
     * 泛型实参只有在运行时才可以确定,编译时 T 不能确定类型(User | Book) 只有在运行时才可以确定
     * 思想:如何在运行时获得实际类型参数?
     */
    public T findById(Integer id){
        return (T) session.get(beanClass, id);
    }
}

注解

介绍

注解就是类,用于修饰对象(类/构造/字段/方法等),常用语取代xml配置.
但是,开发中常用注解+xml混合使用

JDK提供的常用注解

@Override
    在jdk1.5表示子类覆写父类的方法
    在jdk1.6表示子类实现父接口的方法
@Deprecated
    表示被修饰对象已经过期了,过期的方法依旧可以使用.
    以下情况被标记过期:1.安全问题 2.有新的API
@SuppressWarnings
    抑制警告,如果有警告通过编辑器不进行警告
        deprecation:如果过期不警告
        rawtypes:没有泛型
        unused:未使用
        null:空指针
        serial:序列化
        all:所有
    建议:尽量不使用抑制

自定义注解

注解架构

定义注解
使用注解
解析注解

基本语法

定义注解使用关键字 : `@interface` ,和定义类class相似
    public @interface xxx{}
成员格式: 修饰符 返回值 属性名() [default 默认值]
    eg: public String username();
    注意点:
        1.修饰符:默认值 public abstract,且只能是这两个
        2.返回值:只能是基本类型/字符串String/Class/注解/枚举,以及以上类型的一维数组.

使用注解—重点!!!

注解出现的目的就是用来代替xml的

AJAX笔记

Ajax介绍&编写流程

使用异步的方式从浏览器端发送请求,请求服务器端资源,并获得内容一种技术。
ajax 不是新技术,是多个技术整合:javascript、html、css、xml,XMLHttpRequest
a.png

使用步骤:

1.获得核心类(引擎)
2.编写回调函数,获得响应内容
3.建立连接
4发送请求

同步操作 : 访问servlet地址栏改变
异步操作 : 访问servlet地址栏不改变

Jsonlib

需要导入的jar包:

commons-beanutils-1.8.3.jar
commons-collections-3.2.1.jar
commons-lang-2.6.jar
commons-logging-1.1.1.jar
ezmorph-1.0.6.jar
json-lib-2.4-jdk15.jar

Java对象转json:

JSONArray.fromObject(javaObject).toString()

Json转Java对象:

JSONObject obj = JSONObject.fromObject(jsonStr);
Bean bean = (Bean) JSONObject.toBean(obj, Bean.class);

注意:

JsonConfig配置信息,设置排除的字段

User user = new User("u001", "jack", "1234");
// 配置信息 --设置排除
JsonConfig jsonConfig = new JsonConfig();
jsonConfig.setExcludes(new String[]{"username","password"});

// java 对象  转换 json  --> {'k':'v',.....}
String str = JSONObject.fromObject(user,jsonConfig).toString();
System.out.println(str);// {"id":"u001"}

XStream

需要导入的jar包:

xmlpull-1.1.3.1.jar
xpp3_min-1.1.4c.jar
xstream-1.4.7.jar

Java对象转xml:

public void demo02(){
    User user = new User("u007", "路家豪", "123");
    //1 核心类
    XStream xStream = new XStream();
    // * 设置别名,如果不设置的话会使用User类的全限定类名来作为根元素
    xStream.alias("user", User.class);
    //2 转换成xml
    String xmlData = xStream.toXML(user);
    System.out.println(xmlData);
    /*
     * <user>
          <id>u007</id>
          <username>路家豪</username>
          <password>123</password>
        </user>
     */
}

设置属性的值:
public void demo03(){
    User user = new User("u007", "路家豪", "123");
    //1 核心类
    XStream xStream = new XStream();
    // * 设置元素别名
    xStream.alias("user", User.class);
    // * 设置属性别名 , 三个参数分别标售:指定javabean/property属性名/xml中显示的属性名
    xStream.aliasAttribute(User.class, "id", "id");
    //2 转换成xml
    String xmlData = xStream.toXML(user);
    System.out.println(xmlData);

    /*
     * <user id="u007">
          <username>路家豪</username>
          <password>123</password>
        </user>
     */
}

忽略字段:
public void demo06(){
    User user = new User("u007", "路家豪", "123");
    //1 核心类
    XStream xStream = new XStream();
    // * 设置元素别名
    xStream.alias("user", User.class);
    // * 设置属性别名 , 指定javabean property 使用其他别名
    xStream.aliasAttribute(User.class, "id", "id");
    // * 忽略指定的字段
    xStream.omitField(User.class, "username");

    //2 转换成xml
    String xmlData = xStream.toXML(user);
    System.out.println(xmlData);
    /*
     * <user id="u007">
          <password>123</password>
        </user>
     */
}

xml转Java对象:

public void demo04(){
    String str = "<user id='u007'><username>路家豪</username><password>123</password></user>";
    //1 核心类
    XStream xStream = new XStream();
    // * 设置元素别名
    xStream.alias("user", User.class);
    // * 设置属性别名 , 指定javabean property 使用其他别名
    xStream.aliasAttribute(User.class, "id", "id");

    //2 xml转换成 javabean
    User user = (User)xStream.fromXML(str);
    System.out.println(user);
}

案例一:判断用户名是否存在

html代码:

<input id="name" type="text" name="name" onblur="sendData(this)" onfocus="document.getElementById('spanId').innerHTML=''"/>
<span id="spanId"></span>

js代码:

<script type="text/javascript">
    function sendData(obj) {
        // 0.获得输入的数据
        var inputValue = obj.value;
        /* 1 创建核心类
             XMLHttpRequest ajax引擎,不同的浏览器对此对象的创建方式不同。存在浏览器兼容问题
             * 在所有现代浏览器中(包括 IE 7):
                xmlhttp=new XMLHttpRequest()
             * 在 Internet Explorer 5 和 6 中:
                xmlhttp=new ActiveXObject("Microsoft.XMLHTTP")
         */
        var xmlhttp = null;
        if (window.XMLHttpRequest) {// code for all new browsers
            xmlhttp = new XMLHttpRequest();
        } else if (window.ActiveXObject) {// code for IE5 and IE6
            xmlhttp = new ActiveXObject("Microsoft.XMLHTTP");
        }
        /* 2 设置回调函数,通过属性设置 onreadystatechange
             * 目的:发送请求之后可以获得服务器响应内容。
             * 一般情况使用匿名函数实现
             * ajax引擎在不同的阶段都会调用回调函数。
             * 属性:readyState 和 stauts
             2.1 readyState 返回整形数据,表示当前执行某一个阶段。
                 * 0  初始化状态。核心对象创建时默认值
                 * 1 open() 方法已调用,连接创建完成之后,ajax引擎将此状态修改1
                 * 2 send() 方法已调用,发送请求
                 * 3  接受中, 所有响应头部都已经接收到。响应体开始接收但未完成。
                 * 4  响应已经完全接收。【掌握】--服务器发送的所有数据,已经到ajax引擎内部。
             2.2 status 表示  响应的http状态码
                 * 200 正常【掌握】
                 * 302 重定向
                 * 304 缓存
                 * 404 不存在
                 * 500 服务器异常
         */
        xmlhttp.onreadystatechange = function () {
            if (xmlhttp.readyState == 4 && xmlhttp.status == 200){
                // 3.1 接收服务器响应的数据,获得json数据,注意json也是文本
                var data = xmlhttp.responseText;
                // 3.2 将字符串转化成json对象  使用格式:eval("('abc')");
                var jsonData = eval("("+data+")");

                // 3.3 判断  控制按钮是否可用
                var buttonObj = document.getElementById("buttonId");
                var spanObj = document.getElementById("spanId");
                if (jsonData.flag) {// 可用
                    buttonObj.removeAttribute("disabled");
                    spanObj.style.color = "#3D882D";
                } else {// 不可用
                    buttonObj.setAttribute("disabled","disabled");
                    spanObj.style.color = "#CC0000";
                }
                // 3.4 将信息显示到指定位置
                spanObj.innerHTML = jsonData.msg;
            }
        };
        //3 建立连接
        xmlhttp.open("POST","${pageContext.request.contextPath}/CustomerServlet?method=checkName");

        // 4 POST请求需要设置编码
        xmlhttp.setRequestHeader("content-type", "application/x-www-form-urlencoded");

        //5 发送请求  格式: name=lujiahao&password=1234
        xmlhttp.send("name=" + inputValue);
    }
</script>

Servlet端代码:

/**
 * Ajax查询用户名是否存在
 */
private void checkName(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException{
    // 0响应乱码
    response.setContentType("application/json;charset=UTF-8");
    try {
        // 1.获得数据并封装
        String name = request.getParameter("name");
        // 2.通知service查询所有
        //CustomerServeice customerServeice = new CustomerServiceImpl();
        // customerServeice.checkName(name);

        PrintWriter out = response.getWriter();
        if (name == null || "".equals(name)) {
            out.print("{'flag':false,'msg':'用户名不能为空'}");
            return;
        } else if ("lujiahao".equals(name)) {
            out.print("{'flag':false,'msg':'用户名已经存在'}");
        } else {
            out.print("{'flag':true,'msg':'用户名可用'}");
        }
    } catch (Exception e){
        // 1.打印日志
        e.printStackTrace();
    }
}

案例二:省市县三级联动

Listener-监听器

TODO:这一部分不太明白,需要后面在好好看看吧

分页

实现思路

QQ截图20160725151218.png

分页语法

select ..... limit 参数1,参数2;
参数1 ,表示开始索引,从0开始。startIndex  ,算法 (pageNum - 1) * pageSize
参数2 ,表示每页显示个数。pageSize

Servlet的写法

/**
 * 查询所有--分页查询
 */
private void findAllWithPage(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException{
    try {
        // 1.获得数据并封装
        // 获得当前页码
        String pageNumStr = request.getParameter("pageNum");
        int pageNum = 1;
        try {
            // 这种处理方式可以防止在地址栏里面输入非数字导致转化异常
            pageNum = Integer.parseInt(pageNumStr);
        } catch (Exception e){
        }

        // * 每页显示个数(固定值)
        int pageSize = 5;
        // 2.通知service查询所有
        CustomerServeice customerServeice = new CustomerServiceImpl();
        PageBean<Customer> pageBean = customerServeice.findAllCustomerWithPage(pageNum, pageSize);

        // 3.显示
        // 存放到request作用域中--每一次查询都是新的数据
        request.setAttribute("pageBean",pageBean);
        // servlet到jsp中显示,一次请求需要使用请求转发
        request.getRequestDispatcher("/pages/show_all_page.jsp").forward(request,response);
    } catch (Exception e){
        // 1.打印日志
        e.printStackTrace();
        // 2.请求转发到消息界面
        request.setAttribute("msg","查询失败,请稍后重试.");
        request.getRequestDispatcher("/pages/message.jsp").forward(request,response);
    }
}

Service写法

@Override
public PageBean<Customer> findAllCustomerWithPage(int pageNum, int pageSize) {
    // 一般分页的逻辑是要在service里面来写
    // 创建PageBean封装分页所有的数据,并将封装后的数据返回

    // 1.查询总记录数
    int totalRecord = customerDao.getTotalRecord();
    // 2.创建PageBean
    PageBean<Customer> pageBean = new PageBean<>(pageNum,pageSize,totalRecord);
    // 3.查询结果
    List<Customer> data = customerDao.findAll(pageBean.getStartIndex(), pageBean.getPageSize());
    pageBean.setData(data);
    return pageBean;
}

Dao写法

@Override
public List<Customer> findAll(int startIndex, int pageSize) {
    try {
        String sql = "select * from t_customer limit ?,?";
        Object[] params = {startIndex,pageSize};
        return runner.query(sql,new BeanListHandler<Customer>(Customer.class),params);
    } catch (Exception e){
        throw new DaoException(e);
    }
}

JSP界面

当前${pageBean.pageNum}页,共${pageBean.totalPage}页,总条数${pageBean.totalRecord}条<br/>

<c:url value="${pageContext.request.contextPath}/CustomerServlet" var="baseUrl">
    <c:param name="method" value="findAllWithPage"></c:param>
    <c:param name="pageNum" value=""></c:param>
</c:url>
<c:if test="${pageBean.pageNum > 1}">
    <a href="${baseUrl}1">首页</a>
    <a href="${baseUrl}${pageBean.pageNum-1}">上一页</a>
</c:if>
<c:forEach begin="${pageBean.start}" end="${pageBean.end}" var="num">
    <a href="${baseUrl}${num}">${num}</a>
</c:forEach>
<c:if test="${pageBean.pageNum < pageBean.totalPage}">
    <a href="${baseUrl}${pageBean.pageNum+1}">下一页</a>
    <a href="${baseUrl}${pageBean.totalPage}">尾页</a>
</c:if>

PageBean的封装

/**
 * 分页的封装类
 *      封装的两种思路:
 *          -- 集合Map(Android中常见的请求参数的封装)
 *          -- 自定义JavaBean对象
 * Created by lujiahao on 2016/7/25.
 */
public class PageBean<T> {

    // 分页必备选项,为下面计算提供数据
    private int pageNum;            // 当前页(第几页)
    private int pageSize;           // 每页显示个数
    private int totalRecord;        // 总记录数(总条数)---这个数据需要通过查询来获得

    // 通过计算获得的数据
    private int startIndex;         // 分页开始的索引
    private int totalPage;          // 总分页数

    // 分页查询的结果
    private List<T> data;           // 查询分页的数据--使用泛型的目的是为了方便复用

    // 导航条动态显示  首页 上一页 1 2 3 4 下一页 尾页
    private int start;              // 循环开始
    private int end;                // 循环结束

    public PageBean(int pageNum, int pageSize, int totalRecord) {
        // 构造方法中初始化三个必备选项
        this.pageNum = pageNum;
        this.pageSize = pageSize;
        this.totalRecord = totalRecord;

        // 处理地址栏输入负页数
        if (this.pageNum < 1) {
            this.pageNum = 1;
        }

        // 计算分页开始的索引:(当前页 - 1) * 每页显示个数
        this.startIndex = (this.pageNum - 1) * this.pageSize;

        // 计算总分页数
        if (this.totalRecord % this.pageSize == 0){
            // 能整除,总分页数 = 总记录数 / 每页显示个数
            this.totalPage = this.totalRecord / this.pageSize;
        } else {
            // 不能整除,需要加一页用来存不够一页的数据
            this.totalPage = this.totalRecord / this.pageSize + 1;
        }
        // 上面的快捷算法---暂时理解不了啊
        //this.totalPage = (this.totalRecord + (this.pageSize - 1)) / this.pageSize;

        // 导航条动态显示  默认显示10页
        this.start = 1;
        this.end = 10;
        // 总页数不够10页
        if (this.totalPage <= 10){
            this.end = this.totalPage;
        } else {
            // 总页数大于10页
            // 页数要求 前五后四
            this.start = this.pageNum -5;
            this.end = this.pageNum +4;

            // 当pageNum=1时,其实页数至少是1
            if (this.start < 1){
                this.start = 1;
                this.end = 10;
            }

            // 当pageNum到最后一页事
            if (this.end > this.totalPage){
                this.end = this.totalPage;
                this.start = this.totalPage -9; // 9 = 5 + 4
            }
        }
    }
    ...get/set
}

注意点

容错的处理非常巧妙:
这是在Servlet中的处理方式

String pageNumStr = request.getParameter("pageNum");
int pageNum = 1;
try {
    // 这种处理方式可以防止在地址栏里面输入非数字导致转化异常
    pageNum = Integer.parseInt(pageNumStr);
} catch (Exception e){
}

这是在PageBean中的处理方式

// 处理地址栏输入负页数
if (this.pageNum < 1) {
    this.pageNum = 1;
}

MVC介绍&用户管理系统

MVC

MVC基本定义

一种软件设计模式,B/S架构都支持。例如:java、.net、php等
思想:业务逻辑处理与数据显示相分离。

Model:模型,用于封装数据
View:视图,用于显示数据
Controller:控制器,用于控制正常执行。

mvc.png

Web项目分包结构

QQ截图20160624175200.png

面向接口编程

QQ截图20160624175227.png

用户管理系统

功能分析

  • 注册
  • 登录
  • 查询所有用户
  • 查询详情
  • 修改用户
  • 删除用户

技术分析

  • MVC三层架构
  • xml/dom4j
  • servlet/jsp/javabean

导入需要的jar包

解析xml需要

QQ截图20160624173426.png

包结构

com.lujiahao.web.servlet    web层
com.lujiahao.service        service层
com.lujiahao.dao            dao层
com.lujiahao.domain            javabean
com.lujiahao.utils            工具包

数据库

暂定使用xml

xml内容:

<?xml version="1.0" encoding="UTF-8"?>
<users>
    <user id="u001">
        <username>jack</username>
        <password>1234</password>
        <gender>男</gender>
        <age>18</age>
    </user>
</users>

JavaBean

数据库定义完成之后就开始编写JavaBean

dao层实现

随便写点数据的增删改查功能

数据校验

表单校验Bean - UserFormBean

一般都是写在和servlet同级的包里面
这种类型的bean里面所有的字段都是字符串
用于获得浏览器发送的数据,并对数据的有效性进行校验

  1. 提供校验validate()
  2. 记录每一项的校验结果

具体代码实现:

public class UserFormBean {
    private String id;
    private String username;
    private String password;
    private String repassword;
    private String gender;
    private String age;// 因为服务器传过来的数据都是string类型的

    public UserFormBean() {}

    public UserFormBean(String id, String username, String password, String repassword, String gender, String age) {
        this.id = id;
        this.username = username;
        this.password = password;
        this.repassword = repassword;
        this.gender = gender;
        this.age = age;
    }

    public String getId() {
        return id;
    }
    public void setId(String id) {
        this.id = id;
    }
    public String getUsername() {
        return username;
    }
    public void setUsername(String username) {
        this.username = username;
    }
    public String getPassword() {
        return password;
    }
    public void setPassword(String password) {
        this.password = password;
    }
    public String getRepassword() {
        return repassword;
    }
    public void setRepassword(String repassword) {
        this.repassword = repassword;
    }
    public String getGender() {
        return gender;
    }
    public void setGender(String gender) {
        this.gender = gender;
    }
    public String getAge() {
        return age;
    }
    public void setAge(String age) {
        this.age = age;
    }

    // 记录错误信息 key:对应字段  value:提示信息
    private Map<String,String> errorMsg = new HashMap<>();
    /**
     * 校验方法
     */
    public boolean validate() {
        boolean temp = true;
        // 用户名不能为空
        if (username == null || "".equals(username)) {
            errorMsg.put("usernameMsg","用户名不能为空");
            temp = false;
        }
        if (password == null || "".equals(password)) {
            errorMsg.put("passwordMsg","密码不能为空");
            temp = false;
        } else if (! password.equals(repassword)){
            errorMsg.put("repasswordMsg","确认密码和密码不一致");
            temp = false;
        }
        return temp;
    }
    public Map<String, String> getErrorMsg() {
        return errorMsg;
    }
}

使用接口

三层结构每一层都应该是有接口和具体的实现类

QQ截图20160629180224.png

使用Intellj重构代码

QQ截图20160629180029.png

BeanUtils

通过封装的类来简化参数的自动封装
使用了反射和内省

初始代码:

// 1.获取请求参数
String id = request.getParameter("id");
String username = request.getParameter("username");
String password = request.getParameter("password");
String repassword = request.getParameter("repassword");
String gender = request.getParameter("gender");
String age = request.getParameter("age");

/**
 * 数据校验
 */
UserFormBean userFormBean = new UserFormBean(id,username,password,repassword,gender,age);

封装后的代码:

UserFormBean userFormBean = MyBeanUtils.populate(UserFormBean.class,request.getParameterMap());

BeanUtils详细代码:

public class MyBeanUtils {

    /**
     * 创建JavaBean实例,并自动将对应的参数进行封装
     * @param beanClass
     * @param parameterMap
     * @param <T>
     * @return
     */
    public static <T> T populate(Class<T> beanClass, Map<String,String[]> parameterMap){
        try {
            // 1.使用反射创建javabean实例
            T bean = beanClass.newInstance();
            // 2.获得javabean属性(property username-->setUsername()-->执行set方法,数据来自map
            // 2.1获得所有属性--使用内省(java.beans.Introspector):jdk提供工具类,用于操作javabean
            // BeanInfo jdk提供用于对javabean进行描述(封装)对象
            BeanInfo beanInfo = Introspector.getBeanInfo(beanClass, Object.class);
            // 2.2 获得所有的属性描述对象
            PropertyDescriptor[] allPd = beanInfo.getPropertyDescriptors();
            for (PropertyDescriptor pd : allPd) {
                // 2.3 获得属性名称
                String propName = pd.getName();
                // 2.4 获得表单中对应的数据
                String[] allValue = parameterMap.get(propName);
                if (allValue == null) {
                    continue;// 当没有值的时候就跳过这个字段
                }
                String propValue = allValue[0];
                // 2.5 如果有值,将执行set方法
                if (propValue != null && !"".equals(propValue)) {
                    Method writeMethod = pd.getWriteMethod();// 相当于set方法    getReadMethod--相当于get方法
                    if (writeMethod != null) {
                        writeMethod.invoke(bean,propValue);
                    }
                }
            }
            return bean;
        } catch (Exception e){
            throw new RuntimeException(e);
        }
    }
}