什么是单点登录(SSO)?单点登录(SSO)到底什么意思?【附逻辑Demo实例】

article/2025/10/4 19:36:35

     在程序开发中,特别是网站类开发,会接触到单点登录(SSO)什么是单点登录?,单点登录(SSO)有什么用?下面软币网小编来来介绍一下:

一、什么是单点登录?

单点登录的英文名叫做:Single Sign On(简称SSO)。

初学/以前的时候,一般我们就单系统,所有的功能都在同一个系统上。

单点登录(SSO)
单点登录(SSO)

 

后来,我们为了合理利用资源和降低耦合性,于是把单系统拆分成多个子系统。

  • 回顾:分布式基础知识
单点登录(SSO)
单点登录(SSO) - 舆管家

拆分成多个子系统

比如阿里系的淘宝和天猫,很明显地我们可以知道这是两个系统,但是你在使用的时候,登录了天猫,淘宝也会自动登录。

单点登录(SSO)
单点登录(SSO)

 

简单来说,单点登录就是在多个系统中,用户只需一次登录,各个系统即可感知该用户已经登录。

二、回顾单系统登录

在我初学JavaWeb的时候,登录和注册是我做得最多的一个功能了(初学Servlet的时候做过、学SpringMVC的时候做过、跟着做项目的时候做过…),反正我也数不清我做了多少次登录和注册的功能了…这里简单讲述一下我们初学时是怎么做登录功能的。

HTTP是无状态的协议

众所周知,HTTP是无状态的协议,这意味着服务器无法确认用户的信息。于是乎,W3C就提出了:给每一个用户都发一个通行证,无论谁访问的时候都需要携带通行证,这样服务器就可以从通行证上确认用户的信息。通行证就是Cookie

如果说Cookie是检查用户身上的”通行证“来确认用户的身份,那么Session就是通过检查服务器上的”客户明细表“来确认用户的身份的。Session相当于在服务器中建立了一份“客户明细表”

HTTP协议是无状态的,Session不能依据HTTP连接来判断是否为同一个用户。于是乎:服务器向用户浏览器发送了一个名为JESSIONID的Cookie,它的值是Session的id值。其实Session是依据Cookie来识别是否是同一个用户

所以,一般我们单系统实现登录会这样做:

  • 登录:将用户信息保存在Session对象中
  • - 如果在Session对象中能查到,说明已经登录
  • 如果在Session对象中查不到,说明没登录(或者已经退出了登录)
  • 注销(退出登录):从Session中删除用户的信息
  • 记住我(关闭掉浏览器后,重新打开浏览器还能保持登录状态):配合Cookie来用

我之前Demo的代码,可以参考一下:

/*** 用户登陆*/
@PostMapping(value = "/user/session", produces = {"application/json;charset=UTF-8"})
public Result login(String mobileNo, String password, String inputCaptcha, HttpSession session, HttpServletResponse response) {//判断验证码是否正确if (WebUtils.validateCaptcha(inputCaptcha, "captcha", session)) {//判断有没有该用户User user = userService.userLogin(mobileNo, password);if (user != null) {/*设置自动登陆,一个星期. 将token保存在数据库中*/String loginToken = WebUtils.md5(new Date().toString() + session.getId());user.setLoginToken(loginToken);User user1 = userService.userUpload(user);session.setAttribute("user", user1);CookieUtil.addCookie(response,"loginToken",loginToken,604800);return ResultUtil.success(user1);} else {return ResultUtil.error(ResultEnum.LOGIN_ERROR);}} else {return ResultUtil.error(ResultEnum.CAPTCHA_ERROR);}
}
/*** 用户退出*/
@DeleteMapping(value = "/session", produces = {"application/json;charset=UTF-8"})
public Result logout(HttpSession session,HttpServletRequest request,HttpServletResponse response ) {//删除session和cookiesession.removeAttribute("user");CookieUtil.clearCookie(request, response, "loginToken");return ResultUtil.success();
}
/**
* @author ozc
* @version 1.0
* <p>
* 拦截器;实现自动登陆功能
*/
public class UserInterceptor implements HandlerInterceptor {
@Autowired
private UserService userService;
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object o) throws Exception {User sessionUser = (User) request.getSession().getAttribute("user");// 已经登陆了,放行if (sessionUser != null) {return true;} else {//得到带过来cookie是否存在String loginToken = CookieUtil.findCookieByName(request, "loginToken");if (StringUtils.isNotBlank(loginToken)) {//到数据库查询有没有该CookieUser user = userService.findUserByLoginToken(loginToken);if (user != null) {request.getSession().setAttribute("user", user);return true;} else {//没有该Cookie与之对应的用户(Cookie不匹配)CookieUtil.clearCookie(request, response, "loginToken");return false;}} else {//没有cookie、也没有登陆。是index请求获取用户信息,可以放行if (request.getRequestURI().contains("session")) {return true;}//没有cookie凭证response.sendRedirect("/login.html");return false;}}
}
}

总结一下上面代码的思路:

  • 用户登录时,验证用户的账户和密码
  • 生成一个Token保存在数据库中,将Token写到Cookie中
  • 将用户数据保存在Session中
  • 请求时都会带上Cookie,检查有没有登录,如果已经登录则放行

Cookie的作用是什么?和Session有什么区别?

Cookie 和 Session都是用来跟踪浏览器用户身份的会话方式,但是两者的应用场景不太一样。

Cookie 一般用来保存用户信息 比如①我们在 Cookie 中保存已经登录过得用户信息,下次访问网站的时候页面可以自动帮你登录的一些基本信息给填了;②一般的网站都会有保持登录也就是说下次你再访问网站的时候就不需要重新登录了,这是因为用户登录的时候我们可以存放了一个 Token 在 Cookie 中,下次登录的时候只需要根据 Token 值来查找用户即可(为了安全考虑,重新登录一般要将 Token 重写);③登录一次网站后访问网站其他页面不需要重新登录。Session 的主要作用就是通过服务端记录用户的状态。典型的场景是购物车,当你要添加商品到购物车的时候,系统不知道是哪个用户操作的,因为 HTTP 协议是无状态的。服务端给特定的用户创建特定的 Session 之后就可以标识这个用户并且跟踪这个用户了。

Cookie 数据保存在客户端(浏览器端),Session 数据保存在服务器端。

Cookie 存储在客户端中,而Session存储在服务器上,相对来说 Session 安全性更高。如果使用 Cookie 的一些敏感信息不要写入 Cookie 中,最好能将 Cookie 信息加密然后使用到的时候再去服务器端解密。

三、多系统登录的问题与解决

3.1 Session不共享问题

单系统登录功能主要是用Session保存用户信息来实现的,但我们清楚的是:多系统即可能有多个Tomcat,而Session是依赖当前系统的Tomcat,所以系统A的Session和系统B的Session是不共享的。

A的Session和系统B的Session是不共享的
A的Session和系统B的Session是 不共享的 舆情宝

解决系统之间Session不共享问题有一下几种方案:

  • Tomcat集群Session全局复制(集群内每个tomcat的session完全同步)【会影响集群的性能呢,不建议】
  • 根据请求的IP进行Hash映射到对应的机器上(这就相当于请求的IP一直会访问同一个服务器)【如果服务器宕机了,会丢失了一大部分Session的数据,不建议】
  • 把Session数据放在Redis中(使用Redis模拟Session)【建议

我们可以将登录功能单独抽取出来,做成一个子系统。

什么是单点登录(SSO)?单点登录(SSO)到底什么意思?

抽取出来成为子系统

SSO(登录系统)的逻辑如下:

// 登录功能(SSO单独的服务)
@Override
public TaotaoResult login(String username, String password) throws Exception {//根据用户名查询用户信息TbUserExample example = new TbUserExample();Criteria criteria = example.createCriteria();criteria.andUsernameEqualTo(username);List<TbUser> list = userMapper.selectByExample(example);if (null == list || list.isEmpty()) {return TaotaoResult.build(400, "用户不存在");}//核对密码TbUser user = list.get(0);if (!DigestUtils.md5DigestAsHex(password.getBytes()).equals(user.getPassword())) {return TaotaoResult.build(400, "密码错误");}//登录成功,把用户信息写入redis//生成一个用户tokenString token = UUID.randomUUID().toString();jedisCluster.set(USER_TOKEN_KEY + ":" + token, JsonUtils.objectToJson(user));//设置session过期时间jedisCluster.expire(USER_TOKEN_KEY + ":" + token, SESSION_EXPIRE_TIME);return TaotaoResult.ok(token);
}

其他子系统登录时,请求SSO(登录系统)进行登录,将返回的token写到Cookie中,下次访问时则把Cookie带上:

public TaotaoResult login(String username, String password, HttpServletRequest request, HttpServletResponse response) {//请求参数Map<String, String> param = new HashMap<>();param.put("username", username);param.put("password", password);//登录处理String stringResult = HttpClientUtil.doPost(REGISTER_USER_URL + USER_LOGIN_URL, param);TaotaoResult result = TaotaoResult.format(stringResult);//登录出错if (result.getStatus() != 200) {return result;}//登录成功后把取token信息,并写入cookieString token = (String) result.getData();//写入cookieCookieUtils.setCookie(request, response, "TT_TOKEN", token);//返回成功return result;
}

 

总结:

  • SSO系统生成一个token,并将用户信息存到Redis中,并设置过期时间
  • 其他系统请求SSO系统进行登录,得到SSO返回的token,写到Cookie中
  • 每次请求时,Cookie都会带上,拦截器得到token,判断是否已经登录

到这里,其实我们会发现其实就两个变化:

  • 将登陆功能抽取为一个系统(SSO),其他系统请求SSO进行登录
  • 本来将用户信息存到Session,现在将用户信息存到Redis

 

3.2 Cookie跨域的问题

上面我们解决了Session不能共享的问题,但其实还有另一个问题。Cookie是不能跨域的

比如说,我们请求http://www.ibixue.com/时,浏览器会自动把ibixue.com的Cookie带过去给google的服务器,而不会把http://www.gookang.com的Cookie带过去给google的服务器。

这就意味着,由于域名不同,用户向系统A登录后,系统A返回给浏览器的Cookie,用户再请求系统B的时候不会将系统A的Cookie带过去。

针对Cookie存在跨域问题,有几种解决方案:

  1. 服务端将Cookie写到客户端后,客户端对Cookie进行解析,将Token解析出来,此后请求都把这个Token带上就行了
  2. 多个域名共享Cookie,在写到客户端的时候设置Cookie的domain。
  3. 将Token保存在SessionStroage中(不依赖Cookie就没有跨域的问题了)

到这里,我们已经可以实现单点登录了。

3.3 CAS原理

说到单点登录,就肯定会见到这个名词:CAS (Central Authentication Service),下面说说CAS是怎么搞的。

如果已经将登录单独抽取成系统出来,我们还能这样玩。现在我们有两个系统,分别是www.ibixue.com和www.gookang.com,一个 www.ruanally.com

首先,用户想要访问系统A www.ibixue.com受限的资源(比如说购物车功能,购物车功能需要登录后才能访问),系统A www.ibixue.com发现用户并没有登录,于是重定向到sso认证中心,并将自己的地址作为参数。请求的地址如下:

  • www.ruanally.com?service=www.ibixue.com

sso认证中心发现用户未登录,将用户引导至登录页面,用户进行输入用户名和密码进行登录,用户与认证中心建立全局会话(生成一份Token,写到Cookie中,保存在浏览器上)

什么是单点登录(SSO)?单点登录(SSO)到底什么意思?【附逻辑Demo实例】
什么是单点登录(SSO)?单点登录(SSO)到底什么意思?【附逻辑Demo实例】

随后,认证中心重定向回系统A,并把Token携带过去给系统A,重定向的地址如下:

  • www.ibixue.com?token=xxxxxxx

接着,系统A去sso认证中心验证这个Token是否正确,如果正确,则系统A和用户建立局部会话(创建Session)。到此,系统A和用户已经是登录状态了。

什么是单点登录(SSO)?单点登录(SSO)到底什么意思?【附逻辑Demo实例】
什么是单点登录(SSO)?单点登录(SSO)到底什么意思?【附逻辑Demo实例】

 

此时,用户想要访问系统B  www.gookang.com受限的资源(比如说订单功能,订单功能需要登录后才能访问),系统B www.gookang.com发现用户并没有登录,于是重定向到sso认证中心,并将自己的地址作为参数。请求的地址如下:

  • www.ruanally.com?service=www.gookang.com

注意,因为之前用户与认证中心www.ruanally.com已经建立了全局会话(当时已经把Cookie保存到浏览器上了),所以这次系统B重定向到认证中心www.ruanally.com是可以带上Cookie的。

认证中心根据带过来的Cookie发现已经与用户建立了全局会话了,认证中心重定向回系统B,并把Token携带过去给系统B,重定向的地址如下:

  • www.gookang.com?token=xxxxxxx

接着,系统B去sso认证中心验证这个Token是否正确,如果正确,则系统B和用户建立局部会话(创建Session)。到此,系统B和用户已经是登录状态了。

标题

 

看到这里,其实SSO认证中心就类似一个中转站

部分转自:软盟网  

 

 

 

 


http://chatgpt.dhexx.cn/article/LD2BIjxq.shtml

相关文章

分布式系统服务单点问题的探讨

分布式系统服务单点问题的探讨 无状态服务的单点问题有状态服务的单点问题 在分布式系统中&#xff0c;单点问题是一个比较常见的问题&#xff0c;对于单点问题可以分为有状态服务的单点问题和无状态服务的单点问题。 无状态服务的单点问题 对于无状态的服务&#xff0c;单点问…

08-微服务版单点登陆系统(SSO)实践

文章目录 单点登陆系统简介背景分析单点登陆系统概述单点登陆系统解决方案设计 单点登陆系统初步设计服务设计工程结构设计 SSO父工程创建及初始化创建父工程父工程pom文件初始配置 系统基础服务工程设计及实现业务描述表结构设计工程数据初始化创建系统服务工程并初始化Pojo对…

IDEA同时打开多个项目之解决方法

对于我们开发人员来说&#xff0c;有时想在一个窗口中打开多个项目&#xff0c;这时打开新项目时只有两个选项&#xff1a; 覆盖本窗口项目/在新窗口打开 解决方法&#xff1a; 1、点击左上角 File--->Project Structure&#xff1a; 2、然后选择Modules这一栏&#xff0c…

idea一个工作空间打开多个项目

很多时候&#xff0c;多个项目之间并不是parent和module的关系&#xff0c;如果不是parent和module的关系&#xff0c;idea一个工作空间只能打开一个项目&#xff0c;这样的话&#xff0c;如果有多个项目&#xff0c;来回切换视图让人烦不胜烦。下面介绍个在idea一个工作空间打…

Idea打开多个项目文件时,不自动识别maven项目

问题描述 idea打开多个项目文件&#xff08;直接打开的多个项目的文件夹&#xff09;&#xff0c;却没有识别出maven项目。网上查阅&#xff0c;解决方案为&#xff1a; 选中项目的pom.xml文件,右键-" add as maven project"。 但由于项目过多&#xff0c;不方便一个…

idea在一个窗口打开多个项目

1、.将两个项目或多个项目放在同一个文件夹下&#xff08;等同于eclipse中将多个项目放在工作空间workspace下&#xff09;,lz在D盘下创建了ideaCloudProject文件夹&#xff0c;并在该文件夹下创建了一个cloud项目和一个vue项目&#xff0c;如下图&#xff1a; 2、用idea打开该…

一个IDEA界面如何同时打开多个项目

第一步&#xff1a;先导入其中一个工程 第二步&#xff1a;点击File->Project Structure 第三步&#xff1a;导入模块 最后点击Apply即可完成一个IDEA界面同时打开多个项目的需求。

怎么在IDEA的一个窗口中打开多个项目

之前一直使用的是sts,最近比较有空&#xff0c;学习一下idea的使用。 习惯了eclipse,myeclipse,sts等工具&#xff0c;想在一个窗口管理多个项目&#xff0c;只用import就可以了。但今天使用Intellij IDEA时&#xff0c;新建了一个工程&#xff0c;然后我又想从github上把我以前…

IDEA中如何在一个窗口中打开多个项目

idea在一个窗口如何打开多个项目 在使用Spring进行微服务开发时&#xff0c;需要同时打开很多个项目&#xff0c;但是如果每一个项目都单独一个窗口的话&#xff0c;切换就很麻烦&#xff0c;最方便的就是在IDEA一个窗口中管理多个项目。 在IDEA之中&#xff0c;它的一个项目就…

idea同时打开多个项目

想在idea中同时打开多个项目&#xff0c;如下图 打开项目后一般无法识别多个文件&#xff0c;可以按照下面的步骤操作 查看项目中maven插件是否打开 查看maven是否配置 查看右侧是否有maven选项&#xff0c;如果没有就先添加maven选项&#xff0c;添加方案步骤提供下面三种方…

使用IDEA打开eclipse项目

文章目录 点击左上角的File→New→Project from Existing Sources。 选择到要打开的项目路劲&#xff0c;选择好后点击OK 先勾选 Import project from external model 后选择Eclipse后点击Next。 无需设置&#xff0c;直接点击Next。 继续点击Next。 继续Next。 选择运行项目的…

idea关闭多个项目合并在tab栏显示

macos bigsur使用新版idea 2021.2 打开多个项目合并到tab栏&#xff1a; 对于屏幕比较小多了一行感觉比较别扭&#xff0c;以及全屏下暂时没适应……好几次都感觉找不到桌面…… 关闭&#xff1a;这个关闭是mac级别的关闭&#xff0c;并不是idea的setting macos - 系统偏好 …

IDEA设置多窗口打开项目

问题&#xff1a; 解决IDEA打开项目总是在当前窗口下&#xff0c;无法同事打开两个窗口的问题。【可能是我自己默认配置的问题】 解决&#xff1a; 菜单栏File&#xff0c;进入Setting栏 如上图所示&#xff0c;配置即可~~ 转载于:https://www.cnblogs.com/hyc-ana/p/9187153.h…

【Mac】在Idea中打开两个项目在同一个窗口

前置条件&#xff1a; 打开两个项目在不同窗口 步骤&#xff1a; 1.选择一个窗口上的window&#xff08;窗口&#xff09;选项。 2.选择Merge All Project Windows&#xff08;合并所有项目窗口&#xff09; 效果&#xff1a; 注意事项&#xff1a; 一定要打开多个项目&am…

MAC设置idea多个项目合并在tab栏显示

这个是mac级别的设置&#xff0c;并不是idea的设置。系统偏好 - 通用 - 始终 &#xff1a; 另外IDEA打开新项目时应选择new window

idea在一个窗口打开一个文件夹里的多个项目

使用场景&#xff1a;Spring-cloud框架下的项目一般都有多个子项目&#xff0c;就像我们项目由6个子项目&#xff0c;每个子项目单独更新很麻烦&#xff0c;还有相互之间的依赖&#xff0c;所以就把所有的子项目都放到一个文件夹下&#xff0c;只要更新所有的子项目都会更新&am…

Intellij IDEA同时打开多个工程(项目)的方法

这里介绍两中方法&#xff1a; 方法一&#xff1a; 先创建一个文件夹命名为workspace&#xff08;可以自定义名称&#xff09; 将开发项目放在该workspace目录下 打开Intellij idea 开发工具&#xff0c;选择 File -> Open 在Open File or Project 窗口中&#xff0c;找到…

Mac 下IDEA 在同一窗口下打开多个项目

Mac 系统的IDEA 升级之后可以支持在同一个窗口下打开多个项目&#xff0c;就像这样&#xff1a; Mac 设置如下&#xff1a;

Idea-如何像eclipse一样一个窗口打开多个项目

1.打开IDEA → 选择 【file】→【new】→[Project] 2.选择【Empty Project】 创建一个空项目 点击【Next】 3.输入项目空间名   4.选择打开的位置 选择【This Window】 当前工作空间就会变成你创建的那个 我的是‘untitled’ 到这个步骤就把我们需要的工作空间建立好啦&am…

idea设置打开多个窗口

idea默认是打开10个窗口&#xff0c;超过10个后&#xff0c;会关掉前面打开的窗口&#xff0c;这在开发的时候会很难受&#xff0c; File->settings->Editor->General->Editor Tabs->closing Policy->Tab limit 设置窗口个数&#xff0c;调大数值&#xff0…