Qt 综合练习小项目--反金币(2/2)

目录

4 选择关卡场景

4.2 背景设置

4.3 创建返回按钮

4.3 返回按钮

4.4 创建选择关卡按钮

4.5 创建翻金币场景

5 翻金币场景

5.1 场景基本设置

5.2 背景设置

5.3 返回按钮

5.4 显示当前关卡 

5.5 创建金币背景图片

5.6 创建金币类

5.6.1 创建金币类 MyCoin

5.6.2 构造函数

5.6.3 测试

5.7 引入关卡数据

5.7.1 添加现有文件dataConfig

        5.7.2 添加现有文件

5.7.3 完成添加

5.7.4 数据分析 

5.7.5 测试关卡数据

5.8 初始化各个关卡 

5.9 翻金币特效

5.9.1 MyCoin类扩展属性和行为

5.9.2 创建特效

5.9.3 测试

5.9.3 禁用按钮

5.10 翻周围金币

5.11 判断是否胜利

5.12 胜利图片显示

5.13 胜利后禁用按钮

6 音效添加

6.1 开始音效

6.2 选择关卡音效

6.3 返回按钮音效

6.4 翻金币与胜利音效

7 优化项目


4 选择关卡场景

        4.1 场景基本设置

        选择关卡构造函数如下:

 //设置窗口固定大小this->setFixedSize(320,588);//设置图标this->setWindowIcon(QPixmap(":/res/Coin0001.png"));//设置标题this->setWindowTitle("选择关卡");//创建菜单栏QMenuBar * bar = this->menuBar();this->setMenuBar(bar);//创建开始菜单QMenu * startMenu = bar->addMenu("开始");//创建按钮菜单项QAction * quitAction = startMenu->addAction("退出");//点击退出 退出游戏connect(quitAction,&QAction::triggered,[=](){this->close();});

运行效果如图:

4.2 背景设置

void ChooseLevelScene::paintEvent(QPaintEvent *)
{QPainter painter(this);QPixmap pix;pix.load(":/res/OtherSceneBg.png");painter.drawPixmap(0,0,this->width(),this->height(),pix);//加载标题pix.load(":/res/Title.png");painter.drawPixmap( (this->width() - pix.width())*0.5,30,pix.width(),pix.height(),pix);
}

4.3 创建返回按钮

 //返回按钮MyPushButton * closeBtn = new MyPushButton(":/res/BackButton.png",":/res/BackButtonSelected.png");closeBtn->setParent(this);closeBtn->move(this->width()-closeBtn->width(),this->height()-closeBtn->height());

返回按钮是有正常显示图片和点击后显示图片的两种模式,所以我们需要重写MyPushButton中的 MousePressEvent和MouseReleaseEvent。

//鼠标事件
void MyPushButton::mousePressEvent(QMouseEvent *e)
{if(pressedImgPath != "") //选中路径不为空,显示选中图片{QPixmap pixmap;bool ret = pixmap.load(pressedImgPath);if(!ret){qDebug() << pressedImgPath << "加载图片失败!";}this->setFixedSize( pixmap.width(), pixmap.height() );this->setStyleSheet("QPushButton{border:0px;}");this->setIcon(pixmap);this->setIconSize(QSize(pixmap.width(),pixmap.height()));}//交给父类执行按下事件return QPushButton::mousePressEvent(e);
}
void MyPushButton::mouseReleaseEvent(QMouseEvent *e)
{if(normalImgPath != "") //选中路径不为空,显示选中图片{QPixmap pixmap;bool ret = pixmap.load(normalImgPath);if(!ret){qDebug() << normalImgPath << "加载图片失败!";}this->setFixedSize( pixmap.width(), pixmap.height() );this->setStyleSheet("QPushButton{border:0px;}");this->setIcon(pixmap);this->setIconSize(QSize(pixmap.width(),pixmap.height()));}//交给父类执行 释放事件return QPushButton::mouseReleaseEvent(e);
}

4.3 返回按钮

在这里我们点击返回后,延时0.5后隐藏自身,并且发送自定义信号,告诉外界自身已经选择了返回按钮。

//返回按钮功能实现connect(closeBtn,&MyPushButton::clicked,[=](){QTimer::singleShot(500, this,[=](){this->hide();//触发自定义信号,关闭自身,该信号写到 signals下做声明emit this->chooseSceneBack();});});

在主场景MainScene中 点击开始按钮显示选择关卡的同时,监听选择关卡的返回按钮消息

//监听选择场景的返回按钮    connect(chooseScene,&ChooseLevelScene::chooseSceneBack,[=](){this->show();});

测试主场景与选择关卡场景的切换功能。

4.4 创建选择关卡按钮

//创建关卡按钮for(int i = 0 ; i < 20;i++){MyPushButton * menuBtn = new MyPushButton(":/res/LevelIcon.png");menuBtn->setParent(this);menuBtn->move(25 + (i%4)*70 , 130+ (i/4)*70);//按钮上显示的文字QLabel * label = new QLabel;label->setParent(this);label->setFixedSize(menuBtn->width(),menuBtn->height());label->setText(QString::number(i+1));label->setAlignment(Qt::AlignHCenter | Qt::AlignVCenter); //设置居中label->move(25 + (i%4)*70 , 130+ (i/4)*70);label->setAttribute(Qt::WA_TransparentForMouseEvents,true);  //鼠标事件穿透}

运行效果如果:

4.5 创建翻金币场景

点击关卡按钮后,会进入游戏的核心场景,也就是翻金币的场景,首先先创建出该场景的.h和.cpp文件

创建PlayScene

点击选择关卡按钮后会跳入到该场景

建立点击按钮,跳转场景的信号槽连接

ChooseLevelScene.h 中声明

PlayScene *pScene = NULL;

 

//监听选择关卡按钮的信号槽connect(menuBtn,&MyPushButton::clicked,[=](){// qDebug() << "select: " << i;if(pScene == NULL)  //游戏场景最好不用复用,直接移除掉创建新的场景{this->hide();pScene = new PlayScene(i+1); //将选择的关卡号 传入给PlayerScenepScene->show();}});

这里pScene = new PlayScene(i+1); 将用户所选的关卡号发送给pScene,也就是翻金币场景,当然PlayScene 要提供重载的有参构造版本,来接受这个参数。

5 翻金币场景

5.1 场景基本设置

PlayScene.h中 声明成员变量,用于记录当前用户选择的关卡。

//成员变量 记录关卡索引int levalIndex;

PlayScene.cpp中 初始化该场景配置。

PlayScene::PlayScene(int index)
{//qDebug() << "当前关卡为"<< index;this->levalIndex = index;//设置窗口固定大小this->setFixedSize(320,588);//设置图标this->setWindowIcon(QPixmap(":/res/Coin0001.png"));//设置标题this->setWindowTitle("翻金币");//创建菜单栏QMenuBar * bar = this->menuBar();this->setMenuBar(bar);//创建开始菜单QMenu * startMenu = bar->addMenu("开始");//创建按钮菜单项QAction * quitAction = startMenu->addAction("退出");//点击退出 退出游戏connect(quitAction,&QAction::triggered,[=](){this->close();});
}

5.2 背景设置

void PlayScene::paintEvent(QPaintEvent *)
{//加载背景QPainter painter(this);QPixmap pix;pix.load(":/res/PlayLevelSceneBg.png");painter.drawPixmap(0,0,this->width(),this->height(),pix);//加载标题pix.load(":/res/Title.png");pix = pix.scaled(pix.width()*0.5,pix.height()*0.5);painter.drawPixmap( 10,30,pix.width(),pix.height(),pix);
}

5.3 返回按钮

 //返回按钮MyPushButton * closeBtn = new MyPushButton(":/res/BackButton.png",":/res/BackButtonSelected.png");closeBtn->setParent(this);closeBtn->move(this->width()-closeBtn->width(),this->height()-closeBtn->height());//返回按钮功能实现connect(closeBtn,&MyPushButton::clicked,[=](){QTimer::singleShot(500, this,[=](){this->hide();//触发自定义信号,关闭自身,该信号写到 signals下做声明emit this->chooseSceneBack();});});

5.4 在ChooseScene选择关卡场景中,监听PlayScene的返回信号。

connect(pScene,&PlayScene::chooseSceneBack,[=](){this->show();delete pScene;pScene = NULL;});

5.4 显示当前关卡 

//当前关卡标题QLabel * label = new QLabel;label->setParent(this);QFont font;font.setFamily("华文新魏");font.setPointSize(20);label->setFont(font);QString str = QString("Leavel: %1").arg(this->levalIndex);label->setText(str);label->setGeometry(QRect(30, this->height() - 50,120, 50)); //设置大小和位置

假设我们选择了第15关卡,运行效果如果:

5.5 创建金币背景图片

//创建金币的背景图片for(int i = 0 ; i < 4;i++){for(int j = 0 ; j < 4; j++){//绘制背景图片QLabel* label = new QLabel;label->setGeometry(0,0,50,50);label->setPixmap(QPixmap(":/res/BoardNode.png"));label->setParent(this);label->move(57 + i*50,200+j*50);}}

运行效果如图:

5.6 创建金币类

我们知道,金币是本游戏的核心对象,并且在游戏中可以利用二维数组进行维护,拥有支持点击,翻转特效等特殊性,因此不妨将金币单独封装到一个类中,完成金币所需的所有功能。

5.6.1 创建金币类 MyCoin

并修改MyCoin的基类为QPushButton 

5.6.2 构造函数

在资源图片中,我们可以看到,金币翻转的效果原理是多张图片切换而形成的,而以下八张图片中,第一张与最后一张比较特殊,因此我们在给用户看的时候,无非是金币Coin0001或者是银币 Coin0008这两种图。

因此我们在创建一个金币对象时候,应该提供一个参数,代表着传入的是金币资源路径还是银币资源路径,根据路径我们创建不同样式的图案。

在MyCoin.h中声明:

MyCoin(QString butImg); //代表图片路径 

在MyCoin.cpp中进行实现

MyCoin::MyCoin(QString butImg)
{QPixmap pixmap;bool ret = pixmap.load(butImg);if(!ret){qDebug() << butImg << "加载图片失败!";}this->setFixedSize( pixmap.width(), pixmap.height() );this->setStyleSheet("QPushButton{border:0px;}");this->setIcon(pixmap);this->setIconSize(QSize(pixmap.width(),pixmap.height()));}

5.6.3 测试

在翻金币场景 PlayScene中,我们测试下封装的金币类是否可用,可以在创建好的金币背景代码后,添加如下代码:

	//金币对象MyCoin * coin = new MyCoin(":/res/Coin0001.png");coin->setParent(this);coin->move(59 + i*50,204+j*50);

运行效果如图

5.7 引入关卡数据

当然上述的测试只是为了让我们知道提供的对外接口可行,但是每个关卡的初始化界面并非如此,因此需要我们引用一个现有的关卡文件,文件中记录了各个关卡的金币排列清空,也就是二维数组的数值。

5.7.1 添加现有文件dataConfig

首先先将dataConfig.h 和 dataConfig.cpp文件放入到当前项目下:

        5.7.2 添加现有文件

其次在Qt_Creator项目右键,点击添加现有文件。

5.7.3 完成添加

选择当前项目下的文件,并进行添加

5.7.4 数据分析 

我们可以看到,其实dataConfig.h中只有一个数据是对外提供的,如下图

 

在上图中,QMap<int,QVector<QVector<int>>>mData;都记录着每个关卡中的数据。

其中,int代表对应的关卡 ,也就是QMap中的key值,而value值就是对应的二维数组,我们利用的是 QVector<QVector<int>>来记录着其中的二维组。

5.7.5 测试关卡数据

在Main函数可以测试第一关的数据,添加如下代码:

dataConfig config;for(int i = 0 ; i < 4;i++){for(int j = 0 ; j < 4; j++){//打印第一关所有信息qDebug() << config.mData[1][i][j];}qDebug()<< "";}

输出结果如下图:

对应着dataConfig.cpp中第一关数据来看,与之匹配成功,以后我们就可以用dataConfig中的数据来对关卡进行初始化了。

5.8 初始化各个关卡 

首先,可以在playScene中声明一个成员变量,用户记录当前关卡的二维数组。

int gameArray[4][4]; //二维数组数据

之后,在.cpp文件中,初始化这个二维数组

//初始化二维数组dataConfig config;for(int i = 0 ; i < 4;i++){for(int j = 0 ; j < 4; j++){gameArray[i][j] = config.mData[this->levalIndex][i][j];}}

初始化成功后,在金币类 也就是MyCoin类中,扩展属性 posX,posY,以及flag

这三个属性分别代表了,该金币在二维数组中 x的坐标,y的坐标,以及当前的正反标志。

int posX; //x坐标int posY; //y坐标bool flag; //正反标志

然后完成金币初始化,代码如下:

//金币对象QString img;if(gameArray[i][j] == 1){img = ":/res/Coin0001.png";}else{img = ":/res/Coin0008.png";}MyCoin * coin = new MyCoin(img);coin->setParent(this);coin->move(59 + i*50,204+j*50);coin->posX = i; //记录x坐标coin->posY = j; //记录y坐标coin->flag =gameArray[i][j]; //记录正反标志

运行测试各个关卡初始化,例如第一关效果如图:

5.9 翻金币特效

5.9.1 MyCoin类扩展属性和行为

关卡的初始化完成后,下面就应该点击金币,进行翻转的效果了,那么首先我们先在MyCoin类中创建出该方法。

在MyCoin.h中声明:

void changeFlag();//改变标志,执行翻转效果QTimer *timer1; //正面翻反面 定时器QTimer *timer2; //反面翻正面 定时器int min = 1; //最小图片int max = 8; //最大图片

MyCoin.cpp中做实现

void MyCoin::changeFlag()
{if(this->flag) //如果是正面,执行下列代码{timer1->start(30);this->flag = false;}else //反面执行下列代码{timer2->start(30);this->flag = true;}
}

当然在构造函数中,记得创建出两个定时器。

//初始化定时器timer1 = new QTimer(this);timer2 = new QTimer(this);

5.9.2 创建特效

当我们分别启动两个定时器时,需要在构造函数中做监听操作,并且做出响应,翻转金币,然后再结束定时器。

构造函数中 进行下列监听代码:

  //监听正面翻转的信号槽connect(timer1,&QTimer::timeout,[=](){QPixmap pixmap;QString str = QString(":/res/Coin000%1.png").arg(this->min++);pixmap.load(str);this->setFixedSize(pixmap.width(),pixmap.height() );this->setStyleSheet("QPushButton{border:0px;}");this->setIcon(pixmap);this->setIconSize(QSize(pixmap.width(),pixmap.height()));if(this->min > this->max) //如果大于最大值,重置最小值,并停止定时器{this->min = 1;timer1->stop();}});connect(timer2,&QTimer::timeout,[=](){QPixmap pixmap;QString str = QString(":/res/Coin000%1.png").arg((this->max)-- );pixmap.load(str);this->setFixedSize(pixmap.width(),pixmap.height() );this->setStyleSheet("QPushButton{border:0px;}");this->setIcon(pixmap);this->setIconSize(QSize(pixmap.width(),pixmap.height()));if(this->max < this->min) //如果小于最小值,重置最大值,并停止定时器{this->max = 8;timer2->stop();}});

5.9.3 测试

监听每个按钮的点击效果,并翻转金币。

connect(coin,&MyCoin::clicked,[=](){//qDebug() << "点击的位置: x = " <<  coin->posX << " y = " << coin->posY ;coin->changeFlag();gameArray[i][j] = gameArray[i][j] == 0 ? 1 : 0; //数组内部记录的标志同步修改});

5.9.3 禁用按钮

此时,确实已经可以执行翻转金币代码了,但是如果快速点击,会在金币还没有执行一个完整动作之后 ,又继续开始新的动画,我们应该在金币做动画期间,禁止再次点击,并在完成动画后,开启点击。

在MyCoin类中加入一个标志 isAnimation 代表是否正在做翻转动画。

bool isAnimation  = false; //做翻转动画的标志

在MyCoin做动画期间加入

this->isAnimation  = true;

也就是changeFlag函数中将标志设为true

加入位置如下:

并且在做完动画时,将标志改为false

 

重写按钮的按下事件,判断如果正在执行动画,那么直接return掉,不要执行后续代码。

代码如下:

void MyCoin::mousePressEvent(QMouseEvent *e)
{if(this->isAnimation ){return;}else{return QPushButton::mousePressEvent(e);}
}

5.10 翻周围金币

将用户点击的周围 上下左右4个金币也进行延时翻转,代码写到监听点击金币下。

此时我们发现还需要记录住每个按钮的内容,所以我们将所有金币按钮也放到一个二维数组中,在.h中声明

MyCoin * coinBtn[4][4]; //金币按钮数组

并且记录每个按钮的位置

coinBtn[i][j] = coin;

延时翻动其他周围金币 。

QTimer::singleShot(300, this,[=](){if(coin->posX+1 <=3){coinBtn[coin->posX+1][coin->posY]->changeFlag();gameArray[coin->posX+1][coin->posY] = gameArray[coin->posX+1][coin->posY]== 0 ? 1 : 0;}if(coin->posX-1>=0){coinBtn[coin->posX-1][coin->posY]->changeFlag();gameArray[coin->posX-1][coin->posY] = gameArray[coin->posX-1][coin->posY]== 0 ? 1 : 0;}if(coin->posY+1<=3){coinBtn[coin->posX][coin->posY+1]->changeFlag();gameArray[coin->posX][coin->posY+1] = gameArray[coin->posX+1][coin->posY]== 0 ? 1 : 0;}if(coin->posY-1>=0){coinBtn[coin->posX][coin->posY-1]->changeFlag();gameArray[coin->posX][coin->posY-1] = gameArray[coin->posX+1][coin->posY]== 0 ? 1 : 0;}});

5.11 判断是否胜利

在MyCoin.h中加入 isWin标志,代表是否胜利。

bool isWin = true; //是否胜利

默认设置为true,只要有一个反面的金币,就将该值改为false,视为未成功。

代码写到延时翻金币后 进行判断。

//判断是否胜利this->isWin = true;for(int i = 0 ; i < 4;i++){for(int j = 0 ; j < 4; j++){//qDebug() << coinBtn[i][j]->flag ;if( coinBtn[i][j]->flag == false){this->isWin = false;break;}}}

如果isWin依然是true,代表胜利了!

if(this->isWin){qDebug() << "胜利";}

5.12 胜利图片显示

将胜利的图片提前创建好,如果胜利触发了,将图片弹下来即可

QLabel* winLabel = new QLabel;QPixmap tmpPix;tmpPix.load(":/res/LevelCompletedDialogBg.png");winLabel->setGeometry(0,0,tmpPix.width(),tmpPix.height());winLabel->setPixmap(tmpPix);winLabel->setParent(this);winLabel->move( (this->width() - tmpPix.width())*0.5 , -tmpPix.height());

如果胜利了,将上面的图片移动下来

if(this->isWin){qDebug() << "胜利";QPropertyAnimation * animation1 =  new QPropertyAnimation(winLabel,"geometry");animation1->setDuration(1000);animation1->setStartValue(QRect(winLabel->x(),winLabel->y(),winLabel->width(),winLabel->height()));animation1->setEndValue(QRect(winLabel->x(),winLabel->y()+114,winLabel->width(),winLabel->height()));animation1->setEasingCurve(QEasingCurve::OutBounce);animation1->start();}

5.13 胜利后禁用按钮

当胜利后,应该禁用所有按钮的点击状态,可以在每个按钮中加入标志位 isWin,如果isWin为true,MousePressEvent直接return掉即可MyCoin中.h里添加:
bool isWin = false;//胜利标志在鼠标按下事件中修改为
void MyCoin::mousePressEvent(QMouseEvent *e)
{if(this->isAnimation|| isWin == true ){return;}else{return QPushButton::mousePressEvent(e);}
}
//禁用所有按钮点击事件for(int i = 0 ; i < 4;i++){for(int j = 0 ; j < 4; j++){coinBtn[i][j]->isWin = true;}
}

测试,胜利后不可以点击任何的金币。

6 音效添加

6.1 开始音效

QSound *startSound = new QSound(":/res/TapButtonSound.wav",this);

点击开始按钮,播放音效

startSound->play(); //开始音效

6.2 选择关卡音效

在选择关卡场景中,添加音效。

//选择关卡按钮音效QSound *chooseSound = new QSound(":/res/TapButtonSound.wav",this);

选中关卡后,播放音效

chooseSound->play();

6.3 返回按钮音效

在选择关卡场景与翻金币游戏场景中,分别添加返回按钮音效如下:

        

//返回按钮音效QSound *backSound = new QSound(":/res/BackButtonSound.wav",this);

分别在点击返回按钮后,播放该音效

        

backSound->play();

6.4 翻金币与胜利音效

在PlayScene中添加,翻金币的音效以及 胜利的音效

//翻金币音效QSound *flipSound = new QSound(":/res/ConFlipSound.wav",this);//胜利按钮音效QSound *winSound = new QSound(":/res/LevelWinSound.wav",this);

在翻金币时播放 翻金币音效

flipSound->play();

胜利时,播放胜利音效

winSound->play();

测试音效,使音效正常播放。

7 优化项目

当我们移动场景后,如果进入下一个场景,发现场景还在中心位置,如果想设置场景的位置,需要添加如下下图中的代码:

MainScene中添加:

ChooseScene中添加: 

测试切换三个场景的进入与返回都在同一个位置下,优化成功。

至此,本案例全部制作完成。

 

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

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

相关文章

WPF绑定单变量Binding和绑定多变量MultiBinding 字符串格式化 UI绑定数据,数据变化自动更新UI,UI变化自动更新数据

UI绑定数据&#xff0c;数据变化自动更新UI&#xff0c;UI变化自动更新数据。 支持多设备&#xff0c;同时下载。 绑定单变量 在WPF (Windows Presentation Foundation) 中&#xff0c;您可以使用数据绑定来将变量绑定到界面元素。这允许您在界面上显示变量的值&#xff0c;…

SpringCloud-Bus

接上文 SpringCloud-消息组件 1 注册Bus Bus需要基于一个具体的消息队列实现&#xff0c;比如RabbitMQ.还使用最开始的服务拆分项目&#xff0c;比如现在借阅服务的某个接口调用时&#xff0c;能给用户服务和图书服务发送一个通知。 首先父项目导入SpringCloud依赖 <depend…

【网络安全-信息收集】网络安全之信息收集和信息收集工具讲解

一&#xff0c;域名信息收集 1-1 域名信息查询 可以用一些在线网站进行收集&#xff0c;比如站长之家 域名Whois查询 - 站长之家站长之家-站长工具提供whois查询工具&#xff0c;汉化版的域名whois查询工具。https://whois.chinaz.com/ 可以查看一下有没有有用的信息&#xf…

全志ARM926 Melis2.0系统的开发指引⑥

全志ARM926 Melis2.0系统的开发指引⑥ 编写目的9. 系统启动流程9.1. Shell 部分9.2.Orange 和 desktop 部分9.3. app_root 加载部分9.4. home 加载部分 10. 显示相关知识概述10.1. 总体结构10.2. 显示过程10.3. 显示宽高参数关系 -. 全志相关工具和资源-.1 全志固件镜像修改工具…

2023/10/4 QT实现TCP服务器客户端搭建

服务器端&#xff1a; 头文件 #ifndef WIDGET_H #define WIDGET_H#include <QWidget> #include <QTcpServer> #include <QTcpSocket> #include <QList> #include <QMessageBox> #include <QDebug>QT_BEGIN_NAMESPACE namespace Ui { cla…

C++设计模式-生成器(Builder)

目录 C设计模式-生成器&#xff08;Builder&#xff09; 一、意图 二、适用性 三、结构 四、参与者 五、代码 C设计模式-生成器&#xff08;Builder&#xff09; 一、意图 将一个复杂对象的构建与它的表示分离&#xff0c;使得同样的构建过程可以创建不同的表示。 二、…

《计算机视觉中的多视图几何》笔记(13)

13 Scene planes and homographies 本章主要讲述两个摄像机和一个世界平面之间的射影几何关系。 我们假设空间有一平面 π \pi π&#xff0c;平面上的一点为 x π x_{\pi} xπ​。 x π x_{\pi} xπ​分别在两幅图像 P , P ′ P, P P,P′上形成了 x , x ′ x, x x,x′。 那…

JavaEE 网络原理——TCP的工作机制(中篇 三次握手和四次挥手)

文章目录 一、TCP 内部工作机制——连接管理1. 连接(三次握手)(1).有连接和确认应答之间的关系(2). 通过客户端和服务器详细描述三次握手 2. 断开连接(四次挥手)(1)讨论“四次握手”中间步骤的合并问题。(2) 根据简单的 TCP 代码解释断开连接(3) 四次挥手中的两个重要的 TCP 状…

@ConfigurationProperties配置绑定~

ConfigurationProperties注解是Spring Boot中的一个注解&#xff0c;用于将配置文件中的属性值绑定到Java类中的字段上。 ConfigurationProperties注解的作用包括&#xff1a; 实现配置文件属性和Java类字段的映射&#xff0c;简化了读取配置文件的操作。 可以指定配置文件中…

React项目部署 - Nginx配置

写在前面&#xff1a;博主是一只经过实战开发历练后投身培训事业的“小山猪”&#xff0c;昵称取自动画片《狮子王》中的“彭彭”&#xff0c;总是以乐观、积极的心态对待周边的事物。本人的技术路线从Java全栈工程师一路奔向大数据开发、数据挖掘领域&#xff0c;如今终有小成…

GB28181学习(六)——实时视音频点播(数据传输部分)

GB28181系列文章&#xff1a; 总述&#xff1a;https://blog.csdn.net/www_dong/article/details/132515446 注册与注销&#xff1a;https://blog.csdn.net/www_dong/article/details/132654525 心跳保活&#xff1a;https://blog.csdn.net/www_dong/article/details/132796…

GPT系列论文解读:GPT-2

GPT系列 GPT&#xff08;Generative Pre-trained Transformer&#xff09;是一系列基于Transformer架构的预训练语言模型&#xff0c;由OpenAI开发。以下是GPT系列的主要模型&#xff1a; GPT&#xff1a;GPT-1是于2018年发布的第一个版本&#xff0c;它使用了12个Transformer…

企业微信机器人对接GPT

现在网上大部分微信机器人项目都是基于个人微信实现的&#xff0c;常见的类库都是模拟网页版微信接口。 个人微信作为我们自己日常使用的工具&#xff0c;也用于支付场景&#xff0c;很怕因为违规而被封。这时&#xff0c;可以使用我们的企业微信机器人&#xff0c;利用企业微信…

【数据结构】排序(1) ——插入排序 希尔排序

目录 一. 直接插入排序 基本思想 代码实现 时间和空间复杂度 稳定性 二. 希尔排序 基本思想 代码实现 时间和空间复杂度 稳定性 一. 直接插入排序 基本思想 把待排序的记录按其关键码值的大小依次插入到一个已经排好序的有序序列中&#xff0c;直到所有的记录插入完为止&…

程序三高的方法

程序三高的方法 目录概述需求&#xff1a; 设计思路实现思路分析1.1&#xff09;高并发 参考资料和推荐阅读 Survive by day and develop by night. talk for import biz , show your perfect code,full busy&#xff0c;skip hardness,make a better result,wait for change,c…

全志ARM926 Melis2.0系统的开发指引⑤

全志ARM926 Melis2.0系统的开发指引⑤ 编写目的8. 固件修改工具(ImageModify)使用8.1.界面说明8.2.操作步骤8.2.1. 配置平台8.2.2. 选择固件8.2.3. 选择要替换的文件8.2.4. 替换文件8.2.5. 保存固件 8.3.注意事项8.4.增加固件修改权限设置8.4.1. 概述8.4.2. 操作说明8.4.2.1.打…

竞赛选题 机器视觉目标检测 - opencv 深度学习

文章目录 0 前言2 目标检测概念3 目标分类、定位、检测示例4 传统目标检测5 两类目标检测算法5.1 相关研究5.1.1 选择性搜索5.1.2 OverFeat 5.2 基于区域提名的方法5.2.1 R-CNN5.2.2 SPP-net5.2.3 Fast R-CNN 5.3 端到端的方法YOLOSSD 6 人体检测结果7 最后 0 前言 &#x1f5…

Jetpack生命周期感知组件ViewModel

ViewModel Jetpack ViewModel是Android Jetpack组件库中的一个组件&#xff0c;用于帮助开发者管理UI相关的数据和状态。ViewModel的主要作用是存储和管理与UI相关的数据&#xff0c;以及处理UI的状态变化。 使用ViewModel可以解决以下问题&#xff1a; 避免配置变更&#x…

【JavaScript】相等运算符(== 和 ===)

如果x和y的类型相同&#xff0c;JavaScript会用equals 方法比较这两个值或对象。 没有出现在表格中的情况都会返回 false。&#xff08;表格中的方法都是内部规定的&#xff09; 对于 toNumber 方法&#xff1a; 对于 toPrimitive 方法&#xff1a; // 举个例子&#xff1a; c…

用JMeter对HTTP接口进行压测(一)压测脚本的书写、调试思路

文章目录 安装JMeter和Groovy为什么选择Groovy&#xff1f; 压测需求以及思路准备JMeter脚本以及脚本正确性验证使用Test Script Recorder来获取整条业务线上涉及的接口为什么使用Test Script Recorder&#xff1f; 配置Test Script Recorder对接口进行动态化处理处理全局变量以…