Tomcat集群应用同步 —— 源码分析

news2025/1/18 21:10:53

文章目录

  • 前言
  • 一、应用同步的配置与实现原理
  • 二、应用同步源码分析
  • 三、如何获取集群的节点列表
  • 四、通讯模块Tribe
  • 五、集群的Session同步
  • 六、集群的Session共享
  • 总结


前言

相信大家对Tomcat的集群部署都不陌生,以往,我们手动搭建一个Tomcat的集群环境,然后手动部署每个Tomcat上面的应用,保证他们都是相同的应用程序包,以便负载均衡的时候不会出现问题

但是问题来了,如果我们项目源码修改了,重新打包,这时候就要给每个Tomcat单独替换里面的war包,相当麻烦!

Tomcat给我们提供了一种十分简单的解决方案:

应用同步


一、应用同步的配置与实现原理

大家在修改Tomcat的server.xml配置文件的时候,应该都注意到了他这一行注释,
cluster这个单词格外的引人瞩目,没错他就是Tomcat里面集群的配置类,他里面提供了我们想要的应用同步的而功能!

在这里插入图片描述

前提,我们已经配置了多个Tomcat的集群环境,可以用nginx或者Apache做负载均衡器。
然后我们只需在server.xml配置中的 标签里面添加下面红色框里面的配置,就可以在集群环境,开启各个Tomcat之间的应用自动同步的功能。这里的配置是设置了 temp/ 目录作为临时保存同步war文件的目录, webapps/目录作为Tomcat的部署目录, watch/目录是监听目

在这里插入图片描述

Tomcat在启动的时候,会开启一个定时任务,一直检测监听目录文件改变,只要里面的文件保存时间改变了,就触发一系列方法,更新(传输与接收)war包到集群中的其他节点。
集群环境的每个节点都实现了发送端、与接收端的任务,每个节点他们是相等的关系,实际上吗,没有主从之分。

在这里插入图片描述

二、应用同步源码分析

监听watch目录是怎么监听文件变化的,又是怎么处理相关逻辑的,下面我们根据源码来了解!
我们先从集群节点中的,同步请求发送端开始讲起。

tomcat在启动时,开启的定时任务,会调用SimpleTcpCluster类的backgroundProcess(),里面通过clusterDeployer对象,调用其父类FarmWarDeployer的backgroundProcess()

在这里插入图片描述
在FarmWarDeployer的backgroundProcess方法中,调用了WarWatcher的check()方法,这个方法只要就是实现监听文件,做文件的变更判断

在这里插入图片描述

WarWatcher的check()方法中,有两个核心的地方:

  1. 获取配置文件指定的监听目录下的文件列表

在这里插入图片描述

  1. 遍历文件获取上次文件修改时间做对比,判断到文件修改了,就执行war包同步的逻辑

在这里插入图片描述
上面走到的fileModified()方法,是FarmWarDeployer类下的一个方法,这个类主要是实现部署war包的。
触发了FarmWarDeployer的fileModified()方法后,调用里面的copy()复制war包到指定的部署目录(实现这台tomcat的部署)

在这里插入图片描述

下面再调FarmWarDeployer的install()方法,获取集群的节点列表

在这里插入图片描述
再给集群节点发送同步的请求
在这里插入图片描述

下面的源码分析就到了集群同步接收端的节点,如何实现获取应用并实现部署

Tomcat在启动的时候,会开启一个定时任务NioReplicationTask,他会开启通道,定时接收同步请求
在这里插入图片描述
在drainChannel()方法里面,源码一层一层地点进去后发现,调用的是FarmWarDeployer的check()方法
在这里插入图片描述
再通过反射调用HostConfig的check()方法
在这里插入图片描述
然后通过deployApps()再调用deployWAR()方法,在这一行,将已经封装好的应用数据添加到context,里面调用StandardHost的addChild

将这个context(应用)部署到host里面
在这里插入图片描述

三、如何获取集群的节点列表

  1. 通过SimpleTcpCluster的registerMember(),用来注册一个集群节点

在这里插入图片描述

  1. SimpleTcpCluster的unregisterMember(),用来排除一个集群节点
    在这里插入图片描述

  2. tomcat启动的时候通过生命周期Lifecycle执行SimpleTcpCluster的startInternal(),先注册自己作为localMember
    在这里插入图片描述

  3. 启动一个线程(McastServiceImpl里面的线程类ReceiverThread),一直循环调用receive()一直通过socket不停地接收来自局域网等网段的信息(网络中各个ip一直在相互ping)
    在这里插入图片描述
    调用memberDataReceived(),将信息封装成member(包含ip等信息)
    在这里插入图片描述
    当这次调用的member被判断到是不在集群中,就调用里面的service.memberAdded()
    注意:这里虽然调用了memberAdded(),但是不一定会加入到当前集群列表
    在这里插入图片描述

然后里面一层一层调用memberAdded()

当调用到TcpFailureDetector的memberAdded(),如果这里的notify是true,就代表当前member跟localMember(这台tomcat)是属于同一集群的
在这里插入图片描述
notify这个值,是通过Membership来实现多播心跳发送与接收,只要server.xml配置了SimpleTcpCluster,他就会启用Membership来实现多播心跳检测。

同一集群的tomcat如果都配置了SimpleTcpCluster,也就能接收对方的心跳,就将这个notify设置为true,只有属于同一集群的,才会调用到SimpleTcpCluster的memberAdded(),在这里判断,如果该节点还没注册,就调registerMember()注册

四、通讯模块Tribe

Apache Tribes是Tomcat的一个模块,支持服务器集群中的组通信。

Tribes是一个具有组通信能力的消息传递框架,这些是在Tomcat 5容器的集群/session复制代码之外创建的。它是为Tamcat集群实 现提供的通信框架。它的目的之一是简化分布式应用点对点(peer-to-peer)及点对组(peer-to-group)通信。Tribes支持两种 类型的消息传递:可用于两个节点间事件的并发(concurrent)消息传递和可用于发送消息给多个节点的平行(parallel)消息传递

整个Tribes的设计核心可以用下图表示:
在这里插入图片描述

Tribes的设计思路,主要是通过拦截器与监听器来实现的,拦截器则是对底层数据的一种统一额外加工处理,监听器则作为接口提供应用层对数据做业务逻辑处理,组成了一个优雅的设计方案

  • 应用层:

应用层面主要就是一些监听器,Tomcat内置了一些类,实现了这些监听器里面指定的方法,接受IO层传输过来的信息,并对其做响应的逻辑处理。

  • 拦截器栈:

拦截器栈提供了在消息传送到应用层之前对消息进行一些额外的操作,例如对某些信息进行过滤编码等等操作。

  • 在IO层有三个重要的模块:

MembershipService 模块主要负责组成员关系的维护,包括维护现有成员及发现新成员。
ChannelSender 模块负责向组内其他成员发送消息及其各种机制的详细实现
ChannelReceiver 模块用于接收组内其他成员发送过来的消息及其各种机制的详细实现

下面我会从底层往上,逐层通过具体的组件,结合源码来讲解:

MembershipService

一个集群包含若干成员,要对这些成员进行管理就必须要有一张包含所有成员的列表,当要对某个节点做操作时通过这个列表可以准确找到该节点的地址进而对该节点发送操作消息。

Tribes的设计是基于同等节点之间的通信,并不存在主节点选举的问题,每个节点都是相等的关系。它具备自动发现节点,即新节点加入要通知集群其他成员更新成员列表,让每个节点都能及时更新成员列表,每个节点通过交换机各自都维护一份集群成员表,且他们隔一段时间向交换机组播自己节点消息,即心跳操作。

集群的成员列表在 SimpleTcpCluster 类的成员属性 memberOnameMap 里:

在这里插入图片描述
Tribes通过 registerMember()、unregisterMember()方法,实现集群节点的注册和卸载,更新各自节点的集群节点列表
在这里插入图片描述

在这里插入图片描述

举个例子:

一个Tomcat集群,一开始有两个节点,A和B。

A和B各自分别都创建一个节点信息发射器和节点信息接收器,让他们运行于独立的线程中。发射器用于向组内发送自己节点的消息,而接收器则用于接收其他节点发送过来的节点消息并进行处理。

A和B都定时向交换机发送心跳检测,如果超过一定时间交换机没有接受到某个节点的心跳,则会给所有节点发送信息,然后所有节点更新自己的集群节点列表,将他移除。

如果这时候集群新增了节点C,节点C的Tomcat会通过发射器,向交换机发送自己的讯号,交换机负责传递,告诉每个节点,更新他们各自的集群节点列表,新增这个节点。


Tomcat提供了一个接口 MembershipService。我们可以通过他,获取集群环境中的其他成员节点,获取各个节点的信息。

例如,下面的这些方法:

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述
等等方法 …

ChannelSender

刚才也说了,Tribes是个通讯框架,那就肯定有消息的发送者,ChannelSender 就充当这个角色。
ChannelSender 是个接口,Tomcat中他有一个唯一的实现类 ReplicationTransmitter ,他是作为消息发送的实际执行者。

这个类最核心的方法是 sendMessage(),他一目了然,先获取发送消息的承载类,然后利用他,传入集群环境的其他节点,分别对他们发送消息。

在这里插入图片描述
MultiPointSender接口的实现类有多个,实现了sendMessage()方法,作为实际的发送执行者,有以下几个,分别处理不同的场景。

在这里插入图片描述

ChannelReceive

消息的传递,有发送者,自然也有接受者,ChannelReceive就充当这个接收者。他负责接收处理其他节点从消息发送通道发送过来的消息。

本质其实是每个节点暴露一个端口作为服务端,去监听客户端,接受客户端发送的消息;而每个节点又充当客户端,去连接集群中其他节点的服务端,发送自己的消息。

ChannelSender就是充当客户端的,而ChannelReceiver充当服务端。

从ChannelReceive的类继承关系中可以看到,Tribes的消息接受端,有两种处理的方式,NIO和BIO。

在这里插入图片描述
BIO:
同步阻塞,服务器实现模式为一个连接一个线程,常使用于连接数目比较小且固定的架构,程序简单易理解,但是性能较低。

NIO:
同步非阻塞,基于Channel(通道)和Buffer(缓冲区)进行操作的,数据总是从通道读取到缓冲区中,或者从缓冲区写入到通道中,Selector(选择器)用于监听多个通道事件,因此使用单个线程就可以监听多个客户端通道,能够大大提升处理的效率。

NioReceiver 这个接收端处理类中值得一提的是,他引入了任务池的概念,在加载这个类时,已经创建好任务池,把接收任务提前定义好放入内存中,当接收到来自集群节点的消息时,可直接获取使用而不用再实例化。

在这里插入图片描述

ChannelInterceptor

通道拦截器。为了提高系统的可扩展性和灵活性,实现在应用层提供对源消息统一处理,Tribes在通道中引入通道拦截器。

通过类继承关系可以看到,目前Tomcat提供了以下几个通道拦截器:

在这里插入图片描述

这些具体的拦截器都是继承类 ChannelInterceptorBase,源码跟踪发现,启动时,会通过通道的任务池executor,遍历集群成员列表,分别启动一个任务线程,去执行拦截器。

在这里插入图片描述

MembershipListener和ChannelListener

为了更清晰更好地划分职责,Tribes设计了IO层和应用层,IO层专心负责网络传输方面的逻辑处理,把接收到的数据往应用层传送,应用层发送的数据也是通过此IO层发送。

而应用层的处理入口,就是MembershipListener和ChannelListener。

从名字可以看出,这里用了监听器模式,在信息传输时,触发安装好的监听器,使之执行监听器里面的处理逻辑。这些事件主要包含了集群成员的加入和退出、消息报文接收完毕等信息。所以分成两类监听器:

MembershipListener:跟集群成员的变化相关

ChannelListener:是跟集群消息接收发送相关

MembershipListener接口有两个抽象方法:

在这里插入图片描述
在这里插入图片描述
当集群环境添加了节点时,其他节点就会接收信息,这个信息进来Tomcat的通道时,就会触发监听器的 memberAdded()方法(memberAdded一直监听者集群环境过来的而信息),就会根据当前节点Tomcat执行的模式,去调用对应的方法,执行具体的处理逻辑。

移除节点也一样。

ChannelListener接口有两个抽象方法:

在这里插入图片描述

在这里插入图片描述
当消息通过通道传递进来时,触发监听器的 messageReceived(),根据当前节点Tomcat执行的模式,去调用对应的方法,执行具体的处理逻辑。

而accept()方法,则是由通道调用,以确定监听器是否将处理此消息。

在这里插入图片描述

五、集群的Session同步

Session的同步,是指在一个Tomcat的集群环境下,不借助其他第三方的组件,由Tomcat内部通讯机制,实现各个节点之间的Session信息的同步。

如果集群中有一个Tomcat实例的会话变了,它会通过会话管理器将改变的动作消息封装成消息然后调用集群对象Cluster,由Tribes将会话同步请求发出去。集群中会话同步的过程中,通信过程是以ClusterMessage为对象进行传输的,将实际的会话数据载体SessionMessageImpl序列化,进行传输,到其他Tomcat实例的时候会反序列化,消息由Tribes接收之后向Cluster上传。最后达到会话管理器,它根据动作消息同步会话

  • Ssession同步的配置

每个节点的Tomcat,在配置了集群的前提下,在 标签里面,加入 标签,配置上 ClusterSessionListener。

在这里插入图片描述

针对ClusterSessionListener源码进行分析:
里面代码不多,主要就一个方法messageReceived(),用于接受集群环境传输过来的要同步的session信息。

在这里插入图片描述

通过ClusterMessage封装会话信息,传进来,解析这是对应哪个context应用的会话

在这里插入图片描述

拿到集群管理器里面对应的context,将会话信息设置进去

消息的接收方法messageDataReceived(),源码跟踪进去,最终到了DetalManager类的messageReceived()。

从下图可以看到,根据接收的会话消息实体里面的类型,分别做不同的处理,例如会话信息的创建、修改、移除、获取 …

我们点进去created方法,看看创建会话的代码细节

在这里插入图片描述

这个方法是创建一个新的session会话信息,可以看到,他先将全局的接受计数器属性 +1,然后创建一个空的session,把sessionId,会话信息等设置进去。

在这里插入图片描述

这个方法是集群的其他节点请求获取所有session信息的处理方法,他会把当前节点tomcat的所有session信息查询出来,然后通过集群发送出去。

其他剩余的session修改、移除、获取等的方法,这里就不看了,代码不复杂。

在这里插入图片描述

六、集群的Session共享

前面说的session同步,是在Tomcat集群环境下,实现每个节点的会话信息的同步,从而可以使集群的整体,无论访问哪个节点,获取到的会话信息都是一样的。

但是这种方案,有很大的弊端,他是不断通过网络IO去发起请求,接收请求,去不断地在每个节点执行同步,不止存在网络的延迟,还会有性能的损耗。所以自从这种方案诞生,就没几个开发者愿意使用这种方式。

大部分人,还是会选择使用session的共享。共享,就是每个节点使用的session,都是来自同一个地方,拿数据写数据都是在同一个地方操作,这样就不需要同步数据,防止出现数据延迟,数据不一致的问题。

但是Tomcat原生是不支持这种共享方式的,所以有一位开源作者,开发了一个用第三方中间件redis来存储会话信息的的实现方案。

这种方案他的实现方式看以下图:

在这里插入图片描述
他跟用Redis做缓存的方式是一样的,只不过他现在是用Redis来存储Session会话的信息,说白了,他们其实是一个意思,只是如果原本项目代码写了会话的方式,这里可以直接,用会话共享,实现项目的集群部署。

  • 使用方法

把 tomcat8.5-redis-session-manager.jar 和相关的依赖包,放到Tomcat安装目录下的lib目录。

在这里插入图片描述

然后修改Tomcat的 context.xml 配置文件,在 标签里面加入如下配置:
在这里插入图片描述

  • 原理讲解

从上面的修改配置中我们看到,引入了两个类:

com.s.tomcat.redissessions.RedisSessionHandlerValve

com.s.tomcat.redissessions.RedisSessionManager

从他们的包名可以看出来他们都是来自我们给Tomcat引入的第三方依赖包。这个包主要实现的功能是:

在Tomcat里添加一个阀门,当session内容变动时,触发这个阀门,向配置好的redis插入数据,把session的信息同步到redis里面,在具体的应用中,当我们通过代码获取当前的Session时,调用的是这个包里面的方法而不是Tomcat的方法,然后这个方法会去redis,获取我们想要的会话信息并返回,这个过程,应用的程序编写者是完全不感知的,他和原本的获取Tomcat会话信息的代码是一致的。

下面出现的所有代码截图,都是来自tomcat8.5-redis-session-manager.jar的代码

我们打开 RedisSessionManager 类的源码看到,他继承了Tomcat的抽象类ManagerBase,这个类就是用来管理信息的一个父类,他的子类将用来承载例如Session的会话数据的管理。

在这里插入图片描述

而RedisSessionManager继承了ManagerBase,自然也继承了他管理会话的能力,然后他重写了里面的 createSession()、findSession()、add()、remove()等等方法,利用redis,去实现会话信息的创建和获取。

我们以createSession()方法来探索一下,可以看到,他先通过配置文件的redis连接信息,获取redis的连接,以sessionId作为key在redis创建一条作为session的父级数据。

在这里插入图片描述

下面这两段代码,他创建了一个内部类似Session管理的对象(RedisSession继承了Tomcat的StandardSession,StandardSession就是Tomcat的Session信息承载类),然后将会话信息设置进去,调用saveInternal()方法,把对象的信息写进redis。

在这里插入图片描述

在这里插入图片描述

然后我们再分析下findSession(),刚才那个方法,通过第三方的jar包,在里面维护了一个充当session对象的类,然后将会话信息写进redis。
这里,通过传入的sessionId,去redis查询出对应的信息,反序列化成一个对象,才封装成session对象返回。

在这里插入图片描述

到这里就讲解完这个包如何实现redis转存Session信息了。


总结

欢迎指出我的错误!

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

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

相关文章

2023年运动耳机最新排行榜公布、公认最好用的五款运动耳机推荐

建议大家不要轻易尝试那些低价运动耳机,因为这些耳机大多数,只能满足基本听个响,舒适度和蓝牙稳定性都有代提高。通勤使用还勉强,运动强度一大,耳机容易掉落不说,蓝牙连接也很容易掉线,体验感受…

【自学Python】Python算术运算符

Python算术运算符 Python算术运算符教程 Python 中算术运算符是对 数值类型 的 变量 进行运算的,比如:加、减、乘、除、取模、整除和乘方运算等。 Python算数运算符语法 运算符说明实例结果加12.34 43.2155.55-减43.21 - 12.3430.87*乘2 * 3.141596…

springdata个人学习笔记

​ 入门 初始化springboot 依赖引入 <!-- springdata--><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-jpa</artifactId></dependency><dependency><groupId&g…

node与node-sass版本对应

node&#xff1a;14.0.0 npm install sass-loader7.3.1 node-sass4.14.1 --savenode-sass对应node https://github.com/sass/node-sass/releases node-sass 和 sass-loader 的对应关系一般的情况下4.XX.YY&#xff0c;XX就是对应Node XX 版本 参考

Java常用工具类方法(下)

8. IOUtilsIO流在我们日常工作中也用得比较多&#xff0c;尽管java已经给我们提供了丰富的API。但我们不得不每次读取文件&#xff0c;或者写入文件之后&#xff0c;写一些重复的的代码。手动在finally代码块中关闭流&#xff0c;不然可能会造成内存溢出。有个好消息是&#xf…

一文搞懂Linux 环境变量

一文搞懂Linux 环境变量1.环境变量分类2.读取环境变量3.环境变量修改在 Linux 系统中&#xff0c;环境变量是用来定义系统运行环境的一些参数&#xff0c;比如每个用户不同的主目录&#xff08;HOME&#xff09;。 1.环境变量分类 按照作用域来分&#xff0c;环境变量可以简单…

【阶段三】Python机器学习13篇:机器学习项目实战:支持向量机分类的算法原理

本篇的思维导图: 支持向量机分类的算法原理 支持向量机分类算法的基本思路 1.最大间隔 支持向量机说到底就是一种“线性分类器”,它以“间隔”作为损失的度量,目标通过不断调整多维的“直线”——超平面,使得间隔最大化。所谓“支持向量”,就是所有数据点中直接参…

Lichee_RV学习系列---移植dhrystone

系列文章目录 Lichee_RV学习系列—认识Lichee_RV、环境搭建和编译第一个程序 文章目录系列文章目录一、dhrystone简介二、dhrystone源码下载三、dhrystone移植1、移植官方源码2、移植GitHub开源代码a&#xff1a;修改Makefile文件b&#xff1a;编译3、执行dhrstone代码总结一、…

Flink源码解析一之RPC原理解析

在阅读 Flink 源码过程中,如果你见到有这种类型的代码,其实就是在发送 RPC 请求 // resourceManagerGateway 就可以理解成: 当前节点中,对于 ResourceManager 代理对象的封 装 resourceManagerGateway.requestSlot(); // 代码跳转到:resourceManager.requestSlot()…

C++20标准下的左值与右值

C20标准下的左值与右值一、什么是左值与右值二、左值引用三、右值引用四、值类别五、标准库 move 函数5.1 用 static_cast将左值转换为右值5.2 使用 std::move 将左值转换为右值一、什么是左值与右值 左值&#xff1a;左值可以出现在赋值语句的左边或者右边右值&#xff1a;右…

Java设计模式-装饰者模式Decorator

介绍 装饰者模式的核心思想是通过创建一个装饰对象&#xff08;即装饰者&#xff09;&#xff0c;动态扩展目标对象的功能&#xff0c;并且不会改变目标对象的结构&#xff0c;提供了一种比继承更灵活的替代方案。需要注意的是&#xff0c;装饰对象要与目标对象实现相同的接口&…

VisualBox解决CentOS中yum安装失败的问题

怎么说呢&#xff0c;花了一个下午一个上午的时间&#xff0c;总算把这条命令运行成功了&#xff1a; yum install wget -y 打怪兽途中遇到了几个问题&#xff0c;总结一下&#xff1a; 1.ping 啥ip都是unkown&#xff0c;还有一种情况&#xff0c;就是ping之后就一直卡在那…

C++计算机课程设计 学生成绩管理系统 研究报告

课程设计内容 2.1 学生成绩管理系统 2.1.1 内容 主菜单模块 该模块主要用来实现整个系统的流程。主界面提供用户选择并调用各个子模块。 输入模块 当初次使用系统时&#xff0c;学生信息需要从键盘逐个输入。学生信息由学生的学号、姓名、性别、高等数学、英语、计算机和平均…

Qt基于CTK Plugin Framework搭建插件框架--CTK服务工厂

一、前言 注册服务的时候能够用服务工厂来注册&#xff1b; 访问服务getServeice中的plugin参数是执行ctkPluginContext::getService(const ctkServiceReference&)的插件&#xff0c;从而工厂根据执行的不同插件名称返回不同的服务实现 服务工厂的作用 在服务中可以知道…

达摩院2023十大科技趋势发布,生成式AI将进入应用爆发期

1月11日&#xff0c;达摩院2023十大科技趋势发布&#xff0c;生成式AI、Chiplet模块化设计封装、全新云计算体系架构等技术入选。达摩院认为&#xff0c;全球科技日趋显现出交叉融合发展的新态势&#xff0c;尤其在信息与通信技术&#xff08;ICT&#xff09;领域酝酿的新裂变&…

【关于Linux中----进程间通信方式之管道】

文章目录一、什么是管道二、匿名管道三、命名管道一、什么是管道 进程间通信主要目的为以下几点 数据传输&#xff1a;一个进程需要将它的数据发送给另一个进程 资源共享&#xff1a;多个进程之间共享同样的资源。 通知事件&#xff1a;一个进程需要向另一个或一组进程发送消息…

STL forward_list 模拟实现

forward_list 概述 forward_list 是 C 11 新增的容器&#xff0c;它的实现为单链表。 forward_list 是支持从容器中的任何位置快速插入和移除元素的容器&#xff0c;不支持快速随机访问。forward_list 和 list 的主要区别在于&#xff0c;前者的迭代器属于单向的 Forward Ite…

二分法讲解

目录 一、前言 二、二分法理论 1、引导&#xff1a;猜数游戏 2、理论背景&#xff1a;非线性方程的求根问题 1&#xff09;非线性方程的近似解 2&#xff09;搜索法和二分法 3、用二分的两个条件 4、二分法复杂度 三、整数二分 1、在单调递增序列中找 x 或者 x 的后继…

使用python-pptx创建PPT演示文档功能实践

python对PPT演示文档读写&#xff0c;是通过第三方库python-pptx实现的&#xff0c;python-pptx是用于创建和更新 PowerPoint&#xff08;.pptx&#xff09;文件的 Python 库。 关于PPT演示文档与幻灯片模板的内容不是本文的重点&#xff0c;在此略过。 1. PPT基本结构在pyth…

Sophus降维、升维与欧拉角、旋转向量的爱恨情仇

0. 简介 在面对二维与三维之间的转换时&#xff0c;我们常常会困惑该如何去转换&#xff0c;在G2O中存在有理想的坐标转换工具&#xff0c;但是在Sophus中却缺乏这样的手段。之前在Sophus处简要的介绍了一下SE(2)与SE(3)的转换&#xff0c;最近发现之前的文章这部分需要拿出来…