共计 24894 个字符,预计需要花费 63 分钟才能阅读完成。
1.1 业务设计说明
本模块次要是实现对用户行为日志 (例如谁在什么工夫点执行了什么操作, 拜访了哪些办法, 传递的什么参数, 执行时长等) 进行记录、查问、删除等操作。其表设计语句如下:
CREATE TABLE `sys_logs` (`id` bigint(20) NOT NULL AUTO_INCREMENT,
`username` varchar(50) DEFAULT NULL COMMENT '登陆用户名',
`operation` varchar(50) DEFAULT NULL COMMENT '用户操作',
`method` varchar(200) DEFAULT NULL COMMENT '申请办法',
`params` varchar(5000) DEFAULT NULL COMMENT '申请参数',
`time` bigint(20) NOT NULL COMMENT '执行时长(毫秒)',
`ip` varchar(64) DEFAULT NULL COMMENT 'IP 地址',
`createdTime` datetime DEFAULT NULL COMMENT '日志记录时间',
PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8 COMMENT='系统日志';
1.2 原型设计说明
基于用户需要,实现动态页面 (html/css/js),通过动态页面为用户出现根本需要实现,如图 -1 所示
1.3 API 设计说明
日志业务后盾 API 分层架构及调用关系如图 -2 所示:
阐明:分层目标次要将简单问题简单化,实现各司其职,各尽所能。
2 日志治理列表页面出现
2.1 业务时序剖析
当点击首页左侧的 ” 日志治理 ” 菜单时, 其总体时序剖析如图 -3 所示:
2.2 服务端实现
2.2.1 Controller 实现
▪ 业务形容与设计实现
基于日志治理的申请业务,在 PageController 中增加 doLogUI 办法,doPageUI 办法别离用于返回日志列表页面,日志分页页面。
▪ 要害代码设计与实现
第一步:在 PageController 中定义返回日志列表的办法。代码如下:
@RequestMapping("log/log_list")
public String doLogUI() {return "sys/log_list";}
第二步:在 PageController 中定义用于返回分页页面的办法。代码如下:
@RequestMapping("doPageUI")
public String doPageUI() {return "common/page";}
2.3 客户端实现
2.3.1 日志菜单事件处理
▪ 业务形容与设计
首 先 准 备 日 志 列 表 页 面 (/templates/modules/sys/log_list.html),而后在 starter.html 页面中点击日志治理菜单时异步加载日志列表页面。
▪ 要害代码设计与实现
找到我的项目中的 starter.html 页面,页面加载实现当前,注册日志治理菜单项的点击事件,当点击日志治理时,执行事件处理函数。要害代码如下:
$(function(){doLoadUI("load-log-id","log/log_list")
})
function doLoadUI(id,url){$("#"+id).click(function(){$("#mainContentId").load(url);
});
}
其中,load 函数为 jquery 中的 ajax 异步申请函数。
2.3.2 日志列表页面事件处理
▪ 业务形容与设计实现
当日志列表页面加载实现当前异步加载分页页面(page.html)。
▪ 要害代码设计与实现:
在 log_list.html 页面中异步加载 page 页面,这样能够实现分页页面重用,哪里须要分页页面,哪里就进行页面加载即可。要害代码如下:
$(function(){$("#pageId").load("doPageUI");
});
阐明:数据加载通常是一个绝对比拟耗时操作,为了改善用户体验,能够先为用户出现一个页面,数据加载时,显示数据正在加载中, 数据加载实现当前再出现数据。这样也可满足现阶段不同类型客户端需要 (例如手机端, 电脑端, 电视端, 手表端。)
3 日志治理列表数据出现
3.1 数据架构剖析
日志查问服务端数据根本架构,如图 -4 所示。
3.2 服务端 API 架构及业务时序图剖析
服务端日志分页查问代码根本架构,如图 -5 所示:
服务端日志列表数据查问时序图,如图 -6 所示:
3.3 服务端要害业务及代码实现
3.3.1 pojo 类实现
▪ 业务形容及设计实现
构建实体对象(POJO)封装从数据库查问到的记录,一行记录映射为内存中一个的这样的对象。对象属性定义时尽量与表中字段有肯定的映射关系,并增加对应的 set/get/toString 等办法,便于对数据进行更好的操作。
▪ 要害代码剖析及实现
package com.cy.pj.sys.pojo;
import java.io.Serializable;
import java.util.Date;
public class SysLog implements Serializable {
private static final long serialVersionUID = -8799081241453681134L;
private Integer id;
// 用户名
private String username;
// 用户操作
private String operation;
// 申请办法
private String method;
// 申请参数
private String params;
// 执行时长(毫秒)
private Long time;
//IP 地址
private String ip;
// 创立工夫
private Date createdTime;
/** 设置:*/
public void setId(Integer id) {this.id = id;}
/** 获取:*/
public Integer getId() {return id;}
/** 设置:用户名 */
public void setUsername(String username) {this.username = username;}
/** 获取:用户名 */
public String getUsername() {return username;}
/** 设置:用户操作 */
public void setOperation(String operation) {this.operation = operation;}
/** 获取:用户操作 */
public String getOperation() {return operation;}
/** 设置:申请办法 */
public void setMethod(String method) {this.method = method;}
/** 获取:申请办法 */
public String getMethod() {return method;}
/** 设置:申请参数 */
public void setParams(String params) {this.params = params;}
/** 获取:申请参数 */
public String getParams() {return params;}
/** 设置:IP 地址 */
public void setIp(String ip) {this.ip = ip;}
/** 获取:IP 地址 */
public String getIp() {return ip;}
/** 设置:创立工夫 */
public void setCreateDate(Date createdTime) {this.createdTime = createdTime;}
/** 获取:创立工夫 */
public Date getCreatedTime() {return createdTime;}
public Long getTime() {return time;}
public void setTime(Long time) {this.time = time;}
}
阐明:通过此对象除了能够封装从数据库查问的数据,还能够封装客户端申请数据,实现层与层之间数据的传递。
3.3.2 Dao 接口实现
▪ 业务形容及设计实现
通过数据层对象,基于业务层参数数据查问日志记录总数以及当前页要出现的用户行为日志信息。
▪ 要害代码剖析及实现:
第一步:定义数据层接口对象,通过将此对象保障给业务层以提供日志数据拜访。代码如下:
@Mapper
public interface SysLogDao {}
第二步:在 SysLogDao 接口中增加 getRowCount 办法用于按条件统计记录总数。代码如下:
/**
* @param username 查问条件(例如查问哪个用户的日志信息)
* @return 总记录数(基于这个后果能够计算总页数)
*/
int getRowCount(@Param("username") String username);
}
第三步:在 SysLogDao 接口中增加 findPageObjects 办法,基于此办法实现当前页记录的数据查问操作。代码如下:
/**
* @param username 查问条件(例如查问哪个用户的日志信息)
* @param startIndex 当前页的起始地位
* @param pageSize 当前页的页面大小
* @return 当前页的日志记录信息
* 数据库中每条日志信息封装到一个 SysLog 对象中
*/
List<SysLog> findPageObjects(@Param("username")String username,
@Param("startIndex")Integer startIndex,
@Param("pageSize")Integer pageSize);
阐明:
1) 当 DAO 中办法参数多余一个时尽量应用 @Param 注解进行润饰并指定名字,而后在 Mapper 文件中便能够通过相似 #{username}形式进行获取,否则只能通过#{arg0},#{arg1}或者#{param1},#{param2}等形式进行获取。
2) 当 DAO 办法中的参数利用在动静 SQL 中时无论多少个参数,尽量应用 @Param 注解进行润饰并定义。
3.3.3 Mapper 文件实现
▪ 业务形容及设计实现
基于 Dao 接口创立映射文件,在此文件中通过相干元素 (例如 select) 形容要执行的数据操作。
▪ 要害代码设计及实现
第一步:在映射文件的设计目录 (mapper/sys) 中增加 SysLogMapper.xml 映射文件,
代码如下:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.cy.pj.sys.dao.SysLogDao">
</mapper>
第二步:在映射文件中增加 sql 元素实现 SQL 中的共性操作,代码如下:
<sql id="queryWhereId">
from sys_Logs
<where>
<if test="username!=null and username!=''">
username like concat("%",#{username},"%")
</if>
</where>
</sql>
第三步:在映射文件中增加 id 为 getRowCount 元素,按条件统计记录总数,代码如下:
<select id="getRowCount"resultType="int">
select count(*)
<include refid="queryWhereId"/>
</select>
第四步:在映射文件中增加 id 为 findPageObjects 元素,实现分页查问。代码如下:
<select id="findPageObjects"resultType="com.cy.pj.sys.pojo.SysLog">
select *
<include refid="queryWhereId"/>
order by createdTime desc
limit #{startIndex},#{pageSize}
</select>
第五步: 单元测试类 SysLogDaoTests,对数据层办法进行测试。
package com.cy.pj.sys.dao;
import java.util.List;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import com.cy.pj.sys.pojo.SysLog;
@SpringBootTest
public class SysLogDaoTests {
@Autowired
private SysLogDao sysLogDao;
@Test
public void testGetRowCount() {int rows=sysLogDao.getRowCount("admin");
System.out.println("rows="+rows);
}
@Test
public void testFindPageObjects() {List<SysLog> list=sysLogDao.findPageObjects("admin", 0, 3);
for(SysLog log:list) {System.out.println(log);
}
}
}
3.3.4 Service 接口及实现类
▪ 业务形容与设计实现
业务层次要是实现模块中业务逻辑的解决。在日志分页查问中,业务层对象首先要通过业务办法中的参数接管管制层数据 (例 username,pageCurrent) 并校验。而后基于用户名进行总记录数的查问并校验,再基于起始地位及页面大小进行当前页记录的查问,最初对查问后果进行封装并返回。
▪ 要害代码设计及实现
业务值对象定义,基于此对象封装数据层返回的数据以及计算的分页信息,创立在 common 工程下,方便使用,具体代码参考如下:
package com.cy.pj.common.pojo;
import lombok.Data;
import lombok.NoArgsConstructor;
import java.io.Serializable;
import java.util.List;
@Data
//@AllArgsConstructor// 含参结构
@NoArgsConstructor// 无参结构
// 次对象封装分页信息
public class PageObject<T> implements Serializable {
private static final long serialVersionUID = -8770600233998974797L;//alt+enter
// 总记录数
private Integer rowCount=0;
// 当前页记录
private List<T> records;
// 总页数
private Integer pageCount=0;
// 每页显示几条记录
private Integer pageSize=5;
// 以后页面页码值
private Integer pageCurrent=1;
public PageObject(Integer rowCount, List<T> records, Integer pageSize, Integer pageCurrent) {
this.rowCount = rowCount;
this.records = records;
this.pageSize = pageSize;
this.pageCurrent = pageCurrent;
this.pageCount=rowCount/pageSize;
if (rowCount%pageSize!=0)pageCount++;
}
}
在业务层 service 下定义日志业务接口及办法,裸露外界对日志业务数据的拜访,其代码参考如下:
package com.cy.pj.sys.service;
public interface SysLogService {
/**
* @param name 基于条件查问时的参数名
* @param pageCurrent 以后的页码值
* @return 当前页记录 + 分页信息
*/
PageObject<SysLog> findPageObjects(String username,Integer pageCurrent);
}
日志业务接口及实现类,用于具体执行日志业务数据的分页查问操作, 其代码如下:
package com.cy.pj.sys.service.impl;
@Service
public class SysLogServiceImpl implements SysLogService{
@Autowired
private SysLogDao sysLogDao;
@Override
public PageObject<SysLog> findPageObjects(String name, Integer pageCurrent) {
//1. 验证参数合法性
//1.1 验证 pageCurrent 的合法性,不非法抛出 IllegalArgumentException 异样
if(pageCurrent==null||pageCurrent<1)throw new IllegalArgumentException("以后页码不正确");
//2. 基于条件查问总记录数
//2.1) 执行查问
int rowCount=sysLogDao.getRowCount(name);
//2.2) 验证查问后果,如果后果为 0 不再执行如下操作
if(rowCount==0)throw new ServiceException("零碎没有查到对应记录");
//3. 基于条件查问当前页记录(pageSize 定义为 2)
//3.1)定义 pageSize
int pageSize=2;
//3.2)计算 startIndex
int startIndex=(pageCurrent-1)*pageSize;
//3.3)执行以后数据的查问操作
List<SysLog> records=sysLogDao.findPageObjects(name, startIndex, pageSize);
//4. 对分页信息以及当前页记录进行封装
//4.1)构建 PageObject 对象
PageObject<SysLog> pageObject=new PageObject<>();
//4.2)封装数据
pageObject.setPageCurrent(pageCurrent);
pageObject.setPageSize(pageSize);
pageObject.setRowCount(rowCount);
pageObject.setRecords(records);
pageObject.setPageCount((rowCount-1)/pageSize+1);
//5. 返回封装后果。return pageObject;
} }
在以后办法中须要的 ServiceException 是一个本人定义的异样, 通过自定义异样可更好的实现对业务问题的形容,同时能够更好的进步用户体验。创立在 common 工程下,参考代码如下:
package com.cy.pj.common.exception;
// 次对象封装业务层呈现的异样信息
public class ServiceException extends RuntimeException {public ServiceException() { }
public ServiceException(String message) {super(message);
}
public ServiceException(String message, Throwable cause) {super(message, cause);
}
public ServiceException(Throwable cause) {super(cause);
}
public ServiceException(String message, Throwable cause, boolean enableSuppression, boolean writableStackTrace) {super(message, cause, enableSuppression, writableStackTrace);
}
}
定义 Service 对象的单元测试类,代码如下:
package com.cy.pj.sys.service;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import com.cy.pj.common.pojo.PageObject;
import com.cy.pj.sys.entity.SysLog;
@SpringBootTest
public class SysLogServiceTests {
@Autowired
private SysLogService sysLogService;
@Test
public void testFindPageObjects() {PageObject<SysLog> pageObject=sysLogService.findPageObjects("admin", 1);
System.out.println(pageObject);
}
}
3.3.5 Controller 类实现
▪ 业务形容与设计实现
管制层对象次要负责申请和响应数据的解决,例如,本模块首先要通过管制层对象解决申请参数,而后通过业务层对象执行业务逻辑,再通过 pojo 对象封装响应后果 (次要对业务层数据增加状态信息),最初将响应后果转换为 JSON 格局的字符串响应到客户端。
▪ 要害代码设计与实现
在 common 工程下定义管制层值对象(pojo),目标是基于此对象封装管制层响应后果(在此对象中次要是为业务层执行后果增加状态信息)。Spring MVC 框架在响应时能够调用相干 API(例如 jackson)将其对象转换为 JSON 格局字符串。
package com.cy.pj.common.pojo;
import lombok.Data;
import java.io.Serializable;
// 次对象封装管制层响应到客户端的数据
@Data
public class JsonResult implements Serializable {
private static final long serialVersionUID = -8722122492343039602L;
/** 响应状态码(有的人用 code)*/
private Integer state=1;
/** 状态码对应的信息 */
private String message="ok";
/** 正确的响应数据 */
private Object data;
public JsonResult(){}
public JsonResult(String message){this.message=message;}
public JsonResult(Object data){this.data=data;}
public JsonResult(Throwable e){
this.state=0;
this.message=e.getMessage();}
}
定义 Controller 类,并将此类对象应用 Spring 框架中的 @Controller 注解进行标识,示意此类对象要交给 Spring 治理。而后基于 @RequestMapping 注解为此类定义根门路映射。代码参考如下:
package com.cy.pj.sys.controller;
@Controller
@RequestMapping("/log/")
public class SysLogController {
@Autowired
private SysLogService sysLogService;
在 Controller 类中增加分页申请解决办法,代码参考如下:@RequestMapping("doFindPageObjects")
@ResponseBody
public JsonResult doFindPageObjects(String username,Integer pageCurrent){PageObject<SysLog> pageObject=sysLogService.findPageObjects(username,pageCurrent);
return new JsonResult(pageObject);
}
}
定义全局异样解决类,对管制层可能呈现的异样,进行对立异样解决,代码如下:
package com.cy.pj.common.web;
import com.cy.pj.common.pojo.JsonResult;
import lombok.extern.slf4j.Slf4j;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.RestControllerAdvice;
@RestControllerAdvice
@Slf4j
public class GlobalExceptionHandle {@ExceptionHandler(RuntimeException.class)
public JsonResult doHandleRuntimeException(RuntimeException e){e.printStackTrace();
log.error("e.message {}",e.getMessage());
return new JsonResult();}
}
管制层响应数据处理剖析,如图 -7 所示:
3.4 客户端要害业务及代码实现
3.4.1 客户端页面事件剖析
当用户点击首页日志治理时,其页面流转剖析如图 -8 所示:
3.4.2 日志列表信息出现
▪ 业务形容与设计实现
日志分页页面加载实现当前,向服务端发动异步申请加载日志信息,当日志信息加载
实现须要将日志信息、分页信息出现到列表页面上。
▪ 要害代码设计与实现
第一步:分页页面加载实现,向服务端发动异步申请,代码参考如下:
$(function(){
// 为什么要将 doGetObjects 函数写到 load 函数对应的回调外部。$("#pageId").load("doPageUI",function(){doGetObjects();
});
});
第二步:定义异步申请处理函数,代码参考如下:
function doGetObjects(){
//debugger;// 断点调试
//1. 定义 url 和参数
var url="log/doFindPageObjects"
var params={"pageCurrent":1};//pageCurrent=2
//2. 发动异步申请
// 请问如下 ajax 申请的回调函数参数名能够是任意吗?// 能够, 必须合乎标识
符的标准
$.getJSON(url,params,function(result){
// 请问 result 是一个字符串还是 json 格局的 js 对象?对象
doHandleQueryResponseResult(result);
}
);// 非凡的 ajax 函数
}
result 后果对象剖析,如图 -9 所示:
第三步:定义回调函数,解决服务端的响应后果。代码如下:
function doHandleQueryResponseResult (result){ //JsonResult
if(result.state==1){//ok
// 更新 table 中 tbody 外部的数据
doSetTableBodyRows(result.data.records);// 将数据出现在页面上
// 更新页面 page.html 分页数据
//doSetPagination(result.data); // 此办法写到 page.html 中
}else{alert(result.message);
}
}
第四步:将异步响应后果出现在 table 的 tbody 地位。代码参考如下:
function doSetTableBodyRows(records){
//1. 获取 tbody 对象,并清空对象
var tBody=$("#tbodyId");
tBody.empty();
//2. 迭代 records 记录,并将其内容追加到 tbody
for(var i in records){
//2.1 构建 tr 对象
var tr=$("<tr></tr>");
//2.2 构建 tds 对象
var tds=doCreateTds(records[i]);
//2.3 将 tds 追加到 tr 中
tr.append(tds);
//2.4 将 tr 追加到 tbody 中
tBody.append(tr);
}
}
第五步:创立每行中的 td 元素,并填充具体业务数据。代码参考如下:
function doCreateTds(data){
var tds="<td><input type='checkbox'class='cBox'name='cItem'value='"+data.id+"'></td>"+
"<td>"+data.username+"</td>"+
"<td>"+data.operation+"</td>"+
"<td>"+data.method+"</td>"+
"<td>"+data.params+"</td>"+
"<td>"+data.ip+"</td>"+
"<td>"+data.time+"</td>";
return tds;
}
3.4.3 分页数据信息出现
▪ 业务形容与设计实现
日志信息列表初始化实现当前初始化分页数据(调用 setPagination 函数),而后再点击上一页,下一页等操作时,更新页码值,执行基于以后页码值的查问。
▪ 要害代码设计与实现:
第一步:在 page.html 页面中定义 doSetPagination 办法(实现分页数据初始化),代码如下:
function doSetPagination(page){
//1. 始化数据
$(".rowCount").html("总记录数("+page.rowCount+")");
$(".pageCount").html("总页数("+page.pageCount+")");
$(".pageCurrent").html("当前页("+page.pageCurrent+")");
//2. 绑定数据(为后续对此数据的应用提供服务)
$("#pageId").data("pageCurrent",page.pageCurrent);
$("#pageId").data("pageCount",page.pageCount);
}
第二步:分页页面 page.html 中注册点击事件。代码如下:
$(function(){
// 事件注册
$("#pageId").on("click",".first,.pre,.next,.last",doJumpToPage);
})
第三步:定义 doJumpToPage 办法(通过此办法实现以后数据查问)
function doJumpToPage(){
//1. 获取点击对象的 class 值
var cls=$(this).prop("class");//Property
//2. 基于点击的对象执行 pageCurrent 值的批改
//2.1 获取 pageCurrent,pageCount 的以后值
var pageCurrent=$("#pageId").data("pageCurrent");
var pageCount=$("#pageId").data("pageCount");
//2.2 批改 pageCurrent 的值
if(cls=="first"){// 首页
pageCurrent=1;
}else if(cls=="pre"&&pageCurrent>1){// 上一页
pageCurrent--;
}else if(cls=="next"&&pageCurrent<pageCount){// 下一页
pageCurrent++;
}else if(cls=="last"){// 最初一页
pageCurrent=pageCount;
}else{return;}
//3. 对 pageCurrent 值进行从新绑定
$("#pageId").data("pageCurrent",pageCurrent);
//4. 基于新的 pageCurrent 的值进行当前页数据查问
doGetObjects();}
批改分页查询方法:
function doGetObjects(){
//debugger;// 断点调试
//1. 定义 url 和参数
var url="log/doFindPageObjects"
//? 请问 data 函数的含意是什么?(从指定元素上获取绑定的数据)
// 此数据会在何时进行绑定?(setPagination,doQueryObjects)
**var pageCurrent=$("#pageId").data("pageCurrent");
// 为什么要执行如下语句的断定,而后初始化 pageCurrent 的值为 1
//pageCurrent 参数在没有赋值的状况下,默认初始值应该为 1.
if(!pageCurrent) pageCurrent=1;
var params={"pageCurrent":pageCurrent};//pageCurrent=2**
//2. 发动异步申请
// 请问如下 ajax 申请的回调函数参数名能够是任意吗?能够, 必须合乎标识符
的标准
$.getJSON(url,params,function(result){
// 请问 result 是一个字符串还是 json 格局的 js 对象?对象
doHandleQueryResponseResult(result);
}
);// 非凡的 ajax 函数 }
3.4.4 列表页面信息查问实现
▪ 业务形容及设计
当用户点击日志列表的查问按钮时,基于用户输出的用户名进行有条件的分页查问,并将查问后果出现在页面。
▪ 要害代码设计与实现:
第一步:日志列表页面加载实现,在查问按钮上进行事件注册。代码如下:
$(".input-group-btn").on("click",".btn-search",doQueryObjects)
第二步:定义查问按钮对应的点击事件处理函数。代码如下:
function doQueryObjects(){
// 为什么要在此地位初始化 pageCurrent 的值为 1?
// 数据查问时页码的初始地位也应该是第一页
$("#pageId").data("pageCurrent",1);
// 为什么要调用 doGetObjects 函数?// 重用 js 代码,简化 jS 代码编写。doGetObjects();}
第三步:在分页查问函数中追加 name 参数定义,代码如下:
function doGetObjects(){
//debugger;// 断点调试
//1. 定义 url 和参数
var url="log/doFindPageObjects"
//? 请问 data 函数的含意是什么?(从指定元素上获取绑定的数据)
// 此数据会在何时进行绑定?(setPagination,doQueryObjects)
var pageCurrent=$("#pageId").data("pageCurrent");
// 为什么要执行如下语句的断定,而后初始化 pageCurrent 的值为 1
//pageCurrent 参数在没有赋值的状况下,默认初始值应该为 1.
if(!pageCurrent) pageCurrent=1;
var params={"pageCurrent":pageCurrent};
// 为什么此地位要获取查问参数的值?
// 一种冗余的利用办法,目标时让此函数在查问时能够重用。var username=$("#searchNameId").val();
// 如下语句的含意是什么?动静在 json 格局的 js 对象中增加 key/value,
if(username) params.username=username;// 查问时须要
//2. 发动异步申请
// 请问如下 ajax 申请的回调函数参数名能够是任意吗?能够, 必须合乎标识符
的标准
$.getJSON(url,params,function(result){
// 请问 result 是一个字符串还是 json 格局的 js 对象?对象
doHandleQueryResponseResult(result);
}
);
}
4 日志治理删除操作实现
4.1 数据架构剖析
当用户执行日志删除操作时,客户端与服务端交互时的根本数据架构,如图 -10 所示。
4.2 删除业务时序剖析
客户端提交删除申请,服务端对象的工作时序剖析,如图 -11 所示。
4.3 服务端要害业务及代码实现 Dao 接口实现
▪ 业务形容及设计实现
数据层基于业务层提交的日志记录 id,进行日志删除操作。
▪ 要害代码设计及实现:
在 SysLogDao 中增加基于 id 执行日志删除的办法。代码参考如下:
int deleteObjects(@Param("ids")Integer ids);
4.3.2 Mapper 文件实现
▪ 业务形容及设计实现
在 SysLogDao 接口对应的映射文件中增加用于执行删除业务的 delete 元素,此元素外部定义具体的 SQL 实现。
▪ 要害代码设计与实现
在 SysLogMapper.xml 文件增加 delete 元素,要害代码如下:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.cy.pj.sys.dao.SysLogDao">
<delete id="deleteObjects">
delete from sys_Logs
where id in
<foreach collection="ids"open="("close=")"separator=","item="id">
#{id}
</foreach>
</delete>
</mapper>
FAQ 剖析: 如上 SQL 实现可能会存在什么问题?(可靠性问题, 性能问题)从可靠性的角度剖析, 如果 ids 的值为 null 或长度为 0 时,SQL 构建可能会呈现语法问题,可参考如下代码进行改良(先对 ids 的值进行断定):
<delete id="deleteObjects">
delete from sys_logs
<if test="ids!=null and ids.length>0">
where id in
<foreach collection="ids"open="("close=")"separator=","item="id">
#{id}
</foreach>
</if>
<if test="ids==null or ids.length==0">
where 1=2
</if>
</delete>
从 SQL 执行性能角度剖析, 个别在 SQL 语句中不倡议应用 in 表达式, 能够参考如下代码进行实现(重点是 forearch 中 or 运算符的利用):
<delete id="deleteObjects">
delete from sys_logs
<choose>
<when test="ids!=null and ids.length>0">
<where>
<foreach collection="ids"item="id"separator="or">id=#{id}
</foreach>
</where>
</when>
<otherwise>where 1=2
</otherwise>
</choose>
</delete>
阐明: 这里的 choose 元素也为一种抉择构造,when 元素相当于 if,otherwise 相当于 else 的语法。
4.3.3 Service 接口及实现类
▪ 业务形容与设计实现
在日志业务层定义用于执行删除业务的办法,首先通过办法参数接管管制层传递的多个记录的 id,并对参数 id 进行校验。而后基于日志记录 id 执行删除业务实现。最初返回业务执行后果。
▪ 要害代码设计与实现
第一步:在 SysLogService 接口中,增加基于多个 id 进行日志删除的办法。要害代码如下:
int deleteObjects(Integer ids) {}
第二步:在 SysLogServiceImpl 实现类中增加删除业务的具体实现。要害代码如下:
@Override
public int deleteObjects(Integer… ids) {
//1. 断定参数合法性
if(ids==null||ids.length==0)throw new IllegalArgumentException("请抉择一个");
//2. 执行删除操作
int rows;
try{rows=sysLogDao.deleteObjects(ids);
}catch(Throwable e){e.printStackTrace();
// 发出报警信息(例如给运维人员发短信)
throw new ServiceException("系统故障,正在复原中...");
}
//4. 对后果进行验证
if(rows==0)throw new ServiceException("记录可能曾经不存在");
//5. 返回后果
return rows; }
4.3.4 Controller 类实现
▪ 业务形容与设计实现
在日志管制层对象中,增加用于解决日志删除申请的办法。首先在此办法中通过形参接管客户端提交的数据,而后调用业务层对象执行删除操作,最初封装执行后果,并在运行时将响应对象转换为 JSON 格局的字符串,响应到客户端。
▪ 要害代码设计与实现
第一步:在 SysLogController 中增加用于执行删除业务的办法。代码如下:
@RequestMapping("doDeleteObjects")
@ResponseBody
public JsonResult doDeleteObjects(Integer ids){sysLogService.deleteObjects(ids);
return new JsonResult("删除胜利");
}
第二步:启动 tomcat 进行拜访测试,关上浏览器输出如下网址:
http://localhost/log/doDeleteObjects?ids=1,2,3
4.4 客户端要害业务及代码实现
4.4.1 日志列表页面事件处理
▪ 业务形容及设计实现
用户在页面上首先抉择要删除的元素,而后点击删除按钮,将用户抉择的记录 id 异步提交到服务端,最初在服务端执行日志的删除动作。
▪ 要害代码设计与实现
第一步:页面加载实现当前,在删除按钮上进行点击事件注册。要害代码如下:
...
$(".input-group-btn")
.on("click",".btn-delete",doDeleteObjects)
...
第二步:定义删除操作对应的事件处理函数。要害代码如下:
function doDeleteObjects(){
//1. 获取选中的 id 值
var ids=doGetCheckedIds();
if(ids.length==0){alert("至多抉择一个");
return;
}
//2. 发异步申请执行删除操作
var url="log/doDeleteObjects";
var params={"ids":ids.toString()};
console.log(params);
$.post(url,params,function(result){if(result.state==1){alert(result.message);
doGetObjects();}else{alert(result.message);
}
});
}
第三步:定义获取用户选中的记录 id 的函数。要害代码如下:
function doGetCheckedIds(){
// 定义一个数组, 用于存储选中的 checkbox 的 id 值
var array=[];//new Array();
// 获取 tbody 中所有类型为 checkbox 的 input 元素
$("#tbodyId input[type=checkbox]").
// 迭代这些元素,每发现一个元素都会执行如下回调函数
each(function(){
// 假如此元素的 checked 属性的值为 true
if($(this).prop("checked")){
// 调用数组对象的 push 办法将选中对象的值存储到数组
array.push($(this).val());
}
});
return array;
}
第四步:Thead 中全选元素的状态影响 tbody 中 checkbox 对象状态。代码如下:
function doChangeTBodyCheckBoxState(){
//1. 获取以后点击对象的 checked 属性的值
var flag=$(this).prop("checked");//true or false
//2. 将 tbody 中所有 checkbox 元素的值都批改为 flag 对应的值。// 第一种计划
/* $("#tbodyId input[name='cItem']")
.each(function(){$(this).prop("checked",flag);
}); */
// 第二种计划
$("#tbodyId input[type='checkbox']")
.prop("checked",flag);
}
第五步:Tbody 中 checkbox 的状态影响 thead 中全选元素的状态。代码如下:
function doChangeTHeadCheckBoxState(){
//1. 设定默认状态值
var flag=true;
//2. 迭代所有 tbody 中的 checkbox 值并进行与操作
$("#tbodyId input[type='checkbox']")
.each(function(){flag=flag&$(this).prop("checked")
});
//3. 批改全选元素 checkbox 的值为 flag
$("#checkAll").prop("checked",flag);
}
第六步:欠缺业务刷新办法,当在最初一页执行删除操作时,基于全选按钮状态及以后页码值,刷新页面。要害代码如下:
function doRefreshAfterDeleteOK(){var pageCount=$("#pageId").data("pageCount");
var pageCurrent=$("#pageId").data("pageCurrent");
var checked=$("#checkAll").prop("checked");
if(pageCurrent==pageCount&&checked&&pageCurrent>1){
pageCurrent--;
$("#pageId").data("pageCurrent",pageCurrent);
}
doGetObjects();}
阐明:最初将如上办法增加在删除操作胜利当前的代码块中。
5 日志治理数据增加实现
5.1 服务端要害业务及代码实现
这块业务学了 AOP 当前再实现.
5.1.1 Dao 接口实现
▪ 业务形容与设计实现
数据层基于业务层的长久化申请,将业务层提交的用户行为日志信息写入到数据库。
▪ 要害代码设计与实现
在 SysLogDao 接口中增加用于实现日志信息长久化的办法。要害代码如下:
int insertObject(SysLog entity);
5.1.2 Mapper 映射文件
▪ 业务形容与设计实现
基于 SysLogDao 中办法的定义,编写用于数据长久化的 SQL 元素。
▪ 要害代码设计与实现
在 SysLogMapper.xml 中增加 insertObject 元素,用于向日志表写入用户行为日志。要害代码如下:
<insert id="insertObject">
insert into sys_logs
(username,operation,method,params,time,ip,createdTime)
values
(#{username},#{operation},#{method},#{params},#{time},#{ip},#{createdTime})
</insert>
5.1.3 Service 接口及实现类
▪ 业务形容与设计实现
将日志切面中抓取到的用户行为日志信息,通过业务层对象办法长久化到数据库。
▪ 要害代码实现
第一步:在 SysLogService 接口中,增加保留日志信息的办法。要害代码如下:
void saveObject(SysLog entity);
第二步:在 SysLogServiceImpl 类中增加,保留日志的办法实现。要害代码如下:
@Override
public void saveObject(SysLog entity) {sysLogDao.insertObject(entity);
}
5.1.4 日志切面 Aspect 实现
▪ 业务形容与设计实现
在日志切面中,抓取用户行为信息,并将其封装到日志对象而后传递到业务,通过业务层对象对日志日志信息做进一步解决。此局部内容后续联合 AOP 进行实现(临时先理解,不做具体实现)。
▪ 要害代码设计与实现
springboot 工程中利用 AOP 时, 首先要增加如下依赖(如果有则无需增加):
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
</dependency>
定义日志切面类对象,通过盘绕告诉解决日志记录操作。要害代码如下:
package com.cy.pj.common.aspect;
import lombok.extern.slf4j.Slf4j;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.stereotype.Component;
/**@Aspect 注解用于定义切面对象 */
@Aspect
@Component
@Slf4j
public class SysLogAspect {
/**
* 切入点的定义
* 1)应用 @Pointcut 注解进行切入点的形容
* 2)应用 bean 表达式定义切入点, 语法 bean(spring 容器中治理的某个 bean 的名字)
* bean 表达式是一种粗粒度的切入点表达式(不能具体到 bean 中哪个办法)
*/ //@Pointcut("bean(categoryServiceImpl)")
@Pointcut("bean(*ServiceImpl)")
public void doLog() {}// 这里的 doLog()办法, 办法体内不须要写任何内容, 作用是承载切入点
/**
* 在切入点对应的指标办法执行时, 要进行的动作能够以如下形式进行定义
* @param joinPoint 封装了切入点汇合办法中的某个正在执行的指标办法,
* ProceedingJoinPoint 类型的连接点只能利用在 @Around 注解形容的办法参数中
* @return 为指标办法的返回后果
*/
@Around("doLog()") //@Around 注解形容在办法能够在指标办法执行之前, 之后做一些业务拓展
public Object doAround(ProceedingJoinPoint joinPoint) throws Throwable {long t1 = System.currentTimeMillis();
Object result = joinPoint.proceed();// 去调用指标办法
long t2 = System.currentTimeMillis();
System.out.println("execute time" + (t2 - t1));
return result;// 指标办法的执行后果
}
}
办法中用到的 ip 地址获取须要提供一个如下的工具类:(不必本人实现,间接用)
public class IPUtils {
private static Logger logger =
LoggerFactory.getLogger(IPUtils.class);
public static String getIpAddr() {HttpServletRequest request = ((ServletRequestAttributes)
RequestContextHolder.getRequestAttributes()).getRequest();
String ip = null;
try {ip = request.getHeader("x-forwarded-for");
if (StringUtils.isEmpty(ip) ||
"unknown".equalsIgnoreCase(ip)) {ip = request.getHeader("Proxy-Client-IP");
}
if (StringUtils.isEmpty(ip) || ip.length() == 0 ||
"unknown".equalsIgnoreCase(ip)) {ip = request.getHeader("WL-Proxy-Client-IP");
}
if (StringUtils.isEmpty(ip) ||
"unknown".equalsIgnoreCase(ip)) {ip = request.getHeader("HTTP_CLIENT_IP");
}
if (StringUtils.isEmpty(ip) ||
"unknown".equalsIgnoreCase(ip)) {
ip =
request.getHeader("HTTP_X_FORWARDED_FOR");
}
if (StringUtils.isEmpty(ip) ||
"unknown".equalsIgnoreCase(ip)) {ip = request.getRemoteAddr();
}
} catch (Exception e) {logger.error("IPUtils ERROR", e);
}
return ip; }}
原理剖析,如图 -12 所示:
6 总结
6.1 重难点剖析
▪ 日志治理整体业务剖析与实现。
1) 分层架构(应用层 MVC: 基于 spring 的 mvc 模块)。
2) API 架构(SysLogDao,SysLogService,SysLogController)。
3) 业务架构(查问, 删除, 增加用户行为日志)。
4) 数据架构(SysLog,PageObject,JsonResult,..)。
▪ 日志治理模块数据查问操作中的数据封装。
1) 数据层(数据逻辑) 的 SysLog 对象利用 (一行记录一个 log 对象)。
2) 业务层(业务逻辑)PageObject 对象利用(封装每页记录以及对应的分页信息)。
3) 管制层(管制逻辑) 的 JsonResult 对象利用 (对业务数据增加状态信息)。
▪ 日志治理模块数据查问操作中的数据封装。
1) 数据层(数据逻辑) 的 SysLog 对象利用 (一行记录一个 log 对象)。
2) 业务层(业务逻辑)PageObject 对象利用(封装每页记录以及对应的分页信息)。
3) 管制层(管制逻辑) 的 JsonResult 对象利用 (对业务数据增加状态信息)。
▪ 日志管理控制层申请数据映射,响应数据的封装及转换(转换为 json 串)。
1) 申请门路映射, 申请形式映射(GET,POST), 申请参数映射(间接量,POJO)。
2) 响应数据两种(页面,JSON 串)。
▪ 日志治理模块异样解决如何实现的。
1) 申请解决层(管制层) 定义对立 (全局) 异样解决类。
2) 应用注解 @RestControllerAdvice 形容类, 应用 @ExceptionHandler 形容办法.
3) 异样解决规定: 能解决则解决, 不能解决则抛出。
6.2 FAQ 剖析
▪ 用户行为日志表中都有哪些字段?(面试时有时会问)
▪ 用户行为日志是如何实现分页查问的?(limit)
▪ 用户行为数据的封装过程?(数据层,业务层,管制层)
▪ 我的项目中的异样是如何解决的?
▪ 页面中数据乱码, 如何解决?(数据起源,申请数据,响应数据)
▪ 说说的日志删除业务是如何实现?
▪ Spring MVC 响应数据处理?(view,json)
▪ 我的项目你罕用的 JS 函数说几个?(data,prop,ajax,each,..)
▪ MyBatis 中的 @Params 注解的作用?(为参数变量指定其其别名)
▪ Jquery 中 data 函数用于做什么?能够借助 data 函数将数据绑定到指定对象,语法为 data(key[,value]),key 和 value 为本人业务中的任意数据,如果只有 key 示意取值。
▪ Jquery 中的 prop 函数用于获取 html 标签对象中”规范属性”的值或为属性赋值,其语法为 prop(propertyName[,propertyValue]), 如果只有属性名则为获取属性值。
▪ Jquery 中 attr 函数为用户获取 html 标签中任意属性值或为属性赋值的一个办法,其语法为 attr(propertyName[,propertyValue]),如果只有属性名则为获取属性值。
▪ 日志写操作事务的流传个性如何配置?(每次开启新事务, 没学就临时搁置)?
▪ 日志写操作为什么应该是异步的?(用户体验会更好, 不会阻塞用户失常业务)
▪ Spring 中的异步操作如何实现?,(本人间接创立线程或者借助池中线程)
▪ Spring 中的 @Async 如何利用?(没学就临时搁置)
▪ 我的项目中的 BUG 剖析及解决套路?(排除法,打桩(log),断点, 搜索引擎)
-
BUG 剖析
- 无奈找到对应的 Bean 对象(NoSuchBeanDefinitionException),如图 -13 所示:
问题剖析:
- 检测 key 的名字写的是否正确。
- 检测 spring 对此 Bean 对象的扫描,对于 dao 而言。
- 应用有 @Mapper 注解形容或者在 @MapperScan 扫描范畴之内。
- 以上都正确,要检测是否编译了。
- 绑定异样(BindingException),如图 -14 所示。
问题剖析:
- 接口的类全名与对应的映射文件命名空间不同。
- 接口的办法名与对应的映射文件元素名不存在。
- 检测映射文件的门路与 application.properties 或者 application.yml 中的配置是否统一。
- 以上都没有问题时,检测你的类和映射文件是否失常编译。
- 反射异样(ReflectionException),如图 -15 所示
问题剖析:
- 映射文件中动静 sql 中应用的参数在接口办法中没有应用 @Param 注解润饰
- 如果应用了注解润饰还要检测名字是否统一。
阐明:当动静 sql 的参数在接口中没有应用 @Param 注解润饰,还能够借助_parameter 这个变量获取参数的值(mybatis 中的一种标准)。
- 后果映射异样,如图 -16 所示:
问题剖析:getRowCount 元素可能没有写 resultType 或 resultMap。
- 绑定异样,如图 -17 所示:
问题剖析: 绑定异样, 检测 findPageObjects 办法参数与映射文件参数名字是否匹配或者如果版本不是最新版本须要应用 @Param 注解形容。
- Bean 创立异样,如图 -18 所示
问题剖析:应该是查问时的后果映射对的类全名写错了。
- 申请形式不匹配,如图 -19 示
问题剖析:申请形式与管制层解决形式不匹配。
- 响应后果异样,如图 -20 所示:
问题剖析:服务端响应数据不正确,例如服务端没有注册将对象转换为 JSON 串的 Bean
- 申请参数异样,如图 -21 示:
问题剖析:客户端申请参数中不蕴含服务端管制层办法参数或格局不匹配。
- JS 编写谬误,如图 -22 所示:
问题剖析:点击右侧 VM176:64 地位进行查看。
- JS 编写谬误,如图 -23 所示:
问题剖析:找到对应地位,检测 data 的值以及数据起源。
- JS 编写谬误,如图 -24 所示:
问题剖析:找到对应地位,如果无奈确定地位,可排除法或打桩,debug 剖析。
- JS 写谬误,如图 -25 所示:
问题剖析:调用 length 办法的对象有问题,可先检测下对象的值。
- JS 编写谬误,如图 -26 所示:
问题剖析:检测 record 定义或赋值的中央。
- 资源没找到,如图 -27 所示:
问题剖析: 服务端资源没找到, 查看 url 和 controller 映射, 不要点击图中的 jquery。
- 视图解析异样,如图 -28 所示:
问题剖析: 查看服务端要拜访的办法上是否有 @ResponseBody 注解.