java设计模式之观察者模式

. 基本概念

观察者(Observer)模式中包含两种对象,分别是目标对象和观察者对象。在目标对象和观察者对象间存在着一种一对多的对应关系,当这个目标对象的状态发生变化时,所有依赖于它的观察者对象都会得到通知并执行它们各自特有的行为。

通俗地说,就好像这些观察者对象在时刻注视着目标对象(被观察)。无论何时该目标对象的状态发生变化,这些观察者对象都能够马上知道,并根据目标对象的新状态执行相应的任务。

观察者模式又叫发布-订阅(Publish-Subscribe)模式,其中的订阅表示这些观察者对象需要向目标对象进行注册,这样目标对象才知道有哪些对象在观察它。发布指的是当目标对象的状态改变时,它就向它所有的观察者对象发布状态更改的消息,以让这些观察者对象知晓。

一个目标对象的观察者对象数量是不固定的,可以随时增加新的观察者对象或取消已有的观察者对象。观察者模式的主要优点就是极大地降低了目标对象和观察者对象间的耦合,二者可以独自地改变和复用,让对系统增加功能或删除功能都很方便。

2. 应用举例

我们举一个实际的例子来说明对观察者模式的运用,假设我们有一个天气App,它有很多个界面组件,这些组件的作用分别是:显示摄氏温度、显示华氏温度、显示气温感受(比如炎热、凉爽和寒冷等等)。当然该App应该还有另一个对象用于获取实时的天气数据,我们称它为天气对象。

这个天气对象和界面组件之间的依赖关系就可以用观察者模式实现,该天气对象就是目标对象,天气数据就是它的状态。这些界面组件就是观察者对象,当天气对象获取到新的天气数据时(此时它的状态改变了),它就通知所有依赖于它的界面组件,这些组件就更新它们显示的内容。

图2 天气App中的观察者模式

现在我们想扩展这个天气App的功能,让它可以向用户建议如何穿衣。因此我们需要一个新的界面组件,当温度高时显示穿薄点,当温度低时显示穿羽绒服。我们让这个新的界面组件注册成为天气对象(目标对象)的观察者,这样当天气数据改变时,它就会自动得到通知并显示新的穿衣建议。而天气对象和其余的界面组件都不会受到影响,也无需改变。

现在让我们再次修改这个天气App,这次是要删除显示华氏温度的功能。因为我们中国人更习惯摄氏温度,显示华氏温度不但多此一举,还可能让用户误将它当作摄氏温度。此时,我们只需向天气对象(目标对象)取消注册该界面组件再删除它就可以了,天气对象在数据更新时就不会再通知该界面组件了。同样的,该App的其余部分也不会受到本次修改的影响。

3. 结构

我们先直接给出观察者模式的类结构图以及这些类充当的角色和作用,再解释为什么它们是这样的结构。

图3 观察者模式的类结构图

Subject:目标类,它是一个抽象类,也是所有目标对象的父类。它用一个列表记录当前目标对象有哪些观察者对象,并提供增加、删除观察者对象和通知观察者对象的接口。

Observer:观察者类,它也是一个抽象类,是所有观察者对象的父类;它为所有的观察者对象都定义了一个名为update的方法(也叫成员函数)。当目标对象的状态改变时,它就是通过调用它的所有观察者对象的update方法来通知它们的。

ConcreteSubject:具体目标类,可以有多个不同的具体目标类,它们同时继承Subject类。一个目标对象就是某个具体目标类的对象,一个具体目标类负责定义它自身的事务逻辑,并在状态改变时通知它的所有观察者对象。

ConcreteObserver:具体观察者类,可以有多个不同的具体观察者类,它们同时继承Observer类。一个观察者对象就是某个具体观察者类的对象。每个具体观察者类都要重定义Observer类中定义的update方法,在该方法中实现它自己的任务逻辑,当它被通知的时候(目标对象调用它的update方法)就执行自己特有的任务。

注意在Java语言中,Observer类可以用接口(interface)代替,此时所有的具体观察者类都要实现该接口。

3.1 Observer和ConcreteObserver

首先,目标对象需要知道有哪些观察者对象在观察它,这样它才知道状态改变时应该通知哪些观察者对象,并且它还要能随时添加和删除观察者对象。所以目标对象应该要有一个列表,来保存对它的所有观察者对象的引用。但在C++或Java这样的编程语言中,一个列表中的所有项都必须是同一类型的。这说明所有的观察者对象都必须是同一个类,但这样会限制程序的灵活性。因为不同的观察者对象要执行不同的任务,我们应该让它们属于不同的类。解决这一矛盾的途径是让所有的观察者类都有一个共同的父类(Observer),这些观察者类(ConcreteObserver)都是该父类的不同子类。因为所有子类的对象都可以被当作父类的对象,因此它们即可以保持不同又可以保存在同一列表中。

同时,这也解决了另一个问题,那就是目标对象是如何通知它的观察者对象的呢?当然这是通过调用它的观察者对象的一个方法(也叫成员函数)来实现的。目标对象只需要负责通知它的观察者对象它的状态改变了,而对该观察者对象如何处理新状态以及属于哪个具体观察者类都不需要了解。这就是说目标对象只能以同一种方式对待它的所有观察者对象,即这些观察者对象都要有一个相同的方法供目标对象调用,我们称该方法为update方法。

我们在抽象观察者类(Observer)中定义该方法,并在所有具体观察者类(ConcreteObserver)中重定义它。当目标对象的状态改变时,它在它的列表中每遍历到一个观察者对象,就调用该观察者对象的update方法。对目标对象来说,所有的观察者对象都是Observer类的,并且该类确实有一个update方法,所以能调用成功。又因为多态,实际执行的却是该观察者对象所属的具体观察者类中重定义的那个update方法。

要在观察者类中建立这样的继承关系的另一大原因是为了能方便地扩展功能。因为目标对象将所有的观察者对象都当作Observer类的对象来处理,所以要增加某种新的观察者对象时,我们只需创建一个新的类,让它继承Observer类并重定义update方法,在该update方法中实现它自己的任务逻辑就行了。这样所有由该新类创建的观察者对象都可以很容易地融入到当前的观察者模式中,程序的其余部分都不需要改变。

3.2 Subject和ConcreteSubject

那么为什么目标类(目标对象所属的类)也需要有一个抽象目标类(Subject)和多个具体目标类(ConcreteSubject)这样的继承结构呢?如果整个系统中只有一个目标对象,那么确实可以只用一个目标类实现观察者模式。我们之前说的都是多个观察者对象观察一个目标对象,但其实一个观察者对象也可以同时观察多个目标对象。

既然要同时观察多个目标对象,那么它们很可能有不同的状态以及不同的功能,即这些目标对象可能属于不同的类。也就是说一个观察者对象要被多个不同类的目标对象通知到,注意目标对象通知观察者对象是通过调用观察者对象的一个方法实现的。我们先考虑一个笨办法,在观察者类中为每一个目标类都提供一个版本的update方法。这样不仅麻烦而且代码难以维护,想象一下我们要增加一个新的目标类,那么就需要在所有要观察它的观察者类中都增加一个对应版本的update方法;当我们想删掉一个目标类的时候,又要在这些观察者类中删除那个对应版本的update方法。

一个通用的解决方式是让所有的目标类都调用同一个update方法,但是将自身的引用作为该方法的一个参数传递给观察者对象,这样观察者对象就知道是哪一个目标对象在通知它。一个方法(或成员函数)的参数的类型是确定的,也就是说所有的目标类都应该是同一个类。

这就又出现了观察者类中开始的矛盾局面,即所有的目标类既要是同一个类也要是不同的类。解决的方式也是一样的,即所有的目标类都有同一个父类(抽象目标类Subject),而不同的目标类(具体目标类ConcreteSubject)都是该父类的不同子类。

无论一个观察者对象可以同时观察多个目标对象还是只能观察一个目标对象,目标类的这种继承关系都有助于增加新的目标类(即扩展程序的功能)。当我们想增加新的具体目标类时,就创建一个新类,再让它继承Subject类并实现它自己特有的事务逻辑就可以了。这样由该新类所创建的目标对象就可以很轻松地融入到当前的观察者模式中,程序的其余部分都不会受到影响。

抽象目标类(Subject)和具体目标类(ConcreteSubject)以及抽象观察者类(Observer)和具体观察者类(ConcreteObserver)之间的继承关系降低了目标对象(它属于某个具体目标类)和观察者对象(它属于某个具体观察者类)之间的耦合度。

3.3 获取新状态

总的来说,有两种方式可以让观察者对象获取到目标对象的新状态。一是当目标对象调用每个观察者对象的update方法时,将代表它新状态的数据作为该update方法的一个参数传递给该观察者对象。二是目标对象在调用update方法时并不传递新状态,而是该观察者对象在被通知到的时候(在update方法中)再主动去询问该目标对象的新状态是什么。

如果采用第二种方式,那么观察者对象需要拥有它的目标对象的引用,再通过该引用调用目标对象的某个方法,该方法返回目标对象的新状态。该引用可能是目标对象调用观察者对象的update方法时传递来的,也可能是最初将该观察者对象添加进目标对象的列表中(订阅)的时候,设置该观察者对象的某个成员变量让它一直保存对该目标对象的引用。

此时我们可能又要面对既要相同又要不同的问题。先考虑目标对象在调用观察者对象的update方法时传递回它的新状态的方式,我们已知update方法的原型是唯一的(即它的参数数量和类型以及返回值类型是确定了的)。既然所有具体目标类的状态都要作为同一个参数传递,那么这些状态都必须是同一种类型的。但是这些不同的具体目标类的状态很可能不一样,至少在某些细节上有差异,这样看的话它们的状态又很可能是不同类型的。

再考虑上面的第二种方式,即观察者对象在获得通知后再调用目标对象的某个方法,该方法返回目标对象的新状态。我们纠结的是如何在所有的目标类中实现这个状态获取方法,我们称它为getState方法。

3.3.1 用继承方式实现状态获取方法

首先我们可以在目标类的继承关系中实现getState方法,也就是先在抽象目标类(Subject)中定义一个getState方法,然后再在每个具体目标类中重定义该getState方法,使之返回该具体目标类的实际状态。这样做是很容易想到的,因为所有的具体目标类都要有一个功能相似的getState方法,而所有目标类本身就有一个建立好的继承关系。对于所有子类中相似的部分我们应该将它提取到父类中,让子类继承以实现一致性。

另一个原因是,当一个观察者对象同时观察多个目标对象时,这些目标对象都是作为Subject类的对象通过update方法传递给它的。观察者对象将通知它的目标对象都当作Subject类型的对象对待的,可能根本就不知道该目标对象到底属于哪个具体目标类。此时,也要求在Subject父类中为所有类型的目标对象都定义一个相同的状态获取方法。

此时getState方法的实现和update方法是相似的,都是利用了多态的特性让调用父类对象的方法时实际执行的是该对象真正所属的子类中重定义的同名方法。这种实现getState方法的措施限制了getState方法的原型也是唯一的,即它的参数数量和类型以及返回值类型都是确定了的,也就要求所有具体目标类的状态是同一种类型的。

如果所有具体目标类的状态确实是同一种类型或者可以提炼到同一种类中,那么以上两种获取目标对象新状态的方式都是可行的。同时为了保持各个具体目标类的状态的差异化,状态的实现也可以采用继承的方式。定义一个抽象状态类和多个具体状态类,所有具体状态类继承该抽象状态类。每个具体目标类将它对应的具体状态类的对象(代表该具体目标类的状态)作为抽象状态类的对象通过update方法传递给观察者对象或者通过getState方法返回给它们。

3.3.2 每个具体目标类实现不同的状态获取方法

然而现实中也有很多这样的情况:那就是多个具体目标类的状态之间差异太大,根本无法统一为同一种类型。此时只能使用第二种获取目标对象新状态的方式,因为这些不同具体目标类的状态根本不能作为update方法的同一个参数进行传递。

因此各个具体目标类的状态获取方法也会是不一样的,至少它们的返回值类型是不一样的,很可能连方法名也不相同。这样就无法在抽象目标类(Subject)中为所有的具体目标类定义相同的getState方法。

当观察者对象被它的目标对象通知状态改变时,它必须要知道该目标对象是什么具体目标类的,这样它才能调用该具体目标类专有的状态获取方法来获取新状态。但是前文已多次说过所有目标对象都是作为Subject类型传递给观察者对象的,这似乎又是矛盾的。

其实这个问题也可以解决,虽然确实是任意一种具体目标类的目标对象都可以调用任意一个观察者对象的update方法,但实际上一个观察者对象只对某些具体目标类的目标对象感兴趣。因为它需要特定类型的输入数据,而不是任何数据它都可以处理。比如第2小节中,我们那个天气App的例子中的所有观察者对象(界面组件)都只能处理天气数据,而对金融或交通数据都不适用。

通常在设计一个具体观察者类的时候,就已知它所期待的目标对象是属于哪一些具体目标类的,我们也应该只让该类的观察者对象订阅它所期待的那些类的目标对象。当观察者对象的update方法被调用时,它可以检测传递进来的目标对象是否属于它所期待的那些具体目标类,比如Java中的instanceof运算符就可以检测一个对象是否是某种类的对象。如果不是,那么它就忽略本次通知而什么也不做;如果是,它就知道了该目标对象的具体类型,也就能调用该目标对象专有的状态获取方法了。

这样做看似会降低该具体观察者类的复用性,但实际情况是每一个具体观察者类都不是要在任意场景中都可以使用,它本身就是只针对某一领域设计的。上面的观察者模式类图中以及下面我们实现观察者模式的时候,就是用这种方式获取目标对象的新状态的。

4. 实现

现在,让我们来用Java语言和观察者模式实现第2小节中的那个天气App的例子;当然我们不会真的去开发一个功能完整且界面美观的App,我们让这些界面组件打印出它们应该显示的内容来模拟它们的实际功能。

首先,在Observer.java文件中我们定义抽象目标类Subject和接口MyObserver,接口MyObserver充当观察者模式类图结构中的抽象观察者类Observer,我们之前就说过这个抽象父类可以用接口实现。

另一个要注意的是,我们将该接口命名为MyObserver而不是Observer,这是因为Java中本来就有一个自带的名为Observer的接口,它也是用来实现观察者模式的。这里我们想完全实现我们自己的观察者模式,而不使用Java自带的Observer接口。

MyObserver接口只有一个方法update,目标对象通过调用它来通知观察者对象它的状态改变了,该update方法有一个类型为Subject的参数,这允许目标对象将它自己传递给观察者对象,这样一个观察者对象就可以同时观察多个目标对象。

在这个简单例子中只有一个目标对象,因此也只有一个具体目标类,它就是Weather类。为了例子保持简单,这里的天气数据只包含温度;Weather类的状态就是当前的温度值,它通过随机生成一个-80至60的浮点数作为新的温度值来模拟对天气数据的获取,当然实际中的App应该通过一个接口到服务器上获取真实的气象数据。

目标对象必须确保状态确实改变了才通知观察者对象,因此Weather类必须测试新的温度值是否和之前的温度值相等。Weather类的状态获取方法getTemperature是它特有的,没有按照继承关系的方式定义它,即其它具体目标类的状态获取方法是不同的。

在这个例子中,目标对象的状态改变时由它自己调用它的notifyObservers方法通知所有的观察者对象(notifyObservers方法会依次调用每个观察者对象的update方法)。其实也可以在目标对象的状态改变后,由客户代码调用目标对象的notifyObservers方法通知观察者。

在ui.java文件中我们定义两个界面组件类,CelsiusView和WearView,它们都是具体观察者类,因此它们都要实现MyObserver接口。CelsiusView按照摄氏度显示温度值,而WearView则显示穿衣建议。当然这个例子中,它们并不会绘制实际的界面,而是通过一个打印语句来模拟对界面的显示。

虽然我们说过一个观察者对象可以同时观察多个属于不同类的目标对象,但在该例子中的观察者对象只会观察一个目标对象,因为这两个界面组件是用于显示天气信息的,因此对其它类型的目标对象也不感兴趣。所以,当它们被目标对象通知的时候,它们会在检查目标对象确实是一个Weather类的对象后才执行各自的任务。

当这两个观察者对象确实收到来自Weather类的目标对象的通知时,它们就将该目标对象强制转换为Weather类的对象,并通过调用Weather类的专有状态获取方法getTemperature()获取目标对象的新状态。

在App.java文件中,我们定义App类,它模拟该天气App的运行。以上示例代码的执行结果如下图所示,我们可以看到当目标对象的状态改变时(在本例子中是温度值的改变),两个观察者对象都被通知到了并更新了它们的界面显示。

图4 示例代码的执行结果

5. 结语

在某些平台上,可以向目标对象同时注册一个观察者对象和它的某个方法(成员函数)。这样当目标对象的状态改变时,它就调用这个观察者对象的这个注册的方法,而不一定要去调用它的update方法。这就允许这些观察者对象没有update方法,即它们不用都继承自同一个父类。当然一个目标对象可能要求它的所有观察者注册的回调方法都具有相同的函数原型,即这些方法要有相同数量和类型的参数以及相同类型的返回值。

观察者模式还有很多其它的实现方式,但这些方式都只是在一些细节上有所不同。只要理解了观察者模式的主要概念,就能够很容易理解这些细节差异。

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

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

相关文章

1、MongoDb综述

1. MongoDb综述 1.1. 什么是Nosql NoSQL:Not Only SQL ,本质也是一种数据库的技术,相对于传统数据库技术,它不会遵循一些约束,比如:sql标准、ACID属性,表结构等。 Nosql优点 l 满足对数据库的高并发读写…

创建查询系统,提升工作效率

今天我要和大家分享一个非常实用的小技巧,能够让老师们在短短三分钟内创建一个非常方便的查询系统!是的,通过使用易查分这个神奇的工具,你可以轻松满足各种查询需求! 首先,老师们只需要注册一个易查分账号。…

使用两个队列模拟栈

整体思路如下图: 代码实现 import java.util.LinkedList; import java.util.Queue; import java.util.Scanner;/*** author: Arbicoral* Description: 使用2个队列模拟栈的 push() pop() top(), 自己实现打印 print()*/ public class QueueMoniStack2 {public stati…

uniapp 模糊搜索(小白必看)

实现模糊搜索很简单,按照下面的步骤: 1. 搜索栏 <view class"search-box"><uni-search-bar class"uni-mt-10" radius"100" placeholder"请输入移交信息" clearButton"auto" bgColor"#F8F8F8"cancelBut…

SpringMvc决战-【SpringMVC之自定义注解】

目录 一、前言 1.1.什么是注解 1.2.注解的用处 1.3.注解的原理 二.注解父类 1.注解包括那些 2.JDK基本注解 3. JDK元注解 4.自定义注解 5.如何使用自定义注解&#xff08;包括&#xff1a;注解标记【没有任何东西】&#xff0c;元数据注解&#xff09;&#xff1f; 三…

Linux下使用lookbusy加载cpu负载

Lookbusy 是一个用于在 Linux 系统上生成合成负载的简单应用程序。它可以在 CPU 上生成固定的、可预测的负载&#xff0c;保持选定数量的内存处于活动状态&#xff0c;并生成您需要的任意数量的磁盘流量。 官方地址&#xff1a;lookbusy -- a synthetic load generator 编译 …

Article Forge:AI写作文章内容生成器

【产品介绍】 名称 Article Forge 成立/上线时间 2022年 具体描述 Article Forge是一款基于人工智能和深度学习的AI写作文章内容生成器&#xff0c;可以自动写出1500字的文章无论是产品描述&#xff0c;还是整篇博客文章&#xff0c;Article Forge都能在一…

堆与栈的区别

OVERVIEW 栈与堆的区别一、程序内存分区中的堆与栈1.栈2.堆3.堆&栈 二、数据结构中的堆与栈1.栈2.堆 三、堆的深入1.堆插入2.堆删除&#xff1a;3.堆建立&#xff1a;4.堆排序&#xff1a;5.堆实现优先队列&#xff1a;6.堆与栈的相关练习 栈与堆的区别 自整理&#xff0c;…

竞赛 基于机器视觉的车道线检测

文章目录 1 前言2 先上成果3 车道线4 问题抽象(建立模型)5 帧掩码(Frame Mask)6 车道检测的图像预处理7 图像阈值化8 霍夫线变换9 实现车道检测9.1 帧掩码创建9.2 图像预处理9.2.1 图像阈值化9.2.2 霍夫线变换 最后 1 前言 &#x1f525; 优质竞赛项目系列&#xff0c;今天要分…

video 视频编解码一些debug方法

文章目录 一、通过命令去获取一些数据1.2 确定我们xml配置文件: 二、查看我们芯片支持的编码能力三、通过log去获取信息 这个文章的主要内容是为了后期性能方面的debug, 设计到前期的bringup则没有 一、通过命令去获取一些数据 获取媒体相关的参数&#xff1a; # getprop |…

Thinkphp6 配置并使用redis图文详解 小皮面板

这篇文章主要介绍了Thinkphp6 配置并使用redis的方法,结合实例形式详细分析了Redis的安装、配置以及thinkphp6操作Redis的基本技巧,需要的朋友可以参考下 一、安装redis ThinkPHP内置支持的缓存类型包括file、memcache、wincache、sqlite。ThinkPHP默认使用自带的采用think\Ca…

SpringMVC之自定义注解

目录 一.什么是Java注解 1.简介 2.注解的分类 3.JDK元注解 二.自定义注解 1.自定义注解的分类 1.1.标记Annotation: 1.2.元数据Annotation: 2.如何使用自定义注解 3.案例演示 3.1 获取类、方法及属性上的注解值 3.2Inherited 的使用 3.3获取类属性上的注解属性值 3.…

springboot整合mybatis

一、项目结构展示 二、开始整合 1、引入pom依赖 进入Maven中央仓库选择自己所需要的依赖&#xff0c;maven仓库地址&#xff1a;Maven Central 完整Maven依赖如下&#xff1a; <?xml version"1.0" encoding"UTF-8"?> <project xmlns"h…

markdown工具Atom预览与插件安装

​atom是以命令行作为插件选项的入口 打开命令输入框 Windows: ctrl shift p Mac: command shift p 输入命令安装 输入 markdown preview toggle &#xff0c;可以偷懒只输入mdpt(模糊匹配) 按enter键即可看到预览&#xff0c;如图&#xff0c;左边编辑&#xff0c;右…

Vue中的过滤器 Filters

过滤器 Filters 过滤器一般用于格式化文本内容&#xff0c;通常可以在两个地方使用&#xff0c;主要是模板语法、以及 v-bind 表达式中。例如我想对展示的文本进行一些特殊处理&#xff0c;将金额进行四舍五入后再展示。选项 filters 内可以编写多个自定义过滤器。 用法&…

LightDB 23.3 通过GUC参数控制commit fetch

背景 commit游标提交之后&#xff0c;可以继续使用fetch进行结果集的操作。commit和fetch结合使用功能开发时不考虑分布式。后续&#xff0c;又对分布式进行了测试&#xff0c;发现持有portal后&#xff0c;代码中会对querydesc进行非空判断。当querydesc为空时&#xff0c;Li…

工业交换机常见的硬件故障有哪些?

工业交换机常见的硬件故障主要是由于受到供电电源、室内温度、室内湿度、电磁干扰、静电等机房环境的影响&#xff0c;造成工业交换机电源、背板、模块、端口等部件出现故障。具体可以分为以下几类。 1.电力供应故障&#xff1a; 由于外部供电不稳定、电源线路老化或雷击等原因…

LiveNVR监控流媒体Onvif/RTSP功能-支持海康摄像头海康NVR通过EHOME协议ISUP协议接入分发视频流或是转GB28181

LiveNVR支持海康NVR摄像头通EHOME接入ISUP接入LiveNVR分发视频流或是转GB28181 1、海康 ISUP 接入配置2、海康设备接入2.1、海康EHOME接入配置示例2.2、海康ISUP接入配置示例 3、通道配置3.1、直播流接入类型 海康ISUP3.2、海康 ISUP 设备ID3.3、启用保存3.4、接入成功 4、相关…

亚马逊封买家账号的原因有哪些

亚马逊可能封锁买家账号的原因有多种&#xff0c;主要是出于保护市场和维护平台秩序的考虑。以下是一些可能导致亚马逊封锁买家账号的常见原因&#xff1a; 1、涉及违规行为&#xff1a;如果买家违反了亚马逊的使用政策&#xff0c;如发表虚假评价、滥用退货政策、欺诈或盗窃等…

【视觉SLAM入门】7.3.后端优化 基于KF/EKF和基于BA图优化的后端,推导及举例分析

"时间倾诉我的故事" 1. 理论推导2. 主流解法3. 用EKF估计状态3.1. 基于EKF代表解法的感悟 4. 用BA法估计状态4.1 构建最小二乘问题4.2 求解BA推导4.3 H的稀疏结构4.4 根据H稀疏性求解4.5 鲁棒核函数4.6 编程注意 5.总结 引入&#xff1a; 前端里程计能给出一个短时间…