ssh

## crm练习 ##
    
----------
    
**课程内容**
    
----------
    
**课程目标**
    
    1. 通过SSH框架完成基本功能的开发
    
----------
    
### 用户模块 ###
    
----------
    
**功能一:用户注册功能**
    
    1. 可以先判断登录名是否已经存在
    2. 要给密码使用MD5进行加密操作
    
----------
    
**功能二:用户登录功能**
    
    1. 登录功能要注意需要先给密码加密后,再进行查询
        * 密码加密后再查询
        * 用户的状态必须是1,字符串类型的
    
----------
    
**功能三:用户退出功能**
    
    1. 把用户信息从HttpSession中清除
    
----------
    
### 客户模块 ###
    
----------
    
**功能一:查询所有客户功能**
    
    1. 数据字典表的引入
        * 数据字典表的作用:规范开发中数据的写法
        * 字段表与客户表是一对多的关系
        * 修改客户表,添加外键(使用SQLyog进行修改)
    
![](./图片/01-字典表.bmp)
    
    2. 创建字典表的实体和映射的配置文件
        * 编写字典表的JavaBean和映射的配置文件
        * 修改Customer的JavaBean,因为是多方,需要把外键字段换成字典对象
        * 修改Customer.hbm.xml的配置文件,配置多对一
    
    3. 分页查询所有的客户功能实现
    
----------
    
**功能二:按条件查询所有的客户**
    
    1. 使用异步的方式加载客户级别和客户的来源
        * 前端使用JQuery的ajax技术
        * 后端使用fastjson的jar包
            * 导入fastjson的开发jar包fastjson-1.2.8.jar
            * String s = JSON.toJSONString(集合)
            * String s = JSON.toJSONString(对象)
        
        * 如果List集合中存入相同引用的对象
            * fastjson默认的情况下是进行循环检测的,去除掉死循环调用的方式
            * 可以使用JSON.toJSONString(p,SerializerFeature.DisableCircularReferenceDetect) 去除循环检测,但是就会出现死循环的效果
            * 最后可以使用注解:@JSONField(serialize=false)对指定的属性不转换成json
    
    2. 异步获取客户级别
        * ajax的代码
            var url = "${pageContext.request.contextPath }/dict_findByCode.action";
            var param = {"dict_type_code":"006"};
            $.post(url,param,function(data){
                $(data).each(function(){
                    var id = "${model.level.dict_id}";
                    if(id == this.dict_id){
                        $("#levelId").append("<option value='"+this.dict_id+"' selected>"+this.dict_item_name+"</option>");
                    }else{
                        $("#levelId").append("<option value='"+this.dict_id+"'>"+this.dict_item_name+"</option>");
                    }
                });
            },"json");
        
        * Action的代码
            public String findByCode(){
                List<Dict> list = dictService.findByCode(dict.getDict_type_code());
                String jsonString = FastJsonUtil.toJSONString(list);
                HttpServletResponse response = ServletActionContext.getResponse();
                FastJsonUtil.write_json(response, jsonString);
                return NONE;
            }
        
        * CustomerAction的分页查询的代码
            public String findByPage(){
                // 调用service业务层
                DetachedCriteria criteria = DetachedCriteria.forClass(Customer.class);
                // 拼接查询的条件
                String name = customer.getCust_name();
                if(name != null && !name.trim().isEmpty()){
                    criteria.add(Restrictions.like("cust_name", "%"+name+"%"));
                }
                
                // System.out.println(customer.getLevel().getDict_type_code());
                Dict level = customer.getLevel();
                if(level != null && !level.getDict_id().trim().isEmpty()){
                    criteria.add(Restrictions.eq("level.dict_id", level.getDict_id()));
                }
                
                Dict source = customer.getSource();
                if(source != null && !source.getDict_id().trim().isEmpty()){
                    criteria.add(Restrictions.eq("source.dict_id", source.getDict_id()));
                }
                
                // 查询
                PageBean<Customer> page = customerService.findByPage(pageCode,pageSize,criteria);
                // 压栈
                ValueStack vs = ActionContext.getContext().getValueStack();
                // 栈顶是map<"page",page对象>
                vs.set("page", page);
                vs.set("cust_name", name);
                return "page";
            }
    
----------
    
**功能三:添加客户功能(含有文件上传功能)**
    
    1. 跳转到客户的添加页面,需要通过ajax来显示客户的级别,客户的来源和客户的行业。
    2. 添加文件上传的选择项
    3. 客户端三个注意事项
        * method="post"
        * enctype="multipart/form-data"
        * <input type="file" name="myfile">
    
    4. Struts2框架的使用拦截器完成了文件上传,并且底层使用也是FileUpload开源的组件。
        * 提供 FileUpload 拦截器,用于解析 multipart/form-data 编码格式请求,解析上传文件的内容 
        * fileUpload拦截器 默认在 defaultStack 栈中, 默认会执行的 
        
        * 在Action中编写文件上传,需要定义三个属性
            > 文件类型File ,属性名与表单中file的name属性名一致.
            > 字符串类型String , 属性名:前段是name属性名一致 + ContentType;
            > 字符串类型String , 属性名:前段是name属性名一致+FileName;
            
            > 最后需要为上述的三个属性提供set方法。
            > 可以通过FileUtils提供 copyFile 进行文件复制,将上传文件 保存到服务器端 
    
    4. 文件上传中存在的问题
        * 先配置input逻辑视图
        * 在页面中显示错误信息
        
        * 文件上传的总大小默认值是2M,如果超过了2M,程序会报出异常。可以使用<s:actionError>来查看具体信息!
            > 解决总大小的设置,找到常量:
                * struts.multipart.parser=jakarta        -- 默认文件上传解析器,就是FileUpload组件
                * struts.multipart.saveDir=                -- 文件上传的临时文件存储目录
                * struts.multipart.maxSize=2097152        -- 文件上传的最大值(总大小),默认是2M
            
            > 可以在struts.xml中设置常量,修改文件上传的默认总大小!!!
                * <constant name="struts.multipart.maxSize" value="5000000"></constant>
    
    5. 还可以通过配置拦截器来设置文件上传的一些属性
        * 先在<action>标签中引入文件上传的拦截器
            <interceptor-ref name="defaultStack">
                <!-- 设置单个上传文件的大小 -->
                <param name="fileUpload.maximumSize">2097152</param>
                <!-- 设置扩展名 -->
                <param name="fileUpload.allowedExtensions">.txt</param>
            </interceptor-ref>
    
----------
    
**功能三:修改客户的功能**
    
    1. 先通过客户的主键查询出客户的详细信息,显示到修改的页面上
        * 要把客户的主键和上传文件的路径使用隐藏域保存起来
    
    2. 修改客户的信息
        * 如果用户新上传了文件,删除旧的文件,上传新的文件。
        * 如果用户没有上传新文件,正常更新。
    
----------
    
**功能四:删除客户的功能**

    1. 删除上传的文件后,再删除客户信息。

 

## crm练习 ##
    
----------
    
**课程内容**
    
----------
    
**课程目标**
    
    1. 通过SSH框架完成基本功能的开发
    
----------
    
### 用户模块 ###
    
----------
    
**功能一:用户注册功能**
    
    1. 可以先判断登录名是否已经存在
    2. 要给密码使用MD5进行加密操作
    
----------
    
**功能二:用户登录功能**
    
    1. 登录功能要注意需要先给密码加密后,再进行查询
        * 密码加密后再查询
        * 用户的状态必须是1,字符串类型的
    
----------
    
**功能三:用户退出功能**
    
    1. 把用户信息从HttpSession中清除
    
----------
    
### 客户模块 ###
    
----------
    
**功能一:查询所有客户功能**
    
    1. 数据字典表的引入
        * 数据字典表的作用:规范开发中数据的写法
        * 字段表与客户表是一对多的关系
        * 修改客户表,添加外键(使用SQLyog进行修改)
    
![](./图片/01-字典表.bmp)
    
    2. 创建字典表的实体和映射的配置文件
        * 编写字典表的JavaBean和映射的配置文件
        * 修改Customer的JavaBean,因为是多方,需要把外键字段换成字典对象
        * 修改Customer.hbm.xml的配置文件,配置多对一
    
    3. 分页查询所有的客户功能实现
    
----------
    
**功能二:按条件查询所有的客户**
    
    1. 使用异步的方式加载客户级别和客户的来源
        * 前端使用JQuery的ajax技术
        * 后端使用fastjson的jar包
            * 导入fastjson的开发jar包fastjson-1.2.8.jar
            * String s = JSON.toJSONString(集合)
            * String s = JSON.toJSONString(对象)
        
        * 如果List集合中存入相同引用的对象
            * fastjson默认的情况下是进行循环检测的,去除掉死循环调用的方式
            * 可以使用JSON.toJSONString(p,SerializerFeature.DisableCircularReferenceDetect) 去除循环检测,但是就会出现死循环的效果
            * 最后可以使用注解:@JSONField(serialize=false)对指定的属性不转换成json

 

## crm练习 ##
    
----------
    
**课程内容**
    
----------
    
**课程目标**
    
    1. 通过SSH框架完成基本功能的开发
    
----------
    
### 用户模块 ###
    
----------
    
**功能一:用户注册功能**
    
    1. 可以先判断登录名是否已经存在
    2. 要给密码使用MD5进行加密操作
    
----------
    
**功能二:用户登录功能**
    
    1. 登录功能要注意需要先给密码加密后,再进行查询
        * 密码加密后再查询
        * 用户的状态必须是1,字符串类型的
    
----------
    
**功能三:用户退出功能**
    
    1. 把用户信息从HttpSession中清除
    
----------
    
### 客户模块 ###
    
----------
    
**功能一:查询所有客户功能**
    
    1. 数据字典表的引入
        * 数据字典表的作用:规范开发中数据的写法
        * 字段表与客户表是一对多的关系
        * 修改客户表,添加外键(使用SQLyog进行修改)
    
![](./图片/01-字典表.bmp)
    
    2. 创建字典表的实体和映射的配置文件
        * 编写字典表的JavaBean和映射的配置文件
        * 修改Customer的JavaBean,因为是多方,需要把外键字段换成字典对象
        * 修改Customer.hbm.xml的配置文件,配置多对一
    
    3. 分页查询所有的客户功能实现
    
----------
    
**功能二:按条件查询所有的客户**
    
    1. 使用异步的方式加载客户级别和客户的来源
        * 前端使用JQuery的ajax技术
        * 后端使用fastjson的jar包
            * 导入fastjson的开发jar包fastjson-1.2.8.jar
            * String s = JSON.toJSONString(集合)
            * String s = JSON.toJSONString(对象)
        
        * 如果List集合中存入相同引用的对象
            * fastjson默认的情况下是进行循环检测的,去除掉死循环调用的方式
            * 可以使用JSON.toJSONString(p,SerializerFeature.DisableCircularReferenceDetect) 去除循环检测,但是就会出现死循环的效果
            * 最后可以使用注解:@JSONField(serialize=false)对指定的属性不转换成json
    
    2. 异步获取客户级别
        * ajax的代码
            var url = "${pageContext.request.contextPath }/dict_findByCode.action";
            var param = {"dict_type_code":"006"};
            $.post(url,param,function(data){
                $(data).each(function(){
                    var id = "${model.level.dict_id}";
                    if(id == this.dict_id){
                        $("#levelId").append("<option value='"+this.dict_id+"' selected>"+this.dict_item_name+"</option>");
                    }else{
                        $("#levelId").append("<option value='"+this.dict_id+"'>"+this.dict_item_name+"</option>");
                    }
                });
            },"json");
        
        * Action的代码
            public String findByCode(){
                List<Dict> list = dictService.findByCode(dict.getDict_type_code());
                String jsonString = FastJsonUtil.toJSONString(list);
                HttpServletResponse response = ServletActionContext.getResponse();
                FastJsonUtil.write_json(response, jsonString);
                return NONE;
            }
        
        * CustomerAction的分页查询的代码
            public String findByPage(){
                // 调用service业务层
                DetachedCriteria criteria = DetachedCriteria.forClass(Customer.class);
                // 拼接查询的条件
                String name = customer.getCust_name();
                if(name != null && !name.trim().isEmpty()){
                    criteria.add(Restrictions.like("cust_name", "%"+name+"%"));
                }
                
                // System.out.println(customer.getLevel().getDict_type_code());
                Dict level = customer.getLevel();
                if(level != null && !level.getDict_id().trim().isEmpty()){
                    criteria.add(Restrictions.eq("level.dict_id", level.getDict_id()));
                }
                
                Dict source = customer.getSource();
                if(source != null && !source.getDict_id().trim().isEmpty()){
                    criteria.add(Restrictions.eq("source.dict_id", source.getDict_id()));
                }
                
                // 查询
                PageBean<Customer> page = customerService.findByPage(pageCode,pageSize,criteria);
                // 压栈
                ValueStack vs = ActionContext.getContext().getValueStack();
                // 栈顶是map<"page",page对象>
                vs.set("page", page);
                vs.set("cust_name", name);
                return "page";
            }
    
----------
    
**功能三:添加客户功能(含有文件上传功能)**
    
    1. 跳转到客户的添加页面,需要通过ajax来显示客户的级别,客户的来源和客户的行业。
    2. 添加文件上传的选择项
    3. 客户端三个注意事项
        * method="post"
        * enctype="multipart/form-data"
        * <input type="file" name="myfile">
    
    4. Struts2框架的使用拦截器完成了文件上传,并且底层使用也是FileUpload开源的组件。
        * 提供 FileUpload 拦截器,用于解析 multipart/form-data 编码格式请求,解析上传文件的内容 
        * fileUpload拦截器 默认在 defaultStack 栈中, 默认会执行的 
        
        * 在Action中编写文件上传,需要定义三个属性
            > 文件类型File ,属性名与表单中file的name属性名一致.
            > 字符串类型String , 属性名:前段是name属性名一致 + ContentType;
            > 字符串类型String , 属性名:前段是name属性名一致+FileName;
            
            > 最后需要为上述的三个属性提供set方法。
            > 可以通过FileUtils提供 copyFile 进行文件复制,将上传文件 保存到服务器端 
    
    4. 文件上传中存在的问题
        * 先配置input逻辑视图
        * 在页面中显示错误信息
        
        * 文件上传的总大小默认值是2M,如果超过了2M,程序会报出异常。可以使用<s:actionError>来查看具体信息!
            > 解决总大小的设置,找到常量:
                * struts.multipart.parser=jakarta        -- 默认文件上传解析器,就是FileUpload组件
                * struts.multipart.saveDir=                -- 文件上传的临时文件存储目录
                * struts.multipart.maxSize=2097152        -- 文件上传的最大值(总大小),默认是2M
            
            > 可以在struts.xml中设置常量,修改文件上传的默认总大小!!!
                * <constant name="struts.multipart.maxSize" value="5000000"></constant>
    
    5. 还可以通过配置拦截器来设置文件上传的一些属性
        * 先在<action>标签中引入文件上传的拦截器
            <interceptor-ref name="defaultStack">
                <!-- 设置单个上传文件的大小 -->
                <param name="fileUpload.maximumSize">2097152</param>
                <!-- 设置扩展名 -->
                <param name="fileUpload.allowedExtensions">.txt</param>
            </interceptor-ref>
    
----------
    
**功能三:修改客户的功能**
    
    1. 先通过客户的主键查询出客户的详细信息,显示到修改的页面上
        * 要把客户的主键和上传文件的路径使用隐藏域保存起来
    
    2. 修改客户的信息
        * 如果用户新上传了文件,删除旧的文件,上传新的文件。
        * 如果用户没有上传新文件,正常更新。
    
----------
    
**功能四:删除客户的功能**

    1. 删除上传的文件后,再删除客户信息。

## crm练习 ##
    
----------
    
**课程内容**
    
----------
    
**课程目标**
    
    1. 通过SSH框架完成基本功能的开发
    
----------
    
### 用户模块 ###
    
----------
    
**功能一:用户注册功能**
    
    1. 可以先判断登录名是否已经存在
    2. 要给密码使用MD5进行加密操作
    
----------
    
**功能二:用户登录功能**
    
    1. 登录功能要注意需要先给密码加密后,再进行查询
        * 密码加密后再查询
        * 用户的状态必须是1,字符串类型的
    
----------
    
**功能三:用户退出功能**
    
    1. 把用户信息从HttpSession中清除
    
----------
    
### 客户模块 ###
    
----------
    
**功能一:查询所有客户功能**
    
    1. 数据字典表的引入
        * 数据字典表的作用:规范开发中数据的写法
        * 字段表与客户表是一对多的关系
        * 修改客户表,添加外键(使用SQLyog进行修改)
    
![](./图片/01-字典表.bmp)
    
    2. 创建字典表的实体和映射的配置文件
        * 编写字典表的JavaBean和映射的配置文件
        * 修改Customer的JavaBean,因为是多方,需要把外键字段换成字典对象
        * 修改Customer.hbm.xml的配置文件,配置多对一
    
    3. 分页查询所有的客户功能实现
    
----------
    
**功能二:按条件查询所有的客户**
    
    1. 使用异步的方式加载客户级别和客户的来源
        * 前端使用JQuery的ajax技术
        * 后端使用fastjson的jar包
            * 导入fastjson的开发jar包fastjson-1.2.8.jar
            * String s = JSON.toJSONString(集合)
            * String s = JSON.toJSONString(对象)
        
        * 如果List集合中存入相同引用的对象
            * fastjson默认的情况下是进行循环检测的,去除掉死循环调用的方式
            * 可以使用JSON.toJSONString(p,SerializerFeature.DisableCircularReferenceDetect) 去除循环检测,但是就会出现死循环的效果
            * 最后可以使用注解:@JSONField(serialize=false)对指定的属性不转换成json
    
    2. 异步获取客户级别
        * ajax的代码
            var url = "${pageContext.request.contextPath }/dict_findByCode.action";
            var param = {"dict_type_code":"006"};
            $.post(url,param,function(data){
                $(data).each(function(){
                    var id = "${model.level.dict_id}";
                    if(id == this.dict_id){
                        $("#levelId").append("<option value='"+this.dict_id+"' selected>"+this.dict_item_name+"</option>");
                    }else{
                        $("#levelId").append("<option value='"+this.dict_id+"'>"+this.dict_item_name+"</option>");
                    }
                });
            },"json");
        
        * Action的代码
            public String findByCode(){
                List<Dict> list = dictService.findByCode(dict.getDict_type_code());
                String jsonString = FastJsonUtil.toJSONString(list);
                HttpServletResponse response = ServletActionContext.getResponse();
                FastJsonUtil.write_json(response, jsonString);
                return NONE;
            }
        
        * CustomerAction的分页查询的代码
            public String findByPage(){
                // 调用service业务层
                DetachedCriteria criteria = DetachedCriteria.forClass(Customer.class);
                // 拼接查询的条件
                String name = customer.getCust_name();
                if(name != null && !name.trim().isEmpty()){
                    criteria.add(Restrictions.like("cust_name", "%"+name+"%"));
                }
                
                // System.out.println(customer.getLevel().getDict_type_code());
                Dict level = customer.getLevel();
                if(level != null && !level.getDict_id().trim().isEmpty()){
                    criteria.add(Restrictions.eq("level.dict_id", level.getDict_id()));
                }
                
                Dict source = customer.getSource();
                if(source != null && !source.getDict_id().trim().isEmpty()){
                    criteria.add(Restrictions.eq("source.dict_id", source.getDict_id()));
                }
                
                // 查询
                PageBean<Customer> page = customerService.findByPage(pageCode,pageSize,criteria);
                // 压栈
                ValueStack vs = ActionContext.getContext().getValueStack();
                // 栈顶是map<"page",page对象>
                vs.set("page", page);
                vs.set("cust_name", name);
                return "page";
            }
    
----------
    
**功能三:添加客户功能(含有文件上传功能)**
    
    1. 跳转到客户的添加页面,需要通过ajax来显示客户的级别,客户的来源和客户的行业。
    2. 添加文件上传的选择项
    3. 客户端三个注意事项
        * method="post"
        * enctype="multipart/form-data"
        * <input type="file" name="myfile">
    
    4. Struts2框架的使用拦截器完成了文件上传,并且底层使用也是FileUpload开源的组件。
        * 提供 FileUpload 拦截器,用于解析 multipart/form-data 编码格式请求,解析上传文件的内容 
        * fileUpload拦截器 默认在 defaultStack 栈中, 默认会执行的 
        
        * 在Action中编写文件上传,需要定义三个属性
            > 文件类型File ,属性名与表单中file的name属性名一致.
            > 字符串类型String , 属性名:前段是name属性名一致 + ContentType;
            > 字符串类型String , 属性名:前段是name属性名一致+FileName;
            
            > 最后需要为上述的三个属性提供set方法。
            > 可以通过FileUtils提供 copyFile 进行文件复制,将上传文件 保存到服务器端 
    
    4. 文件上传中存在的问题
        * 先配置input逻辑视图
        * 在页面中显示错误信息
        
        * 文件上传的总大小默认值是2M,如果超过了2M,程序会报出异常。可以使用<s:actionError>来查看具体信息!
            > 解决总大小的设置,找到常量:
                * struts.multipart.parser=jakarta        -- 默认文件上传解析器,就是FileUpload组件
                * struts.multipart.saveDir=                -- 文件上传的临时文件存储目录
                * struts.multipart.maxSize=2097152        -- 文件上传的最大值(总大小),默认是2M
            
            > 可以在struts.xml中设置常量,修改文件上传的默认总大小!!!
                * <constant name="struts.multipart.maxSize" value="5000000"></constant>
    
    5. 还可以通过配置拦截器来设置文件上传的一些属性
        * 先在<action>标签中引入文件上传的拦截器
            <interceptor-ref name="defaultStack">
                <!-- 设置单个上传文件的大小 -->
                <param name="fileUpload.maximumSize">2097152</param>
                <!-- 设置扩展名 -->
                <param name="fileUpload.allowedExtensions">.txt</param>
            </interceptor-ref>
    
----------
    
**功能三:修改客户的功能**
    
    1. 先通过客户的主键查询出客户的详细信息,显示到修改的页面上
        * 要把客户的主键和上传文件的路径使用隐藏域保存起来
        * 在edit.jsp中,把客户的网络地址等信息删除掉,没有用这些字段。
    
    2. 修改客户的信息
        * 修改表单的enctype属性(enctype="multipart/form-data")
        * 给edit.jsp页面添加文件上传项()
        * 如果用户新上传了文件,删除旧的文件,上传新的文件。
        * 如果用户没有上传新文件,正常更新。
    
    3. 如果要客户和联系人配置了一对多
        * 再修改客户的时候,由于Customer对象中linkmans的set中没有值,所以在默认修改Customer的时候,会把set集合中的Linkman的外键设置成null
        * 创建linkman的SQL语句中,要求外键是不能为null的
            * <set name="linkmans" inverse="true">
    
----------
    
**功能四:删除客户的功能**
    
    1. 删除上传的文件后,再删除客户信息。
    
----------
    
**抽取通用的BaseDao功能**
    
    1. 通过上面编写的一些功能,DAO层的代码相对比较固定,所以可以想办法来抽取出通用的方法!!
    2. 代码如下
        private Class clazz;
        public BaseDaoImpl(){
            Class c = this.getClass();
            Type type = c.getGenericSuperclass();
            // 判断
            if(type instanceof ParameterizedType){
                ParameterizedType ptype = (ParameterizedType) type;
                // 获取实际类型参数
                Type[] types = ptype.getActualTypeArguments();
                // 获取0位置的值
                clazz = (Class) types[0];
            }
        }
    
----------
    
**抽取BaseAction的功能**
    
    1. Action需要完成分页的代码,需要接收pageCode和pageSize的请求参数,可以编写BaseAction用来接收分页的请求参数
        private Integer pageCode = 1;
        public void setPageCode(Integer pageCode) {
            if(pageCode == null){
                pageCode = 1;
            }
            this.pageCode = pageCode;
        }
        public Integer getPageCode() {
            return pageCode;
        }
        
        // 每页显示的数据的条数
        private Integer pageSize = 2;
        public void setPageSize(Integer pageSize) {
            this.pageSize = pageSize;
        }
        public Integer getPageSize() {
            return pageSize;
        }
        
        public void setVs(String key,Object obj){
            ActionContext.getContext().getValueStack().set(key, obj);
        }
        
        public void pushVs(Object obj){
            ActionContext.getContext().getValueStack().push(obj);
        }
    
----------
    
### 联系人模块 ###
    
----------
    
**功能一:查询联系人功能**
    
    1. 分页显示所有的联系人的数据
    
----------
    
**功能二:添加联系人功能**
    
    1. 自己完成
    
----------
    
**功能三:修改联系人功能**    

    1. 自己完成
    
----------
    
**功能四:删除联系人功能**

    1. 自己完成
 

 

## crm练习 ##
    
----------
    
**课程内容**
    
----------
    
**课程目标**
    
    1. 通过SSH框架完成基本功能的开发
    
----------
    
### 用户模块 ###
    
----------
    
**功能一:用户注册功能**
    
    1. 可以先判断登录名是否已经存在
    2. 要给密码使用MD5进行加密操作
    
----------
    
**功能二:用户登录功能**
    
    1. 登录功能要注意需要先给密码加密后,再进行查询
        * 密码加密后再查询
        * 用户的状态必须是1,字符串类型的
    
----------
    
**功能三:用户退出功能**
    
    1. 把用户信息从HttpSession中清除
    
----------
    
### 客户模块 ###
    
----------
    
**功能一:查询所有客户功能**
    
    1. 数据字典表的引入
        * 数据字典表的作用:规范开发中数据的写法
        * 字段表与客户表是一对多的关系
        * 修改客户表,添加外键(使用SQLyog进行修改)
    
![](./图片/01-字典表.bmp)
    
    2. 创建字典表的实体和映射的配置文件
        * 编写字典表的JavaBean和映射的配置文件
        * 修改Customer的JavaBean,因为是多方,需要把外键字段换成字典对象
        * 修改Customer.hbm.xml的配置文件,配置多对一
    
    3. 分页查询所有的客户功能实现
    
----------
    
**功能二:按条件查询所有的客户**
    
    1. 使用异步的方式加载客户级别和客户的来源
        * 前端使用JQuery的ajax技术
        * 后端使用fastjson的jar包
            * 导入fastjson的开发jar包fastjson-1.2.8.jar
            * String s = JSON.toJSONString(集合)
            * String s = JSON.toJSONString(对象)
        
        * 如果List集合中存入相同引用的对象
            * fastjson默认的情况下是进行循环检测的,去除掉死循环调用的方式
            * 可以使用JSON.toJSONString(p,SerializerFeature.DisableCircularReferenceDetect) 去除循环检测,但是就会出现死循环的效果
            * 最后可以使用注解:@JSONField(serialize=false)对指定的属性不转换成json
    
    2. 异步获取客户级别
        * ajax的代码
            var url = "${pageContext.request.contextPath }/dict_findByCode.action";
            var param = {"dict_type_code":"006"};
            $.post(url,param,function(data){
                $(data).each(function(){
                    var id = "${model.level.dict_id}";
                    if(id == this.dict_id){
                        $("#levelId").append("<option value='"+this.dict_id+"' selected>"+this.dict_item_name+"</option>");
                    }else{
                        $("#levelId").append("<option value='"+this.dict_id+"'>"+this.dict_item_name+"</option>");
                    }
                });
            },"json");
        
        * Action的代码
            public String findByCode(){
                List<Dict> list = dictService.findByCode(dict.getDict_type_code());
                String jsonString = FastJsonUtil.toJSONString(list);
                HttpServletResponse response = ServletActionContext.getResponse();
                FastJsonUtil.write_json(response, jsonString);
                return NONE;
            }
        
        * CustomerAction的分页查询的代码
            public String findByPage(){
                // 调用service业务层
                DetachedCriteria criteria = DetachedCriteria.forClass(Customer.class);
                // 拼接查询的条件
                String name = customer.getCust_name();
                if(name != null && !name.trim().isEmpty()){
                    criteria.add(Restrictions.like("cust_name", "%"+name+"%"));
                }
                
                // System.out.println(customer.getLevel().getDict_type_code());
                Dict level = customer.getLevel();
                if(level != null && !level.getDict_id().trim().isEmpty()){
                    criteria.add(Restrictions.eq("level.dict_id", level.getDict_id()));
                }
                
                Dict source = customer.getSource();
                if(source != null && !source.getDict_id().trim().isEmpty()){
                    criteria.add(Restrictions.eq("source.dict_id", source.getDict_id()));
                }
                
                // 查询
                PageBean<Customer> page = customerService.findByPage(pageCode,pageSize,criteria);
                // 压栈
                ValueStack vs = ActionContext.getContext().getValueStack();
                // 栈顶是map<"page",page对象>
                vs.set("page", page);
                vs.set("cust_name", name);
                return "page";
            }
    
----------
    
**功能三:添加客户功能(含有文件上传功能)**
    
    1. 跳转到客户的添加页面,需要通过ajax来显示客户的级别,客户的来源和客户的行业。
    2. 添加文件上传的选择项
    3. 客户端三个注意事项
        * method="post"
        * enctype="multipart/form-data"
        * <input type="file" name="myfile">
    
    4. Struts2框架的使用拦截器完成了文件上传,并且底层使用也是FileUpload开源的组件。
        * 提供 FileUpload 拦截器,用于解析 multipart/form-data 编码格式请求,解析上传文件的内容 
        * fileUpload拦截器 默认在 defaultStack 栈中, 默认会执行的 
        
        * 在Action中编写文件上传,需要定义三个属性
            > 文件类型File ,属性名与表单中file的name属性名一致.
            > 字符串类型String , 属性名:前段是name属性名一致 + ContentType;
            > 字符串类型String , 属性名:前段是name属性名一致+FileName;
            
            > 最后需要为上述的三个属性提供set方法。
            > 可以通过FileUtils提供 copyFile 进行文件复制,将上传文件 保存到服务器端 
    
    4. 文件上传中存在的问题
        * 先配置input逻辑视图
        * 在页面中显示错误信息
        
        * 文件上传的总大小默认值是2M,如果超过了2M,程序会报出异常。可以使用<s:actionError>来查看具体信息!
            > 解决总大小的设置,找到常量:
                * struts.multipart.parser=jakarta        -- 默认文件上传解析器,就是FileUpload组件
                * struts.multipart.saveDir=                -- 文件上传的临时文件存储目录
                * struts.multipart.maxSize=2097152        -- 文件上传的最大值(总大小),默认是2M
            
            > 可以在struts.xml中设置常量,修改文件上传的默认总大小!!!
                * <constant name="struts.multipart.maxSize" value="5000000"></constant>
    
    5. 还可以通过配置拦截器来设置文件上传的一些属性
        * 先在<action>标签中引入文件上传的拦截器
            <interceptor-ref name="defaultStack">
                <!-- 设置单个上传文件的大小 -->
                <param name="fileUpload.maximumSize">2097152</param>
                <!-- 设置扩展名 -->
                <param name="fileUpload.allowedExtensions">.txt</param>
            </interceptor-ref>
    
----------
    
**功能三:修改客户的功能**
    
    1. 先通过客户的主键查询出客户的详细信息,显示到修改的页面上
        * 要把客户的主键和上传文件的路径使用隐藏域保存起来
        * 在edit.jsp中,把客户的网络地址等信息删除掉,没有用这些字段。
    
    2. 修改客户的信息
        * 修改表单的enctype属性(enctype="multipart/form-data")
        * 给edit.jsp页面添加文件上传项()
        * 如果用户新上传了文件,删除旧的文件,上传新的文件。
        * 如果用户没有上传新文件,正常更新。
    
    3. 如果要客户和联系人配置了一对多
        * 再修改客户的时候,由于Customer对象中linkmans的set中没有值,所以在默认修改Customer的时候,会把set集合中的Linkman的外键设置成null
        * 创建linkman的SQL语句中,要求外键是不能为null的
            * <set name="linkmans" inverse="true">
    
----------
    
**功能四:删除客户的功能**
    
    1. 删除上传的文件后,再删除客户信息。
    
----------
    
**抽取通用的BaseDao功能**
    
    1. 通过上面编写的一些功能,DAO层的代码相对比较固定,所以可以想办法来抽取出通用的方法!!
    2. 代码如下
        private Class clazz;
        public BaseDaoImpl(){
            Class c = this.getClass();
            Type type = c.getGenericSuperclass();
            // 判断
            if(type instanceof ParameterizedType){
                ParameterizedType ptype = (ParameterizedType) type;
                // 获取实际类型参数
                Type[] types = ptype.getActualTypeArguments();
                // 获取0位置的值
                clazz = (Class) types[0];
            }
        }
    
----------
    
**抽取BaseAction的功能**
    
    1. Action需要完成分页的代码,需要接收pageCode和pageSize的请求参数,可以编写BaseAction用来接收分页的请求参数
        private Integer pageCode = 1;
        public void setPageCode(Integer pageCode) {
            if(pageCode == null){
                pageCode = 1;
            }
            this.pageCode = pageCode;
        }
        public Integer getPageCode() {
            return pageCode;
        }
        
        // 每页显示的数据的条数
        private Integer pageSize = 2;
        public void setPageSize(Integer pageSize) {
            this.pageSize = pageSize;
        }
        public Integer getPageSize() {
            return pageSize;
        }
        
        public void setVs(String key,Object obj){
            ActionContext.getContext().getValueStack().set(key, obj);
        }
        
        public void pushVs(Object obj){
            ActionContext.getContext().getValueStack().push(obj);
        }
    
----------
    
### 联系人模块 ###
    
----------
    
**功能一:查询联系人功能**
    
    1. 分页显示所有的联系人的数据
    
----------
    
**功能二:添加联系人功能**
    
    1. 自己完成
    
----------
    
**功能三:修改联系人功能**    
    
    1. 自己完成
    
----------
    
**功能四:删除联系人功能**
    
    1. 自己完成
    
----------
    
### 客户拜访模块 ###
    
----------
    
**功能一:搭建客户拜访表的开发环境(导入资料中的拜访客户的SQL语句)**
    
    1. 客户关系拜访表是该系统的用户和客户之间的关系建立表
        * 用户可以拜访多个客户
        * 客户也可以被多个用户所拜访
        * 所以:用户和客户之间应该是多对多的关系,那么客户拜访表就是用户和客户的中间表。
        * 正常的情况下,在用户和客户中添加set集合,在映射的配置文件中配置<set>标签即可。
        * 但是现在客户拜访中间表中存在其他的字段,默认的情况下,中间表只能维护外键。而不能维护其他的字段。所以需要把一对多拆开成两个一对多。
    
    2. 用户与客户拜访表是一对多的关系
    3. 客户与客户拜访表是一对多的关系
    4. 创建客户拜访表的实体类和映射配置文件
    5. 编写客户拜访的Action等类和完成配置
        * 先开启注解的扫描
            * <context:component-scan base-package="com.itheima"/>
        
        * Action编写(@RestController(value="visitAction") @Scope(value="prototype"))
        * Service编写(@Service(value="visitService") @Transactional)
        * Dao编写(@Repository(value="visitDao"))
            * 重点是dao中注入SessionFactory对象
                @Resource(name="sessionFactory")
                public void sSessionFactory(SessionFactory sessionFactory){
                    // 重点的代码
                    super.setSessionFactory(sessionFactory);
                }
    
----------
    
**功能二:客户拜访列表查询功能**
    
    1. 先导入客户拜访的页面
        * 在资料中的(visit文件夹和jquery文件夹)
            * visit文件夹复制到jsp的目录下
            * jquery文件夹复制到WebContent目录下
    
    2. 查询我的客户拜访记录
        * 登录的用户,点击客户拜访列表,查询该用户下的所有的拜访记录
        * 通过用户的主键查询该用户下的所有的拜访记录
    
----------
    
**功能三:新增客户拜访记录功能**
    
    1. 点击新增客户拜访功能菜单,跳转到新增页面,输入信息,保存数据
        * 从HttpSession中获取到用户的信息,设置到拜访记录中,保存到数据库中
    
----------
    
**功能四:按条件查询客户信息列表功能**

    1. 修改list.jsp的页面,添加开始和结束日期的选项
        <TD>拜访时间:</TD>
        <TD>
            <INPUT class=textbox id="beginDate" style="WIDTH: 80px" maxLength=50 name="beginDate">
            至
            <INPUT class=textbox id="endDate" style="WIDTH: 80px" maxLength=50 name="endDate">
        </TD>
    
----------
    
### 统计分析模块 ###
    
----------
    
**功能一:客户来源统计**
    
    1. 想要统计客户的来源,即该来源下有多少个客户
        * SQL语句:SELECT d.dict_item_name,COUNT(*) FROM base_dict d,cst_customer c WHERE d.dict_id = c.cust_source GROUP BY d.dict_id;
        * HQL语句:String hql = "select c.source.dict_item_name,COUNT(*) from Customer c inner join c.source GROUP BY c.source";
    
----------
    
### 用户登录的拦截器 ###
    
----------
    
**用户登录的拦截器功能实现**

    1. 功能:如果用户没有登录,是不能操作后台的功能的!!
    2. 代码如下
        public class UserInterceptor extends MethodFilterInterceptor{

            private static final long serialVersionUID = 335018670739692955L;
            
            /**
             * 进行拦截的方法
             */
            protected String doIntercept(ActionInvocation invocation) throws Exception {
                // 获取session对象
                User user = (User) ServletActionContext.getRequest().getSession().getAttribute("existUser");
                if(user == null){
                    // 说明,没有登录,后面就不会执行了
                    return "login";
                }
                return invocation.invoke();
            }
        }
    
    3. 配置如下
        <interceptors>
            <interceptor name="UserInterceptor" class="com.itheima.web.interceptor.UserInterceptor"/>
        </interceptors>
        <interceptor-ref name="UserInterceptor">
            <!-- login方法不拦截 -->
            <param name="excludeMethods">login</param>
        </interceptor-ref>
        <interceptor-ref name="defaultStack"/>

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
/* * @(#)PageControl.java 1.00 2004-9-22 * * Copyright 2004 2004 . All rights reserved. * PROPRIETARY/CONFIDENTIAL. Use is subject to license terms. */ package com.hexiang.utils; /** * PageControl, 分页控制, 可以判断总页数和是否有上下页. * * 2008-07-22 加入输出上下分页HTML代码功能 * * @author HX * @version 1.1 2008-9-22 */ public class PageBean { /** 每页显示记录数 */ private int pageCount; /** 是否有上一页 */ private boolean hasPrevPage; /** 记录总数 */ private int recordCount; /** 是否有下一页 */ private boolean hasNextPage; /**总页面数 */ private int totalPage; /** 当前页码数 */ private int currentPage; /** * 分页前的页面地址 */ private String pageUrl; /** * 输出分页 HTML 页面跳转代码, 分链接和静态文字两种. * 2008-07-22 * @return HTML 代码 */ public String getPageJumpLinkHtml() { if(StringUtil.isEmpty(pageUrl)) { return ""; } // 检查是否有参数符号, 没有就加上一个? if(pageUrl.indexOf('?') == -1) { pageUrl = pageUrl + '?'; } StringBuffer buff = new StringBuffer("<span id='pageText'>"); // 上一页翻页标记 if(currentPage > 1) { buff.append("[ <a href='" + pageUrl + "&page=" + (currentPage - 1) + "' title='转到第 " + (currentPage - 1) + " 页'>上一页</a> ] "); } else { buff.append("[ 上一页 ] "); } // 下一页翻页标记 if(currentPage < getTotalPage()) { buff.append("[ <a href='" + pageUrl + "&page=" + (currentPage + 1)+ "' title='转到第 " + (currentPage + 1) + " 页'>下一页</a> ] "); } else { buff.append("[ 下一页 ] "); } buff.append("</span>"); return buff.toString(); } /** * 输出页码信息: 第${currentPage}页/共${totalPage}页 * @return */ public String getPageCountHtml() { return "第" + currentPage + "页/共" + getTotalPage() + "页"; } /** * 输出 JavaScript 跳转函数代码 * @return */ public String getJavaScriptJumpCode() { if(StringUtil.isEmpty(pageUrl)) { return ""; } // 检查是否有参数符号, 没有就加上一个? if(pageUrl.indexOf("?") == -1) { pageUrl = pageUrl + '?'; } return "<script>" + "// 页面跳转函数\n" + "// 参数: 包含页码的表单元素,例如输入框,下拉框等\n" + "function jumpPage(input) {\n" + " // 页码相同就不做跳转\n" + " if(input.value == " + currentPage + ") {" + " return;\n" + " }" + " var newUrl = '" + pageUrl + "&page=' + input.value;\n" + " document.location = newUrl;\n" + " }\n" + " </script>"; } /** * 输出页面跳转的选择框和输入框. 示例输出: * <pre> 转到 <!-- 输出 HTML SELECT 元素, 并选中当前页面编码 --> <select onchange='jumpPage(this);'> <c:forEach var="i" begin="1" end="${totalPage}"> <option value="${i}" <c:if test="${currentPage == i}"> selected </c:if> >第${i}页</option> </c:forEach> </select> 输入页码:<input type="text" value="${currentPage}" id="jumpPageBox" size="3"> <input type="button" value="跳转" onclick="jumpPage(document.getElementById('jumpPageBox'))"> </pre> * @return */ public String getPageFormJumpHtml() { String s = "转到\n" + "\t <!-- 输出 HTML SELECT 元素, 并选中当前页面编码 -->\n" + " <select onchange='jumpPage(this);'>\n" + " \n"; for(int i = 1; i <= getTotalPage(); i++ ) { s += "<option value=" + i + "\n"; if(currentPage == i) { s+= " selected "; } s += "\t>第" + i + "页</option>\n"; } s+= " </select>\n" + " 输入页码:<input type=\"text\" value=\"" + currentPage + "\" id=\"jumpPageBox\" size=\"3\"> \n" + " <input type=\"button\" value=\"跳转\" onclick=\"jumpPage(document.getElementById('jumpPageBox'))\"> "; return s; } /** * 进行分页计算. */ private void calculate() { if (getPageCount() == 0) { setPageCount(1); } totalPage = (int) Math.ceil(1.0 * getRecordCount() / getPageCount()); // 总页面数 if (totalPage == 0) totalPage = 1; // Check current page range, 2006-08-03 if(currentPage > totalPage) { currentPage = totalPage; } // System.out.println("currentPage=" + currentPage); // System.out.println("maxPage=" + maxPage); // // Fixed logic error at 2004-09-25 hasNextPage = currentPage < totalPage; hasPrevPage = currentPage > 1; return; } /** * @return Returns the 最大页面数. */ public int getTotalPage() { calculate(); return totalPage; } /** * @param currentPage * The 最大页面数 to set. */ @SuppressWarnings("unused") private void setTotalPage(int maxPage) { this.totalPage = maxPage; } /** * 是否有上一页数据 */ public boolean hasPrevPage() { calculate(); return hasPrevPage; } /** * 是否有下一页数据 */ public boolean hasNextPage() { calculate(); return hasNextPage; } // Test public static void main(String[] args) { PageBean pc = new PageBean(); pc.setCurrentPage(2); pc.setPageCount(4); pc.setRecordCount(5); pc.setPageUrl("product/list.do"); System.out.println("当前页 " + pc.getCurrentPage()); System.out.println("有上一页 " + pc.hasPrevPage()); System.out.println("有下一页 " + pc.hasNextPage()); System.out.println("总页面数 " + pc.getTotalPage()); System.out.println("分页 HTML 代码 " + pc.getPageJumpLinkHtml()); } /** * @return Returns the 当前页码数. */ public int getCurrentPage() { return currentPage; } /** * 设置当前页码, 从 1 开始. * @param currentPage * The 当前页码数 to set. */ public void setCurrentPage(int currentPage) { if (currentPage <= 0) { currentPage = 1; } this.currentPage = currentPage; } /** * @return Returns the recordCount. */ public int getRecordCount() { return recordCount; } /** * @param recordCount * The recordCount to set. */ public void setRecordCount(int property1) { this.recordCount = property1; } /** * @return Returns the 每页显示记录数. */ public int getPageCount() { return pageCount; } /** * @param pageCount * The 每页显示记录数 to set. */ public void setPageCount(int pageCount) { this.pageCount = pageCount; } public String getPageUrl() { return pageUrl; } public void setPageUrl(String value) { pageUrl = value; } }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值