由代码可以看到,SimpleSession 实现了和 Serializable,下面再看一下 ValidatingSession 接口
// IntelliJ API Decompiler stub source generated from a class file
// Implementation of methods is not available
package org.apache.shiro.session.mgt;
public interface ValidatingSession extends org.apache.shiro.session.Session {
boolean isValid();
void validate() throws org.apache.shiro.session.InvalidSessionException;
}
可以看到,ValidatingSession 接口继承了 org.apache.shiro.session.Session 接口,下面再看看 org.apache.shiro.session.Session 接口
// IntelliJ API Decompiler stub source generated from a class file
// Implementation of methods is not available
package org.apache.shiro.session;
public interface Session {
java.io.Serializable getId();
java.util.Date getStartTimestamp();
java.util.Date getLastAccessTime();
long getTimeout() throws org.apache.shiro.session.InvalidSessionException;
void setTimeout(long l) throws org.apache.shiro.session.InvalidSessionException;
java.lang.String getHost();
void touch() throws org.apache.shiro.session.InvalidSessionException;
void stop() throws org.apache.shiro.session.InvalidSessionException;
java.util.Collection<java.lang.Object> getAttributeKeys() throws org.apache.shiro.session.InvalidSessionException;
java.lang.Object getAttribute(java.lang.Object o) throws org.apache.shiro.session.InvalidSessionException;
void setAttribute(java.lang.Object o, java.lang.Object o1) throws org.apache.shiro.session.InvalidSessionException;
java.lang.Object removeAttribute(java.lang.Object o) throws org.apache.shiro.session.InvalidSessionException;
}
有上述代码可以看到,SimpleSession 对象是一个比较复杂的对象,既有很多个属性,又有多个复杂方法,如果直接使用Jackson工具转化为 json 格式字符串存入数据库,最终是无法解析出来的
我的解决方案是:
存储方案
1.1 将对象序列化到内存中,并生成一个字节数据
1.2 将1.1中得到的字节数组转化为 base64 位的字符串
1.3 将1.2中得到的字符串存入数据库即可
读取方案
2.1 从数据库中读取 base64 加密后的字符串
2.2 将2.1中的加密字符串解密成字节数组
2.3 将2.3中的字节数组反序列化
具体实现
我这里借助 org.apache.commons.lang3.SerializationUtils 工具类来实现,序列化功能代码如下:
package com.lixiaohao.test.springshiro.util;
import org.apache.commons.lang3.SerializationUtils;
import org.apache.shiro.codec.Base64;
import org.apache.shiro.session.Session;
import java.io.Serializable;
/**
@program: shiroTest
@description:
@author: xiaohao.li
@create: 2018-07-30 14:55
**/
public class SerializeUtils extends SerializationUtils {
public static String serializeToString(Serializable obj) {
try {
byte[] value = serialize(obj);
return Base64.encodeToString(value);
} catch (Exception e) {
throw new RuntimeException("serialize session error", e);
}
}
public static Session deserializeFromString(String base64) {
try {
byte[] objectData = Base64.decode(base64);
return deserialize(objectData);
} catch (Exception e) {
throw new RuntimeException("deserialize session error", e);
}
}
}
session管理功能如下:
package com.lixiaohao.test.springshiro.session;
import com.lixiaohao.test.springshiro.dao.SessionModelDao;
import com.lixiaohao.test.springshiro.dao.UserDao;
import com.lixiaohao.test.springshiro.model.SessionModel;
import com.lixiaohao.test.springshiro.model.User;
import com.lixiaohao.test.springshiro.util.JsonUtils;
import com.lixiaohao.test.springshiro.util.SerializeUtils;
import org.apache.shiro.session.Session;
import org.apache.shiro.session.mgt.SimpleSession;
import org.apache.shiro.session.mgt.eis.CachingSessionDAO;
import org.springframework.beans.factory.annotation.Autowired;
import java.io.Serializable;
import com.lixiaohao.test.springshiro.util.StringUtils;
/**
@program: shiroTest
@description:
@author: xiaohao.li
@create: 2018-07-26 12:40
**/
public class UserCacheSessionDao extends CachingSessionDAO{
private final String PRINCIPALS_SESSION_KEY = "org.apache.shiro.subject.support.DefaultSubjectContext_PRINCIPALS_SESSION_KEY";
@Autowired
private SessionModelDao sessionModelDao;
@Autowired
private UserDao userDao;
protected void doUpdate(Session session) {
}
protected void doDelete(Session session) {
if ( session == null ) {
throw new NullPointerException(" session argument can not be null ");
}
String sessionId = (String)session.getId();
if ( StringUtils.isEmpty(sessionId) ) {
throw new NullPointerException(" property sessionId can not be null in session argument ");
}
sessionModelDao.delete(sessionId);
}
protected Serializable doCreate(Session session) {
if ( session == null ) {
throw new NullPointerException(" session argument can not be null ");
}
String sessionId = (String) generateSessionId(session);
assignSessionId(session,sessionId);
storeSession(session);
return sessionId;
}
private void storeSession(Session session){
String sessionId = (String) session.getId();
SessionModel sessionModel = new SessionModel();
sessionModel.setSessionId(sessionId);
sessionModel.setStatus(1);
//将对象序列化
String sessionStr = SerializeUtils.serializeToString((SimpleSession)session);
sessionModel.setSessionStr(sessionStr);
Object principals = session.getAttribute(PRINCIPALS_SESSION_KEY);
if ( principals != null ) {
User user = userDao.findByUserName((String) principals);
sessionModel.setValue(JsonUtils.objectToJson(user));
}
System.out.println("保存session -------->> "+JsonUtils.objectToJson(sessionModel));
sessionModelDao.insert(sessionModel);
}
protected Session doReadSession(Serializable sessionId) {
if ( sessionId == null ) {
throw new NullPointerException(" sessionId argument can not be null. ");
}
String id = (String) sessionId;
SessionModel sessionMode = sessionModelDao.findBySessionId(id);
if ( sessionMode == null ) {
return null;
}
java.lang.String sessionStr = sessionMode.getSessionStr();
//反序列化对象
Session session = SerializeUtils.deserializeFromString(sessionStr);
return session;
}
}
分享到:
相关推荐
本文将深入探讨如何使用Java来实现多个数据库之间的数据同步。 首先,我们需要理解数据同步的含义。数据同步是指在两个或多个数据库之间,当某个数据库中的数据发生改变时,这些变化能够被实时或者近实时地反映到...
本文将深入探讨如何使用Java编程语言实现从一个数据库中定时自动抽取数据并复制到另一个数据库,以达到数据库间的实时或近实时同步。 首先,我们需要了解基础概念。Java是一种广泛使用的面向对象的编程语言,具有...
在Java编程中,将图片上传并存储到数据库是一项常见的任务,尤其在开发Web应用时。这一过程涉及到文件处理、数据库操作以及可能的图像处理。以下是对这个主题的详细阐述: 首先,我们需要理解基本的文件上传流程。...
接下来,为了将数据持久化存储,你需要配置一个数据库。常见的选择有MySQL、PostgreSQL或者MongoDB。在Java程序中,你可以使用JDBC(Java Database Connectivity)来连接数据库。在数据库配置中,你需要提供数据库...
数据库迁移是指将数据从一个数据库系统迁移到另一个数据库系统的过程,或者在同一系统中不同版本之间进行数据转移。这通常涉及到数据的抽取、转换和加载(ETL)过程,以及确保数据的一致性和完整性。 Java作为广泛...
本项目关注的是如何使用Java来实现读取Microsoft Word文件,并将其中的数据上传到数据库,以此提升数据录入的效率。这一过程涉及到多个技术点,包括文件I/O、Word文档解析、数据库操作以及可能的数据转换。 首先,...
Java实现从excel中批量高效导入数据到数据库中,包括图片的导入存储,读取图片显示在jsp页面等,需要下载JspSmartUpload.jar和jxl.jar两个jar包,将两个包导入项目才能运行
在本文中,我们将深入探讨如何使用Java开发一个串口助手应用程序,该程序能接收来自串行端口的数据并将其解析后存储到MySQL数据库中。首先,我们需要了解串口通信的基本概念,然后学习Java如何处理串口输入,接着是...
在Java编程中,将文件保存到数据库是一种常见的需求,特别是在处理图像、文档和其他非结构化数据时。这个过程涉及到将文件转换为二进制数据,然后将这些数据存储到数据库的特定字段中,通常是一个BLOB(Binary Large...
在IT领域,它被广泛用于生成报告、发票、证书等,特别是在需要将数据动态地从数据库填充到预定义模板的情况。在这个场景中,"java iText 实现从数据库中导出数据到已有的PDF模板"是一个常见的需求,它涉及到多个关键...
1、java解析读取excel文件中的数据,并写入数据库。 2、java读取数据库数据,并导出为excel文件。 3、README.md中有详细的操作步骤示例。 使用说明: 1. 先使用postman导入:other/excel相关.postman_collection....
Java建立数据库连接并实现数据库查询,初学者通过本Java源代码,了解如何加载驱动器,获取连接,定义SQL查询语句,获取Statement对象,执行查询、遍历ResultSet、获取数据、在控制台输出数据,关闭连接等。
- 当检测到数据库有新数据时,后端通过已经建立的WebSocket连接,将更新的消息推送给前端。 - 前端页面接收到WebSocket消息后,刷新显示最新的数据库内容,无需手动刷新整个页面。 总结来说,这个项目通过...
博客地址:https://blog.csdn.net/fukaiit/article/details/92853086 1. 启动后访问http://localhost:8848/code,可见... 2. 新建一个只有key和value两个字段的表code 3. 不要轻易浪费积分下载代码,没什么实质内容
数据库系统原理实验 数据库管理系统 javase java 大学数据库实验 主要利用JAVA序列化和反序列化 注解挺全的,DBMS, 文件存储表、库,根据sql语句实现建表,建库 可以建立索引(B+树) 可以做笛卡尔积(hash) 自然...
获取到数据后,使用EasyExcel的write方法将数据写入到Excel文件中。 接着,我们讨论如何将Excel数据保存到Mysql数据库。首先,你需要解析Excel文件,EasyExcel提供了read方法来读取Excel内容,将其转化为Java对象。...
可以先将数据收集到内存中的集合,然后一次性提交到数据库,而不是逐条插入。 6. **事务管理**: 为了保证数据的一致性,操作数据库时可能需要使用事务。比如,如果在插入过程中发生错误,可以回滚事务,确保数据的...
进销存管理系统源码java实现毕业设计项目(含数据库脚本)进销存管理系统源码java实现毕业设计项目(含数据库脚本)进销存管理系统源码java实现毕业设计项目(含数据库脚本)进销存管理系统源码java实现毕业设计项目...
Hibernate通过XML配置文件或注解将Java实体类映射到数据库表,然后提供了Session接口来处理数据持久化。例如,我们可以使用`Session.save()`方法来保存对象,`Session.get()`来读取,`Session.update()`来更新,以及...