Shiro在springboot中如何快速实现

其他教程   发布日期:2023年06月12日   浏览次数:392

这篇“Shiro在springboot中如何快速实现”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Shiro在springboot中如何快速实现”文章吧。

    一、shiro使用必须了解的知识

    1、shiro是什么?

    • 1、Apache Shiro是一个Java的安全(权限)框架

    • 2、可以容易的开发出足够好的应用,既可以在JavaEE中使用,也可以在JavaSE中使用

    • 3、shiro可以完成,认证、授权、加密、会话管理,web集成、缓存等

    2、shiro架构三个常用三大核心对象

    • Subject:用户

    • SecurityManager:管理所有用户

    • Readim:连接数据

    3、在springboot中使用时,主要可将其看作两个模块(请求过滤模块、认证授权模块)

    1、认证授权模块:在认证授权模块中主要包含两个方面,分别是认证和授权。认证就是指对用户登录的情况进行判定;授权就是指对当前用户所拥有的角色、权限进行获取并将其交给AuthoriztionInfo,使其能够将相关信息交给Shiro
    2、请求过滤模块:根据当前用户所拥有的权限、角色等信息来进行判断是否具有请求的权限(即是否能够请求当前要访问的地址),如果该用户具有访问当前请求地址的权限,则放行,否则进行拦截
    3、以上是使用shiro框架进行权限认证拦截的最基本实现,此外还可以通过对密码进行加密,登录次数限流(redis)等功能重写来按照自己实际业务情况进行学习

    4、依赖

    1. <!-- 后台拦截-->
    2. <dependency>
    3. <groupId>org.apache.shiro</groupId>
    4. <artifactId>shiro-spring</artifactId>
    5. <version>1.4.0</version>
    6. </dependency>

    二、具体使用

    1、编写配置类(config)

    1.1、Shiro过滤对象(ShiroFilterFactoryBean)

    1. @Bean
    2. public ShiroFilterFactoryBean getShiroFilterFactoryBean(@Qualifier(SecurityManager) DefaultWebSecurityManager securityManager){
    3. ShiroFilterFactiryBean bean = new ShiroFilterFactoryBean()
    4. //关联SecurityManager设置安全管理器
    5. bean.setSecurityManager(securityManager)
    6. //添加内置过滤器
    7. /*
    8. anon:无需过滤就可以访问
    9. authc:必须认证了才可访问(登录后才可访问)
    10. user:必须拥有"记住我"功能才可访问
    11. perms:拥有对某个资源的权限才可以访问
    12. role:拥有某个角色权限才可访问
    13. */
    14. Map<String,String> filterMap = new LinkedHashMap<>();
    15. //拦截
    16. //filterMap.put("页面地址","内置过滤器")
    17. //filterMap.put("/user/name","anon")
    18. //filterMap.put("/user/book","authc")
    19. //具有user:add权限时才可以访问/user/name
    20. //perms中的“user:add”与数据库中对应权限要一致
    21. filterMap.put("/user/name","perms[user:add]")
    22. //授权,正常情况下,没有授权会跳转到未授权页面
    23. bean.setUnauthorizedUrl("未授权时跳转的页面")
    24. //创建一个过滤器链(其中内容通过Map存储)
    25. bean.setFilterChainDefinitionMap(FilterMap);
    26. //设置登录请求(登录的地址添加,当使用"authc"时,如果未登录,则跳转到登录页面)
    27. bean.setLoginUrl("/login")
    28. return bean;
    29. }

    1.2、Shiro安全对象(DefaultWebSecurity)

    1. //@Qualifier:引入bena对象
    2. @Bean(name="SecurityManager")
    3. public DefaultWebSecurityManager getDefaultWebSecurityManager(@Qualifier("MyRealm") MyRealm myRealm){
    4. DefaultWebSecurityManager securityManager = new DefaultWebSecurotyManager();
    5. //关联MyRealm
    6. securityManager.setRealm(myRealm);
    7. return securityManager;
    8. }

    1.3、创建realm对象(自定义)

    1. //将自定义的realm对象交给spring
    2. //@Bean(name="MyRealm")中name属性不加默认名称为方法名
    3. @Bean(name="MyRealm")
    4. public MyRealm MyRealm(){
    5. return new MyRealm();
    6. }

    2、创建realm对象

    2.1、自定义realm类去继承AuthorizingRealm类

    1. project AthorizationInfo doGetAuthorizationInfo(PrincipalCollection principals){
    2. //1、权限信息对象info,用来存放查出的用户的所有的角色(role)及权限(permission)
    3. SimpleAuthorizationInfo info = new SimpleAuthorizationInfo();
    4. //2、拿到当前登录的对象信息,通过认证方法SimpleAuthenticationInfo(第一个参数)已经进行存入
    5. User user =(userSecurityUtils.getSubject().getPrincipal();
    6. //3、将该对象的角色信息进行存入
    7. // 赋予角色
    8. List<Role> roleList = roleService.listRolesByUserId(userId);
    9. for (Role role : roleList) {
    10. info.addRole(role.getName());
    11. }
    12. //4、设置该用户的权限
    13. infO.addStringPermission(user.getPerms())
    14. //5、将该对象的权限信息进行存入(permissionSet一个权限信息的集合)
    15. info.setStringPermissions(permissionSet);
    16. return info;
    17. }

    认证:

    1. project AuthenticationInfo doGetAuthorizationInfo(AuthenticationToken token){
    2. //1、拿到用户登陆的信息
    3. UsernamePasswordToken userToken =(UsernamePasswordToken) token;
    4. //2、通过用户名(userToken.getUsername)获取数据库中的对象user
    5. //如果获取对象user为空则该用户不从在,返回return null(抛出用户不存在异常)
    6. if (user == null) {
    7. throw new UnknownAccountException("账号不存在!");
    8. //或直接 return null;
    9. }
    10. //3、密码认证,有shiro完成(AuthenticationInfo是一个接口,SimpleAuthenticationInfo是其接口的实现类)
    11. //也可对密码进行加密 如MD5 MD5盐值
    12. return new SimpleAuthenticationInfo("用户对象信息(user)","通过用户从数据库中获得的用户密码(user.password)","")
    13. }

    3、登录用户的信息传入(通过controller获取登录的请求信息)

    1. //获取当前用户
    2. Subject subject = SecurityUtils.getSubject();
    3. //封装用户的登录数据(username:用户登陆时传入的账号;password:用户登陆时传入的密码)
    4. UsernamePasswordToken token = new UsernamePasswordToken(username,password);
    5. //执行登录(如果有异常则登录失败,没有异常则登录成功,在Shiro中已经为我们封装了登录相关的异常,直接使用即可)
    6. try{
    7. subject.login(token);//执行登录成功后
    8. return "首页"
    9. }catch(UnknowAccountException e){//用户名不存在
    10. return "login"
    11. }catch(IncorrectCredentialsException e){//密码不存在
    12. return "login"
    13. }
    14. 注意:该方法中登录失败后返回的是跳转的页面,故不可用@ResponseBody

    三、具体实现

    1、realm实现

    1. package com.lingmeng.shiro;
    2. import com.lingmeng.pojo.entity.Admin;
    3. import com.lingmeng.pojo.entity.Permission;
    4. import com.lingmeng.pojo.entity.Role;
    5. import com.lingmeng.pojo.resp.BaseResp;
    6. import com.lingmeng.service.AdminService;
    7. import com.lingmeng.service.RoleService;
    8. import org.apache.shiro.SecurityUtils;
    9. import org.apache.shiro.authc.*;
    10. import org.apache.shiro.authz.AuthorizationInfo;
    11. import org.apache.shiro.authz.SimpleAuthorizationInfo;
    12. import org.apache.shiro.realm.AuthorizingRealm;
    13. import org.apache.shiro.subject.PrincipalCollection;
    14. import org.apache.shiro.subject.Subject;
    15. import org.springframework.beans.factory.annotation.Autowired;
    16. import java.util.HashSet;
    17. import java.util.Set;
    18. public class MyRealm extends AuthorizingRealm {
    19. @Autowired
    20. RoleService roleService;
    21. @Autowired
    22. AdminService adminService;
    23. //授权
    24. @Override
    25. protected AuthorizationInfo doGetAuthorizationInfo(PrincipalCollection principalCollection) {
    26. SimpleAuthorizationInfo info = new SimpleAuthorizationInfo();
    27. //获取用户信息
    28. Subject subject = SecurityUtils.getSubject();
    29. Admin admin =(Admin) subject.getPrincipal();
    30. //获取用户的权限及角色信息
    31. BaseResp baseResp = roleService.selectOne(admin.getUsername());
    32. Role role = (Role) baseResp.getData();
    33. //将获取的角色及权限进行存入
    34. if (role!=null){
    35. //角色存入
    36. info.addRole(role.getName());
    37. //权限信息进行存入
    38. Set<String> perms = new HashSet<>();
    39. for (Permission perm : role.getPerms()) {
    40. perms.add(perm.getUrl());
    41. }
    42. info.setStringPermissions(perms);
    43. }
    44. return info;
    45. }
    46. //认证
    47. @Override
    48. protected AuthenticationInfo doGetAuthenticationInfo(AuthenticationToken authenticationToken) throws AuthenticationException {
    49. //获取登录信息(登录的账号)
    50. String username =(String)authenticationToken.getPrincipal();
    51. // UsernamePasswordToken userToken =(UsernamePasswordToken) authenticationToken;拿到登录时传入的账号和密码对象
    52. //从数据库中查询该对象的信息
    53. Admin admin = adminService.selectOne(username);
    54. if (admin==null){
    55. throw new UnknownAccountException("账号不存在");
    56. }
    57. return new SimpleAuthenticationInfo(admin,admin.getPassword(),this.getName());
    58. }
    59. }

    2、controller实现

    1. package com.lingmeng.controller;
    2. import com.lingmeng.pojo.entity.Admin;
    3. import com.lingmeng.pojo.resp.BaseResp;
    4. import org.apache.shiro.SecurityUtils;
    5. import org.apache.shiro.authc.IncorrectCredentialsException;
    6. import org.apache.shiro.authc.UnknownAccountException;
    7. import org.apache.shiro.authc.UsernamePasswordToken;
    8. import org.apache.shiro.subject.Subject;
    9. import org.springframework.web.bind.annotation.*;
    10. @RestController
    11. public class AdminController {
    12. @PostMapping("background/login")
    13. public BaseResp login(@RequestBody Admin admin){
    14. Subject subject = SecurityUtils.getSubject();
    15. UsernamePasswordToken token = new UsernamePasswordToken(admin.getUsername(), admin.getPassword());
    16. try{
    17. subject.login(token);
    18. return BaseResp.SUCCESS("登录成功",null,null);
    19. }catch (UnknownAccountException e){//账号不存在
    20. return BaseResp.FAIL(201,"账号不存在");
    21. }catch(IncorrectCredentialsException incorrectCredentialsException){//密码错误
    22. return BaseResp.FAIL(201,"密码错误") ;
    23. }
    24. }
    25. @GetMapping("/background/exitLogin")
    26. public BaseResp exitLogin(){
    27. Subject subject = SecurityUtils.getSubject();
    28. System.out.println(subject.getPrincipal());
    29. try{
    30. subject.logout();//退出登录
    31. return BaseResp.SUCCESS("退出登录",null,null);
    32. }catch(Exception e){
    33. return BaseResp.FAIL(202,"退出失败");
    34. }
    35. }
    36. }

    3、config实现

    1. package com.lingmeng.config;
    2. import com.lingmeng.shiro.MyRealm;
    3. import org.apache.shiro.spring.web.ShiroFilterFactoryBean;
    4. import org.apache.shiro.web.mgt.DefaultWebSecurityManager;
    5. import org.springframework.beans.factory.annotation.Qualifier;
    6. import org.springframework.context.annotation.Bean;
    7. import org.springframework.context.annotation.Configuration;
    8. import java.util.LinkedHashMap;
    9. import java.util.Map;
    10. @Configuration
    11. public class ShiroConfig {
    12. @Bean
    13. public ShiroFilterFactoryBean shiroFilterFactoryBean(@Qualifier("securityManager") DefaultWebSecurityManager securityManager){
    14. ShiroFilterFactoryBean bean = new ShiroFilterFactoryBean();
    15. bean.setSecurityManager(securityManager);
    16. //配置请求拦截并存入map中
    17. /*
    18. anon:无需过滤就可以访问
    19. authc:必须认证了才可访问(登录后才可访问)
    20. user:必须拥有"记住我"功能才可访问
    21. perms:拥有对某个资源的权限才可以访问
    22. role:拥有某个角色权限才可访问
    23. */
    24. Map<String, String> map = new LinkedHashMap<>();
    25. map.put("/background/**","authc");
    26. map.put("background/login","anon");
    27. bean.setFilterChainDefinitionMap(map);
    28. //设置未授权跳转地址
    29. bean.setUnauthorizedUrl("");
    30. //设置登录地址
    31. bean.setLoginUrl("/background/login");
    32. return bean;
    33. }
    34. @Bean("securityManager")
    35. public DefaultWebSecurityManager defaultWebSecurityManager(@Qualifier("myRealm") MyRealm myRealm){
    36. return new DefaultWebSecurityManager(myRealm);
    37. }
    38. @Bean()
    39. public MyRealm myRealm(){
    40. return new MyRealm();
    41. }
    42. }

    以上是一些shiro在springboot中的基本用法,希望能够对大家学习有所帮助(代码中的实体,角色,权限根据自己数据库查询结果进行替换即可)。

    以上就是Shiro在springboot中如何快速实现的详细内容,更多关于Shiro在springboot中如何快速实现的资料请关注九品源码其它相关文章!