嵌入式代码升级——IAP

目录

 IAP的特点

实现 IAP 功能

STM32 正常的程序运行流程

STM32 加入IAP后的运行流程

 程序执行流程

BootLoader程序

APP1程序

APP2程序

验证操作步骤


        IAP(In-Application Programming)指的是在应用程序运行时对其自身的Flash存储器进行编程的操作。这种技术允许嵌入式设备在不需要外部编程器或调试器的情况下,通过其自身的程序来更新、修改或升级存储在Flash存储器中的代码或数据。

        OTA:空中下载技术--通过联网模块下载程序,更新本地运行的程序。

 IAP的特点

1.传统的嵌入式系统更新通常需要连接外部编程器或使用特定的调试接口,这增加了开发的复杂性和设备部署后的维护难度。而IAP技术使得设备可以通过预留的通信接口(如串口、USB、网口等)接收新的代码或数据,并在运行时写入Flash存储器,从而实现了免拆机壳的升级。

2.对于具备网络通信功能的嵌入式设备,IAP技术还可以通过网络实现远程升级。

3.IAP技术减少了因频繁拆装机壳和连接外部设备而带来的成本和时间消耗,提高了升级效率。

实现 IAP 功能

        想要实现程序的更新操作,需要我们在编写两部分程序代码,第一个项目程序不执行正常的功能操作,而只是通过某种通信方式(如 USB、 USART)接收程序或数据,执行对第二部分代码的更新;第二个项目代码才是真正的功能代码。这两部分程序都烧录在单片机的FLASH中,芯片上电后,第一部分的代码先执行,检测是否对第二部分的代码更新,如果不需要更新则直接运行第二部分的代码;如果需要更新,执行更新的相关操作,再运行第二部分的代码。

        其中第一部分的代码通过ST_Link、JTAG、SWD等方式烧录;第二部分的代码则通过第一部分代码的IAP来烧录进单片机中,或者在首次烧录的时候和第一部分的代码一块烧录,后续需要跟新的时候,再利用IAP进行更新。

        在上面的过程中,将第一个部分代码称之为 Bootloader 程序(引导加载程序),第二部分代码称为 APP 程序,他们存放在 STM32的FLASH 的不同地址范围,一般从最低地址区开始存放 Bootloader,紧跟其后的就是 APP 程序(注意,如果 FLASH 容量足够,是可以设计很多 APP 程序的,我们按最常用的两个 APP 程序的情况来学习IAP)。这样我们就是要实现 3 个程序:Bootloader 和 APP1和APP2。

STM32 正常的程序运行流程

        STM32 的内部闪存(FLASH)地址起始于 0x08000000,一般情况下,程序文件就从此地址开始写入。此外 STM32 是基于 Cortex-M3 内核的微控制器,其内部通过一张“中断向量表”来响应中断,程序启动后,将首先从“中断向量表”取出复位中断向量执行复位中断程序完成启动,而这张“中断向量表”的起始地址是 0x08000004,当中断来临,STM32 的内部硬件机制亦会自动将 PC 指针定位到“中断向量表”处,并根据中断源取出对应的中断向量执行中断服务程序

        STM32 在复位后,先从 0X08000004 地址取出复位中断向量的地址,并跳转到复位中断服务程序,如图标号①所示;在复位中断服务程序执行完之后,会跳转到我们的main 函数,如图标号②所示;而我们的 main 函数一般都是一个死循环,在 main 函数执行过程中,如果收到中断请求(发生重中断),此时 STM32 强制将 PC 指针指回中断向量表处,如图标号③所示;然后,根据中断源进入相应的中断服务程序,如图标号④所示;在执行完中断服务程序以后,程序再次返回 main 函数执行,如图标号⑤所示。(PC指针,用于存储CPU接下来要执行的指令的内存地址。换句话说,它指向了当前指令序列中的下一条指令

STM32 加入IAP后的运行流程

        在加入 IAP 之后程序运行流程图中,可以看到,STM32 在复位后,还是从 0X08000004 地址取出复位中断向量的地址,并跳转到复位中断服务程序,在运行完复位中断服务程序之后跳转到 IAP 的 main 函数,如图标号①所示,在执行完 IAP 以后(即将新的 APP 代码写入 STM32的 FLASH,灰底部分。新程序的复位中断向量起始地址为 0X08000004+N+M),跳转至新写入程序的复位向量表,取出新程序的复位中断向量的地址,并跳转执行新程序的复位中断服务程序,随后跳转至新程序的 main 函数,如图标号②和③所示,同样 main 函数为一个死循环,并且注意到此时 STM32 的 FLASH,在不同位置上,共有两个中断向量表。

        在 main 函数执行过程中,如果 CPU 得到一个中断请求,PC 指针仍强制跳转到地址0X08000004 中断向量表处,而不是新程序的中断向量表,如图标号④所示;程序再根据我们设置的中断向量表偏移量,跳转到对应中断源新的中断服务程序中,如图标号⑤所示;在执行完中断服务程序后,程序返回 main 函数继续运行,如图标号⑥所示。

(IAP程序须满足两个要求:新程序必须在 IAP 程序之后的某个偏移量为 x 的地址开始,必须将新程序的中断向量表相应的移动,移动的偏移量为 x)

STM32的闪存模块由:主存储器、信息块和闪存存储器接口寄存器等 3 部分组成。代码最终都会被编译成二进制文件hex并保存在Flash中。

对FLASH的主存储器进行分区,使用的是STM32F103ZE,共512K的Flash大小,我们将它分成三个区,BootLoader区存放启动代码、App1区存放应用代码、App2区(备份区)存放暂存的升级代码,最好是升级的代码限制在250K以内(不要求)

 程序执行流程

        在程序运行开始时,单片机先执行BootLoader程序,同时检测APP2备份区有没有用于升级的代码(检查标志位,一般设置在此区域的最后四字节)。如果检测到备份区有需要升级的代码,就将APP2部分的代码拷贝到APP1区域中,再去运行APP1区域的代码程序;如果检测到备份区没有相关的代码,就直接去执行APP1的代码;如果APP1也没有可执行的代码,则就只能执行Bootloader区域的代码。

        通过上面的图看到,BootLoader和App1这两个程序的中断向量表位置不一样, 所以跳转到App1区域内首先去更改程序的向量表,然后再去执行其他的应用程序。需要执行升级的代码部分放到APP2内,重启时就可以按上述内容去更新程序了。

BootLoader程序

在编写此部分的代码时,主要的内容就是:读取到APP2备份区的标志位,将APP2的代码写入到APP1中,然后执行APP1。

#include "update.h"
#include "stmflash.h"
#include "stdio.h"

//检查是否有更新标志
//1.有更新标志  将APP2区域的拷贝到APP1区域,并且跳转到APP1区域执行
//2.没有更新标志  执行原有APP1区域的代码
//3.APP1和APP2区域都没有代码   不跳转,执行bootloader
void Check_UPdate_Flag(void)
{
	uint16_t App2FlagBuff[2] = {0};
//1.读APP2区域存放的标志位,如果有0xAAAA,表示APP2中有待更新的程序
	STMFLASH_Read(APP2_FLAG_ADDR, App2FlagBuff, 2);
	if(App2FlagBuff[0] == 0xAAAA && App2FlagBuff[1] == 0xAAAA) {
		//APP2区域有更新程序
		printf("有更新程序,正在执行代码升级\r\n");
		UpdateFun();   //2.将程序从APP2搬运到APP1
	}
	else {
		//APP2区域没有新的程序
		printf("没有新的程序,执行原有APP\r\n");
		UserFlashAppRun(); //3.没有新的APP2程序,执行原有的APP1
	}	
}


//擦除APP1区域,方便接收新的代码  FLASH必须先擦除才能写入
void Erase_APP1(void)
{
	STMFLASH_Erase(FLASH_APP1_ADDR, APP_MAX_SIZE);
	printf("APP1备份区域擦除成功\r\n");
}

//擦除APP2区域,方便接收新的代码  FLASH必须先擦除才能写入
void Erase_APP2(void)
{
	STMFLASH_Erase(FLASH_APP2_ADDR, APP_MAX_SIZE);
	printf("APP2备份区域擦除成功\r\n");
}

//固件更新函数  将APP2区域的代码搬运到APP1区域  每次搬运2K
uint16_t ReadBuff[STM_SECTOR_SIZE/2] = {0};  
void UpdateFun(void)
{
//	uint16_t App2FlagBuff[2] = {0xFFFF, 0xFFFF};  
	printf("开始更新固件...\r\n");
	Erase_APP1();   //擦除APP1区域250K
	for(uint16_t i=0; i<APP_SIZE; i++) {   //每次读2K
		printf("正在更新固件%d...\r\n", i);
		STMFLASH_Read(FLASH_APP2_ADDR+i*STM_SECTOR_SIZE, ReadBuff, STM_SECTOR_SIZE/2);   //从APP2的起始地址开始读2K
		STMFLASH_Write_NoCheck(FLASH_APP1_ADDR+i*STM_SECTOR_SIZE, ReadBuff, STM_SECTOR_SIZE/2);	 //将读到的数据写到APP1的区域
	}
	printf("固件更新完成!\r\n");	
//	STMFLASH_Write_NoCheck(APP2_FLAG_ADDR, App2FlagBuff, 2);  //清除APP2区域的标志位
	Erase_APP2();   //擦除APP2区域
	UserFlashAppRun();   //跳转到APP1区域去执行
}

//跳转到Flash中用户代码执行
void UserFlashAppRun(void)
{
	printf("开始执行FLASH用户代码!!\r\n");  
								//0x08003000     0x08003004
	if(((*(vu32*)(FLASH_APP1_ADDR+4))&0xFF000000)==0x08000000)//判断是否为0X08XXXXXX.  如果不是表示地址不合法
	{	 
		printf("成功跳转APP1区域执行\r\n");
		IAP_Load_App(FLASH_APP1_ADDR);//执行FLASH APP1代码
	}else 
	{
		printf("APP程序加载失败!\r\n");   
	}									 
}

pFunction Jump_To_Application;
//跳转到应用程序段
//appxaddr:用户代码起始地址.
void IAP_Load_App(u32 AppxAddr)
{
	if(((*(__IO uint32_t*)AppxAddr)&0x2FFE0000)==0x20000000)	//检查栈顶地址是否合法.
	{ 
		Jump_To_Application=(pFunction)*(uint32_t*)(AppxAddr+4);		//用户代码区第二个字为程序开始地址(复位地址)		
		__set_MSP(*(__IO uint32_t*)AppxAddr);					//初始化APP堆栈指针(用户代码区的第一个字用于存放栈顶地址)   在core_cm3.c  28行
		Jump_To_Application();									//跳转到APP.
	}	
}
#ifndef __UPDATE_H_
#define __UPDATE_H_

#include "stm32f10x.h"

//我们使用的FLASH大小:512K
//0x08000000-0x0807FFFF
//Boot -- 12K			0x08000000-0x08002FFF		0x3000
//APP1 -- 250K		0x08003000-0x080417FF		0x3E800
//APP2 -- 250K		0x08041800-0x0807FFFF		0x3E800
#define FLASH_APP1_ADDR		0x08003000  		//第一个应用程序APP1起始地址(存放在FLASH)
#define FLASH_APP2_ADDR		0x08041800      //第二个应用程序APP2的起始地址
#define APP1_FLAG_ADDR		(FLASH_APP2_ADDR-4)	//APP1是否有更新程序标志位
#define APP2_FLAG_ADDR		(0x08080000-4)	//APP2是否有更新程序标志位
#define APP_SIZE	(0x3E800/STM_SECTOR_SIZE)	//分给APP的页数量
#define APP_MAX_SIZE	0x3E800   //250K

typedef  void (*pFunction)(void);   //函数指针  函数指针是一个指针,指向1个函数
//char *pFunction(void); //指针函数   指针函数是一个函数,返回的是1个指针(地址)

void Check_UPdate_Flag(void);
void UpdateFun(void);
void UserFlashAppRun(void);
void IAP_Load_App(u32 AppxAddr);
void Erase_APP1(void);
void Erase_APP2(void);

#endif

APP1程序

        进入该部分,首先修改向量表, 因为本程序是由BootLoader跳转过来的, 不修改向量表后面会出现问题;需要在APP的基本功能上加入串口接收数据并保存到APP2(备份区)的功能代码。

(生成的hex文件中包含地址信息,在生成app部分代码的hex文件时,注意更改ROM的地址位置,RAM的地址在0X2000xxxx上位置才算合理)

如果APP的代码使用下载器下载,那么我们就需要修改下载的属性

#include "update.h"
#include "stmflash.h"
#include "stdio.h"

//生成二进制文件
//D:\Keil5\ARM\ARMCC\bin\fromelf.exe --bin -o .\Objects\Demo.bin .\Objects\Demo.axf

uint8_t RecvBuff[2] = {0};   //存放准备写入APP2的数据,在串口中断中调用,没收到2个字节,写入一次
uint32_t RecvNum = 0;   //接收的数量  标记升级文件的大小
uint32_t Addr = FLASH_APP2_ADDR;   //写入APP2的地址   最开始是APP2区域的起始地址
uint8_t RecvTime = 0;  //用来判断是否接收完成
uint8_t RecvOver = 0;   //升级文件接收完成  1接收完成
uint16_t App2FlagBuff[2] = {0xAAAA, 0xAAAA};  //APP2区域是否有升级文件的标志  获取升级文件完成之后,写入APP2区域有升级文件的标记

//擦除APP2区域,方便接收新的代码  FLASH必须先擦除才能写入
void Erase_APP2(void)
{
	STMFLASH_Erase(FLASH_APP2_ADDR, APP_MAX_SIZE);
	printf("APP2备份区域擦除成功\r\n");
}

//判断从串口发送的升级文件是否发送完成
//如何确定  最后一个字节收到之后,计时会溢出
void RecvOverFun(void)
{
	if(RecvOver == 1) {
		printf("APP数据接收完成:%d\r\n", RecvNum);
		RecvNum = 0;
		RecvOver = 0;
		RecvTime = 0;
		Addr = FLASH_APP2_ADDR;
		
		STMFLASH_WriteHalfWord(APP2_FLAG_ADDR, App2FlagBuff[0]);    //写APP2区域有升级文件的标志
		STMFLASH_WriteHalfWord(APP2_FLAG_ADDR+2, App2FlagBuff[0]);
		printf("核对数据无误后,请按下复位按键进行数据更新\r\n");   //也可以选择调用复位函数   看门狗复位   NVIC_SystemReset();
	}
}

//修改中断向量表的地址偏移
void NVIC_SETVectorTable(void)
{
//	SCB->VTOR = FLASH_BASE | 0x3000;//中断向量表的地址偏移,寄存器写法
//	void NVIC_SetVectorTable(uint32_t NVIC_VectTab, uint32_t Offset);   //库函数写法  misc.h 198行
	NVIC_SetVectorTable(NVIC_VectTab_FLASH,0x3000);
}

void RecvTimeOut(void)	//1ms一次
{
	if(RecvTime) {
		RecvTime++;
		if(RecvTime >= 100) {
			RecvOver = 1;
			RecvTime = 0;
		}
	}
}

APP2程序

该程序只需要写需要升级的代码即可。然后生成bin文件即可。

#include "update.h"
#include "stmflash.h"
#include "stdio.h"

//生成二进制文件
//D:\MDK5\ARM\ARMCC\bin\fromelf.exe --bin -o .\Objects\Demo.bin .\Objects\Demo.axf

uint8_t RecvBuff[2] = {0};   //存放准备写入APP2的数据,在串口中断中调用,没收到2个字节,写入一次
uint32_t RecvNum = 0;   //接收的数量  标记升级文件的大小
uint32_t Addr = FLASH_APP2_ADDR;   //写入APP2的地址   最开始是APP2区域的起始地址
uint8_t RecvTime = 0;  //用来判断是否接收完成
uint8_t RecvOver = 0;   //升级文件接收完成  1接收完成
uint16_t App2FlagBuff[2] = {0xAAAA, 0xAAAA};  //APP2区域是否有升级文件的标志  获取升级文件完成之后,写入APP2区域有升级文件的标记

//擦除APP2区域,方便接收新的代码  FLASH必须先擦除才能写入
void Erase_APP2(void)
{
	STMFLASH_Erase(FLASH_APP2_ADDR, APP_MAX_SIZE);
	printf("APP2备份区域擦除成功\r\n");
}

//判断从串口发送的升级文件是否发送完成
//如何确定  最后一个字节收到之后,计时会溢出
void RecvOverFun(void)
{
	if(RecvOver == 1) {
		printf("APP数据接收完成:%d\r\n", RecvNum);
		RecvNum = 0;
		RecvOver = 0;
		RecvTime = 0;
		Addr = FLASH_APP2_ADDR;
		
		STMFLASH_WriteHalfWord(APP2_FLAG_ADDR, App2FlagBuff[0]);    //写APP2区域有升级文件的标志
		STMFLASH_WriteHalfWord(APP2_FLAG_ADDR+2, App2FlagBuff[0]);
		printf("核对数据无误后,请按下复位按键进行数据更新\r\n");   //也可以选择调用复位函数   看门狗复位   NVIC_SystemReset();
	}
}

//修改中断向量表的地址偏移
void NVIC_SETVectorTable(void)
{
//	SCB->VTOR = FLASH_BASE | 0x3000;//中断向量表的地址偏移,寄存器写法
//	void NVIC_SetVectorTable(uint32_t NVIC_VectTab, uint32_t Offset);   //库函数写法  misc.h 198行
	NVIC_SetVectorTable(NVIC_VectTab_FLASH,0x3000);
}

void RecvTimeOut(void)	//1ms一次
{
	if(RecvTime) {
		RecvTime++;
		if(RecvTime >= 100) {
			RecvOver = 1;
			RecvTime = 0;
		}
	}
}

D:\Keil5\ARM\ARMCC\bin\fromelf.exe是你的KEIL5安装路径下的romelf.exe是一个keil自带的生成bin文件的工具绝对路径;

--bin -o .\Objects\Demo.bin .\Objects\Demo.axf将这部分的可执行程序改成自己的可执行程序文件名;

然后将D:\Keil5\ARM\ARMCC\bin\fromelf.exe --bin -o .\Objects\Demo.bin .\Objects\Demo.axf复制到下图位置上。

        在 MDK 编译成功之后,调用 fromelf.exe(注意,我的 MDK 是安装在 D盘文件夹下,如果你是安装在其他目录,请根据自己的目录修改fromelf.exe 的路径),根据当前工程的 Demo.axf(如果是其他的名字,请记住修改,这个文件存放在 Objects 目录下面,格式为 xxx.axf),生成一个 .bin 的文件。并存放在 axf 文件相同的目录下,即工程的 Objects 文件夹里面。

        在得到.bin 文件之后,我们只需要将这个 bin 文件传送给单片机,即可执行 IAP 升级。(我们也可以将bin文件无线发送,存放在SD卡内,存放在外部FLASH内等等方式进行代码升级,其中无线发送的形式叫OTA)

        把APP2生成的bin文件,通过串口,发送到APP1的运行设备上,就会自动的保存APP2的代码数据到对应的Flash地址下,那么按下复位按键后(也可以软件复位),再次运行bootloader代码,就会加载APP2的数据到APP1的地址下,并运行新的程序。

最后重启或者按下复位键即可。

验证操作——步骤

1.将BOOTLoader程序编译后下载到单片机中,打开串口助手显示bootloader执行,led1、led2同时闪烁。

2.下载APP1程序到单片机中,观察现象。led3、led4同时闪烁。

3.编译APP2程序生成bin文件。

按下复位键后,更新代码,蜂鸣器响。

同理,可以先下载APP2,在发送APP1的bin文件。验证IAP的功能。

另外利用STM32ST—LINK Utility也可以将程序烧录到单片机中。将hex文件直接托拽到软件界面中去然后烧录即可。

拓展:

hex文件:包含地址信息;bin文件:不包含地址信息。HEX文件比BIN文件大,HEX文件有地址信息,BIN文件没有地址信息。HEX文件和BIN文件都可以是程序文件,但是HEX文件放的信息比BIN多,所以代码会比较大。一般远程升级用bin文件。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

LLM - Transformer 的 多头自注意力(MHSA) 理解与源码

欢迎关注我的CSDN:https://spike.blog.csdn.net/ 本文地址:https://spike.blog.csdn.net/article/details/140281680 免责声明:本文来源于个人知识与公开资料,仅用于学术交流,欢迎讨论,不支持转载。 在 Transformer 中,多头自注意力机制 (MHSA, Multi-Head Self-Attenti…

【pytorch18】Logistic Regression

回忆线性回归 for continuous:y xwbfor probability output:yσ(xwb) σ:sigmoid or logistic 线性回归是简单的线性模型&#xff0c;输入是x&#xff0c;网络参数是w和b&#xff0c;输出是连续的y的值 如何把它转化为分类问题?加了sigmoid函数&#xff0c;输出的值不再是…

Gen4Gen:多概念个性化图像生成的数据驱动革新

个性化文本到图像生成模型在用户控制生成过程方面取得了重要进展。这些模型能够通过少量训练样本学习并合成包含新颖个性化概念的图像&#xff0c;例如用户的宠物或特定物品。然而&#xff0c;现有技术在处理多概念个性化时存在局限性&#xff0c;尤其是在生成包含多个相似概念…

挂耳式耳机哪款比较好、挂耳式耳机推荐高性价比

近年来&#xff0c;开放式耳机行业蓬勃发展&#xff0c;受到了越来越多消费者的喜爱&#xff0c;然而&#xff0c;这里边也夹着不专业的产品&#xff0c;低质量的生产不仅不能带来舒适的体验&#xff0c;甚至可能对耳朵造成潜在的伤害。挂耳式耳机哪款比较好&#xff1f;为了帮…

初识STM32:寄存器编程 × 库函数编程 × 开发环境

STM32的编程模型 假如使用C语言的方式写了一段程序&#xff0c;这段程序首先会被烧录到芯片当中&#xff08;Flash存储器中&#xff09;&#xff0c;Flash存储器中的程序会逐条的进入CPU里面去执行。 CPU相当于人的一个大脑&#xff0c;虽然能执行运算和执行指令&#xff0c;…

WPF依赖附加属性

依赖附加属性的定义 基本过程&#xff1a;声明、注册、包装 依赖附加属性必须在依赖对象&#xff0c;附加属性不一定&#xff0c;关注的是被附加的对象是否是依赖对象 快捷方式&#xff1a;propa tab 关键字&#xff1a;RegisterAttached // 方法封装 public static int …

Java客户端调用SOAP方式的WebService服务实现方式分析

简介 在多系统交互中&#xff0c;有时候需要以Java作为客户端来调用SOAP方式的WebService服务&#xff0c;本文通过分析不同的调用方式&#xff0c;以Demo的形式&#xff0c;帮助读者在生产实践中选择合适的调用方式。 本文JDK环境为JDK17。 结论 推荐使用Axis2或者Jaxws&#…

推出全新的无线通讯模块(1SJ型、2DT-158型、2GT-001型、1YN型、2AE型)助力物联网新发展

相关型号&#xff1a;LBAA0QB1SJ-296 LBAA0XV2DT-158 LBAA0XV2GT-001 LBEE5KL1YN-814 LBEE5PK2AE-564 全新的无线通讯模块&#xff08;1SJ型、2DT-158型、2GT-001型、1YN型、2AE型&#xff09;助力物联网新发展&#xff08;明佳达&#xff09; 1、1SJ型集成LoRaWAN调制解调器…

优劣分析:重启路由器 vs 使用IP代理

目前更换IP主要有两种常见方法&#xff0c;一种是重启路由器&#xff0c;另一种是使用代理IP&#xff0c;那么&#xff0c;这两种方法有什么优缺点呢&#xff1f;下面我们一起来探讨一下。 方法一&#xff1a;重启路由器变换IP 优点 1. 操作简单&#xff1a;只需断开路由器电…

『C + ⒈』‘\‘

&#x1f942;在反斜杠(\)有⒉种最常用的功能如下所示&#x1f44b; #define _CRT_SECURE_NO_WARNINGS 1 #include<stdio.h> int main(void) {int a 10;int b 20;int c 30;if (a 10 &&\b 20 &&\c 30){printf("Your print\n");}else{prin…

解锁AI大模型潜能:预训练、迁移学习与中间件编程的协同艺术

在人工智能的浩瀚星空中&#xff0c;大型预训练模型&#xff08;Large Language Models, LLMs&#xff09;犹如璀璨的星辰&#xff0c;引领着技术革新的浪潮。这些模型通过海量数据的滋养&#xff0c;学会了理解语言、生成文本乃至执行复杂任务的能力。然而&#xff0c;要让这些…

可以拖拽的富文本编辑器(VueDragResize,quill-editor)

该功能实现一个帮助文档的展示和编辑功能&#xff0c;默认进去只能查看帮助文档的内容&#xff0c;点击编辑可以进行富文本编辑器的编辑功能。 出现的问题1.如何隐藏富文本编辑的工具栏并且禁止编辑 //隐藏工具栏this.toolbar this.$refs.myTextEditor.quill.getModule(toolb…

LeetCode之无重复字符的最长子串

1.题目链接 3. 无重复字符的最长子串 - 力扣&#xff08;LeetCode&#xff09;https://leetcode.cn/problems/longest-substring-without-repeating-characters/description/ 2.题目解析 题目主要思路其实是滑动窗口&#xff0c;使用两个指针维护一个动态区间&#xff0c;使…

价格疑云?格行WiFi创始人亲解谜团,性价比之王如何炼成?

随身wifi行业乱象频出&#xff0c;作为行业领跑品牌的格行随身wifi&#xff0c;关于价格问题一直备受质疑。关于设备上的“格行自有格行的骄傲”也被外界认定为是自大&#xff0c;甚至发展的线下一万多家门店也被同行不认可。近日&#xff0c;企业财经专访记者有幸采访了格行随…

实时消息推送系统,写得太好了!

websocket 协议是在 http 协议上的一种补充协议&#xff0c;是 html5 的新特性&#xff0c;是一种持久化的协议。其实 websocket 和 http 关系并不是很大&#xff0c;不过都是属于应用层的协议&#xff0c;接下来我们就开始实战。 websocket 定时推送 本教程基于 springboot …

华为od相关信息分享

2024年OD统一考试&#xff08;D卷&#xff09;完整题库&#xff1a;华为OD机试2024年最新题库&#xff08;Python、JAVA、C合集&#xff09; 问 1.什么是华为od&#xff1f; 答&#xff1a;OD全称是Outsourcing Dispacth&#xff0c;即外包派遣&#xff0c;是华为和外企德科…

HTML 标签简写及全称:表格内容将通过JavaScript动态生成

<!DOCTYPE html> <html lang"zh-CN"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>HTML 标签简写及全称</title><style>…

2025考研~数据结构试卷

作者主页&#xff1a;知孤云出岫 数据结构试题 [TOC](数据结构试题)数据结构试卷一、选择题&#xff08;每题2分&#xff0c;共20分&#xff09;二、填空题&#xff08;每题3分&#xff0c;共15分&#xff09;三、简答题&#xff08;每题10分&#xff0c;共40分&#xff09;四…

卷技术还是卷应用?李彦宏给出了明确答案

如何理解李彦宏说的“不要卷模型&#xff0c;要卷应用” 引言 7月4日&#xff0c;2024世界人工智能大会在上海世博中心召开。百度创始人兼CEO李彦宏在产业发展主论坛上呼吁&#xff1a;“大家不要卷模型&#xff0c;要卷应用&#xff01;”这句话引起了广泛讨论。李彦宏认为&a…

【python】PyQt5对象类型的判定,对象删除操作详细解读

✨✨ 欢迎大家来到景天科技苑✨✨ &#x1f388;&#x1f388; 养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; &#x1f3c6; 作者简介&#xff1a;景天科技苑 &#x1f3c6;《头衔》&#xff1a;大厂架构师&#xff0c;华为云开发者社区专家博主&#xff0c;…