- 浏览: 69808 次
- 性别:
- 来自: 绍兴
最新评论
埋点的作用是把客户端每次访问服务端的操作记录下来,包括请求连接 请求者ip 请求参数 请求结果等等,以便于在出现异常的情况下排查,在用户量庞大的情况下还可以对这些记录做数据分析.这个功能我还是坚持一直以来的原则,一次封装终身受用,使用拦截器的方式全局做埋点记录.
首先创建一个注解类,这个注解类的作用是在action方法上标注当前方法的中文信息,记录到日志以后有便于查阅
这个方法的缺点是需要在每个action上进行配置,对我这种比较爱偷懒的猿来说还是有点不怎么好忍受的,还好我有权限系统(上一篇),记录每个权限对应的链接,也就是我知道客户端访问的链接后就能通过权限数据获取到对应的权限(就是菜单),菜单的名称其实就是我需要的备注,这样我就不需要在每个action方法做备注了,当然有些没进行权限配置的action方法还是需要注解的,先来看下注解的代码
很简单的注解,然后在来看下核心代码,action拦截器
简单的一个拦截器记录操作信息到单独的日志文件,你也可以记录到数据库,我是感觉如果记录到数据库的话多多少少会影响一点效率,而且这个数据量也是挺大的.这边还值得一提的是结果值是被调用的action的toString方法,还记不记得上一篇讲的action基类,所有的action类都从它派生,在这里他又发挥作用了,看一下基类里面的toString代码
因为我所有的返回值都是基于entity和searchList实现的,所以可以很方便的把返回值获取到,其实程序中有很多模块的实现不是随便想想的,大多数情况下这些设计都是可以复用的,这也体现出了设计模式的重要性.
最后只要在struts文件中配置拦截器就可以了
首先创建一个注解类,这个注解类的作用是在action方法上标注当前方法的中文信息,记录到日志以后有便于查阅
package cn.sdh.common.annotation; import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target; @Target(ElementType.METHOD) @Retention(RetentionPolicy.RUNTIME) public @interface WebOperateAnno { String memo() default "无定义"; }
这个方法的缺点是需要在每个action上进行配置,对我这种比较爱偷懒的猿来说还是有点不怎么好忍受的,还好我有权限系统(上一篇),记录每个权限对应的链接,也就是我知道客户端访问的链接后就能通过权限数据获取到对应的权限(就是菜单),菜单的名称其实就是我需要的备注,这样我就不需要在每个action方法做备注了,当然有些没进行权限配置的action方法还是需要注解的,先来看下注解的代码
/** * 跳转到主页 * * @throws JSONException */ @WebOperateAnno(memo="主页") public String index() throws JSONException { entity = SpringSecurityUtils.getCurrentUser(); List<Permission> perList = entity.getSignPer(); HttpServletRequest request = ServletActionContext.getRequest(); request.setAttribute("perList", perList); request.setAttribute("perListJson", JSONUtil.serialize(perList)); return "index"; }
很简单的注解,然后在来看下核心代码,action拦截器
package cn.sdh.common.intercepter; import javax.servlet.http.HttpServletRequest; import org.apache.log4j.Logger; import org.apache.struts2.ServletActionContext; import cn.sdh.common.annotation.WebOperateAnno; import cn.sdh.entity.Account; import cn.sdh.utils.MyCacheUtil; import cn.sdh.utils.MyJsonUtil; import cn.sdh.utils.SpringSecurityUtils; import com.opensymphony.xwork2.ActionInvocation; import com.opensymphony.xwork2.interceptor.MethodFilterInterceptor; @SuppressWarnings("serial") public class WebOperateIntercepter extends MethodFilterInterceptor { private static final Logger logger = Logger.getLogger("oplog"); private static final Logger sysLogger = Logger.getLogger(WebOperateIntercepter.class); @Override protected String doIntercept(ActionInvocation invocation) throws Exception { String result = null; try { //先执行action方法 result = invocation.invoke(); String enterMethod = invocation.getProxy().getMethod(); //获取action方法上的WebOperateAnno 注解 WebOperateAnno webOperateAnno = invocation.getAction().getClass().getMethod(enterMethod, new Class[0]).getAnnotation(WebOperateAnno.class); Object returnData = invocation.getAction(); writeLog(webOperateAnno,returnData); } catch (Exception e) { sysLogger.error("操作日志记录失败",e); } return result; } private void writeLog(WebOperateAnno ano, Object returnData){ //当前登录用户 Account account = SpringSecurityUtils.getCurrentUser(); String operate = ""; if(account != null){ operate = account.getUsername(); } //从注解中获取备注 String memo = ""; if(ano != null){ memo = ano.memo(); } HttpServletRequest request = ServletActionContext.getRequest(); //请求者ip String ip = request.getRemoteAddr(); //请求类型 post or get String method = request.getMethod(); //请求的链接 String url = request.getRequestURI(); //通过请求的链接从权限系统获取到当前模块 String module = MyCacheUtil.getPerNameByUrl(url); //对请求的参数进行json格式化 String entityString = MyJsonUtil.getJsonByObject(request.getParameterMap()); //做日志记录 logger.info(module+"|"+operate+"|"+ip+"|"+memo+"|"+method+"|"+url+"|"+entityString+"|"+returnData.toString()); } }
简单的一个拦截器记录操作信息到单独的日志文件,你也可以记录到数据库,我是感觉如果记录到数据库的话多多少少会影响一点效率,而且这个数据量也是挺大的.这边还值得一提的是结果值是被调用的action的toString方法,还记不记得上一篇讲的action基类,所有的action类都从它派生,在这里他又发挥作用了,看一下基类里面的toString代码
public String toString(){ StringBuilder builder = new StringBuilder(); builder.append(",success:"); builder.append(this.success); if(this.entity != null){ builder.append("entity:"); builder.append(this.entity.toString()); }if(this.searchList != null){ builder.append("searchList:"); builder.append(this.searchList.toString()); } return builder.toString(); }
因为我所有的返回值都是基于entity和searchList实现的,所以可以很方便的把返回值获取到,其实程序中有很多模块的实现不是随便想想的,大多数情况下这些设计都是可以复用的,这也体现出了设计模式的重要性.
最后只要在struts文件中配置拦截器就可以了
<interceptors> <interceptor name="webOperate" class="cn.sdh.common.intercepter.WebOperateIntercepter" /> <interceptor-stack name="crudStack"> <interceptor-ref name="defaultStack"/> <interceptor-ref name="webOperate" /> </interceptor-stack> </interceptors> <default-interceptor-ref name="crudStack" />
发表评论
-
Web开发学习13聊聊java反射
2016-07-12 18:41 769很喜欢一本叫《走出软件作坊》的书,其中有一句话让我较为深刻,“ ... -
Web开发学习12 浅谈设计模式
2016-07-01 13:33 1330在我刚接触 ... -
Web开发学习11 全局缓存控制
2015-10-12 14:27 750缓存用的好可以减少数据库的压力,在大并发的情况下极大的提升服务 ... -
Web开发学习10 hadoop实战
2015-10-09 14:02 767前一篇说到项目记录了很多埋点日志,当有一天项目需求需要对这些日 ... -
Web开发学习8Struts2基类封装
2015-07-24 12:03 711一个好的基类可以帮助开发者减少很多开发工作,像我这种爱偷懒的人 ... -
Web开发学习7按钮级别权限控制
2015-07-23 16:21 13368这个功能的思路是自己原创的,没看过其他例子,其实是在做之前网上 ... -
Web开发学习6添加liqiubase
2015-07-22 14:29 1480开发过程中经常碰到数据库更改的情况,在日常环境下如果每位开发人 ... -
浅列JavaEnum
2014-01-05 16:42 795enum Province{ ZHEJIANG,SHANGHA ... -
maven常用命令
2013-07-08 10:20 7951、创建普通Java项目: ... -
canvas绘制科赫雪花
2012-12-20 17:24 2166<html> <head> ... -
html5 audio与video方法属性事件概括
2012-12-20 14:46 1127play() 继续播放 pause() ... -
spring 整合mina
2012-12-20 11:48 1127首先定义自定义过滤器 <bean id=" ... -
Web开发学习(5)添加springsecurity应用
2011-12-14 15:00 976我学习公司这个架构多半就是为了把springsecurity搞 ... -
Web开发学习(4)添加spring应用
2011-12-14 14:45 802印象中似乎没做过不用spring的项目,因为它在web开发中的 ... -
Web开发学习(3)配置struts-tiles插件(附带json插件)
2011-12-05 15:22 2577struts的tiles插件是非常实用的,从某些方面可以很好的 ... -
使用线程删除导出临时文件
2011-12-01 16:33 2285项目支持大数据量导出e ... -
jQuery实现表格行点击选中复选框
2011-11-23 12:36 6015这个需求是在项目完成后客户提出的要求,看似简单但是还需要一点小 ... -
Web开发学习(2)配置convention插件
2011-11-04 10:28 1401注解的方式某些程度上能减少xml的配置量,个人感觉使程序更加清 ... -
Web开发学习(1)使用eclipse搭建maven项目
2011-10-20 10:04 5753首先用eclipse创建工程 ne ... -
数据库连接泛型基类创建
2011-06-30 11:49 1004记录 以便以后参考---- 基类代码 public cl ...
相关推荐
Asp.Net MVC数据埋点是一种在Web应用中收集用户行为数据的技术,用于分析用户与网站交互的情况,从而优化用户体验和业务决策。在这个场景下,我们使用了...记住,数据驱动的决策是现代Web开发不可或缺的一部分。
在前端开发中,数据采集和行为上报是至关重要的环节,特别是在现代互联网产品中,数据分析对于优化用户体验、提高转化率和实现精细化运营具有决定性的作用。"前端数据采集+行为上报(数据埋点)"这个主题主要关注...
在现代Web开发中,数据分析和用户行为追踪成为衡量应用性能和优化用户体验的关键因素。"vtrack"是一个专为Vue.js框架设计的埋点插件,它利用Vue的自定义指令机制,实现了对用户交互的无缝跟踪。这种解耦的设计使得...
Spring Cloud Gateway作为Spring Cloud生态系中的网关,目标是替代Netflix ZUUL,其不仅提供统一的路由方式,并且基于Filter链的方式提供了网关基本的功能,例如:安全,监控/埋点,和限流等。 二、Spring Cloud ...
- **提供统一的Web开发框架**:解决前端技术栈分散的问题。 - **提供统一的SSR解决方案**:针对SEO和性能优化。 - **与现有无线研发支撑平台打通**:确保与现有系统的兼容性和互操作性。 - **方便应对技术出海场景**...
在蚂蚁前端核心面经中,我们可以看到对前端开发者的面试考察主要围绕React、Web开发、跨端技术、安全、性能优化、大型项目经验、Node.js以及SSR(服务器端渲染)等方面的知识点。 1. **ReactsetState的使用和原理**...
上述知识点涉及前端开发的多个方面,包括但不限于:React和Vue的状态管理、JavaScript ES5和ES6的新特性、Web前端性能优化、Web安全、异步编程模型、JavaScript数组操作和函数式编程概念等。这些内容不仅适用于面试...
根据文件内容,我们可以得知这是一份关于Web前端技术面试的题目列表,涵盖了广泛的技术知识点。下面我将逐个解析这些题目中涉及到的知识点: 1. 在React或Vue项目中编写列表组件时使用key的原因及其作用: - key...
Nuxt.js是一个基于Vue.js开发的服务端渲染框架,它通过预设各种配置和简化API使用,极大地提升了开发基于Vue的服务端渲染应用的效率。Nuxt.js不仅支持服务端渲染,还支持静态站点生成,提供了热加载等开发便捷功能,...
首先,项目的开发架构包括前端页面由Vue.js构建,网络请求通过Axios库进行,使用Vue.addAxios进行集成,并利用Vuex进行全局状态管理,如token的存储。同时,localStorage用于在浏览器端持久化存储token。开发者还...