Java阶段四Day08

news2024/10/7 4:29:42

Java阶段四Day08

文章目录

  • Java阶段四Day08
    • 关于pom.xml中的版本
    • 关于Session
    • 关于Token
    • 关于JWT
    • 在项目中使用JWT
      • CustomUserDetails
      • UserDetailServiceImpl
      • UserServiceImpl

关于pom.xml中的版本

  • 查看<groupId> 是同一家的只需配一个版本号<version>
  • <artifactId>spring-boot-starter开头的可不写版本号<version>
  • springBoot保证了添加的依赖“开箱即用”即 约定大于配置

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-j6TePRMv-1687687094756)(../../../AppData/Roaming/Typora/typora-user-images/image-20230625110039742.png)]

关于Session

  • 服务器端的应用程序通常是基于HTTP协议的,HTTP协议本身是一种“无状态"协议,所以,它并不能保存客户端的状态,例如,无法识别客户端的身份,所以,即使同一个客户端多次访问同一个服务器,服务器并不能识别出它就是此前来访的客户端

  • 在开发实践中,大多是需要能够识别客户端身份的,通常可以使用Session机制来解决

  • 当某个客户端首次访问某个服务器端时,将直接发起请求,当服务器端收到此请求时,会在响应时返回一个Session lD值(本质上是一个UUID值),当客户端收到Session lD后,后续的访问都会自动携带此Session ID到服务器端,则服务器端可以根据这个Session ID值来识别客户端的身份。

  • 在服务器端,使用K-V结构的数据表示Session,客户端携带的Session ID就是K-V结构中的Key,所以,每个客户端都可以访问到不同的value,即每个客户端对应的Session数据。

  • Session是存储在服务器端的内存中的数据,而内存资源是相对有限的资源,存储空间相对较小,所以,必然存在清除Session的机制,默认的清除机制是"超时自动清除",即某个客户端最后一次提交请求之后,在多长时间之内没有再次提交请求,服务器端就会清除此客户端对应的Session数据,至于过多久清除Session,没有明确的要求,大多软件的默认时间是15~30分钟,但是,也可以设置为更短或更长的时间。

  • 基于Session的特征,必有一些不足:

    • 不适合存储较大的数据,可以通过规范的开发来避免此问题
    • 不易于应用到集群或分布式系统中,可以通过共享Session来解决此问题
    • 不可以长时间存储数据,无解

关于Token

Token:票据、令牌

当某个客户端向服务器端发起登录的请求时, 将直接发起请求,当服务器端收到此请求时,如果判断登录成功,会在响应时返回一个Token值,当客户端收到Token后,后续的访问都会自动携带此Token到服务器端,则服务器端可以根据这个Token值来识别客户端的身份。

与Session不同,Token是由服务器端的程序(开发者自行编写)生成的一段有意义的数据。例如可以将用户的ID、用户名都存放到Token中,则在后续访问中,客户端携带了Token后,服务器端可以直接从Token中找到相关信息,如用户ID、用户名,从而服务器端的内存中,并不需要持续保存相关信息,所以,Token可以被设置一段非常长的有效期,且不用担心持续性的消耗服务器端内存的问题。

基于Token的特征,可以解决Session能解决的问题,且天生适用于集群或分布式系统,只要集群或分布式系统中的各个服务器具有相同的检查Token和解析Token的程序即可。

关于JWT

JWT:(JSON Web Token) 官网;

每个JWT数据都是由3大部分组成的:

  • Header:声明算法与Token类型
  • Payload:数据
  • Verify Signature:验证签名

在尝试生成和解析JWT之前,需要添加依赖项:

<!--    JJWT (Java JWT)-->
<dependency>
    <groupId>io.jsonwebtoken</groupId>
    <artifactId>jjwt</artifactId>
    <version>0.9.1</version>
</dependency>

测试

public class JWTTest {
    String secretKey = "ladnamdsm2ne12[d;mas;camd;a,d[]12ke[12ke";

    @Test
    void generateToken() {
        Date date = new Date(System.currentTimeMillis() + 24L * 60 * 60 * 1000 *30);//注 超出Integer上限 加入L

        Map<String, Object> claims = new HashMap<>();
        claims.put("id", 81290);
        claims.put("username", "BoB Doe");

        String jwt = Jwts.builder()
                //Header
                .setHeaderParam("alg", "HS256")
                .setHeaderParam("typ", "JWT")
                //Payload
                .setClaims(claims)
                .setExpiration(date)
                //Verify Signature
                .signWith(SignatureAlgorithm.HS256, secretKey)
                .compact();
        System.out.println("jwt = " + jwt);
    }

    //eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpZCI6OTk3LCJleHAiOjE2ODc3NjMyMDUsInVzZXJuYW1lIjoiSm9obiBEb2UifQ.neXB-DvVoV7YU_etCvgFqCodSk25mv8hAiq2KYyTpV0

    @Test
    void parseToken() {
        //String jwt = "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpZCI6OTk3LCJleHAiOjE2ODc3NjMyMDUsInVzZXJuYW1lIjoiSm9obiBEb2UifQ.neXB-DvVoV7YU_etCvgFqCodSk25mv8hAiq2KYyTpV0";
        String jwt = "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpZCI6ODEyOTAsImV4cCI6MTY4Nzc2MzgxMCwidXNlcm5hbWUiOiJCb0IgRG9lIn0.rtO2Ifae7B-NtAvZS9Rkm8omzuOaTt-giApzV8giMcQ";
        //可见请求头不变

        //JWT 并不是一种可以保护隐私的数据,即使不知道SecretKey JWT数据也会被解析 所以不要存放敏感信息

        //io.jsonwebtoken.ExpiredJwtException 过期异常
        //io.jsonwebtoken.SignatureException 签名错误异常
        //io.jsonwebtoken.MalformedJwtException  格式错误异常

        Claims claims = Jwts.parser().setSigningKey(secretKey).parseClaimsJws(jwt).getBody();
        System.out.println("claims = " + claims);
        System.out.println("username = " +claims.get("username",String.class));
        System.out.println("id = " + claims.get("id"));
    }
}

在项目中使用JWT

大致分为两个步骤:

  1. 当验证客户端登录信息成功之后,生成JWT数据,并响应到客户端去 >> 相当于“买票”过程,
  2. 客户端自主携带JWT 向服务器端发请求,服务器端尝试接收并解析JWT >> 相当于“验票”过程
  3. 解析成功后得到JWT中的数据信息,将这些信息创建为Authentication对象并存入到SecurityContext中 >> 相当于“进站”过程,

CustomUserDetails

@Getter
@ToString(callSuper = true) //调用父类的toString
@EqualsAndHashCode(callSuper = true)
public class CustomUserDetails extends User {
    private Long id;
    private String avatar;

    public CustomUserDetails(Long id, String username, String password, String avatar,boolean enabled, Collection<? extends GrantedAuthority> authorities) {
        super(username, password, enabled, true, true, true, authorities);
        this.id = id;
        this.avatar = avatar;
    }
}

UserDetailServiceImpl

@Override
public UserDetails loadUserByUsername(String username) throws UsernameNotFoundException {
    log.debug("Spring Security框架自动调用UserDetailsService对象,根据用户名获取用户详情");
    UserLoginInfoVO loginInfo = userRepository.getLoginInfoByUsername(username);
    log.debug("根据用户名【{}】从数据库中查询用户详情,查询结果:{}", username, loginInfo);

    if (loginInfo == null) {
        return null;
    }

    List<GrantedAuthority> authorities = new ArrayList<>();
    List<String> permissions = loginInfo.getPermissionValue();
    for (String permission : permissions) {
        SimpleGrantedAuthority authority = new SimpleGrantedAuthority(permission);
        authorities.add(authority);
    }

    //自定义的UserDetails
    CustomUserDetails userDetails = new CustomUserDetails(
        loginInfo.getId(),
        loginInfo.getUsername(),
        loginInfo.getPassword(),
        loginInfo.getAvatar(),
        loginInfo.getEnable() == 1,
        authorities
    );


    /*    UserDetails userDetails = User.builder()
                .password(loginInfo.getPassword())
                .username(loginInfo.getUsername())
                .disabled(loginInfo.getEnable() == 0)     // 账号是否被禁用   优先判断状态,再校验密码
                .accountLocked(false)   // 账号是否被锁定 当前项目中无此概念,则所以账号都是false
                .accountExpired(false) //账号是否过期
                .credentialsExpired(false) //凭证是否过期
                .authorities(authorities) //后续添加
                .build();
    */


    log.debug("即将向Spring Security框架返回UserDetails类型的结果:{}" ,userDetails);
    log.debug("接下来,将由Spring Security框架自动验证用户状态、密码等,以判断是否成功登录!");
    return userDetails;
}

UserServiceImpl

@Override
public String login(UserLoginInfoParam userLoginInfoParam) {
    log.debug("开始处理【用户登录】业务,参数:{}", userLoginInfoParam);
    Authentication authentication = new UsernamePasswordAuthenticationToken(
        userLoginInfoParam.getUsername(), userLoginInfoParam.getPassword());

    log.debug("准备调用 AuthenticationManager 的认证方法,判断用户名、密码 是否成功登录");
    Authentication authenticate = authenticationManager.authenticate(authentication);
    log.debug("验证用户登录成功,返回认证结果:{}", authenticate);

    Object principal = authenticate.getPrincipal(); //认证结果的当事人就是 userDetail的人
    log.debug("从认证结果中获取当事人:{}", principal);
    CustomUserDetails userDetails = (CustomUserDetails) principal;
    Long id = userDetails.getId();
    log.debug("从认证结果中的当事人中获取ID:{}", id);
    String username = userDetails.getUsername();
    log.debug("从认证结果中的当事人中获取用户名:{}", username);
    String avatar = userDetails.getAvatar();
    log.debug("从认证结果中的当事人中获取头像:{}", avatar);
    Collection<? extends GrantedAuthority> authorities = userDetails.getAuthorities();
    log.debug("从认证结果中的当事人中获取权限列表:{}", authorities);
    String authoritiesJsonString = JSON.toJSONString(authorities);
    log.debug("将权限列表对象转换为JSON格式的字符串:{}", authoritiesJsonString);

    Date date = new Date(System.currentTimeMillis() + 30L * 24 * 60 * 60 * 1000);
    //                                                  ↑ 注意加L,避免int溢出为负数
    String secretKey = "fNesMDkqrJFdsfDSwAbFLJ8SnsHJ438AF72D73aKJSmfdsafdLKKAFKDSJ";
    Map<String, Object> claims = new HashMap<>();
    claims.put("id", id);
    claims.put("username", username);
    claims.put("avatar", avatar);
    //        claims.put("authorities", authorities);
    claims.put("authoritiesJsonString", authoritiesJsonString);

    return Jwts.builder()
        .setHeaderParam("alg", "HS256")
        .setHeaderParam("typ", "JWT")
        .setClaims(claims)
        .setExpiration(date)
        .signWith(SignatureAlgorithm.HS256, secretKey)
        .compact();

    //改为使用JWT后,不必在登录成功后就将认证信息存入 SecurityContext 中
    /*log.debug("准备将认证信息结果存入 SecurityContext 中...");
        SecurityContextHolder.getContext().setAuthentication(authenticate);
        log.debug("已经将认证信息存入到 SecurityContext 中 登录业务处理完成");*/
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/683507.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

C# 线程基础

目录 一、概述 二、线程的创建 三、线程的休眠 四、线程的等待 五、线程的终止 六、线程的状态 七、线程的优先级 一、概述 线程&#xff08;Thread&#xff09;是进程中的基本执行单元&#xff0c;是操作系统分配CPU时间的基本单位&#xff0c;一个进程可以包含若干个…

【FPGA入门】第七篇、FPGA驱动VGA实现动态图像移动

目录 第一部分、实现效果 第二部分、动态VGA显示的原理 1、将动态显示的区域提前进行赋值 2、图像块的移动是每张图片叠加后的效果 3、如何实现图像块位置的改变 第三部分、系统结构和驱动波形 1、系统的Top-down结构 2、图像块移动的驱动波形 第四部分、代码 1、同步…

大语模型前世今生

引言&#xff1a;席卷世界的大语言模型浪潮 2022年11月30日&#xff0c;OpenAI公司发布了ChatGPT。这迅速成为了社会各界关注的焦点&#xff0c;ChatGPT能够如此快速&#xff0c;准确的完成文本生成&#xff0c;信息抽取&#xff0c;机器翻译&#xff0c;甚至代码生成等复杂任务…

数字化转型|银行业数据中心数字化转型之模型篇 01

导语&#xff1a; 银行业数据中心数字化转型是一项系统性工程&#xff0c;既涉及管理层面转型——包括数字化转型战略、基础架构和技术架构转型、技术创新和知识体系转型&#xff0c;又涉及执行层面转型——包括人员管理&#xff08;P&#xff09;、流程管理&#xff08;P&…

突破官方限制!最强TV观影神器我都给你找来了!

随着移动互联网的兴起&#xff0c;我想很多人家里的电视机都积起了灰&#xff0c;大家追剧的设备都从电视机变成了手机、平板、电脑 但这两年&#xff0c;我发现这个事情又慢慢有在转变了&#xff1a;随着大家&#xff08;尤其是年轻人&#xff09;对观看体验的追求&#xff0…

接口的学习

接口 接口可以理解为一种规则&#xff0c;是对行为的抽象 如何定义一个接口 使用关键词interface定义 public interface 接口名{} 接口不能实例化 接口和类之间是实现关系&#xff0c;通过关键词implements关键字表示 public class 类名 implements 接口名{} 接口的子类…

三个数据恢复方法解决移动硬盘数据丢失问题!

移动硬盘容量大、写入和读取速度快&#xff0c;受到很多人的欢迎。但是&#xff0c;无论数据存储在何处&#xff0c;都有数据丢失的风险。今天&#xff0c;小编来介绍一下移动硬盘数据恢复的方法&#xff0c;以免大家不慎删除移动硬盘数据而陷入无助的境地! 方法1.使用命令恢复…

全网最详细,性能测试-测试方法总结(压力/负载)超详细

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 并发/负载/压力理…

leetcode123. 买卖股票的最佳时机 III(java)

买卖股票的最佳时机 leetcode123. 买卖股票的最佳时机 III题目描述动态规划代码演示 动态规划专题 leetcode123. 买卖股票的最佳时机 III 来源&#xff1a;力扣&#xff08;LeetCode&#xff09; 链接&#xff1a;https://leetcode.cn/problems/best-time-to-buy-and-sell-sto…

API手册使用方式说明

API手册使用方式说明 其实我们在API阶段,更多是要去学习别人已有内容,比如方法的使用 但是这么多的方法对于新手来说其实是不太友好的,刚开始根本记不住呀 所以API手册就是我们的一个好帮手,我们可以在API手册查到目标内容的介绍 类似于小学刚学字的时候,不会的字就可以去查字…

连接器信号完整性仿真教程 四

本文详细讲解了CST做连接器信号完整性仿真时,如何从材料库中载入材料,如何新增材料、如何编辑材料属性、如何将材料添加到库中,以及如何设置仿真模型材料、并以实例逐步做了详细演示。 一 从材料库中载入材料 从材料库中载入材料有两种方法。 方法一 点击菜单"Modelin…

Android 12 以上PendingIntent使用注意FLAG_IMMUTABLE

遇到如下报错&#xff1a; Fatal Exception: java.langlllegalArgumentException : Targeting S (version 31 and above) reures that one of FLAG_MMUTABLE r FLA-MUTABLE be specfed when creating a Pendinglntent. Strongly consider using FLAG_JMMUTABLE only use FLAG_M…

Redis的缓存类型分析

HashMap/ConcurrentHashMap HashMap 是一种基于哈希表的集合类&#xff0c;它提供了快速的插入、查找和删除操作。是很多程序员接触的第一种缓存 , 因为现实业务场景里&#xff0c;我们可能需要给缓存添加缓存统计、过期失效、淘汰策略等功能&#xff0c;HashMap 的功能就显得…

如何搭建产品知识库?让产品知识库管理更有序高效!

在现代企业中&#xff0c;一个完善的产品知识库对于提升团队的工作效率和产品质量至关重要。本文将介绍如何搭建一个高效的产品知识库&#xff0c;并提供一些管理方法&#xff0c;以使知识库的管理更有序、高效。 随着科技的不断进步和市场竞争的加剧&#xff0c;企业对于高效…

python基础学习--01

1.python环境的安装&#xff1a; 1.安装 Python 解释器&#xff1a;https://www.python.org/ 1.选择下载&#xff1a; 2.选择windows x86 -64 可执行的安装文件 (根据自己电脑的操作系统选择&#xff09; 3.安装完成后 左下角点击开始地方能看到这些说明安装好了。 4.安装…

SpringBoot原理(1)--@SpringBootApplication注解使用和原理

文章目录 前言主启动类的配置SpringBootConfiguration注解验证启动类是否被注入到spring容器中 ComponentScan 注解ComponentScan 注解解析与路径扫描 EnableAutoConfiguration注解 问题解答1.AutoConfigurationPackage和ComponentScan的作用是否冲突起因回答 2.为什么能实现自…

双路高速 AD 实验

目录 双路高速 AD 实验 1、简介 3PA1030 芯片 2、实验任务 3、程序设计 3.1、hs_dual_ad 模块代码 clk_wiz IP 核 的添加方法 ILA IP 核&#xff08;集成逻辑分析器&#xff1a;Integrated Logic Analyzer&#xff0c;ILA&#xff09; 4、硬件设计 4.1、添加.xdc约束…

23年软考网络工程师是什么?主要是考什么,有什么用?

网络工程师每年考两次&#xff0c;相比其他的软考考试一年中考的机会又多了一次 网络工程师证书考到后&#xff0c;通过本级考试的合格人员能根据应用部门的要求进行网络系统的规划、设计和网络设备的软硬件安装调试工作&#xff0c;能进行网络系统的运行、维护和管理&#xf…

汽车行业项目管理面临的5个挑战及解决方案

汽车行业正跨越式地迈向新的未来。其目前的发展主要由四大趋势驱动&#xff1a;连接性、自动驾驶汽车、共享出行和电气化。这给汽车企业带来了诸多挑战&#xff1a;竞争加剧&#xff0c;快速发展带来的频繁变化&#xff0c;与软件公司建立伙伴关系&#xff0c;以及其他相关问题…

GIS 功能模块实现

文章目录 1. GIS 模块流程图2. 网页端地图缓存的实现3. GIS 图形操作功能实现1 &#xff09;地图漫游2 &#xff09;对象删除3 &#xff09;选择复制属性查看 GIS 基本功能模块主要是在表现层开发的&#xff0c;是在OpenLayers 开发框架提供的接口上&#xff0c;通过Geo Server…