<Qt> 信号和槽

news2025/1/12 20:46:01

目录

一、信号和槽概述

二、信号和槽的使用​​​​​​

(一)connect函数

(二)实现一个点击按钮关闭窗口的功能

(三)再谈connect

三、自定义槽函数

四、自定义信号

五、带参数的信号和槽

六、信号和槽存在的意义

七、信号和槽的断开

八、使用 Lambda 表达式来定义槽函数


一、信号和槽概述

        “信号”这个词在我们的日常生活中随处可见,比如:信号灯变绿,我们行人进行通行、鸡叫表示天亮了、下课铃响了代表着下课了、王者连跪表示我们该充钱了等等信号这个概念,在我们的生活中随处可见。从以上的例子中我们仔细观察的话其实是会发现,每一个信号后面都对应着我们的一个动作,比如:绿灯了,我们要通过人行道、鸡叫了我们要起床了、下课了我们要去干饭、王者连跪了我们要打开微信充钱…这些信号后面伴随的动作是怎么来到?或者说我们怎么知道接收到这些信号后该做什么?当然是生活经验、和老师学校教的反正不可能是临时起意嘛!换而言之就是这些信号后面代表的动作都是我们早已知道的!

        在Linux系统中,我们也介绍了信号的产生、信号的检测以及信号的处理机制,它就是系统内部的通知机制,也可以是一种进程间通信的方式。在系统中有很多信号,我们可以通过signal()函数捕捉信号,重写一个信号处理函数。在Qt中的信号也和系统中的信号有相似之处。

在QT中也是同理,信号就代表着用户发出了一个指令,而槽就代表着对于这个指令的处理方法!


  • 在Qt中,用户和控件的每次交互过程称为一个事件。比如"用户点击按钮"是一个事件,"用户关闭窗口"也是一个事件。每个事件都会发出一个信号,例如用户点击按钮会发出"按钮被点击"的信号,用户关闭窗口会发出"窗口被关闭"的信号。
  • Qt中的所有控件都具有接收信号的能力,一个控件还可以接收多个不同的信号。对于接收到的每个信号,控件都会做出相应的响应动作。例如,按钮所在的窗口接收到"按钮被点击"的信号后,会做出"关闭自己"的响应动作;再比如输入框自己接收到"输入框被点击"的信号后,会做出"显示闪烁的光标,等待用户输入数据"的响应动作。在Qt中,对信号做出的响应动作就称之为槽。
  • 信号和槽是Qt特有的消息传输机制,它能将相互独立的控件关联起来。比如,"按钮"和"窗口"本身是两个独立的控件,点击"按钮"并不会对"窗口"造成任何影响。通过信号和槽机制,可以将"按钮"和"窗口"关联起来,实现"点击按钮会使窗口关闭"的效果。

信号的本质
信号是由于用户对窗口或控件进行了某些操作,导致窗口或控件产生了某个特定事件,这时Qt对应的窗口类会发出某个信号,以此对用户的操作做出反应。因此,信号的本质就是事件。如:

  • 按钮单击、双击
  • 窗口刷新
  • 鼠标移动、鼠标按下、鼠标释放,
  • 键盘输入

在 Qt 中信号是通过什么形式呈现给使用者呢?

  • 我们对哪个窗口进行操作,哪个窗口就可以捕捉到这些被触发的事件。
  • 对于使用者来说触发了一个事件我们就可以得到Qt框架给我们发出的某个特定信号。
  • 信号的呈现形式就是函数,也就是说某个事件产生了,Qt 框架就会调用某个对应的信号函数,通知使用者。

槽的本质

(Slot)就是对信号响应的函数。槽就是一个函数,与一般的C++函数是一样的,可以定义在类的任何位置( public、protected 或private), 可以具有任何参数,可以被重载,也可以被直接调用(但是不能有默认参数)。槽函数与一般的函数不同的是:槽函数可以与一个信号关联,当信号被发射时,关联的槽函数被自动执行。

说明
(1) 信号和槽机制底层是通过函数间的相互调用实现的。每个信号都可以用函数来表示,称为信号函数;每个槽也可以用函数表示,称为槽函数。 例如: "按钮被按下"这个信号可以用clicked()函数表示,"窗口关闭"这个槽可以用close()函数表示,假如使用信号和槽机制实现: "点击按钮会关闭窗口"的功能,其实就是clicked()函数调用close()函数的效果。

(2) 信号函数和槽函数通常位于某个类中,和普通的成员函数相比,它们的特别之处在于:

  • 信号函数用 signals 关键字修饰,槽函数用public slots、protected slots或者private slots修饰。signals 和 slots 是Qt在C++的基础上扩展的关键字,专门用来指明信号函数和槽函数;
  • 信号函数只需要声明,不需要定义(实现) , 而槽函数需要定义(实现)。

信号函数的定义是Qt自动在编译程序之前生成的.编写Qt应用程序的程序员无需关注。这种自动生成代码的机制称为元编程(Meta Programming) .这种操作在很多场景中都能见到。


Qt中的信号也要涉及信号三要素:信号源、信号类型和信号处理方式:

  • 信号源:Qt中的信号是由某个控件发出的,Linux系统中可以是一个进程发送的信号。
  • 信号类型:用户不同的操作会触发不同的信号,例如点击按钮就会触发点击信号(clicked)、在输入框中移动光标也会触发相应的信号,我们写这样的GUI程序就是为了和用户交互,所以必须知道当前用户的具体操作,通过不同的操作进行不同的处理
  • 信号处理方式:在Qt中就引入了一个概念就是slot),说白了就是一个函数,再使用connect这样的函数把一个信号和一个槽关联起来,之后只要触发了信号,Qt就会自动执行槽函数,这种槽函数本质上也是一种回调函数

所以在Qt中一定要先关联信号和槽,也就是先要有槽函数并使用connect函数将二者关联起来,然后再触发这个信号,顺序不能颠倒

二、信号和槽的使用​​​​​​

(一)connect函数

上面既然说了,信号的三要素是:发送源、信号类型、处理方法,可是对于QT来说,它怎么知道哪一个信号该调用哪一个处理方法呢?也就是说,对于QT来说,它怎么知道某个信号该调用哪个处理方法呢?
当然,是我们开发人员提前将信号与处理方法之间的关系绑定好,到时候QT直接进行调用即可!

在 Qt 中,QObject类 中提供了一个静态成员函数 connect() ,该函数专门用来关联指定的信号函数和槽函数。

关于QObject类:
QObject类实际上是QT中所有内置类型的祖宗类!也就是说在QT中大部分类要么直接继承自QObject类,要么间接继承自QObject类,比如 QWidget 就继承自 QObject。与Java中的继承十分相似!
在QT中大概的继承关系如下(不准确的继承):

以下是 connect 的函数原型:

connect (const QObject *sender,
		const char *signal,
		const QObject *receiver,
		const char *method,
		Qt::ConnectionType type = Qt::AutoConnection)

sender:   哪个控件发出的信号
signal:   发出的什么信号(信号函数) 
receiver: 谁接收这个信号
method:   怎么处理这个信号(接收信号的槽函数)
type:     用户指定关联方式,常用默认值,一般不需要手动设置.

connect 要求第一个参数和第二个参数是匹配的,比如第一个参数的类型如果是 QPushButton* ,那么第二个参数的信号必须是 QPushButton 内置的信号(父类的信号),不能是其他的一个类。

(二)实现一个点击按钮关闭窗口的功能

我们在窗口上设置一个按钮,当我们按下这个按钮就可关闭窗口。

分析: 当用户按下按钮控件过后,实际上是向QT发送了一个叫clicked的信号,为此我们现在要做的就是捕捉该信号,为该信号关联一个关闭函数,也就是对应Widget类中的 close() 函数

具体代码如下:

#include "widget.h"
#include "ui_widget.h"
#include <QPushButton>

Widget::Widget(QWidget *parent)
    : QWidget(parent)
    , ui(new Ui::Widget)
{
    ui->setupUi(this);

    // 点击按钮,窗口关闭
    QPushButton *button = new QPushButton(this); // 创建一个button对象
    button->setText("关闭"); // 设置按钮上的文本
    button->move(200, 200); // 设置按钮的位置
    connect(button, &QPushButton::clicked, this, &Widget::close);
}

Widget::~Widget()
{
    delete ui;
}
  1. 创建一个QPushButton对象,想要点击这个按钮后就关闭这个窗口,发出信号的一定是QPushButton对象。
  2. 发出的信号一定是QPushButton内置的clicked信号。
  3. 要交给的对象就是this也就是Widget对象。
  4. 处理函数就可以使用Widget继承的QWidget内置的close()槽函数。

注意这里是 clicked,而不是 click:

运行程序后,点击按钮成功关闭窗口: 

(三)再谈connect

但是对以上的代码我们有两个疑问?

我们怎么知道点击一下按钮就会发出一个clicked信号?或者我们怎么知道QT有哪些信号或槽函数?

答:多看官方文档;当我们不了解某一个知识点的时候,官方文档是最好的学习工具,况且QT自己在也已经内置了官方文档:

如果我们要查找某一个类的信号的话,我们可以直接对这个类进行搜索,当然可能这个类中并没有包含整个信号,那么我们可以对这个类的父类、祖宗类进行搜索,这样的话我们一定可以找到这个信号,就比如上图的clicked信号,在QPushButton中就没有,但是在其父类QAbstractButton就存在:

找了一遍后发现没有找到clicked这样的信号,既然这样就要向它的父类中查找。 

QPushButton的父类就是QAbstractButton,通过类名我们也可以得知这是一个抽象类,在这个页面向下查找就找到了Signals这块,这里就有clicked等信号,也就类似于函数。

 对于槽函数我们也是如此:


在查看官方文档的时候,connect的原型如下 :

 第一个和第三个参数我们传入的是 QObject 子类指针,所以没有问题,但是第二个和第四个参数传入的是函数指针,但是为什么使用 const char* 来接收呢?( 并且这在C++中是绝对禁止的!) 就拿上面connect的例子举例,clicked的类型应该是void ( * ) () ,而close的类型应该是 bool ( * ) (),可是为什么这两个函数指针类型可以直接赋值给 const char*? 这两个的意义可不一样。

  • 原因就是这是旧版本的 connect 函数声明,是在对于信号传参的时候需要用SIGNAL宏修饰一下,对于槽函数传参的时候需要使用SLOT宏修饰一下,通过宏替换就可以把函数指针转换成 char*,这样就不会有类型不匹配的问题了。
  • 官方文档里的这个函数声明是旧版本的 connect 声明 ,正确写法是:
  • connect(button1, SIGNAL(&QPushButton::clicked), this, SLOT(&Widget::close))

从 Qt5 开始,对该写法进行了简化,不再需要写这两个宏了,所以上面的代码才能直接传入函数指针。Qt5 开始,给 connect 提供了重载版本,第二个参数和第四个参数成了泛型参数,允许传入任意类型的函数指针了。
点击 ctrl 然后鼠标点击 connect ,跳转到源码中查看:

这样的话,对于一个函数指针来说,它就不用再使用SIGNAL宏和SLOT宏来进行修饰了,根据模板的特性,它会自动推导传递过来的函数指针的类型,极其方便!同时新版的connect还使用了“类型萃取”技术,它可以帮助我们检查参数,发送源控件与信号类型是否匹配:若传入的第一个参数和第二个参数,或者第三个参数和第四个参数不匹配,(不匹配是指:2、4参数传入的函数指针,不是1、3参数的成员函数的指针。)代码出现编译错误。

三、自定义槽函数

纯代码

虽然QT已经给我们内置了许多对应的槽函数,但是在我们实际的开发中,我们更多的情况下,是结合自己的业务场景,针对对应的信号量身定做一个槽函数,也就是自定义槽函数。

自定义参函数的语法:

  1. 所谓的 slot 就是一个普通的成员函数,所以所谓的自定义一个槽函数,操作过程和自定义一个普通的成员函数没啥区别。
  2. 该成员函数需要使用public signals、private signals、protected signals来进行修饰。(在QT 5及以上版本中可以不用signal修饰)

为此,为了演示自定义槽函数,这也是上一篇也用过的,先使用纯代码的方式实现一下自定义槽。我们设计出一个具体的场景:

当用户点击按钮的时候就可以切换窗口标题。

分析: 当用户点击按钮的时候,实际上会发出一个clicked信号,现在对应的处理动作是将窗口标题改为:“捕捉到按钮信号”,为此我们需要捕捉clicked信号,并且自定义处理函数。

  • 还是要先new一个QPushButton对象,把点击信号和槽函数关联起来。
  • 槽函数中就设置为,捕捉到了信号就把Widget界面的标题修改一下。

具体代码如下:

widget.h:记得声明函数

#ifndef WIDGET_H
#define WIDGET_H

#include <QWidget>

QT_BEGIN_NAMESPACE
namespace Ui { class Widget; }
QT_END_NAMESPACE

class Widget : public QWidget
{
    Q_OBJECT

public:
    Widget(QWidget *parent = nullptr);
    ~Widget();

// 自定义槽函数
//public slots:
//    void handleClicked();

    void handleClicked(); // 这两者自定义槽函数的方式都是可以的

private:
    Ui::Widget *ui;
};
#endif // WIDGET_H

widget.cpp

#include "widget.h"
#include "ui_widget.h"
#include <QPushButton>

Widget::Widget(QWidget *parent)
    : QWidget(parent)
    , ui(new Ui::Widget)
{
    ui->setupUi(this);

    QPushButton *button = new QPushButton(this); // 创建一个button对象
    button->setText("按钮"); // 设置按钮上的文本
    button->move(200, 200); // 设置按钮的位置
    connect(button, &QPushButton::clicked, this, &Widget::handleClicked);
}

Widget::~Widget()
{
    delete ui;
}

void Widget::handleClicked()
{
    // 按下按钮,修改窗口标题
    this->setWindowTitle("捕捉到按钮信号");
}

运行程序,查看效果:

窗口标题成功修改: 

图形化界面

在 ui 界面,拖动 pushbutton 到编辑界面然后输入文字后,右键单击该按钮,选择 “转到槽” 

这个窗口为我们提供了 QPushButton 的所有信号,还包括了 QPushButton的父类信号。选择第一个信号:

点击确定或者双击就会跳转到编辑widget.cpp的界面,此时函数的声明和定义都已经自动生成好了,编写代码即可:

【注意】:这种方式是不会出现 connect() 函数的,Qt中除了connect可以连接信号和槽外,还可以通过函数名字的方式来自动连接,如上图所示函数名格式:on_ XXX_ SSS,其中:

  • 以"on"开头,中间使用下划线连接起来;
  • " XXX "表示的是对象名(控件的objectName)。
  • " SSS "表示的是对应的信号。
  • 如: " on_ pushButton_ _clicked() " , pushButton代表的是对象名,clicked 是对应的信号。

点击按钮,改变窗口的标题:


这里名字故意写错,就不能达到我们想要的效果:

这都是Qt中调用connectSlotsByName这个函数触发自动连接信号和槽,这个函数是在自动生成的ui_widget.h中setupUi函数中调用的。 

所以,如果是通过图形化界面创建控件,还是使用第二种方式快速连接信号和槽;如果是通过代码的方式创建控件,还是得手动调用connect函数,原因就是没有调用connectSlotsByName函数。

四、自定义信号

自定义信号很少用到。因为在GUI中,用户的操作行为是可以穷举的,Qt内置的信号已经覆盖到了大部分可能的用户操作,对于某些场景还是需要我们使用自定义信号的!

  • 信号是一种特殊的函数,程序员只需写出函数声明,并告诉Qt(需要使用slots关键字进行修饰),这是一个信号即可。这个函数的定义,是Qt在编译过程中,自动生成的(无法干预)。
  • 信号函数的返回值必须是void,有没有参数都可以,也支持函数重载。
  • 信号可以使用emit关键字进行发射(Qt5 emit不写也行)。

为此,我们根据该场景设计出一个具体的业务场景:

当用户点击按钮就可以切换窗口标题

signals是Qt自己扩展的关键字,qmake会调用代码分析和生成的工具,识别到signals这个关键字时就会把下面的函数声明认为是信号,并自动生成函数定义:

通过connect函数连接信号和槽函数就可以了,但是光连接还是不够的,还需要发送我们自定义的信号,这个emit也是Qt中的关键字,作用就是发送信号:

当点击按钮,就会触发on_pushButton_clicked函数,函数中就会发送mySignal信号,收到信号就会执行handleSignal自定义函数,之后就会修改窗口标题:

五、带参数的信号和槽

在自定义信号和槽函数的时候可以带上参数,并且QT允许重载信号和槽函数。信号在和槽函数进行绑定的时候,主要表现在以下两点:

  • 槽函数的参数类型必须和它绑定的信号的参数类型一致。
  • 槽函数的参数个数可以少于等于它绑定的信号的参数的个数。

参数的类型必须要一致,个数不一致是可以的,但是要求信号的参数的个数必须要比槽的参数多。

通过这一次连接信号和槽,并搭配不同参数就可以实现不同结果,可以让代码复用,就比如:

点击按按钮1,左上角标题改为标题1,点击按钮2,左上角标题改为标题2:

而且信号的参数比槽函数多也是可以的,但是不允许槽函数中的参数比信号中的多,原因就是:

  • 一个槽函数可能绑定多个信号。
  • 如果严格要求就意味着信号绑定到槽的要求变高了。
  • 所以信号的参数个数可以大于槽的参数个数,这样让信号和槽之间的绑定变得更灵活,更多的信号可以绑定到这个槽函数上。
  • 虽然个数不一致,但是槽函数会按照参数顺序拿前N个参数,这样就可以确保槽函数的每个参数都有值。

 还有一点就是,如果想要使用信号和槽的机制,就必须在类的一开始写上Q_OBJECT宏,否则会出现编译错误

当我们转到定义就可以看到这个宏中展开的代码,这里也有很多的宏,也会继续展开,最终实现相应的功能:

六、信号和槽存在的意义

信号和槽最主要就是要解决响应用户的操作,它在现在GUI框架中是一个有特色的机制,但是它的实现就没有那么简洁,但是它还要这样做就是为了:

  • 把触发用户操作的控件 和 处理用户的操作逻辑 解耦合。信号发送者不需要知道发出的信号被哪个对象的槽函数接收,槽函数也不需要知道哪些信号关联了自己,Qt的信号槽机制保证了信号与槽函数的调用。支持信号槽机制的类或者父类必须继承于QObject类。
  • 也想起到多对多的效果,一个信号可以connect到多个槽函数上,一个槽函数也可以被多个信号connect。(实际开发中,这种情况极少)

这就是可以把多个信号和多个槽函数绑定到一起:

缺点:

与回调函数相比,信号和槽稍微慢⼀些,因为它们提供了更高的灵活性,尽管在实际应用程序中差别不大。通过信号调用的槽函数比直接调用的速度慢约10倍(这是定位信号的接收对象所需的开销;遍历所有关联;编组/解组传递的参数;多线程时,信号可能需要排队),这种调用速度对性能要求不是非常高的场景是可以忽略的,是可以满足绝大部分场景。

七、信号和槽的断开

可以使用disconnect断开信号和槽的连接。但是大部分情况下,把信号和槽连接好就不用管了,主动断开的情况就是要把信号重新绑定到另一个槽函数上,如果不断开,这个信号发送就会触发两个槽函数。

  • 点击按钮修改标题:

  • 先点击断开原信号按钮,建立新连接,再点击修改标题按钮:

断开连接前,标题修改为旧标题,当点击下方pushButton后,上方原来的信号和槽断开了连接,重新绑定后,再点击上方pushButton,标题修改为新标题。

八、使用 Lambda 表达式来定义槽函数

Qt5在Qt4的基础上提高了信号与槽的灵活性,允许使用任意函数作为槽函数。
但如果想方便的编写槽函数,比如在编写函数时连函数名都不想定义,则可以通过Lambda表达式来达到这个目的。

Lambda表达式是C++11增加的特性。C++11 中的Lambda表达式用于定义并创建匿名的函数对
象,以简化编程工作。
Lambda表达式的语法格式如下:

[ capture ] ( params ) opt -> ret { 
 Function body; 
};

局部变量引入方式 [ ]
[ ]:标识一个Lambda表达式的开始。不可省略。

  • 由于使用引用方式捕获对象会有局部变量释放了而Lambda函数还没有被调用的情况。如果执行Lambda函数,那么引用传递方式捕获进来的局部变量的值不可预知。所以绝大多数场合使用的形式为:={}
  • 早期版本的Qt,若要使用Lambda表达式,要在"pro"文件中添加:
    CONFIG += C++11
    

使用 Lambda 表达式的方式填写槽函数。这里的功能是点击一下按钮,按钮就移动到相应位置:

上述代码传值捕获没问题,传引用捕获会崩溃。原因是button局部变量(它指向的空间位于堆区,但其本身是一个局部变量的指针),构造函数结束后button变量即被销毁,生命周期结束,若传引用对这块非法空间进行访问就会造成程序崩溃。

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

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

相关文章

Clonezilla 备份还原过程推送日志到 syslog

Clonezilla 备份、还原过程中&#xff0c;系统的运行日志只能显示到客户端显示器上&#xff0c;如果出现错误&#xff0c;无法在服务端查询到对应的日志&#xff0c;一是故障判断不太方便&#xff1b;另一方面&#xff0c;实现日志推送&#xff0c;也可以将 Clonezilla 运行进度…

【前端】ikun-qrcode:极简的二维码生成组件,使用view而非canvas避免层级问题

文章目录 背景ikun-qrcode界面效果如何发布一款自己的插件到uniapp市场。&#xff08;5分钟搞定&#xff09; 背景 之前在uniapp上100行搞定二维码生成&#xff0c; 现在封装为vue组件分享出来&#xff1a; 下载地址&#xff1a; https://ext.dcloud.net.cn/plugin?id19351 …

吐血整理如何在Google Earth Engine上写循环 五个代码实例详细拆解

引言 这篇文章主要解答GEE中.map()和.iterate()函数的用法。 首先解答一个疑问&#xff0c;为什么需要自己写循环&#xff1f;确实&#xff0c;GEE 为各种数据类型提供了无数常用的内置函数&#xff0c;对这些方法做排列组合足以应对大多数使用场景&#xff0c;算法效率也颇佳。…

台风预警新选择:太阳能LED宣传信号杆

台风预警新选择&#xff1a;太阳能LED宣传信号杆 以下是对台风灾害的严重性、传统预警方式的不足以及太阳能台风预警宣传信号杆的出现和优势等方面进行分析和归纳&#xff1a; 一、台风灾害的严重性 台风作为一种强烈的自然灾害&#xff0c;给沿海地区带来了极大的威胁。台风…

【数学建模】——【线性规划】及其在资源优化中的应用

目录 线性规划问题的两类主要应用&#xff1a; 线性规划的数学模型的三要素&#xff1a; 线性规划的一般步骤&#xff1a; 例1&#xff1a; 人数选择 例2 &#xff1a;任务分配问题 例3: 饮食问题 线性规划模型 线性规划的模型一般可表示为 线性规划的模型标准型&…

论文学习——基于自适应选择的动态多目标进化优化有效响应策略

论文题目&#xff1a;Effective response strategies based on adaptive selection for dynamic multi-objective evolutionary optimization 基于自适应选择的动态多目标进化优化有效响应策略&#xff08;Xiaoli Li a,b,c, Anran Cao a,∗, Kang Wang a&#xff09;Applied S…

活动报名 | 智源研究院数据与行业应用Workshop

7月25日周四下午14点&#xff0c;智源人工智能研究院将联合中国互联网协会人工智能工委会、中国AIIA联盟数据委员会、共同举办数据与行业应用 Workshop&#xff5e; 届时&#xff0c;智源的技术专家将介绍行业数据集和千万级指令微调数据集的构建思路和获取方法。更有来自北京…

【QGroundControl二次开发】二.使用QT编译QGC(Windows)

【QGroundControl二次开发】一.开发环境准备&#xff08;Windows&#xff09; 二. 使用QT编译QGC&#xff08;Windows&#xff09; 2.1 打开QT Creator&#xff0c;选择打开项目&#xff0c;打开之前下载的QGC项目源码。 编译器选择Desktop Qt 6.6.3 MSVC2019 64bit。 点击运…

Java项目打包成exe

文章目录 1.使用 exe4j 工具2.导出 jar 包3.转化 1.使用 exe4j 工具 安装激活&#xff1a; https://www.cnblogs.com/jepson6669/p/9211208.html 2.导出 jar 包 使用 mvn 的 package 导出默认没有依赖包&#xff0c;这里从 IDEA 的 Artifacts 导出。 // 导出路径为了方便…

鱼眼相机变普通相机,利用Transform进行球面变换

Abstract 高分辨率广角鱼眼图像在自动驾驶等机器人应用中变得越来越重要。然而&#xff0c;使用普通的卷积神经网络或视觉变换器处理这类数据时会遇到问题&#xff0c;因为在将其投影到平面上的矩形网格时会引入投影和失真损失。为了解决这个问题&#xff0c;我们引入了HEAL-S…

强化学习的数学原理(2)

Value iteration & Policy itreation Value iteration algorithm 之前我们已经讲过怎么去求解贝尔曼最优公式&#xff0c;是利用contraction mapping theorem 来进行求解&#xff0c;我们知道这个contraction mapping theorem是一个迭代算法&#xff0c;实际上这个算法他有…

在mybatis-plus中关于@insert注解自定义批处理sql导致其雪花算法失效而无法自动生成id的解决方法

受到这位作者的启发 > 原文在点这里 为了自己实现批量插入&#xff0c;我在mapper层使用insert注解写了一段自定义sql //自定义的批量插入方法 Insert("<script>" "insert into rpt_material_hour(id,sample_time,rounding_time,cur_month,machine_no…

【技术追踪】TeethDreamer:从 5 张口腔照片实现三维牙齿重建(MICCAI-2024)

三维重建搞起来~ TeethDreamer&#xff1a;一种3D牙齿重建新框架&#xff0c;旨在恢复上下牙齿的形状和位置&#xff0c;引入大型扩散模型的先验知识和3D感知特征注意力机制&#xff0c;重建性能表现SOTA&#xff01; 论文&#xff1a;TeethDreamer: 3D Teeth Reconstruction f…

elasticsearch 聚合 : 指标聚合、桶聚合、管道聚合解析使用总结

❃博主首页 &#xff1a; <码到三十五> ☠博主专栏 &#xff1a; <mysql高手> <elasticsearch高手> <源码解读> <java核心> <面试攻关> ♝博主的话 &#xff1a; <搬的每块砖&#xff0c;皆为峰峦之基&#xff1b;公众号搜索(码到…

FedAvg的简单实现(详解)

对于联邦学习正在学习中&#xff0c;下文中若有错误出现&#xff0c;望指正 介绍 本文在简单实现联邦平均算法时&#xff0c;使用客户-服务器架构&#xff0c;其基本流程是&#xff1a; 1、server初始化模型参数&#xff0c;所有clients将这个初始模型下载到本地 2、clien…

RK3568笔记三十六:LED驱动开发(设备树)

若该文为原创文章&#xff0c;转载请注明原文出处。 记录使用设备树编写一个简单的 LED 灯驱动程序 一、编程思路 程序编写的主要内容为添加 LED 灯的设备树节点、在驱动程序中使用 of 函数获取设备节点中的 属性&#xff0c;编写测试应用程序。 • 首先向设备树添加 LED 设备…

Python基础语法篇(下)+ 数据可视化

Python基础语法&#xff08;下&#xff09; 数据可视化 一、函数&#xff08;一&#xff09;函数的定义&#xff08;二&#xff09;函数的调用和传参 二、文件操作&#xff08;一&#xff09;文件读取和写入&#xff08;二&#xff09;文件对象及方法&#xff08;三&#xff09…

探寻大模型回答9.9和9.11犯错的根本原因

大家好,我是herosunly。985院校硕士毕业,现担任算法研究员一职,热衷于机器学习算法研究与应用。曾获得阿里云天池比赛第一名,CCF比赛第二名,科大讯飞比赛第三名。拥有多项发明专利。对机器学习和深度学习拥有自己独到的见解。曾经辅导过若干个非计算机专业的学生进入到算法…

WinOS下获取dll中的方法列表

开发的Windows应用程序的安装环境从Win11 23H2切换到24H2时&#xff0c;出现获取电源模式不正确的问题&#xff0c;通过debug代码发现获取电源模式的方法是走的方法编号。由于Win11 24H2中增加了对外提供的方法&#xff0c;而增加的方法放在方法列表中间&#xff0c;导致其后面…

生成式AI的未来:对话的艺术与代理的实践

生成式 AI 的发展方向&#xff0c;是 Chat 还是 Agent&#xff1f; 随着生成式AI技术的不断进步&#xff0c;关于其未来发展方向的讨论也愈发激烈。究竟生成式AI的未来是在对话系统&#xff08;Chat&#xff09;中展现智慧&#xff0c;还是在自主代理&#xff08;Agent&#x…