首先,这个问题困扰了好久,一直想着是否能有更合适的方式来解决,下面我就先描述下需求吧,如有不理解的地方,欢迎提出疑问,并且给出良好的建议,谢谢!
项目框架是SSH,大家再熟悉不过,用户登录操作后,将用户登录信息压入session中,这是常规做法,一般用户在维护数据的时候都会用到userSession的一些基本信息,如新增、修改和删除等。一般这样的操作大家都会认为请求大都是从action入口,所以必然能获取到request对象。但是我们现在的项目有点特殊,先介绍下我们项目的基本情况吧:
系统分为浏览版和维护版的开发,并且这两块的开发是由两个独立的开发团队进行的,只有一个技术经理在中间进行协调工作,所以在开发的过程当中有些维护版产生的bug不得不由我们这些在现场开发浏览版的人来给他们维护,修改bug。
(以上是项目大概的背景资料,大家可以重点看下面)
用户进行新增、修改和删除数据的时候记录日志,记录日志是利用Hibernate的拦截器来进行操作的,但是Hibernate的拦截器只会对使用Hibernate指定的方法,如save()、update()、delete()等方法进行拦截,据我所测试而不会对nativate的sql语句进行拦截,所以问题就出现了,在通过sql语句如(delete from table where id=1)删除数据的时候,Hibernate事件拦截器并不会对其进行拦截,恰恰巧合的是,我们项目里所有实体删除数据的时候都是用sql语句来执行操作的,并不会记录日志,直到后来我们发现了这个问题,因为涉及的地方比较多,不可能将之前的代码全部更新,只能在现有的代码中进行少量的修改。
之前的实现大概是下面这样的情况:
创建一个包含所有实体删除方法的类AllDeleteSQL.java,用户删除数据时只需要调用这个类的静态方法,传入数据的ID即可完成删除操作(他们认为这同样能被Hibernate操作事件拦截器拦截并记录日志),此时并没有传入request参数,实际结果记录日志失败。
我现在想到一种解决思想是:
新增一个类ThreadLocalUtil.java类,里面有个ThreadLocal的全局静态变量,用户在进行登录操作的时候,将用户的session信息存放到这个类的ThreadLocal变量当中,下次再直接从这个变量中获取出来,具体的代码如下:
ThreadLocalUtil.java
package **.**;
import **.UserSession;
/**
* 功能:存放用户登录后的session信息类
* 每个用户登录调用静态方法存储相应的用户session信息
* @author Administrator
*/
public class ThreadLocalUtil {
private static volatile ThreadLocal<UserSession> localSession = new ThreadLocal<UserSession>(); //存放session信息的ThreadLocal全局变量
public ThreadLocalUtil() {
}
/**
* 将用户的session信息set到该类的localSession变量当中
* @param us
*/
public static synchronized void setUserSessionToThreadLocal(UserSession us) {
localSession.set(us);
}
/**
* 获取对应的localSession中保存的用户session信息
* @return localSession
*/
public static synchronized UserSession getUserSessionToThreadLocal() {
return localSession.get();
}
/**
* 用户注销时将localSession中保存的用户信息清除
*/
public static void removeUserSession() {
if(localSession.get()!=null) {
localSession.remove();
}
}
}
AllDeleteSQL.java的部分方法:
import **.UserSession;
import **.ThreadLocalUtil;
public class AllDeleteSQL {
static CommDelete commDelete = CommDelete.getCommDelete();
/**
* 删除业务需求
*/
public static String[] getRequirementDeleteSql(Long entityId, Long dataId){
String sql = commDelete.delCommInfo(entityId, dataId);
sql = sql + "update REQUIREMENT set DEL_FLAG = 1 where ID = " + dataId;
sql = sql + ";";
sql = sql + "update REQ_COMM set DEL_FLAG = 1 where REQ_ID = " + dataId;
sql = sql + ";";
sql = sql + getDelelteLogSQL(entityId,dataId);
return sql.split(";");
}
/**
* 删除实体数据时拼接记录日志删除SQL
*/
public static String getDelelteLogSQL(Long entityId,Long dataId) {
UserSession us = ThreadLocalUtil.getUserSessionToThreadLocal();
StringBuilder dSql = new StringBuilder();
if(us==null) {
return "";
} else {
dSql.append("INSERT INTO EAM_LOG");
dSql.append("(ID, LOG_ENTITY, LOG_DATA_ID, LOG_TYPE, LOG_ORG_ID, LOG_PERSON_ID, LOG_DATE)");
dSql.append(" VALUES (");
dSql.append(Long.parseLong(SequencerUtil.getSequence()));
dSql.append(", ");
dSql.append("(SELECT T.ENTITY_NAME_EN_FULL FROM EAM_ENTITY T WHERE T.ID=").append(entityId).append(")");
dSql.append(", ");
dSql.append("'" + dataId + "'");
dSql.append(", ");
dSql.append("'删除数据'");
dSql.append(", ");
dSql.append("'" + us.getUserOrgId() + "'");
dSql.append(", ");
dSql.append("'" + us.getUserArchPersonId() + "'");
dSql.append(", ");
dSql.append("'" + DateUtil.getCurrentDateTime("yyyy-MM-dd HH:mm:ss") + "'");
dSql.append(" )");
return dSql.toString();
}
}
}
用户登录时请求的action操作部分代码:
//将用户session信息存放到ThreadLocal中
UserSession us = (UserSession)request.getSession().getAttribute(Constants.USER_KEY);
ThreadLocalUtil.setUserSessionToThreadLocal(us);
以上只贴出部分代码,但是主体部分全部出来了,不知道大家有没有看懂我解决该问题的思路,因为struts的action请求都是线程不安全的,这里也是我为什么在ThreadLocalUtil.java中采用同步方法来进行访问,这样使得在多线程下能够实时的访问同步变量更新后的值。
按理说做到这里我觉得问题应该是解决了,当我测试的时候发现,结果并不然,在AllDeleteSQL.java中getDelelteLogSQL(args)方法中通过UserSession us = ThreadLocalUtil.getUserSessionToThreadLocal();语句来获取用户session信息,实际是这样获取的us信息并不稳定,有时候为空,有时候正确,当我以多个浏览器登录不同的用户时,获取的us信息时为空,时为当前用户的session信息,时为其他用户的session信息,记录的日志并不准确,问题依然没能彻底解决。
思考,得不到解决办法,也查不到相关的资料如何处理这种情况,说到根源,还是因为我自己的基础知识不扎实导致的结果。
在这里首先要说明一下,用ThreadLocal来存取用户session信息的方案是由我们技术经理提出的,说这种方式能够解决现在的问题,但是我并没有把事情做好,所以希望在此描述清楚自己的需求,能够和大家讨论下,看你们有没有碰到过类似的情况,你们是如何解决的,不妨说来听听,让我学习学习吧。
如果你愿意耐心看完,对我的表达能力提出疑问,欢迎你提出宝贵的意见,并向你说声谢谢;
如果你看完了还能发表下对我目前所碰到的问题给出你的意见,向你深表感谢;
最后祝大家工作顺利^_^
分享到:
相关推荐
金煤婚恋 92版本, 去授权,仅供学习,商业使用请支持正版
传统办法管理信息首先需要花费的时间比较多,其次数据出错率比较高,而且对错误的数据进行更改也比较困难,最后,检索数据费事费力。因此,在计算机上安装小区团购管理软件来发挥其高效地信息处理的作用,可以规范信息管理流程,让管理工作可以系统化和程序化,同时,小区团购管理的有效运用可以帮助管理人员准确快速地处理信息。 小区团购管理在对开发工具的选择上也很慎重,为了便于开发实现,选择的开发工具为Eclipse,选择的数据库工具为Mysql。以此搭建开发环境实现小区团购管理的功能。其中管理员管理用户,新闻公告。 小区团购管理是一款运用软件开发技术设计实现的应用系统,在信息处理上可以达到快速的目的,不管是针对数据添加,数据维护和统计,以及数据查询等处理要求,小区团购管理都可以轻松应对。 关键词:小区团购管理;SSM框架,系统分析,数据库设计
数据集是一个关于初创企业失败案例的详细数据集,由Daglox Kankwanda于2025年2月27日发布在Kaggle上。该数据集包含483家初创企业的失败信息,数据来源于CB Insights的“初创企业失败后分析”汇编。 数据集涵盖了多个行业的初创企业,提供了丰富的字段信息,包括公司名称、行业领域、失败原因、资金筹集情况、运营时长、地理位置等。这些字段为研究者提供了多维度的视角,可以深入分析初创企业失败的共性和差异。 通过该数据集,研究者可以探索不同行业初创企业的失败模式,例如,某些行业可能因市场竞争激烈而失败,而另一些行业可能因技术瓶颈或资金不足而终止。此外,数据集还提供了失败原因的详细分类,如产品市场契合度不足、团队问题、资金链断裂等,为创业者和投资者提供了宝贵的经验教训。 该数据集不仅适用于商业分析和研究,还可以用于机器学习模型的训练,例如预测初创企业的成功概率或识别潜在的失败风险因素。对于希望深入了解创业生态和风险的研究者、创业者以及投资者来说,“Startup Failures”数据集是一个极具价值的资源。
Swift-Button封装
内容概要:本文档详尽介绍了AUTOSAR TcpIp模块的功能与设计,作为AUTOSAR通信栈的核心部分,它提供了完整的TCP/IP协议栈解决方案,包括TCP和UDP在内的多种协议以及IPv4和IPv6的支持。文档涵盖模块的总体架构设计,详细描绘其状态管理和数据传输机制,并阐述与其它相关模块之间的交互协作。同时,对初始化流程、Socket操作以及数据发送接收的具体过程进行了逐步拆解与解释。还介绍了针对可能出现故障时的错误处理措施。 适合人群:汽车电子工程师、嵌入式系统开发者、研究AUTOSAR规范及其网络协议栈实现的专业人士。 使用场景及目标:本文件主要用于深入了解和支持汽车行业内基于AUTOSAR平台的网络协议部署情况;为设计符合工业标准的车载信息系统打下坚实的基础;对于提升产品性能和安全性具有重要的指导价值。 其他说明:作者提供了一个详细的参考资料链接,感兴趣的读者可以进一步访问获取更多信息。此外,在实际工作中遇到的问题也可以参考文中所提到的各种处理方法来进行有效的排查和解决。
Thinkpad T480s的BIOS升级软件 【版本n22uj39w】
特易通国产对讲机TYT-T3 v1.0中英写频软件
Swift-ViewController+SP
Swift-PDFImage
VMware-Workstation-Full-17.6.2-24409262.x86_64.rar Linux版本 Vmware是一款领先的虚拟化软件,为用户提供强大的虚拟机平台。通过使用Vmware,用户可以在一台物理计算机上同时运行多个虚拟操作系统,实现资源的高效利用和隔离。它提供了灵活的配置选项、快速的性能和可靠的安全性,适用于个人用户、企业和数据中心。无论是开发测试、应用部署还是服务器管理,Vmware都是一个强大而可靠的工具,为用户提供了简单且可扩展的虚拟化解决方案。
内容概要:本文首先介绍了智能座舱的概念及其组成结构,详细解释了硬件、软件及交互三个部分的功能,并阐述了智能座舱从电子座舱到智能移动空间的不同发展阶段,重点讨论了语音交互和AR-HUD两大核心技术的发展路径及未来发展方向。接下来介绍车载基础软件的重要性及其分类,强调其在智能汽车发展中扮演的角色。文中指出,在软件定义汽车的趋势下,车载基础软件成为衔接硬件和应用软件的核心枢纽,特别是在软硬件分层解耦背景下,其作用日益凸显。最后探讨了当前国内车载基础软件行业的竞争情况与发展趋势。文章还分析了行业发展面临的高技术壁垒、高转换成本和高退出壁垒等问题,指出了车载基础软件对未来汽车产业变革的意义。 适用人群:从事汽车电子产品设计的研发人员、相关专业在校学生以及对汽车行业新技术感兴趣的爱好者。 使用场景及目标:该文档适合用作学习资料或研究参考资料,旨在帮助读者深入了解智能座舱的构成要素和技术演进,掌握车载基础软件的架构特点,洞悉该领域的市场动向。 其他说明:文中部分内容带有作者个人感悟和思考,但这并不影响其专业价值,反而增加了人文气息,有助于激发工程师的人文关怀和社会责任感。
JAVA面试最新最有效的全网顶级资料,免费提供,码字不易,你的关注就是博主最大的动力。
Swift-String-Extension
社会发展日新月异,用计算机应用实现数据管理功能已经算是很完善的了,但是随着移动互联网的到来,处理信息不再受制于地理位置的限制,处理信息及时高效,备受人们的喜爱。本次开发一套物流管理系统有管理员和用户两个角色。管理员功能有个人中心,用户管理,车辆信息管理,公告信息管理,司机管理,物流信息管理,运单信息管理,车辆类型管理,车辆状态管理,公告类型管理,物流状态管理,运单状态管理。用户可以注册登录,查看公告信息,查看物流信息,可以添加运单信息。物流管理系统服务端用Java开发,用Spring Boot框架开发的网站后台,数据库用到了MySQL数据库作为数据的存储。这样就让用户用着方便快捷,都通过同一个后台进行业务处理,而后台又可以根据并发量做好部署,用硬件和软件进行协作,满足于数据的交互式处理,让用户的数据存储更安全,得到数据更方便。 关键字:物流管理系统;Spring Boot框架;Java;MySQL
数据结构学习
KaihongOS_System_Component 4.1.2.17(RT00E000C000M68A).part2.rar 请勿下载,请联系对应销售获取。
数据结构学习
最长上升子序列(Longest Increasing Subsequence,LIS)问题是指在一个给定的无序序列中,找到一个最长的单调递增子序列的长度。
数据结构学习
网课专注度监测预警系统基于yolov5目标检测的网课专注度检测系统源码+模型+pyqt5界面,含有代码注释,新手也可看懂,个人手打98分项目,导师非常认可的高分项目,毕业设计、期末大作业和课程设计高分必看,下载下来,简单部署,就可以使用。该项目可以直接作为毕设、期末大作业使用,代码都在里面,系统功能完善、界面美观、操作简单、功能齐全、管理便捷,具有很高的实际应用价值,项目都经过严格调试,确保可以运行! 网课专注度监测预警系统基于yolov5目标检测的网课专注度检测系统源码+模型+pyqt5界面.zip网课专注度监测预警系统基于yolov5目标检测的网课专注度检测系统源码+模型+pyqt5界面.zip网课专注度监测预警系统基于yolov5目标检测的网课专注度检测系统源码+模型+pyqt5界面.zip网课专注度监测预警系统基于yolov5目标检测的网课专注度检测系统源码+模型+pyqt5界面.zip网课专注度监测预警系统基于yolov5目标检测的网课专注度检测系统源码+模型+pyqt5界面.zip网课专注度监测预警系统基于yolov5目标检测的网课专注度检测系统源码+模型+pyqt5界面