• 前言

  最近开发了一个项目,由于项目在整个开发过程中处于赶时间状态(每个项目都差不多如此)所以项目在收尾阶段发现缺少记录系统日志功能,以前系统都是直接写在每个模块的代码中,然后存入表单,在页面可以查看部分日志。这个系统并没有此要求,因此便想到了做一个系统通用的日志记录,主要记录数据有:操作时间、操作对象、操作方法、操作的一些参数以及操作结果。刚开始直接想到的是利用aspect实现aop记录日志,但实际应用中发现aspect并不能友好的拦截action,而是主要用作拦截service层业务。由于系统框架是基于ssh框架的,所以最终考虑使用struts2的自带拦截器Interceptor。

  • 什么拦截器?

  拦截器是动态拦截Action调用的对象。它提供了一种机制可以使开发者可以定义在一个action执行的前后执行的代码,也可以在一个action执行前阻止其执行。同时也是提供了一种可以提取action中可重用的部分的方式。谈到拦截器,还有一个词大家应该知道——拦截器链(Interceptor Chain,在Struts 2中称为拦截器栈Interceptor Stack)。拦截器链就是将拦截器按一定的顺序联结成一条链。在访问被拦截的方法或字段时,拦截器链中的拦截器就会按其之前定义的顺序被调用。

  •  拦截器的原理和大致流程

  1.ActionInvocation初始化时,根据配置,加载Action相关的所有Interc eptor。

   2. 通过ActionInvocation.invoke方法调用Action实现时,执行Interceptor。

Interceptor将很多功能从我们的Action中独立出来,大量减少了我们Action的代码,独立出来的行为具有很好的重用性。XWork、WebWork的许多功能都是有Interceptor实现,可以在配置文件中组装Action用到的Interceptor,它会按照你指定的顺序,在Action执行前后运行。

  • 项目实现拦截器代码
 package com.mnsn.utils;

 import java.io.BufferedWriter;
import java.io.File;
import java.io.FileWriter;
import java.util.Calendar;
import java.util.Map;
import java.util.Set; import org.apache.struts2.ServletActionContext; import com.mnsn.project.user.User;
import com.opensymphony.xwork2.ActionInvocation;
import com.opensymphony.xwork2.interceptor.Interceptor;
import com.opensymphony.xwork2.interceptor.PreResultListener; /**
* 系统日志拦截器
* @AUTHER LiuLonglong
* @Motto Goals determine what you are going to be.
* @URL http://www.cnblogs.com/mvilplss/
* @Time 下午04:09:37
* @Version
*/
public class Dolog implements Interceptor { private static final long serialVersionUID = 1L; public String intercept(ActionInvocation ai) throws Exception { ai.addPreResultListener(new PreResultListener() { public void beforeResult(ActionInvocation ai, String arg1) {
try {
StringBuffer sb = new StringBuffer();
sb.append(MyUtils.getDataYmdhms2() + ":");
Map<String, Object> session = ai.getInvocationContext().getSession();
User user = (User) session.get("loginUser");
if (user != null) {
sb.append("操作人:" + user.getName());
} else {
sb.append("操作人:系统未获取");
}
sb.append("类名:" + ai.getAction() + " ");
sb.append("方法名:" + ai.getInvocationContext().getName()+ " ");
Map<String, Object> map = ai.getInvocationContext().getParameters();
Set<String> keys = map.keySet();
sb.append("参数:");
for (String key : keys) {
sb.append(key + "=" + ((Object[]) map.get(key))[0]+ "#");
}
sb.append(" ");
sb.append("执行结果:" + ai.getResultCode() + " ");
String appPath = ServletActionContext.getServletContext().getRealPath("/");
saveLog(appPath + "operLog", sb.toString());
} catch (Exception e) {
e.printStackTrace();
} }
}); return ai.invoke();
} public static void saveLog(String dir, String content) {
try {
File path = new File(dir);
if (!path.exists()) {
path.mkdir();
}
File LogDir = new File(path + "/"
+ (Calendar.getInstance().get(Calendar.MONTH) + 1));
if (!LogDir.exists()) {
LogDir.mkdir();
}
File file = new File(LogDir + "/"
+ Calendar.getInstance().get(Calendar.DAY_OF_MONTH)
+ ".log");
if (!file.exists()) {
file.createNewFile();
}
BufferedWriter br = new BufferedWriter(new FileWriter(file, true));
br.write(content);
br.newLine();
br.flush();
br.close(); File LogDirOld = new File(
path
+ "/"
+ (Calendar.getInstance().get(Calendar.MONTH) - 2 > 0 ? (Calendar
.getInstance().get(Calendar.MONTH) - 2)
: Calendar.getInstance()
.get(Calendar.MONTH) + 10));
if (LogDirOld.exists()) {
File[] fileOlds = LogDirOld.listFiles();
for (File f : fileOlds) {
f.delete();
}
LogDirOld.delete();
}
} catch (Exception e) {
e.printStackTrace();
} } public void destroy() { } public void init() { }
}
  • 项目部分配置
<!-- 配置拦截器 -->
<interceptors>
<interceptor name="dolog" class="com.mnsn.utils.Dolog"></interceptor>
<interceptor-stack name="defaultStack">
<interceptor-ref name="dolog"></interceptor-ref>
<interceptor-ref name="defaultStack"></interceptor-ref>
</interceptor-stack>
</interceptors>
  • 执行结果

取自于文件:C:\Program Files (x86)\tomcat-6.0.43-myeclipse\webapps\qzdl\operLog\3\10.log

20150310173309:操作人:系统未获取类名:com.mnsn.project.user.UserAction@b2c940 方法名:login 参数:loginname=ss#password=ss#x=31#y=11# 执行结果:pass
20150310173313:操作人:钦州老大类名:com.mnsn.project.user.UserAction@1a2157e 方法名:list 参数: 执行结果:list
20150310173322:操作人:钦州老大类名:com.mnsn.project.user.UserAction@108a2d1 方法名:delete 参数:users[0].id=402881ec4b38cf8f014b38d4510f0001# 执行结果:toList
20150310173323:操作人:钦州老大类名:com.mnsn.project.user.UserAction@13e3101 方法名:list 参数: 执行结果:list
20150310173326:操作人:钦州老大类名:com.mnsn.project.user.UserAction@158a9b 方法名:toOper 参数:user.id=402881fd4becf4d5014becf8253c0000# 执行结果:oper
20150310173328:操作人:钦州老大类名:com.mnsn.project.user.UserAction@174edfd 方法名:update 参数:toWhere=#user.createtime=2015-03-06 10:44:20.0#user.group.id=402881ef4a9f4536014a9f69be6c0005#user.id=402881fd4becf4d5014becf8253c0000#user.job=部门主管#user.loginname=wml#user.mobilephone=18758010019#user.name=王明路#user.office_telephone=0571-6856156#user.password=123456#user.remark=此人为部门管理员#user.unit.id=402881fd4becf4d5014bed053ce80001# 执行结果:toList
20150310173328:操作人:钦州老大类名:com.mnsn.project.user.UserAction@11efcc2 方法名:list 参数: 执行结果:list
  • 总结

  struts2的拦截器相对aspect来说还是比较容易上手和理解的,对于要求一般的系统操作日志可以采用拦截器,而aspect重要用于细粒度的控制方法的出口和入口,实现逻辑层的增强,例如声明式事物。

  • 备注

  本文在工作繁忙中码出来的,如有错误在所难免,望读者指出,谢谢!

Struts2拦截器记录系统操作日志的更多相关文章

  1. 运用Unity结合PolicyInjection实现拦截器[结合操作日志实例]

    上一篇文章我们通过Unity自身Unity.InterceptionExtension.IInterceptionBehavior实现一个有系统关异常日志记录:解决代码中到处充满的异常记录的代码: 本 ...

  2. 防止未登录用户操作—struts2拦截器简单实现(转)

    原文地址:http://blog.csdn.net/zhutulang/article/details/38351629 尊重原创,请访问原地址 一般,我们的web应用都是只有在用户登录之后才允许操作 ...

  3. 从struts2拦截器到自定义拦截器

    拦截器可谓struts2的核心了,最基本的bean的注入就是通过默认的拦截器实现的,一般在struts2.xml的配置中,package内直接或间接继承了struts-default.xml,这样st ...

  4. struts2 拦截器

    拦截器:对Action的访问.可以拦截到Action中某个方法.与过滤器不同,过滤器过滤的是请求.过滤JSP.html.但是拦截器不能拦截jsp.html的访问. Struts2 拦截器在访问某个 A ...

  5. Struts2拦截器的使用 (详解)

    Struts2拦截器的使用 (详解) 如何使用struts2拦截器,或者自定义拦截器.特别注意,在使用拦截器的时候,在Action里面必须最后一定要引用struts2自带的拦截器缺省堆栈default ...

  6. 基于SSH2框架Struts2拦截器的登录验证实现(转)

        大象在这里假设你已经弄清楚了Struts2拦截器的基本概念,可以进入实际运用了.那么我们在之前的基础上只需要做下小小的改变,就可以使用Struts2的拦截器机制实现登录的验证.     修改数 ...

  7. Struts2拦截器配置实例

    拦截器介绍 拦截器 的使用 ,源自Spring AOP(面向切面编程)思想 拦截器 采用 责任链 模式 * 在责任链模式里,很多对象由每一个对象对其下家的引用而连接起来形成一条链. * 责任链每一个节 ...

  8. struts2拦截器interceptor的配置方法及使用

    转: struts2拦截器interceptor的配置方法及使用 (2015-11-09 10:22:28) 转载▼ 标签: it 365 分类: Struts2  NormalText Code  ...

  9. Struts2拦截器详解

    一.Struts2拦截器原理: Struts2拦截器的实现原理相对简单,当请求struts2的action时,Struts 2会查找配置文件,并根据其配置实例化相对的    拦截器对象,然后串成一个列 ...

随机推荐

  1. C#设置richtextbox某一段文本颜色

    假设 RichTextBox1 文本是"你好,我爱你中国",想要把中国变为红色,则 可以先找到中的位置是 7 :国的位置是8 设置 RichTextBox1.SelectionSt ...

  2. GTD:让大脑用来思考,而不是用来记事!

    前段时间听刘润大师分享了一套GTD时间管理方法理论,感觉非常受用!现拿来跟大家分享下,这套方法是刘润老师践行20多年总结提炼的精华,经亲自实践确实行之有效. 俗话说:工欲善其事,必先利其器!人生也是如 ...

  3. React配合Webpack实现代码分割与异步加载

    这是Webpack+React系列配置过程记录的第四篇.其他内容请参考: 第一篇:使用webpack.babel.react.antdesign配置单页面应用开发环境 第二篇:使用react-rout ...

  4. 《安卓网络编程》之第四篇 处理URL地址

    在Android手机系统中,可以通过URL地址获取网络资源.在URL类的众多方法中,可以使用openStream()方法来读取该URL资源的输入流InputStream.在此方法的基础上可以引申出很多 ...

  5. C#访问ORALCE数据库

    随着时间的推移知识也在更新,原来可用的技术也会被淘汰或更新. framework4.0开始不再支持System.Data.OracleClient了,但是令人欣慰的是ORACLE公司自己出了一个Ora ...

  6. grunt基础配置

    grunt基础配置 要使用grunt来管理项目,一般需要如下的几个步骤: 安装grunt命令行工具grunt-cli 在项目中安装grunt 安装grunt插件 建立并配置Gruntfile.js 安 ...

  7. 编写原生的Node.js模块

    导语:当Javascript的性能遭遇瓶颈,或者需要增强Javascript能力的时候,就需要依赖native模块来实现了. 应用场景 日常工作中,我们经常需要将原生的Node.js模块做为依赖并在项 ...

  8. nodeJS之进程process对象

    前面的话 process对象是一个全局对象,在任何地方都能访问到它,通过这个对象提供的属性和方法,使我们可以对当前运行的程序的进程进行访问和控制.本文将详细介绍process对象 概述 process ...

  9. 使用wget下载JDK8

    每次去官网下载JDK有点烦 但是直接使用wget 又得同意协议所以 使用如下的wget就好了(注意是64位的哦) 先去官网看一下地址变化 没有如下 :修改后面的下载地址即可 注意哦~ 2.然后使用下面 ...

  10. Coursera 机器学习笔记(四)

    主要为第六周内容机器学习应用建议以及系统设计. 下一步做什么 当训练好一个模型,预测未知数据,发现结果不如人意,该如何提高呢? 1.获得更多的训练实例 2.尝试减少特征的数量 3.尝试获得更多的特征 ...