Spring Boot和Shiro教程-RememberMe功能实现

发布时间: 2024-01-09 04:42:10 阅读量: 59 订阅数: 40
RAR

springmvc+shiro实现记住我功能以及权限缓存.rar

# 1. 简介 ## 1.1 Spring Boot和Shiro简介 Spring Boot 是一个基于 Spring 的轻量级框架,用于快速搭建基于 Spring 的应用程序。它简化了基于 Spring 的应用程序的搭建过程,提供了一种快速、便捷的方式来构建应用程序。 Apache Shiro 是一个强大且易于使用的Java安全框架,提供了身份验证、授权、加密和会话管理等功能,适用于任何应用程序。Shiro 的设计目标是使安全易于使用,它提供了一种简单直接的方式,来满足应用程序中的安全需求。 ## 1.2 RememberMe功能概述 RememberMe 功能是指用户登录后,即使关闭浏览器再次打开时,也能保持登录状态。该功能通过在用户浏览器中保存一个持久化的令牌来实现。当用户再次访问网站时,令牌会自动登录用户,无需重新输入账号密码。 接下来我们将介绍如何在Spring Boot项目中集成Shiro,并实现RememberMe功能。 # 2. 环境搭建 在本章节中,我们将介绍如何搭建环境来集成Spring Boot和Shiro,并配置数据库用于存储用户信息。 ### 2.1 安装Spring Boot 首先,我们需要安装Spring Boot。可以通过以下步骤来完成安装: 1. 访问Spring Boot的官方网站(https://spring.io/projects/spring-boot)。 2. 在网站上找到最新版的Spring Boot,并下载对应的压缩包。 3. 解压缩下载的压缩包到指定的目录。 4. 配置环境变量,将Spring Boot的bin目录添加到系统的PATH中。 完成以上步骤后,我们就成功安装了Spring Boot。 ### 2.2 引入Shiro依赖 接下来,我们将引入Shiro的依赖,以便在Spring Boot项目中使用Shiro功能。可以按照以下步骤进行操作: 1. 打开你的Spring Boot项目,找到项目的pom.xml文件。 2. 在pom.xml文件的dependencies节中添加以下代码: ```xml <dependency> <groupId>org.apache.shiro</groupId> <artifactId>shiro-spring-boot-starter</artifactId> <version>1.7.1</version> </dependency> ``` 3. 保存并关闭pom.xml文件。重新构建项目,以使Shiro的依赖生效。 ### 2.3 配置数据库 在使用Shiro进行用户认证和授权时,我们需要一个持久化存储来保存用户信息、角色和权限等数据。通常情况下,我们会选择使用数据库来存储这些数据。 以下是配置数据库的步骤: 1. 在你的Spring Boot项目的配置文件(比如application.yml或application.properties)中添加以下配置: ```yaml spring: datasource: url: jdbc:mysql://localhost:3306/mydb username: your_username password: your_password driver-class-name: com.mysql.jdbc.Driver ``` 请将上述配置中的`your_username`替换为你的数据库用户名,`your_password`替换为你的数据库密码。 2. 确保你的项目中已经引入了MySQL数据库的相关依赖。如果没有引入,你可以在pom.xml文件中添加以下代码: ```xml <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> </dependency> ``` 3. 运行你的Spring Boot项目,确保数据库配置生效并能够成功连接到数据库。 完成以上步骤后,我们就完成了环境搭建的过程。接下来,我们可以开始实现用户认证和授权的功能了。 # 3. 用户认证和授权 用户认证和授权是一个系统中非常重要的部分,能够保障系统的安全性。在Spring Boot项目中集成Shiro可以很方便地实现用户认证和授权功能。 #### 3.1 用户登录功能实现 在使用Shiro进行用户认证之前,首先需要实现用户登录功能。通过收集用户输入的用户名和密码,然后验证其合法性,最后将用户信息存储到会话中以表示用户已登录。 ```java // 示例代码,实现用户登录功能 @Controller public class LoginController { @RequestMapping("/login") public String login(HttpServletRequest request, String username, String password, boolean rememberMe) { Subject subject = SecurityUtils.getSubject(); UsernamePasswordToken token = new UsernamePasswordToken(username, password, rememberMe); try { subject.login(token); // 登录成功后的处理 return "index"; } catch (AuthenticationException e) { // 登录失败后的处理 return "login"; } } } ``` #### 3.2 用户角色和权限管理 在Shiro中,可以使用不同的注解和配置来实现用户的角色和权限管理。通过给用户分配角色和权限,可以控制用户在系统中可以执行的操作。 ```java // 示例代码,实现用户角色和权限管理 @RequiresRoles("admin") public class AdminController { @RequiresPermissions("user:delete") @RequestMapping("/deleteUser") public String deleteUser(Long userId) { // 删除用户操作 return "success"; } } ``` #### 3.3 使用Shiro进行用户认证 Shiro提供了多种方式来进行用户认证,可以通过编程的方式或配置文件的方式来实现。在Spring Boot中,通常会选择使用注解和配置类的方式来集成Shiro进行用户认证。 ```java // 示例代码,使用Shiro进行用户认证 @Configuration public class ShiroConfig { @Bean public ShiroFilterFactoryBean shiroFilterFactoryBean(DefaultWebSecurityManager securityManager) { ShiroFilterFactoryBean shiroFilterFactoryBean = new ShiroFilterFactoryBean(); // 设置其他配置项 return shiroFilterFactoryBean; } @Bean public DefaultWebSecurityManager securityManager(Realm realm) { DefaultWebSecurityManager securityManager = new DefaultWebSecurityManager(); securityManager.setRealm(realm); // 设置其他配置项 return securityManager; } @Bean public Realm realm() { // 创建自定义的Realm return new CustomRealm(); } } ``` 以上是用户认证和授权在Spring Boot项目中集成Shiro的基本实现方式,通过以上配置和代码,可以实现基本的用户登录、角色和权限管理以及用户认证功能。 # 4. RememberMe功能实现 RememberMe功能是Shiro提供的一种方便的自动登录功能,允许用户在关闭网站或浏览器后,重新打开页面时能够自动登录。本章将介绍RememberMe的原理、在Spring Boot中启用RememberMe的方法以及配置RememberMe的持久化方式。 #### 4.1 RememberMe的原理介绍 RememberMe的原理是在用户登录成功后,将用户的身份信息(通常是用户ID或用户名)保存到客户端的Cookie中,并在下次访问网站时,通过解析Cookie中的身份信息来自动登录用户。 #### 4.2 在Spring Boot中启用RememberMe 要在Spring Boot项目中启用RememberMe功能,需要进行如下配置: 首先,在Spring Boot的配置文件(通常是application.yml或application.properties)中添加如下配置: ```yaml shiro: rememberMe: enabled: true cipherKey: kPH+bIxk5D2deZiIxcaaaA== ``` 其中,`enabled`用于启用RememberMe功能,`cipherKey`用于指定加密密钥,可以使用任意长度的字符串,但必须是Base64编码的字符串。 然后,在Spring Boot的配置类或配置文件中,将RememberMeManager注入到Shiro的SecurityManager中: ```java @Bean public SecurityManager securityManager() { DefaultWebSecurityManager securityManager = new DefaultWebSecurityManager(); // 配置其他Realm和RealNamePasswordRetry用于登录认证 securityManager.setRememberMeManager(rememberMeManager()); return securityManager; } @Bean public CookieRememberMeManager rememberMeManager() { CookieRememberMeManager rememberMeManager = new CookieRememberMeManager(); rememberMeManager.setCipherKey(Base64.decode("kPH+bIxk5D2deZiIxcaaaA==")); return rememberMeManager; } ``` #### 4.3 配置RememberMe的持久化方式 通过以上步骤,已经启用了RememberMe功能,但默认情况下,RememberMe是使用Cookie存储持久化信息的。如果需要将RememberMe持久化到数据库中,可以自定义实现RememberMeManager接口,并进行相应的配置。 首先,创建一个实现RememberMeManager接口的类,并实现相关的方法。例如,使用JDBC来持久化RememberMe信息的实现代码如下: ```java public class JdbcRememberMeManager implements RememberMeManager { private JdbcTemplate jdbcTemplate; public JdbcRememberMeManager(JdbcTemplate jdbcTemplate) { this.jdbcTemplate = jdbcTemplate; } @Override public void rememberIdentity(PrincipalCollection principals, AuthenticationToken token, RememberMeConfig rememberMeConfig) { // 获取用户信息 String username = (String) principals.getPrimaryPrincipal(); // 获取RememberMe的加密密钥 byte[] cipherKey = rememberMeConfig.getCipherKey().getBytes(); // 生成RememberMe的持久化数据 // ... // 将数据存储至数据库 // ... } @Override public PrincipalCollection forgetIdentity(AuthenticationToken token, RememberMeConfig rememberMeConfig) { // 获取RememberMe的加密密钥 byte[] cipherKey = rememberMeConfig.getCipherKey().getBytes(); // 获取持久化数据 // ... // 将数据从数据库中读取并解析为PrincipalCollection对象 // ... return principalCollection; } // 其他方法的实现 // ... } ``` 然后,在Spring Boot的配置类中,注入相应的Bean,并配置到SecurityManager中: ```java @Bean public SecurityManager securityManager() { DefaultWebSecurityManager securityManager = new DefaultWebSecurityManager(); // 配置其他Realm和RealNamePasswordRetry用于登录认证 securityManager.setRememberMeManager(rememberMeManager()); return securityManager; } @Bean public JdbcRememberMeManager rememberMeManager(JdbcTemplate jdbcTemplate) { return new JdbcRememberMeManager(jdbcTemplate); } ``` 通过以上配置,RememberMe功能将使用自定义的RememberMeManager实现进行持久化。 在本章中,我们介绍了RememberMe的原理、启用RememberMe的方法以及配置RememberMe的持久化方式。通过正确的配置,我们可以在Spring Boot项目中实现方便的自动登录功能。在下一章节中,将介绍RememberMe功能的安全性相关内容。 # 5. RememberMe功能的安全性 RememberMe功能虽然方便用户登录,但也存在一定的安全风险。在使用RememberMe功能时,需要注意以下几点来提高系统的安全性。 ### 5.1 RememberMe的安全风险 RememberMe功能的安全风险主要包括以下几个方面: - 持久化方式:如果RememberMe的凭证信息被持久化保存在客户端,那么一旦客户端被攻击者获取,攻击者就能通过这些凭证信息来伪造身份,并访问用户的账户。 - 凭证信息的泄露:如果RememberMe的凭证信息被泄露,比如存储在浏览器的Cookie中,那么攻击者可以通过获取该凭证来绕过登录认证,直接访问用户的账户。 - RememberMe的失效问题:如果RememberMe的凭证信息在用户主动注销登录后没有失效,那么别人可以利用这些凭证信息来登录用户的账户。 - 弱密码问题:如果用户设置的RememberMe的密码比较弱,容易被猜测或撞库破解,那么攻击者也能利用这些密码来登录用户的账户。 ### 5.2 RememberMe的加固方法 为了加强RememberMe功能的安全性,可以采取以下几种措施: - 使用安全的持久化方式:避免将凭证信息直接保存在客户端,而是使用服务器上的持久化存储来保存凭证信息,比如使用数据库、缓存等。同时,可以对凭证信息进行加密存储,提高难度。 - 增加凭证信息的安全性:可以对凭证信息进行加密处理,确保即使被泄露也难以解密。可以使用密钥、散列函数等技术来增加凭证信息的安全性。 - 设置RememberMe的有效期:在用户注销登录后,需要及时失效RememberMe的凭证信息,避免被滥用。可以根据具体业务需求和安全性要求来设置有效期的时长。 - 强化用户密码策略:用户在设置RememberMe的密码时,应该遵循一定的密码策略,比如密码长度、复杂度要求等,以增加破解密码的难度。 - 添加额外的安全措施:可以基于RememberMe功能加入额外的安全措施,比如验证码、双因素认证等,提高系统的安全性。 ### 5.3 使用RememberMe时的注意事项 使用RememberMe功能时,需要注意以下几点: - 需要在系统的登录页面明确提示用户是否启用RememberMe功能,并在用户授权前明确告知用户RememberMe的安全风险,让用户自主选择是否启用该功能。 - 应该定期检查和更新系统的RememberMe功能,及时修复可能存在的安全漏洞。 - 如果发现RememberMe的凭证信息被泄露或存在安全风险,应及时通知用户,并参考相关安全建议进行处理。 总之,RememberMe功能在提高用户体验的同时也带来了一定的安全风险。在使用该功能时,需要综合考虑业务需求和安全性要求,采取适当的安全措施来保护用户的账户安全。 # 6. 实战演练 在本章中,我们将通过创建一个简单的Spring Boot项目来演示如何集成Shiro,并实现RememberMe功能。 ### 6.1 创建一个简单的Spring Boot项目 首先,我们需要创建一个简单的Spring Boot项目。可以通过使用Spring Initializer(https://start.spring.io/)或使用IDE的项目创建向导来完成。 创建完成后,我们可以在项目的入口类中添加一些基本的配置。 ```java @SpringBootApplication public class Application { public static void main(String[] args) { SpringApplication.run(Application.class, args); } } ``` ### 6.2 添加Shiro的配置和用户认证功能 接下来,我们需要添加Shiro的配置和用户认证功能。 首先,我们需要在项目的配置文件(application.properties或application.yaml)中配置Shiro相关的属性。 ```yaml # Shiro配置 shiro: ini-config: classpath:shiro.ini rememberMe: encryptionKey: your-encryption-key cookieName: remember-me cookieMaxAge: 2592000 ``` 然后,我们需要创建一个`ShiroConfig`类,并在该类中添加Shiro的配置。 ```java @Configuration public class ShiroConfig { @Value("${shiro.ini-config}") private String shiroIniConfig; // Shiro拦截器配置 @Bean public ShiroFilterFactoryBean shiroFilterFactoryBean(DefaultWebSecurityManager securityManager) { ShiroFilterFactoryBean shiroFilterFactoryBean = new ShiroFilterFactoryBean(); shiroFilterFactoryBean.setSecurityManager(securityManager); // 添加Shiro过滤器链 Map<String, String> filterChainDefinitionMap = new LinkedHashMap<>(); filterChainDefinitionMap.put("/login", "anon"); filterChainDefinitionMap.put("/**", "user"); shiroFilterFactoryBean.setFilterChainDefinitionMap(filterChainDefinitionMap); return shiroFilterFactoryBean; } // 创建SecurityManager @Bean public DefaultWebSecurityManager securityManager(Realm realm) { DefaultWebSecurityManager securityManager = new DefaultWebSecurityManager(); securityManager.setRealm(realm); // 配置RememberMeManager securityManager.setRememberMeManager(rememberMeManager()); return securityManager; } // 创建Realm @Bean public Realm realm() { IniRealm realm = new IniRealm(shiroIniConfig); return realm; } // 创建RememberMeManager @Bean public RememberMeManager rememberMeManager() { CookieRememberMeManager rememberMeManager = new CookieRememberMeManager(); rememberMeManager.setCookie(rememberMeCookie()); rememberMeManager.setCipherKey(Base64.decode("your-base64-encoded-key")); return rememberMeManager; } // 创建RememberMeCookie @Bean public SimpleCookie rememberMeCookie() { SimpleCookie simpleCookie = new SimpleCookie(); simpleCookie.setName("remember-me"); simpleCookie.setMaxAge(2592000); return simpleCookie; } } ``` 在上述代码中,我们通过配置`shiro.ini-config`属性来指定Shiro的INI配置文件。我们还通过`ShiroFilterFactoryBean`来配置Shiro的过滤器链,以及通过`DefaultWebSecurityManager`配置Shiro的SecurityManager。 ### 6.3 实现RememberMe功能的演示 在完成上述配置后,我们可以创建一个简单的Controller类来演示RememberMe功能的使用。 ```java @RestController public class UserController { @PostMapping("/login") public String login(String username, String password, boolean rememberMe) { Subject currentUser = SecurityUtils.getSubject(); // 使用UsernamePasswordToken进行登录 UsernamePasswordToken token = new UsernamePasswordToken(username, password, rememberMe); currentUser.login(token); return "Login success"; } @GetMapping("/profile") public String profile() { return "User profile"; } @GetMapping("/logout") public String logout() { Subject currentUser = SecurityUtils.getSubject(); currentUser.logout(); return "Logout success"; } } ``` 在上述代码中,我们定义了三个接口:`/login`用于用户登录,`/profile`用于查看用户信息,`/logout`用于用户退出登录。 使用Postman或浏览器访问`/login`接口,提交用户名、密码和是否勾选RememberMe的参数,即可完成用户登录。在登录成功后,访问`/profile`接口即可查看用户信息。如果用户勾选了RememberMe,则关闭浏览器再次打开时,用户将自动登录。 至此,我们已经成功创建了一个简单的Spring Boot项目,并集成了Shiro并实现了RememberMe功能。 ## 结语 本文通过详细介绍了如何在Spring Boot项目中集成Shiro,并实现RememberMe功能。通过实战演练和相关的安全性问题的讲解,希望读者能够掌握相关的知识,并在实际项目中正确使用RememberMe功能,提高系统的用户体验。
corwn 最低0.47元/天 解锁专栏
买1年送3月
点击查看下一篇
profit 百万级 高质量VIP文章无限畅学
profit 千万级 优质资源任意下载
profit C知道 免费提问 ( 生成式Al产品 )

相关推荐

李_涛

知名公司架构师
拥有多年在大型科技公司的工作经验,曾在多个大厂担任技术主管和架构师一职。擅长设计和开发高效稳定的后端系统,熟练掌握多种后端开发语言和框架,包括Java、Python、Spring、Django等。精通关系型数据库和NoSQL数据库的设计和优化,能够有效地处理海量数据和复杂查询。
专栏简介
本专栏是一系列关于Spring Boot和Shiro整合的权限教程。通过学习这些教程,您将掌握从基础概念到环境搭建的整体流程,深入了解用户认证和权限控制的实现方式,以及使用JWT和无状态认证的技巧。我们还会介绍基于URL的权限控制、RememberMe功能的实现、Session管理和在分布式环境下的应用,以及多Realm的配置和使用。此外,我们还将探讨密码加密和解密、集成OAuth2实现第三方登录、集成CAS实现单点登录、集成数据库实现动态权限管理,以及集成Ehcache实现缓存管理和Logback实现日志管理的方法。通过这些教程的学习,您将拥有构建安全可靠的Spring Boot和Shiro应用的能力。
最低0.47元/天 解锁专栏
买1年送3月
百万级 高质量VIP文章无限畅学
千万级 优质资源任意下载
C知道 免费提问 ( 生成式Al产品 )

最新推荐

【跨模块协同效应】:SAP MM与PP结合优化库存管理的5大策略

![【跨模块协同效应】:SAP MM与PP结合优化库存管理的5大策略](https://community.sap.com/legacyfs/online/storage/blog_attachments/2013/02/3_189632.jpg) # 摘要 本文旨在探讨SAP MM(物料管理)和PP(生产计划)模块在库存管理中的核心应用与协同策略。首先介绍了库存管理的基础理论,重点阐述了SAP MM模块在材料管理和库存控制方面的作用,以及PP模块如何与库存管理紧密结合实现生产计划的优化。接着,文章分析了SAP MM与PP结合的协同策略,包括集成供应链管理和需求驱动的库存管理方法,以减少库存

【接口保护与电源管理】:RS232通信接口的维护与优化

![【接口保护与电源管理】:RS232通信接口的维护与优化](https://e2e.ti.com/resized-image/__size/1230x0/__key/communityserver-discussions-components-files/138/8551.232.png) # 摘要 本文全面探讨了RS232通信接口的设计、保护策略、电源管理和优化实践。首先,概述了RS232的基本概念和电气特性,包括电压标准和物理连接方式。随后,文章详细分析了接口的保护措施,如静电和过电压防护、物理防护以及软件层面的错误检测机制。此外,探讨了电源管理技术,包括低功耗设计和远程通信设备的案例

零基础Pycharm教程:如何添加Pypi以外的源和库

![零基础Pycharm教程:如何添加Pypi以外的源和库](https://datascientest.com/wp-content/uploads/2022/05/pycharm-1-1024x443.jpg) # 摘要 Pycharm作为一款流行的Python集成开发环境(IDE),为开发人员提供了丰富的功能以提升工作效率和项目管理能力。本文从初识Pycharm开始,详细介绍了环境配置、自定义源与库安装、项目实战应用以及高级功能的使用技巧。通过系统地讲解Pycharm的安装、界面布局、版本控制集成,以及如何添加第三方源和手动安装第三方库,本文旨在帮助读者全面掌握Pycharm的使用,特

【ArcEngine进阶攻略】:实现高级功能与地图管理(专业技能提升)

![【ArcEngine进阶攻略】:实现高级功能与地图管理(专业技能提升)](https://www.a2hosting.com/blog/content/uploads/2019/05/dynamic-rendering.png) # 摘要 本文深入介绍了ArcEngine的基本应用、地图管理与编辑、空间分析功能、网络和数据管理以及高级功能应用。首先,本文概述了ArcEngine的介绍和基础使用,然后详细探讨了地图管理和编辑的关键操作,如图层管理、高级编辑和样式设置。接着,文章着重分析了空间分析的基础理论和实际应用,包括缓冲区分析和网络分析。在此基础上,文章继续阐述了网络和数据库的基本操作

【VTK跨平台部署】:确保高性能与兼容性的秘诀

![【VTK跨平台部署】:确保高性能与兼容性的秘诀](https://opengraph.githubassets.com/6e92ff618ae4b2a046478eb7071feaa58bf735b501d11fce9fe8ed24a197c089/HadyKh/VTK-Examples) # 摘要 本文详细探讨了VTK(Visualization Toolkit)跨平台部署的关键方面。首先概述了VTK的基本架构和渲染引擎,然后分析了在不同操作系统间进行部署时面临的挑战和优势。接着,本文提供了一系列跨平台部署策略,包括环境准备、依赖管理、编译和优化以及应用分发。此外,通过高级跨平台功能的

函数内联的权衡:编译器优化的利与弊全解

![pg140-cic-compiler.pdf](https://releases.llvm.org/10.0.0/tools/polly/docs/_images/LLVM-Passes-all.png) # 摘要 函数内联是编译技术中的一个优化手段,通过将函数调用替换为函数体本身来减少函数调用的开销,并有可能提高程序的执行效率。本文从基础理论到实践应用,全面介绍了函数内联的概念、工作机制以及与程序性能之间的关系。通过分析不同编译器的内联机制和优化选项,本文进一步探讨了函数内联在简单和复杂场景下的实际应用案例。同时,文章也对函数内联带来的优势和潜在风险进行了权衡分析,并给出了相关的优化技

【数据处理差异揭秘】

![【数据处理差异揭秘】](https://static.packt-cdn.com/products/9781838642365/graphics/image/C14197_01_10.jpg) # 摘要 数据处理是一个涵盖从数据收集到数据分析和应用的广泛领域,对于支持决策过程和知识发现至关重要。本文综述了数据处理的基本概念和理论基础,并探讨了数据处理中的传统与现代技术手段。文章还分析了数据处理在实践应用中的工具和案例,尤其关注了金融与医疗健康行业中的数据处理实践。此外,本文展望了数据处理的未来趋势,包括人工智能、大数据、云计算、边缘计算和区块链技术如何塑造数据处理的未来。通过对数据治理和

C++安全编程:防范ASCII文件操作中的3个主要安全陷阱

![C++安全编程:防范ASCII文件操作中的3个主要安全陷阱](https://ask.qcloudimg.com/http-save/yehe-4308965/8c6be1c8b333d88a538d7057537c61ef.png) # 摘要 本文全面介绍了C++安全编程的核心概念、ASCII文件操作基础以及面临的主要安全陷阱,并提供了一系列实用的安全编程实践指导。文章首先概述C++安全编程的重要性,随后深入探讨ASCII文件与二进制文件的区别、C++文件I/O操作原理和标准库中的文件处理方法。接着,重点分析了C++安全编程中的缓冲区溢出、格式化字符串漏洞和字符编码问题,提出相应的防范

时间序列自回归移动平均模型(ARMA)综合攻略:与S命令的完美结合

![时间序列自回归移动平均模型(ARMA)综合攻略:与S命令的完美结合](https://cdn.educba.com/academy/wp-content/uploads/2021/05/Arima-Model-in-R.jpg) # 摘要 时间序列分析是理解和预测数据序列变化的关键技术,在多个领域如金融、环境科学和行为经济学中具有广泛的应用。本文首先介绍了时间序列分析的基础知识,特别是自回归移动平均(ARMA)模型的定义、组件和理论架构。随后,详细探讨了ARMA模型参数的估计、选择标准、模型平稳性检验,以及S命令语言在实现ARMA模型中的应用和案例分析。进一步,本文探讨了季节性ARMA模