详解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,如需转载请自行联系原作者
目录
相关文章
|
3月前
|
设计模式 Java 开发者
如何快速上手【Spring AOP】?从动态代理到源码剖析(下篇)
Spring AOP的实现本质上依赖于代理模式这一经典设计模式。代理模式通过引入代理对象作为目标对象的中间层,实现了对目标对象访问的控制与增强,其核心价值在于解耦核心业务逻辑与横切关注点。在框架设计中,这种模式广泛用于实现功能扩展(如远程调用、延迟加载)、行为拦截(如权限校验、异常处理)等场景,为系统提供了更高的灵活性和可维护性。
|
7月前
|
前端开发 Java 物联网
智慧班牌源码,采用Java + Spring Boot后端框架,搭配Vue2前端技术,支持SaaS云部署
智慧班牌系统是一款基于信息化与物联网技术的校园管理工具,集成电子屏显示、人脸识别及数据交互功能,实现班级信息展示、智能考勤与家校互通。系统采用Java + Spring Boot后端框架,搭配Vue2前端技术,支持SaaS云部署与私有化定制。核心功能涵盖信息发布、考勤管理、教务处理及数据分析,助力校园文化建设与教学优化。其综合性和可扩展性有效打破数据孤岛,提升交互体验并降低管理成本,适用于日常教学、考试管理和应急场景,为智慧校园建设提供全面解决方案。
491 70
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
346 2
|
8月前
|
存储 监控 数据可视化
SaaS云计算技术的智慧工地源码,基于Java+Spring Cloud框架开发
智慧工地源码基于微服务+Java+Spring Cloud +UniApp +MySql架构,利用传感器、监控摄像头、AI、大数据等技术,实现施工现场的实时监测、数据分析与智能决策。平台涵盖人员、车辆、视频监控、施工质量、设备、环境和能耗管理七大维度,提供可视化管理、智能化报警、移动智能办公及分布计算存储等功能,全面提升工地的安全性、效率和质量。
218 0
|
10月前
|
监控 JavaScript 数据可视化
建筑施工一体化信息管理平台源码,支持微服务架构,采用Java、Spring Cloud、Vue等技术开发。
智慧工地云平台是专为建筑施工领域打造的一体化信息管理平台,利用大数据、云计算、物联网等技术,实现施工区域各系统数据汇总与可视化管理。平台涵盖人员、设备、物料、环境等关键因素的实时监控与数据分析,提供远程指挥、决策支持等功能,提升工作效率,促进产业信息化发展。系统由PC端、APP移动端及项目、监管、数据屏三大平台组成,支持微服务架构,采用Java、Spring Cloud、Vue等技术开发。
394 7
|
11月前
|
存储 缓存 Java
Spring面试必问:手写Spring IoC 循环依赖底层源码剖析
在Spring框架中,IoC(Inversion of Control,控制反转)是一个核心概念,它允许容器管理对象的生命周期和依赖关系。然而,在实际应用中,我们可能会遇到对象间的循环依赖问题。本文将深入探讨Spring如何解决IoC中的循环依赖问题,并通过手写源码的方式,让你对其底层原理有一个全新的认识。
276 2
|
前端开发 Java 开发者
Spring生态学习路径与源码深度探讨
【11月更文挑战第13天】Spring框架作为Java企业级开发中的核心框架,其丰富的生态系统和强大的功能吸引了无数开发者的关注。学习Spring生态不仅仅是掌握Spring Framework本身,更需要深入理解其周边组件和工具,以及源码的底层实现逻辑。本文将从Spring生态的学习路径入手,详细探讨如何系统地学习Spring,并深入解析各个重点的底层实现逻辑。
287 9
|
存储 开发框架 Java
什么是Spring?什么是IOC?什么是DI?IOC和DI的关系? —— 零基础可无压力学习,带源码
文章详细介绍了Spring、IOC、DI的概念和关系,解释了控制反转(IOC)和依赖注入(DI)的原理,并提供了IOC的代码示例,阐述了Spring框架作为IOC容器的应用。
863 1
什么是Spring?什么是IOC?什么是DI?IOC和DI的关系? —— 零基础可无压力学习,带源码
|
Java Spring
Spring底层架构源码解析(三)
Spring底层架构源码解析(三)
511 5