Qt_事件的介绍

目录

1、理解事件

2、处理事件QEvent 

3、键盘事件QKeyEvent 

4、鼠标事件QMouseEvent 

4.1 鼠标点击事件

4.2 鼠标释放事件 

4.3 鼠标移动事件 

5、滚轮事件QWheelEvent 

6、定时器事件QTimerEvent 

7、窗口事件QMoveEvent

8、事件分发器event 

9、事件过滤器eventfilter

结语 


前言:

        在Qt中,事件指的是由程序内部或者外部设备产生的事情或动作,比如当程序中定时器的时间到了,就会发出一个事件,这就是内部产生的事件,当鼠标、键盘进行某些操作所产生的动作,这就是外部产生的事件。在Qt中事件被抽象成一个类,所有的事件都继承自QEvent类,如下图:

1、理解事件

        Qt中有一个重要的概念叫做信号与槽,指的是:用户的特定操作会产生信号,将该信号与槽函数连接,就能实现产生信号的同时自动去调用槽函数了。事件的用法和信号是一样的,当用户进行某些操作就会产生事件,给事件关联一个处理函数,这样事件产生的同时就会去调用该处理函数。实际上,由于事件使用起来较麻烦,因此Qt将事件做了一层封装就成了信号与槽,即信号与槽就是对于事件的进一步封装,事件是信号槽的底层机制。

2、处理事件QEvent 

         当产生了一个事件后,要做的就是对该事件进行处理。因为在Qt中事件被分成多个类型,比如鼠标事件、键盘事件、窗口事件,就拿鼠标事件来说,当点击鼠标或者移动鼠标,其实就已经产生了事件,而Qt系统默认产生事件后会自动调用相关的处理函数,只不过不同的事件会调用不同的处理函数,比如点击鼠标调用的是mousePressEvent()函数,而移动鼠标调用的是mouseMoveEvent()函数。最重要的是这些处理函数都是虚函数,开发者可以重写这些虚函数,就可以在事件产生时实现不同的效果了


        举一个例子,首先创建一个标签(QLabel),当鼠标进入标签和离开标签时会触发QEvent事件,为了验证这个过程,在触发事件时会打印提示信息。实现步骤如下:1、首先新建一个QWidget项目,2、由于要自定义标签的事件的处理函数,而所以需要手动自定义一个类,让该类继承标签,目的是可以手动操作继承类的构造函数,并且该继承类还拥有标签的功能,如下图:

        然后给自定义类起一个名字,并且选择要继承的类:

         创建完成后就会自动生成对应的文件和代码:

        3、至此,就可以在mylabel类中重写鼠标进入标签和( enterEvent)离开标签(leaveEvent)的虚函数了,然后就可以实现鼠标进入和离开标签时执行我们自定义的功能,上述虚函数介绍如下:

        因为触发的是QEvent事件,所以该虚函数的形参类型是QEvent*。重写虚函数的mylabel.cpp代码如下:

#include "mylabel.h"
#include <QDebug>mylabel::mylabel(QWidget* parent):QLabel(parent)
{}void mylabel::enterEvent(QEvent *event)
{qDebug()<<"enterEvent";
}void mylabel::leaveEvent(QEvent *event)
{qDebug()<<"leaveEvent";
}

        现在自定义类的工作已经完成了,当然还需要在widget.cpp中将这个类创建出来,这样才能在最终的界面上显示标签,widget.cpp代码如下:

#include "widget.h"
#include "ui_widget.h"
#include "mylabel.h"Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget)
{ui->setupUi(this);mylabel* m = new mylabel(this);m->setText("这一个自定义的标签");
}Widget::~Widget()
{delete ui;
}

        运行结果:

        将鼠标进入和移除标签后就会打印以上信息。 

3、键盘事件QKeyEvent 

        Qt中的按键事件被抽象看成QKeyEvent类,当键盘上的按键被按下或松开时,键盘事件便会产生。该事件的处理函数主要是让系统知道用户按下了什么键,以便进行后续的工作。当键盘按下按键后产生的事件会自动调用keyPressEvent函数,因此只需要重写该函数就能自定义按键事件的处理动作了。


        1、测试单个按键,Qt中内置了一个枚举类型key,该类型中枚举了大部分键盘按键,并且QKeyEvent中提供了key方法,调用此方法就能让系统知道我们按下的具体是哪个按键了,测试代码如下(节选widget.cpp文件,.h文件的声明代码就不展示了):

#include "widget.h"
#include "ui_widget.h"
#include <QDebug>
#include <QKeyEvent>Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget)
{ui->setupUi(this);
}Widget::~Widget()
{delete ui;
}void Widget::keyPressEvent(QKeyEvent* e)
{if(e->key()==Qt::Key_A){qDebug()<<"按下的按键是A";}}

        运行结果:

         只有当鼠标点击界面后(即让界面获取到焦点),并按下按键A才会触发打印信息。


        2、测试组合按键,同样的,Qt中也内置了一个枚举类型KeyboardModifier,该类型中枚举了组合键中的修改键(组合键=修改键+普通按键),常用的枚举量有如下几种:

Qt::NoModifier
⽆修改键
Qt::ShiftModifier
Shift键
Qt::ControlModifier
Ctrl 键
Qt::AltModifier
Alt 键

        并且QKeyEvent中提供了modifiers方法,调用该方式就能识别按键是否为修改键了,代码测试如下: 

#include "widget.h"
#include "ui_widget.h"
#include <QDebug>
#include <QKeyEvent>Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget)
{ui->setupUi(this);
}Widget::~Widget()
{delete ui;
}void Widget::keyPressEvent(QKeyEvent* e)
{if(e->modifiers()==Qt::ControlModifier&&e->key()==Qt::Key_A){qDebug()<<"按下的按键是A";}}

        运行结果:

4、鼠标事件QMouseEvent 

         在Qt中,鼠标事件被抽象成QMouseEvent类,当在控件中按下⿏标或者移动⿏标时,都会以该控件为基础产生鼠标事件。

4.1 鼠标点击事件

         ⿏标点击产生的事件对应虚函数mousePressEvent。mousePressEvent函数原型如下:

[virtual] void QWidget::mousePressEvent(QMouseEvent *event)

        传统的鼠标有三个按键,分别是:鼠标左键、鼠标右键、鼠标滚轮键。这三个键都可以触发mousePressEvent,Qt中也为这三个键提供了对应的枚举常量,分别是:Qt::LeftButton 鼠标左键 、Qt::RightButton 鼠标右键 、Qt::MidButton 鼠标滚轮并且QMouseEvent也提供了一个方法button,调用该方法就能让系统知道我们按下的具体是哪个键了。


        测试代码如下:

#include "widget.h"
#include "ui_widget.h"
#include <QMouseEvent>
#include <QDebug>Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget)
{ui->setupUi(this);
}Widget::~Widget()
{delete ui;
}void Widget::mousePressEvent(QMouseEvent *e)
{if(e->button()==Qt::LeftButton){qDebug()<<"鼠标左键进行点击";}
}

        运行结果:

         鼠标右键点击和滚轮点击就在上述虚函数中添加条件判断即可。


        上述的点击事件为单击,除了单击触发可以事件,双击触发也可以触发事件,双击事件的虚函数为:mouseDoubleClickEvent。mouseDoubleClickEvent() 函数原型如下:

 [virtual] void QWidget::mouseDoubleClickEvent(QMouseEvent *event)

        测试代码如下:

#include "widget.h"
#include "ui_widget.h"
#include <QMouseEvent>
#include <QDebug>Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget)
{ui->setupUi(this);
}Widget::~Widget()
{delete ui;
}void Widget::mouseDoubleClickEvent(QMouseEvent *e)
{if(e->button()==Qt::LeftButton){qDebug()<<"鼠标左键进行点击";}
}

        测试结果就是用鼠标任意一键双击界面后才会调用此函数,并且也可以通过button函数得知双击键的具体键。

4.2 鼠标释放事件 

        ⿏标释放事件是虚函数mouseReleaseEvent来处理的。mouseReleaseEvent函数介绍如下:

[virtual] void QWidget::mouseReleaseEvent(QMouseEvent *event) 

        使用逻辑和上述点击事件是一样的,只不过这里是在鼠标释放后调用,并且可以通过button函数得知释放的鼠标键,测试代码如下:

#include "widget.h"
#include "ui_widget.h"
#include <QMouseEvent>
#include <QDebug>Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget)
{ui->setupUi(this);
}Widget::~Widget()
{delete ui;
}void Widget::mousePressEvent(QMouseEvent *e)
{if(e->button()==Qt::LeftButton){qDebug()<<"鼠标左键进行点击";}
}void Widget::mouseReleaseEvent(QMouseEvent *e)
{if(e->button()==Qt::LeftButton){qDebug()<<"鼠标左键释放";}
}

        运行结果:

4.3 鼠标移动事件 

         鼠标移动事件的处理函数是mouseMoveEvent,表示只要鼠标一做移动操作就会调用该函数,这可能对于大部分人来说难以相信,因为实际生活中使用计算机,鼠标肯定是会大量的进行移动的,因此难以相信一个微不足道的移动鼠标操作背后竟然可以有如此复杂的逻辑,函数介绍如下:

[virtual] void QWidget::mouseMoveEvent(QMouseEvent *event)

        也正是因为使用计算机会造成大量的鼠标移动,若每次移动都调用对应事件处理函数,则会造成卡顿,因此Qt为了保证程序的流畅性,默认情况下不会让鼠标移动产生事件,即鼠标移动的时候不会调用mouseMoveEvent,除非显式告诉Qt就要产生事件,方法是调用函数setMouseTracking,该函数原型如下:

void setMouseTracking(bool enable)
//setMouseTracking函数默认是false,需要设置为true才能让鼠标移动产生事件

        测试代码如下:

#include "widget.h"
#include "ui_widget.h"
#include <QDebug>
#include <QMouseEvent>Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget)
{ui->setupUi(this);setMouseTracking(true);
}Widget::~Widget()
{delete ui;
}void Widget::mouseMoveEvent(QMouseEvent *e)
{//获取鼠标的左边,以窗口左上角为原点qDebug()<<"["<<e->x()<<","<<e->y()<<"]";
}

        运行结果:

5、滚轮事件QWheelEvent 

        注意此处的滚轮事件和鼠标事件中的滚动点击是不一样的,滚轮事件表示用滚轮进行上下滑动时产生的事件,滚轮滑动所产生的事件调用的处理函数是WheelEvent,介绍如下:

[virtual] void QWidget::wheelEvent(QWheelEvent *event)

        滚轮事件着重在于滚轮滑动距离计算,滚轮滑动的距离可以通过delta函数获取。该函数介绍如下:

int QGraphicsSceneWheelEvent::delta() const
//其中返回值代表滚轮滑动的距离
//正数表⽰滚轮相对于⽤⼾向前滑动,负数表⽰滚轮相对于⽤⼾向后滑动。

        测试代码如下:

#include "widget.h"
#include "ui_widget.h"
#include <QWheelEvent>
#include <QDebug>Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget)
{ui->setupUi(this);
}Widget::~Widget()
{delete ui;
}void Widget::wheelEvent(QWheelEvent *e)
{static int count = 0;count+=e->delta();if(e->delta()>0)qDebug()<<"向前滑动"<<count;elseqDebug()<<"向后移动"<<count;
}

        运行结果:

6、定时器事件QTimerEvent 

        在一些需要周期性的执行某项任务的场景下,定时器事件尤为重要,他表示当设定的时间到后,会产生一个QTimerEvent事件,该事件的处理函数是timerEvent(即定时器“响了”就会去调用该函数),该函数介绍如下:

[virtual] void QTimer::timerEvent(QTimerEvent *e)

        因此只要重写timerEvent函数,就能在定时器到时后执行我们期望的工作,前提当然是要启动该定时器,需要通过startTimer函数来开启⼀个定时器,这个函数需要输入⼀个以毫秒为单位的整数作为参数来表明定时器设定的时间,他的返回值表示这个定时器的编号,因为实际运用中有多个定时器,他们都调用同一个函数,为了区分多个定时器对应不同的任务,因此需要用编号的形式来区分他们。startTimer函数介绍如下:

int QObject::startTimer(int interval, 
Qt::TimerType timerType = Qt::CoarseTimer)//interval表示时间

        测试代码如下,需要定义两个定时器,widget.h文件代码如下:

#ifndef WIDGET_H
#define WIDGET_H#include <QWidget>QT_BEGIN_NAMESPACE
namespace Ui { class Widget; }
QT_END_NAMESPACEclass Widget : public QWidget
{Q_OBJECTpublic:Widget(QWidget *parent = nullptr);~Widget();void timerEvent(QTimerEvent *e);private:Ui::Widget *ui;int time1;//定时器1int time2;//定时器2
};
#endif // WIDGET_H

        widget.cpp文件代码如下:

#include "widget.h"
#include "ui_widget.h"
#include <QDebug>Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget)
{ui->setupUi(this);time1 = startTimer(2000);//2秒后定时器1触发time2 = startTimer(1000);//1秒后定时器2触发
}Widget::~Widget()
{delete ui;
}void Widget::timerEvent(QTimerEvent *e)
{if(time1==e->timerId())//定时器1执行的任务{static int t1 = 0;t1++;qDebug()<<"定时器1每两秒加一次"<<t1;}if(time2==e->timerId())//定时器2执行的任务{static int t2 = 0;t2++;qDebug()<<"定时器2每一秒加一次"<<t2;}
}

        运行结果:

7、窗口事件QMoveEvent

        窗口产生的是被抽象成QMoveEvent类,当窗口进行移动或者窗口大小发生改变时都产生窗口事件,这两个操作对应的处理函数分别是:

[virtual] void QWidget::moveEvent(QMoveEvent *event)//窗口移动[virtual] void QWidget::resizeEvent(QResizeEvent *event)//窗口大小发生改变

        测试代码如下:

#include "widget.h"
#include "ui_widget.h"
#include <QDebug>
#include <QMoveEvent>Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget)
{ui->setupUi(this);
}Widget::~Widget()
{delete ui;
}void Widget::moveEvent(QMoveEvent *e)
{qDebug()<<e->pos();
}void Widget::resizeEvent(QResizeEvent *e)
{qDebug()<<e->size();
}

        运行结果:

8、事件分发器event 

        产生事件后之所以可以自动执行处理函数,是因为中间有一个事件分发器,他接收所有的事件,然后对该事件的类型做分析,再去向下传递给具体的事件,再由具体的事件去调用合适的处理函数(事件分发器本身不直接调用处理函数),如下图所示:

        事件分发器实际上就是QObject对象里的event函数,产生的事件都是发送给该函数的,并且可以重写该函数,重写该函数表示由我们实现的事件分发器来接收事件,这样做还能起到一个拦截事件的作用。该函数介绍如下:

bool event(QEvent *e)
//其返回值为布尔类型,若为ture,代表拦截了该事件,不向下分发

        可以看到该函数参数是一个QEvent*的指针,QEvent是所有事件的基类,因此该函数可以接收所有的事件。


        重写event函数,实现一个事件拦截器,比如鼠标点击会产生事件,这个事件按理来说会被事件分发器传给下层,然后调用我们自己实现的处理函数,但是有了事件拦截器后,就会拦截这个点击事件,导致该事件不会被传给下一层也就无法调用我们自己实现的处理函数了,代码如下:

#include "widget.h"
#include "ui_widget.h"
#include <QMouseEvent>
#include <QDebug>
#include <QEvent>Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget)
{ui->setupUi(this);
}Widget::~Widget()
{delete ui;
}void Widget::mousePressEvent(QMouseEvent *e)
{if(e->button()==Qt::LeftButton){qDebug()<<"鼠标左键进行点击";}
}bool Widget::event(QEvent *e)
{if(e->type()==QEvent::MouseButtonPress){qDebug()<<"拦截鼠标点击事件";return true;//返回true表示不会将该事件传给下一层}//转交给下一层return QWidget::event(e);
}

        运行结果:

9、事件过滤器eventfilter

        从上面例子中可以发现一个逻辑,在哪个控件的类里面重写事件处理函数,则只有在该控件中产生事件才会调用该处理函数,比如一个界面widget中有一个label控件,如果在label控件中重写鼠标点击事件的处理函数,那么鼠标只有在该label中点击才会调用该函数,在界面的其他位置点击是不会调用该函数的。这样又会导致一个问题,即上述的event拦截功能只能拦截当前控件的事件,比如在label控件中实现的拦截器只能拦截label控件中产生的事件,不能拦截界面产生的事件,若想拦截多个控件的事件,只能在每个控件的类内都重写event,这样过于麻烦,并且代码内聚性不强。


        针对上面问题,Qt推出事件过滤器来解决,事件过滤器是在事件分发器的上一层,如下图:

        事件过滤器的创建也是重写eventFilter函数,该函数介绍如下:

//其返回值为布尔类型,若为ture,代表拦截了该事件,不向下分发
[virtual] bool eventFilter(QObject *obj, QEvent *event)//obj表示事件是由哪个控件产生的
//event表示事件

        并且事件过滤器的使用逻辑和事件分发器是一样的,简单来说可以把事件过滤器看成是事件分发器之上的”事件分发器“,正因为事件过滤器在”源头“,因此只需要重写一个事件过滤器就可以接收窗口界面上所有控件产生的事件(前提是将事件过滤器安装到这些控件上,具体操作如下文)。


        举个例子,若一个界面widget上有一个控件label,在widget上实现事件过滤器,则可以捕捉到label的事件。首先先自定义一个mylabel继承自QLabel,然后在该类中重写事件处理函数和事件分发器,这时候按理来说触发mylabel的事件则会调用mylabel中重写的事件分发器,但是我们又在widget中重写了一个事件过滤器并安装到mylabel中,就会导致触发mylabel的事件时调用的是widget下的事件过滤器。实现代码如下(节选自widget.cpp):

#include "widget.h"
#include "ui_widget.h"
#include <QDebug>Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget)
{ui->setupUi(this);my = new mylabel(this);my->setText("自定义标签");//给自定义标签安装事件过滤器my->installEventFilter(this);
}Widget::~Widget()
{delete ui;
}bool Widget::eventFilter(QObject *obj, QEvent *e)//widget的事件过滤器
{if(obj == my)//可以接收来自其他控件的事件{if(e->type()==QEvent::MouseButtonPress){qDebug()<<"由widget的事件过滤器捕捉";return true;}}//交给下一层处理return QWidget::eventFilter(obj,e);
}

        运行结果:

        结果是点击标签时,调用的函数是widget中的事件过滤器而不是mylabel中的事件分发器,说明事件过滤器可以拦截其他控件的事件,并且事件过滤器的优先级在事件分发器之上

结语 

        以上就是关于Qt事件的讲解,Qt事件是Qt的一个底层机制,他是信号与槽的基础,虽然信号与槽使用起来很方便,并且可以满足大部分的开发场景,但是在一些特殊的场景下只能使用事件的方式进行界面DIY的操作,并且事件给予了开发者更多的开发空间。

        最后如果本文有遗漏或者有误的地方欢迎大家在评论区补充,谢谢大家!! 

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

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

相关文章

峟思助力堤防工程安全:构建多功能防洪屏障

堤防工程&#xff0c;作为水利建设中至关重要的防护体系&#xff0c;不仅守护着江河、湖泊及滨海区域的安全&#xff0c;更是确保人民生命财产安全的坚固防线。在现代社会&#xff0c;随着技术的进步与安全意识的提升&#xff0c;堤防工程不仅限于传统的防洪功能&#xff0c;更…

CVPR最牛图像评价算法!

本文所涉及所有资源均在 传知代码平台可获取。 目录 概述 一、论文思路 1.多任务学习框架&#xff1a; 2.视觉-语言对应关系&#xff1a; 3.动态损失权重&#xff1a; 4.模型优化和评估&#xff1a; 二、模型介绍 三、详细实现方法 1.图像编码器和语言编码器&#xff08;Image…

Solidity语言:重点学习Solidity编程语言,这是EVM上最常用的智能合约语言。

Solidity是一种面向合约的编程语言,用于在以太坊虚拟机(EVM)上编写智能合约。它是Solidity开发者在以太坊平台上创建智能合约的主要选择之一。 学习Solidity的重点包括以下几方面: 语法和数据类型:学习Solidity的基本语法、数据类型、变量声明和函数定义等。 智能合约:了…

刷完这个笔记,17K不能再少了....

大家好&#xff0c;最近有不少小伙伴在后台留言&#xff0c;得准备面试了&#xff0c;又不知道从何下手&#xff01;为了帮大家节约时间&#xff0c;特意准备了一份面试相关的资料&#xff0c;内容非常的全面&#xff0c;真的可以好好补一补&#xff0c;希望大家在都能拿到理想…

cobaltstrike之execute-assembly内存加载—后渗透利用

通过execute-assembly内存加载来执行文件&#xff0c;从而避免后渗透中被杀毒软件静态报毒&#xff0c;使更多的工具能够继续利用&#xff0c;常见的方式有权限维持&#xff0c;代理上线等操作 远程bin文件加载 首先尝试远程加载bin文件 使用项目https://github.com/shanekha…

IO 多路转接之 epoll

文章目录 IO 多路转接之 epoll1、IO 多路转接之 poll1.1、poll 函数1.2、poll 函数返回值1.3、Socket 就绪条件1.3.1、读就绪1.3.2、写就绪1.3.3、异常就绪 1.4、poll 的优点1.5、poll 的缺点1.6、poll 改写 select 2、IO 多路转接之 epoll2.1、epoll 函数2.2、epoll_create2.3…

视频字幕生成:分享6款专业易操作的工具,让创作更简单!

​视频字幕如何添加&#xff1f;日常剪辑Vlog视频时&#xff0c;就需要给视频添加上字幕了。字幕是一个比较重要的元素&#xff0c;它不仅可以帮助听力受损或语言障碍的人士理解内容&#xff0c;还可以让你的视频更加易于理解和吸引观众。 那么如何实现视频字幕生成&#xff0c…

Linux 进程与进程状态

目录 1.进程。 1.进程的概念 2.并行和并发 3.并行和并发的区别&#xff1a; 4.PCB&#xff08;程序控制块&#xff09; 5.进程组与会话。 6.进程状态。 1.进程。 1.进程的概念 进程是操作系统进行资源分配和调度的一个独立单位。每个进程都运行在操作系统的控制之下&…

8.进销存系统(基于springboot的进销存系统)

目录 1.系统的受众说明 2.开发技术与环境配置 2.1 SpringBoot框架 2.2 Java语言简介 2.3 MySQL环境配置 2.4 idea介绍 2.5 mysql数据库介绍 2.6 B/S架构 3.系统分析与设计 3.1 可行性分析 3.1.1 技术可行性 3.1.2 操作可行性 3.1.3经济可行性 3.4.1 数据库…

一些做题中总结的零散的python函数的简单运用

输出保留两位数的小数 将16进制&#xff08;可修改&#xff09;的数进制转换成十进制并输出 大小写转化&#xff0c;第一个是搞成全部大写的&#xff0c;第二个高成全部小写的&#xff0c;最后一个是搞成第一个是大写的其他全部是小写的 将这个n的两边空格去掉 使用print(n,end…

叶国富学得会胖东来吗?

“大家都看不懂就对了&#xff0c;如果都看得懂我就没有机会了。”昨晚&#xff0c;实体零售迎来一则重磅消息&#xff0c;名创优品获得了全国第二大连锁超市永辉超市的大股东身份。在资本市场负反馈的压力下&#xff0c;名创优品创始人叶国富有了上述回应。 消息公布后&#x…

2.以太网

局域网 局域网: Local Area Networks (LAN) 网络大小分类 局域网园区网(可以理解为企业网)城域网 广域网是一个网络连接的技术&#xff0c;并非多大范围的网络 网关 为局域网内的用户提供了一扇门&#xff0c;通过网关可以访问到别的网络。这个门&#xff0c;就叫网关 以…

解决你的IDE在使用的时候测试单元@Test在创建Scanner对象是键盘键入不了的问题;

插播一条快讯&#xff0c;我在我的ide中新创建 了project后发现我的测试单元不好使了&#xff0c;即 import org.junit.Test; 这个包在创建Scanner对象接受键盘时&#xff0c;控制台输入时没有任何反应&#xff0c;键入不了了&#xff0c;我的问题出现原因可能是我导入了JDBC…

BitSet-解决数据压缩问题

一、问题引入 假设QQ音乐服务器上有9000万首音乐&#xff0c;用户按照歌名来搜索歌曲&#xff0c;如何使得满足这一需求所需的数据占用的内存空间最小以及用户搜索歌曲速度更快 二、分析问题 1、为了满足使得数据占用的内存更小&#xff0c;可以采用映射的思路&#xff0c;按…

【2024W35】肖恩技术周刊(第 13 期):肉,好次!

周刊内容: 对一周内阅读的资讯或技术内容精品&#xff08;个人向&#xff09;进行总结&#xff0c;分类大致包含“业界资讯”、“技术博客”、“开源项目”和“工具分享”等。为减少阅读负担提高记忆留存率&#xff0c;每类下内容数一般不超过3条。 更新时间: 星期天 历史收录:…

【C++算法】链表

知识总结 常用技术&#xff1a; 1.画图&#xff01;&#xff01;——>直观形象便于理解 2.引入虚拟”头结点“ 便于处理边界情况方便对链表操作 3.不要吝啬空间&#xff0c;大胆定义变量 4.快慢双指针——判环、找链表中环的入口、找链表中倒数第n个节点 链表中的常用…

移动数组中数字的方法(c语言)

1.移动一维数组中的内容;若数组中有n个整数&#xff0c;要求把下标从0到p(含p&#xff0c;p小于等于n-1&#xff09;的数组元素平移到数组的最后。 例如&#xff0c;一维数组中的原始内容为:1,2,3,4,5,6,7,8,9,10;p的值为3。 移动后&#xff0c;一维数组中的内容应为:5,6,7,8…

融会贯通记单词,绝对丝滑,一天轻松记几百

如果我将flower(花&#xff09;、flat(公寓)、floor(地板)、plane(飞机)几个单词放在一起&#xff0c;你会怎么来记忆这样的一些单词呢&#xff1f; 我们会发现&#xff0c;我们首先可以将plane去掉&#xff0c;因为它看上去几乎就是一个异类。这样&#xff0c;我们首先就可以将…

力扣958:判断二叉树是否为完全二叉树

给你一棵二叉树的根节点 root &#xff0c;请你判断这棵树是否是一棵 完全二叉树 。 在一棵 完全二叉树 中&#xff0c;除了最后一层外&#xff0c;所有层都被完全填满&#xff0c;并且最后一层中的所有节点都尽可能靠左。最后一层&#xff08;第 h 层&#xff09;中可以包含 …