开始时间:7月17日
技术架构
(一)Web开发4层开发
- 视图层(view):展示数据,跟用户交互。《html,css,js,jquery,bootstrap(ext|easyUI),jsp》
- 控制层(controller):控制业务处理流程(接受请求,接受参数,封装参数,调用业务层,响应数据)。《SpringMVC,servlet,struts1,struts2,webwork》
- 业务层(service):处理业务逻辑(处理业务的步骤以及操作的原子性)。《javase(工作流:activiti | JBPM)》
- 持久层(Dao或Mapper):操作数据库。《传统jdbc,MyBatis,hibernate,ibatis》
- 整合层:粘合各层框架,维护类资源(ioc),维护数据库资源(aop)。《Spring,ejb,corba》
教学目的
(二)综合应用:
学编程,就要学以致用,否则学得再熟练也没有
(三)教学方式:26天(线下),打算20天做完。
做web项目开发(完全模拟企业开发),包装2到3年的开发经验。
(四)教学目的:
- 对软件公司和软件开发有一定的了解
- 了解CRM项目的核心业务
- 能够独立完成CRM项目核心业务的开发
- 对前期所学技术进行回顾,熟练,加深和扩展
- 掌握互联网基础课:linux,redis,git
(五)软件公司的组织结构:
下面部门都和程序员相关
研发部:程序员,美工(前端,设计图片),DBA(数据库调优员)
测试部:测试工程师(测试每个版本,找出bug,记录bug,国内大多数都是黑盒测试,只看功能,不看代码,而白盒都看)
产品部:了解软件市场和业务(提需求)
实施部:部署服务器,了解软硬件服务器(兼职性)
运维部:懂服务器(一般在机房隔壁办公)
市场部:重要!负责去拉项目来做(工资高,有提成,找程序员做标书)
其他:人力部和财务部和程序员关系不大
软件开发周期
(六)软件开发生命周期
1–
招标:政府和金融机构等,需要做哪些项目后,会发布招标(工期,投入等)
投标: 市场部做标书,和程序员沟通。然后投给招标
甲方:招标方
乙方:投标方
2–可行性分析:—————————— 可行性分析报告
技术总监或者架构师负责
技术(本公司技术能不能匹配要求)
经济(是否还需招程序员,招哪些层次的程序员)
(如果中标了,而可行性较低,可以转给外包公司(所以说,中标稳赚不赔))
3—-需求分析:——————————–《需求文档》
产品经理进行分析,需要和客户沟通(需求调研,有可能和程序员一起去)
项目原型(美工做,完成通用的一些业务功能模块页面):去需要调研时,都是带项目原型去和客户沟通的(给客户看静态的页面,确定是否是这些需求),容易确定需求。
确定需求之后,写好需求文档,还需双方签字,才能进行开发。
4—-分析与设计:
负责人是架构师或者项目经理,就相当于盖一个房子,里面负责设计图纸的建筑师。而这时候负责搬砖的工人还没有招进来,就相当于程序员,所以说程序员是“码农
5—-架构设计:整体性设计(服务器,选用技术)—————————————–架构师
物理架构设计:项目将来运行所在的硬件服务器,用的硬盘
应用服务器:(Tomcat(免费,中小企业使用),weblogic(收费,功能强大,支持javaee:13种协议),websphere(ibm,收费),jboss(redhat公司,收费),resin(微软))
数据库服务器:mysql(小巧灵活,免费),oracle(笨重,收费),DB2,sqlservet,达梦
逻辑架构设计:代码分层
技术选型:java,.net(微软) ,框架
6—-项目设计:项目经理负责 ————-项目设计文档
物理模型设计(项目经理设计):哪些表,哪些字段,字符的类型和长度,以及表和表之间的关系。工具:powerdesigner —-.pdm(能转换成表sql语句)
逻辑模型设计(实际中,由程序员自己写):哪些类,哪些属性和方法,方法的参数和返回值,以及类和类之间的关系。 工具:rationnal rose—pdl
界面设计:企业级应用(朴素,用户群体固定,并发数有限),互联网应用(炫酷)
算法设计:复杂业务,企业级应用不特别在意算法设计,而互联网应用会经常涉及(高并发)
7–搭建开发环境:项目经理搭建
创建项目,添加jar包,添加配置文件,添加静态页面,添加公共类以及其他资源;能够正常启动运行
8—–编码实现:HR开始忙起来,去面试程序员。———–注释文档
9—–测试:并不是完全开发完项目才开始测试,而是开发和测试同步进行,一般每周都会提交一个版本交给测试部测试,测试找到bug,放在bug记录平台,程序员再去修改自己的bug。—-测试用例—————–测试用例
10—–试运行:实施部帮客户或者客户找人去买对应性能的服务器。运行大概一周,没问题就可以去上线了。程序员开始逐渐转到下一个项目你 ———–使用手册
11——上线:——–实施文档
12—-运维:软件周期一般最长在5年,就要下架了(随着使用,系统会变慢) ——–运维手册
13—-文档编纂:包含在整个开发过程,例如;需求文档,项目设计文档,可行性分析文档和注释文档。
servlet,jsp(jspl) springMVC
CRM项目的核心业务
CRM项目的简介:Customer Relationship Management 客户关系管理系统
类别为企业级应用,传统应用。给销售或者贸易型公司使用,在市场,销售,服务各个环节中维护客户关系。
CRM项目的宗旨:增加新客户,留住老客户,把已有客户转化成忠诚客户。
CRM是一类项目,每种行业,需要的CRM的实现都不一样,因为业务不一样,本CRM是给一个大型的进出口贸易公司来使用的,做大宗商品的进出口贸易;商品受管制
CRM项目的核心业务:
系统管理功能:不是直接处理业务数据,为了保证业务管理的功能正常安全运行而设计的功能。例如:登录功能,安全退出,登录验证(给超级管理员,系统管理员用的)
业务管理功能:处理业务数据
市场活动:市场部,设计市场活动营销活动
线索:销售部(初级销售),增加线索
客户和联系人:销售部(高级销售),有效地区分和跟踪客户和联系人
交易:销售部(高级销售),更好地区分和统计交易的各个阶段
销售回放:客户部,妥善安排销售回访。主动提醒(防止客户忘记)
统计图表:管理层用,统计交易表中各个阶段数据量。
CRM-物理模型设计
系统管理功能表:
tbl_user 用户表
tbl_dic_type 数据字典类型表
tbl_dic_value 数据字典值
业务管理功能相关表:
tbl_activity 市场活动表
tbl_activeity_remark 市场活动备注表
tbl_clue_activity_relation 线索和市场活动的关联关系表
tbl_clue 线索表
tbl_clue_remark 线索备注表
tbl_customer 客户表
tbl_customer_remark 客户备注表
tbl_contacts 联系人表
tbl_contacts_remark 联系人备注表
tbl_contacts_activity_relation 联系人和市场活动的关联关系表
tbl_tran 交易表
tbl_tran_history 交易历史表
tbl_tran_remark 交易备注表
tbl_task 任务表
主键字段
在表中,如果有一组字段能够唯一确认一条记录,则可以把它们设计成表的主键。推荐使用一个字段做主键,而且推荐使用没有业务含义的字段做主键。比如:id等。主键字段的类型和长度由主键值的生成方式来决定:
- 自增:借助数据库自身主键生成机制。数值类型,长度由数据量来决定(企业中,一般不用,底层实现用到了锁,高并发时,运行效率低,但开发效率高)
- assigned(手动):程序员手动生成主键值,唯一非空,算法。hi/low:数值型,长度由数据量决定。UUID:字符串,长度固定为32位(企业常用,效率高,适合多线程场景,没有加锁)
- 共享主键:由另一张表的类型和长度决定,例如:一个人的信息表,和驾照表。驾照表的id就引用一个人的id就行
- 联合主键:由多个字段的类型和长度来决定的
外键字段
用来确定表和表之间的关系
一(主表)对多(父表):
一张表A中的一条记录可以对应另一张表B中的多条记录;而另一张表B中的一条记录只能对应一张表A中的一个记录。例如:学生表(子表)和班级表(父表),一般多的那张表引用一的那个表的主键
添加记录:先添加父表记录,才能添加子表记录
删除记录:先删除子表记录,才能删除父表记录
内连接([inner] join on :外键不能为空):查询所有符合条件的数据,并且要求结果在两张表都有对应的
左外连接(left login on)(外键可以为空,按需求选择左还是右):查询左侧表中所有符合条件的数据,即使在右表中没有对应的数据,右外连接则相反(right login on)
*如果外键不能为空,优先使用内连接
*如果外键能为空,按需求选择左右外连接
一对一:
一张表A中的一条记录可以对应另一张表B中的一条记录;而另一张表B中的一条记录对应另一张表A中的一个记录。例如:个人信息表和驾照表,可以使用共享主键(不经常用)
添加记录:先添加先产生的表的记录,后再添加后产生的表的记录
删除记录:先删除后后产生的表记录,在删除先产生的表记录
查询数据:无需进行连接查询
一般都是用:唯一外键。
多对多:
一张表A中的一条记录可以对应另一张表B中的多条记录;而另一张表B中的一条记录对应另一张表A中的多个记录。例如:学生表和课程表。
添加记录:先添加父表记录,再添加子表记录(中间表)
删除记录:先删除子表(中间表)记录,再删除父表记录。
查询记录:可能会进行关联查询
借助中间表来维护关系
关于日期和时间的字段
java
Date
数据库
date:只有日期
time:只有时间
datetime:日期和时间
基于以上格式转换问题,都直接按照字符串处理:
- char(10)yyyy-MM-dd
- char(19)yyyy-MM-dd HH:mm:ss
导入表sql,物理模型设计完成。
搭建开发环境
创建项目,创建工程:crm
设置编码格式:统一UTF-8
添加jar包依赖:需求文档上有(不需要背)
添加配置文件:不需要背,前期搭建环境时,需要用而已。
添加静态页面:防止非法访问,造成的恶意破坏,一般都把页面放在WEB-INF下
合理的项目文件目录:
生成的Tomcat服务器项目文件目录:
编码开发顺序:先做系统管理功能,后做业务管理功能。
做模块功能必须走的流程:
- 分析需求:
- 分析与设计:
- 编码实现
- 测试
登录功能的实现
1.流程图
2.写代码,先写底层,再写高层(高层调用底层)
一个资源目录,创建一个controller进行响应
同步请求:页面全局刷新
异步请求:全局或者局部刷新都可以,存在两个刷新方式时,只能用异步请求。
一个表,对应的一个service,一个对象
mybatis逆向工程
简介:根据表生成mapper层三层部分代码:实体类,mapper接口,映射文件
创建工程:crm-mybatis-generator
添加插件:
org.mybatis.generatormybatis-generator-maven-plugin1.3.2truetrue
需要配置:
数据库连接信息
代码保存的目录
表的信息
运行mybatis的逆向工程,根据指定的表
选择器.attr(“属性值名”);//用来获取哪些值不是true或者false的属性的值
选择器.prop(“属性名”);//用来获取值是true/false 的属性值
选择器.click(function(){})//给指定的元素添加事件
选择器.click();//给指定的元素上模拟发生一次点击事件
把控制层(controller)处理好的数据传到视图层(jsp),使用作用域传递;
四大作用域:
pageContext:同页面内传参数,用来在同一个页面不同标签之间传递数据
request:在同一个请求之间传输数据。(请求完成后,会消失(不能重定向))
session:同一个浏览器,不同请求之间传递数据(服务器停止前,都有效)
application:所有用户共享的数据,而且长久频繁使用的数据。
实现点击回车,确认登录功能:
//给整个登录窗口添加键盘按下事件-------------实现点击回车,自动登录$(window).keydown(function (e){//判断是否按键为回车键,,是,则提交登录请求,否则不做响应if(e.keyCode===13){//回车的Ascall码为13$("#loginBtn").click()}})
记住密码:
//如果需要记住密码,则往外写Cookieif("true".equals(isRemember)){//选择记住密码System.out.println("到了!!");//账号Cookie cookie1 = new Cookie("loginAct", user.getLoginAct());cookie1.setMaxAge(10*24*60*60);//设置声明周期,10天后自动删除response.addCookie(cookie1);//密码Cookie cookie2 = new Cookie("loginPwd", user.getLoginPwd());cookie2.setMaxAge(10*24*60*60);//设置声明周期,10天后自动删除response.addCookie(cookie2);}else{//选择不记住密码//把没有过期cookie删除,不能直接删除客户端的Cookie,通过覆盖,设置生命周期,来达到同样效果Cookie cookie1 = new Cookie("loginAct", "1");cookie1.setMaxAge(0);response.addCookie(cookie1);Cookie cookie2 = new Cookie("loginPwd", "1");cookie2.setMaxAge(0);response.addCookie(cookie2);}
通过判断和记录Cookie记录
实现思路:每次登录,判断是否需要记住密码,如果需要,则写Cookie。而且把用户的账号和密码保存在Cookie里面
下次登录,判断该用户有没有Cookie,没有,则不填,有,则填写Cookie值,即自动填写账号和密码,
页面获取Cookie的方法:
安全退出:
P38集
业务页面的登录验证功能
借助SpringMVC的拦截器HandlerInterceptor实现验证功能
前期映射路径命名规则有利于拦截器使用:
1–写拦截类
public class LoginInterceptor implements HandlerInterceptor {//访问目标资源前,执行@Overridepublic boolean preHandle(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o) throws Exception {//验证用户是否处于登录状态HttpSession session = httpServletRequest.getSession();//获取session对象User user = (User) session.getAttribute(Contants.SESSION_USER);if(user==null){//未登录,跳转到登录页面//重定向--需要加项目的名称(这里不是MVC框架范围,视图解析器不起作用,不帮忙加前后缀)httpServletResponse.sendRedirect(httpServletRequest.getContextPath());//手动重定向,url必须加项目的名称,httpServletRequest.getContextPath()=="/crm"return false;}else{//已登录,放行资源return true;}}@Overridepublic void postHandle(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, ModelAndView modelAndView) throws Exception {}@Overridepublic void afterCompletion(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, Exception e) throws Exception {}}
2–配置拦截器
<!--不需要加这个,因为没有用到全局拦截“/**”-->
页面切割技术
一个页面,可以由多个页面组成
技术1:和(重标签,性能低)
: 用来切割页面,只能按行或者列来切
: 显示页面,每一个标签就是一个独立的浏览器窗口
技术2:和(轻量标签,性能高)
: 用来切割页面,即和之前用div一样就行
:显示页面
市场活动
1–创建(先实现创建模块,在做删改查)
模态窗口:模拟的窗口,不是真实弹出来的,本质上是div,通过设置z-index大小来实现;
初始时,z-index初始参数小于0,不显示;需要显示时,设置大于0,即可1显示
借助bootstrap框架来控制z-index的大小
方式1–通过框架的标签的属性date-toggle=”modal“ :触发显示模态窗口,date-target=”模态窗口的id“ :指定显示某个模态窗口。
方式2–通过js函数控制
方式3–通过标签的属性data-dismiss=”“,通过点击添加了属性data-dismiss=”“的按钮,来关闭当前的模态窗口
模态窗口有利于获取要处理的数据
显示市场活动页面
p49
清空form表单里面所有输入框内容
//清空表单.get[0].reset(),, .get(0)为获取表单的全部DOM对象
$(“#createActivityForm”).get[0].reset()
日历功能的实现:
满足这两个特定,说明有插件,现成的功能的直接cv就行
- 实现起来比较复杂
- 更业务无关
日历插件:bootstrap-datetimepicker
前端插使用步骤:
- 下载开发包
- 拷贝到webapp目录下
- 引入开发包到使用页面上
- 创建容器:组件运行结果为字符串
- 当容器加载之后,对容器调用工具函数(传参数)
引入css和js等依赖资源,先引入被依赖的资源
$(function(){}) :为入口函数,只有页面其他资源加载完后,才执行
分页查询功能的实现
- 查询的时候,能分开为一个独立方法,就分开为独立方法(service层),这样更灵活。
- 涉及多个表的增删改时,最好放在service层的一个方法执行,且controller层只调用service一次,保证操作都在一个事务内,可以把全部操作回滚,保持原子性和数据一致性。而不是在controller层调用多次service方法。
P63
在指定的标签中,显示jsp页面片段:
$(“#tBody”).html(html语句)//覆盖显示
$(“#tBody”).text(文本)//覆盖,只能添加文本
$(“#tBody”).append(html语句)//在尾部追加显示
$(“#tBody”).after(html语句)//追加显示
$(“#tBody”).before(html语句)//追加显示
分页插件:bs_pagination
使用步骤:
- 引入开发包bs_pagination
- 创建容器
- 当容器加载完后,对容器调用工具函数(注意:要容器加载完后,再调用函数,否则失效)
P74
给页面中元素添加事件方式有:
1–使用元素的事件属性:onxxx=“函数名()”//例如:οnclick=“ 函数名()”(一般不用,不易维护)
2–使用jquery对象:选择器.xxx(function(){ //js语句 })(只能为固有元素添加事件)
3–使用jquery的on函数:父选择器.on(‘事件类型名称’,’添加事件元素名的判定依据’,function()){ //js语句 };(不但能给固有元素添加事件,也能给动态生成的元素添加事件)
(父元素:必须时固有元素,可以是直接父元素,也可以是间接父元素)
(事件类型:跟事件数据属性和事件函数意义对应)
(子元素:目标元素,跟父选择器构成一个父子选择器)
注意:使用使用jquery对象判定事件时,对于用在动态元素上,事件可能会失效,而固有元素则没问题
P85
js的截取字符串函数:
js求长度:
ajax向后台发送请求时,通过data提交参数,data的数据格式有三种格式:
第一种:json对象
data:{
k1:v1,
k2:v2,
……
}
*优势:操作简单
*劣势:不能向后台提交一个参数名对应多个参数值的数据。否则会覆盖,只能提交字符串。
data:{
id:v1,
id:v2,
……}//后台只能获取最后一个id的值
第二种:字符串型
data:k1=v1&k2:v2$……
*优势:可以向后台提交一个参数名对应多个参数值的数据
*劣势:需要拼接字符串,操作麻烦,只能提交字符串。
第三种:FormData对象
优势:不但能提交字符串数,还能提交二进制数据
劣势:操作更复杂
封装参数原则规范:
导出市场活动(导出Excel文件)
关键技术准备:
(一)使用java生产Excel文件:iText(收费,功能强大),Apache-poi(免费)
关于办公文档插件使用的基本思想:把办公文档的所有元素封装成普通的java类,程序员通过操作这些类达到操作办公文档的目的。
最基础的办公文档类:
- Excel文件—–HSSFWorkbook对象
- 页——HSSFSheet对象
- 行——-HSSFRow对象
- 列——-HSSFCell对象
- 样式—–HSSFCellStyle
使用Apache-poi生成excel文件:
1–添加依赖
org.apache.poipoi3.15
2–使用封装类
- Excel文件—–HSSFWorkbook对象
- 页——HSSFSheet对象
- 行——-HSSFRow对象
- 列——-HSSFCell对象
- 样式—–HSSFCellStyle
(二)文件下载:
web技术,涉及前后台技术
//所有的文件下载请求,只能发同步请求
//所有的文件下载请求,只能发同步请求window.location.href = "workbench/activity/exportAllActivity.do"//同步请求
//浏览器收到响应数据后,默认情况下,直接在显示窗口打开响应信息
//即使打不开,也会调用其他应用程序打开,只有实在打不开,才回激活下载
//可以设置响应头信息,使浏览器默认直接下载。
//设置响应头,是浏览器默认直接下载响应信息,且设置文件名为myStudent.xslresponse.addHeader("Content-Disposition","attachment;filename=ActivityList.xls");
P108
导入市场活动
技术准备:
文件上传:
前端上传文件三要素:
(一)选择文件选择按钮,type值为“file”
(二)form表单提交方式,method值为“post”
请求方式只能用:post
get:参数通过请求头提交到后台,参数放在url后边;只能向后台提交文本数据;对参数长度有限制;数据不安全;效率高;自动将得到的数据放在缓存(重复访问,用缓存,不需发新请求)。
post:参数通过请求体提交到后台,即能提交文件数据,又能够提交二进制数据(图片,视频等);理论上对参数长度没有限制;相对更安全;效率相对较低。
(三)form表单提交编码格式,enctype值为“mutipart/form-data”
根据HTTP协议的规定,浏览器每次向后台提交参数,都会对参数进行统一编码;默认采用的的编码格式为urlencode,这种编码格式只能为文本数据进行编码;所以文件上传的表单编码格式只能用ultipart/form-data。
使用java解析excel文件: iText,apache-poi(这里工具对于excel文件操作都是双向的)
- Excel文件—–HSSFWorkbook对象
- 页——HSSFSheet对象
- 行——-HSSFRow对象
- 列——-HSSFCell对象
- 样式不提供解析
文件上传同步和异步都行
p125
要注意的时,用mvc框架接受页面传来的文件,需要去mvc核心配置文件配置文件上传解析器才能正常使用MultipartFile类
文件上传,如果需要用到文件里面的数据,则文件格式一定要跟用户约定好
技术上的问题解决不了,可以在业务上选择更好的方案
js获取文件名中的后缀子串(全部转成小写):
let fileType = activityFile.substr(activityFile.lastIndexOf(".")+1).toLocaleLowerCase()
jquery对象转DOM对象
&(“#id”).[0] 或者&(“#id”).get(0)
用ajax上传文件:
//发送请求----//FormData是ajax提供的接口,可以模拟键值对向后台提交参数//FormData不但能提交文本数据,还是提交二进制数据let formData = new FormData();//把文件数据放在formData中的键值对中formData.append("activityFile",File)//key要和后台获取文件的参数名对应$.ajax({url:"workbench/activity/importActivity.do",data:formData,//type:"post",//上传文件,必须为postprocessData:false,//设置ajax向后台提交参数前,是否把参数统一转为字符串,true--是,默认为true,当上传文件时,要设置为falsecontentType:false,//设置ajax向后台提交参数之前,是否把参数统一按urlencoded编码,默认为true,上传文件时,要设置为false,等同于form表单上的enctype=“mutipart/form-data”设置dataType:"json",success:function (data) {if (data.code=="1"){//导入成功alert("成功导入:"+data.retData+"条记录!")//关闭模态窗口$("#importActivityModal").modal("hide")//刷新页面queryActivityByConditionForPage(1, $("#page").bs_pagination('getOption', 'rowsPerPage'))}else{//导入成功//提示信息alert(data.message)//模态窗口不关闭$("#importActivityModal").modal("show")}}})
使用标签保存数据(以便在需要的时候能够获取这些数据)
给标签添加属性:
- 如果时表单组件标签,优先使用value属性,只有value不方便使用时,使用自定义属性(随便在标签上,定义属性名和属性值);
- 如果不是表单组件标签,不推荐是使用value,推荐使用自定义属性;
获取属性值:
- 如果获取表单组件标签的value属性值时,可以用:dom对象.value或者jquery.val();
- 如果是自定义的属性,不管是什么标签,只能用:jquery对象.attr(‘属性名’);
P 130
el表达式在js中使用:
//获取表单数据let activityId = '${activity.id}'//el表达式在js使用,用''括起来,否则浏览器识别为变量
把页面片段动态地显示在页面:5种方法
选择器.append(html):追加显示在指定标签的内部的后面
选择器.text(字符串):覆盖显示在标签内部
选择器.html(html):覆盖显示在标签内部
选择器.after(html):追加显示在指定标签的外部的后面(同级标签)
选择器.before(html):追加显示在指定标签的外部的前面(同级标签)
给元素扩展属性
html页面是可拓展的标记语句,可以给指定的标签任意扩展属性,只要属性名名称不重复即可。且有扩展有两个主要目的;
使用标签保存数据:
- 如果时表单组件标签,优先使用value属性,只有value不方便使用时,使用自定义属性(随便在标签上,定义属性名和属性值);
- 如果不是表单组件标签,不推荐是使用value,推荐使用自定义属性;
定位标签:
优先考虑id属性,其次考虑name属性,只有id和name属性都不方便使用时,才考虑使用自定义属性。
dom对象转jquery对象
$("#remarkDivList").on('click',"a[name='deleteA']",function () {//收集参数( this是dom对象,要放在$()转为jquery对象 )---自定义属性let id = $(this).attr('remarkId')}
删除页面指定的元素
//刷新页面(删除元素)$("#div_"+id).remove()//从页面中移除对应元素
P135–P158还没看
P158
统计图表:以更专业,更形象的形式展示系统中的数据。
销售漏斗图:展示商品销售数据,销售业绩
—————————————-2022/7/25转去学MyBatis Plus先——————————–