详解Spring Security进阶身份认证之UserDetailsService(附源码)

简介:

  在上一篇Spring Security身份认证博文中,我们采用了配置文件的方式从数据库中读取用户进行登录。虽然该方式的灵活性相较于静态账号密码的方式灵活了许多,但是将数据库的结构暴露在明显的位置上,绝对不是一个明智的做法。本文通过Java代码实现UserDetailsService接口来实现身份认证。


    1.1 UserDetailsService在身份认证中的作用


    Spring Security中进行身份验证的是AuthenticationManager接口,ProviderManager是它的一个默认实现,但它并不用来处理身份认证,而是委托给配置好的AuthenticationProvider,每个AuthenticationProvider会轮流检查身份认证。检查后或者返回Authentication对象或者抛出异常。


    验证身份就是加载响应的UserDetails,看看是否和用户输入的账号、密码、权限等信息匹配。此步骤由实现AuthenticationProvider的DaoAuthenticationProvider(它利用UserDetailsService验证用户名、密码和授权)处理。包含 GrantedAuthority 的 UserDetails对象在构建 Authentication对象时填入数据。


wKiom1TJ_G-QennHAALbrutxzyc077.jpg


    

    1.2 配置UserDetailsService


    1.2.1 更改Spring-Security.xml中身份的方式,使用自定义的UserDetailsService。


1
< span  style = "font-family:arial, helvetica, sans-serif;" >< security:authentication-manager >< br >  < security:authentication-provider  user-service-ref = "favUserDetailService" >< br >      </ security:authentication-provider >< br > </ security:authentication-manager >< br >< br > < bean  id = "favUserDetailService"  class = "com.favccxx.favsecurity.security.FavUserDetailService"  />< br ></ span >


    

    1.2.2 新建FavUserDetailsService.java,实现UserDetailsService接口。为了降低学习的难度,这里并没有与数据库进行集成,而是采用模拟从数据库中获取用户的方式进行身份验证。示例代码如下:


1
<span style= "font-family:arial, helvetica, sans-serif;" > package  com.favccxx.favsecurity.security;<br><br> import  java.util.ArrayList;<br> import  java.util.Collection;<br> import  java.util.List;<br><br> import  org.apache.logging.log4j.LogManager;<br> import  org.apache.logging.log4j.Logger;<br> import  org.springframework.security.core.GrantedAuthority;<br> import  org.springframework.security.core.authority.SimpleGrantedAuthority;<br> import  org.springframework.security.core.userdetails.User;<br> import  org.springframework.security.core.userdetails.UserDetails;<br> import  org.springframework.security.core.userdetails.UserDetailsService;<br> import  org.springframework.security.core.userdetails.UsernameNotFoundException;<br><br> public  class  FavUserDetailService  implements  UserDetailsService {<br><br>  private  static  final  Logger logger = LogManager.getLogger(FavUserDetailService. class );<br><br>   /**<br>  * 根据用户名获取用户 - 用户的角色、权限等信息<br>   */ <br>  public  UserDetails loadUserByUsername(String username)<br>           throws  UsernameNotFoundException {<br>      UserDetails userDetails =  null ;<br>      try  {<br>           com.favccxx.favsecurity.pojo.User favUser =  new  com.favccxx.favsecurity.pojo.User();<br>            favUser.setUsername( "favccxx" );<br>         favUser.setPassword( "favccxx" );<br>         Collection<GrantedAuthority> authList = getAuthorities();<br>         userDetails =  new  User(username, favUser.getPassword().toLowerCase(), true , true , true , true ,authList);<br>     }  catch  (Exception e) {<br>         e.printStackTrace();<br>        }<br><br><br>        return  userDetails;<br> }<br><br>    /**<br>  * 获取用户的角色权限,为了降低实验的难度,这里去掉了根据用户名获取角色的步骤<br>     * @param <br>   * @return<br>   */ <br>  private  Collection<GrantedAuthority> getAuthorities(){<br>        List<GrantedAuthority> authList =  new  ArrayList<GrantedAuthority>();  <br>     authList.add( new  SimpleGrantedAuthority( "ROLE_USER" )); <br>     authList.add( new  SimpleGrantedAuthority( "ROLE_ADMIN" ));<br><br>      return  authList;<br>    }<br><br><br><br>}<br></span>


    1.2.3 启动应用服务器,只要用户名和密码不全是favccxx,就会产生下面的错误。


wKioL1TJ_giBpILOAAF2QL-ASRg527.jpg


    用户名和密码都输入favccxx,则登陆成功


wKiom1TJ_SexYSkWAAHYMLuVe1E670.jpg


    1.3 跟踪UserDetailsService。


    身份认证的调用流程图如下,用户可下载Spring Security源代码跟踪调试。


wKiom1TJ_WvBCs7aAAHZn6syY1Y174.jpg


    1.4 如不能正常运行,点这里看看源代码吧





本文转自 genuinecx 51CTO博客,原文链接:http://blog.51cto.com/favccxx/1609692,如需转载请自行联系原作者
目录
相关文章
|
10天前
|
数据采集 监控 前端开发
二级公立医院绩效考核系统源码,B/S架构,前后端分别基于Spring Boot和Avue框架
医院绩效管理系统通过与HIS系统的无缝对接,实现数据网络化采集、评价结果透明化管理及奖金分配自动化生成。系统涵盖科室和个人绩效考核、医疗质量考核、数据采集、绩效工资核算、收支核算、工作量统计、单项奖惩等功能,提升绩效评估的全面性、准确性和公正性。技术栈采用B/S架构,前后端分别基于Spring Boot和Avue框架。
|
2月前
|
缓存 Java 开发工具
Spring是如何解决循环依赖的?从底层源码入手,详细解读Spring框架的三级缓存
三级缓存是Spring框架里,一个经典的技术点,它很好地解决了循环依赖的问题,也是很多面试中会被问到的问题,本文从源码入手,详细剖析Spring三级缓存的来龙去脉。
169 24
Spring是如何解决循环依赖的?从底层源码入手,详细解读Spring框架的三级缓存
|
2月前
|
缓存 安全 Java
Spring框架中Bean是如何加载的?从底层源码入手,详细解读Bean的创建流程
从底层源码入手,通过代码示例,追踪AnnotationConfigApplicationContext加载配置类、启动Spring容器的整个流程,并对IOC、BeanDefinition、PostProcesser等相关概念进行解释
168 24
Spring框架中Bean是如何加载的?从底层源码入手,详细解读Bean的创建流程
|
2月前
|
XML 缓存 Java
手写Spring源码(简化版)
Spring包下的类、手写@ComponentScan注解、@Component注解、@Autowired注解、@Scope注解、手写BeanDefinition、BeanNameAware、InitializingBean、BeanPostProcessor 、手写AnnotationConfigApplicationContext
手写Spring源码(简化版)
|
26天前
|
Java Spring
Spring底层架构源码解析(三)
Spring底层架构源码解析(三)
|
26天前
|
XML Java 数据格式
Spring底层架构源码解析(二)
Spring底层架构源码解析(二)
|
1月前
|
Java Spring 容器
Spring IOC、AOP与事务管理底层原理及源码解析
【10月更文挑战第1天】Spring框架以其强大的控制反转(IOC)和面向切面编程(AOP)功能,成为Java企业级开发中的首选框架。本文将深入探讨Spring IOC和AOP的底层原理,并通过源码解析来揭示其实现机制。同时,我们还将探讨Spring事务管理的核心原理,并给出相应的源码示例。
117 9
|
1月前
|
设计模式 JavaScript Java
Spring 事件监听机制源码
Spring 提供了事件发布订阅机制,广泛应用于项目中。本文介绍了如何通过自定义事件类、订阅类和发布类实现这一机制,并展示了如何监听 SpringBoot 启动过程中的多个事件(如 `ApplicationStartingEvent`、`ApplicationEnvironmentPreparedEvent` 等)。通过掌握这些事件,可以更好地理解 SpringBoot 的启动流程。示例代码展示了从事件发布到接收的完整过程。
|
1月前
|
缓存 Java Spring
源码解读:Spring如何解决构造器注入的循环依赖?
本文详细探讨了Spring框架中的循环依赖问题,包括构造器注入和字段注入两种情况,并重点分析了构造器注入循环依赖的解决方案。文章通过具体示例展示了循环依赖的错误信息及常见场景,提出了三种解决方法:重构代码、使用字段依赖注入以及使用`@Lazy`注解。其中,`@Lazy`注解通过延迟初始化和动态代理机制有效解决了循环依赖问题。作者建议优先使用`@Lazy`注解,并提供了详细的源码解析和调试截图,帮助读者深入理解其实现机制。
20 1
|
1月前
|
存储 开发框架 Java
什么是Spring?什么是IOC?什么是DI?IOC和DI的关系? —— 零基础可无压力学习,带源码
文章详细介绍了Spring、IOC、DI的概念和关系,解释了控制反转(IOC)和依赖注入(DI)的原理,并提供了IOC的代码示例,阐述了Spring框架作为IOC容器的应用。
25 0
什么是Spring?什么是IOC?什么是DI?IOC和DI的关系? —— 零基础可无压力学习,带源码