嵌入式代码升级——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区域250Kfor(uint16_t i=0; i<APP_SIZE; i++) {   //每次读2Kprintf("正在更新固件%d...\r\n", i);STMFLASH_Read(FLASH_APP2_ADDR+i*STM_SECTOR_SIZE, ReadBuff, STM_SECTOR_SIZE/2);   //从APP2的起始地址开始读2KSTMFLASH_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     0x08003004if(((*(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   //250Ktypedef  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.axfuint8_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.axfuint8_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文件。

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

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

相关文章

招投标信息采集系统:让您的企业始终站在行业前沿

一、为何招投标信息如此关键&#xff1f; 在经济全球化的大背景下&#xff0c;招投标活动日益频繁&#xff0c;成为企业获取项目、拓展市场的主流方式之一。招投标信息采集&#xff0c;作为企业战略决策的前置环节&#xff0c;其重要性不言而喻。它不仅关乎企业能否第一时间发…

如何网页在线编辑微软Office Word,并导出为PDF格式。

随着互联网技术的不断发展&#xff0c;越来越多的企业开始采用在线办公模式&#xff0c;微软Office Word 是最好用的文档编辑工具&#xff0c;然而doc、docx、xls、xlsx、ppt、pptx等格式的Office文档是无法直接在浏览器中直接打开的&#xff0c;如果可以实现Web在线预览编辑Of…

git只列出本地分支

git只列出本地分支 git branch --list git强制删除本地分支 git branch -D_error: the branch dlx-test is not fully merged. -CSDN博客文章浏览阅读648次。git branch -d 可以通过: git branch 查看所有本地分支及其名字&#xff0c;然后删除特定分支。git删除远程remote分支…

互联网医院系统,开发互联网医院设计哪些功能?

随着科技的进步和数字化转型的推动&#xff0c;互联网医院系统已成为现代医疗服务的重要组成部分。这一系统通过整合信息技术与医疗资源&#xff0c;为用户提供便捷、高效的医疗服务。以下是互联网医院系统的主要功能介绍。 1、在线咨询与诊断 互联网医院系统允许患者通过网络平…

IEC62056标准体系简介-2.IEC62056标准体系及对象标识系统(OBIS)

1. IEC 62056标准体系 IEC 62056标准体系目前共包括六部分&#xff0c;见图1&#xff1a; 第61部分&#xff1a;对象标识系统第62部分&#xff1a;接口类第53部分&#xff1a;COSEM应用层第46部分&#xff1a;使用HDLC&#xff08;High Level Data Link Control&#xff09;协…

54、一维和二维自组织映射(matlab)

1、一维和二维自组织映射原理 一维和二维自组织映射&#xff08;Self-Organizing Maps, SOM&#xff09;是一种无监督的机器学习算法&#xff0c;通过学习输入数据的拓扑结构&#xff0c;将高维输入数据映射到低维的网格结构中&#xff0c;使得相似的输入数据点在映射空间中也…

阿尔泰科技与西安交通大学陕西省某技术重点实验室共谋未来!

近日&#xff0c;阿尔泰科技的电子工程师&#xff08;熊工&#xff09;应邀前往西安交通大学陕西省某技术重点实验室&#xff0c;参与课题组项目的测试与调试工作。此次合作不仅成功推动了项目的进展&#xff0c;还为未来的深入合作奠定了坚实基础。 阿尔泰科技作为领先的测控技…

刷题之删除有序数组中的重复项(leetcode)

删除有序数组中的重复项 这题简单题&#xff0c;双指针&#xff0c;一个指针记录未重复的数的个数&#xff0c;另一个记录遍历的位置。 以下是简单模拟&#xff0c;可以优化&#xff1a; class Solution { public:int removeDuplicates(vector<int>& nums) {int l0…

Debezium报错处理系列之第114篇:No TableMapEventData has been found for table id:256.

Debezium报错处理系列之第114篇:Caused by: com.github.shyiko.mysql.binlog.event.deserialization.MissingTableMapEventException: No TableMapEventData has been found for table id:256. Usually that means that you have started reading binary log within the logic…

博美犬插画:成都亚恒丰创教育科技有限公司

​博美犬插画&#xff1a;萌动心灵的细腻笔触 在浩瀚的艺术海洋中&#xff0c;有一种艺术形式总能以它独有的温柔与细腻&#xff0c;触动人心最柔软的部分——那便是插画。而当插画遇上博美犬这一萌宠界的明星&#xff0c;便诞生了一幅幅令人爱不释手的作品&#xff0c;成都亚…

代码随想录打卡第十八天

代码随想录–二叉树部分 day 17 休息日 day 18 二叉树第五天 文章目录 代码随想录--二叉树部分一、力扣654--最大二叉树二、力扣617--合并二叉树三、力扣700--二乘树中的搜素四、力扣98--验证二叉搜索树 一、力扣654–最大二叉树 代码随想录题目链接&#xff1a;代码随想录 给…

美容师有什么话术技巧?美业人如何提升自己的销售技巧?博弈美业门店管理系统分享经验

作为一名美容师&#xff0c;有一些话术和销售技巧可以帮助你提升服务质量和销售业绩。以下是博弈美业收银系统分享的一些建议&#xff1a; 1.建立信任&#xff1a; 在与客户交流时&#xff0c;表现出真诚、友好和专业的态度。倾听客户的需求&#xff0c;并给予针对性的建议&a…

随笔(一)

1.即时通信软件原理&#xff08;发展&#xff09; 即时通信软件实现原理_即时通讯原理-CSDN博客 笔记&#xff1a; 2.泛洪算法&#xff1a; 算法介绍 | 泛洪算法&#xff08;Flood fill Algorithm&#xff09;-CSDN博客 漫水填充算法实现最常见有四邻域像素填充法&#xf…

Redis代替Session实现共享

集群的session共享问题 session共享问题&#xff1a;多台tomcat并不共享session存储空间&#xff0c;当请求切换到不同的tomcat服务时导致数据丢失的问题。 session的替代方案&#xff1a; 数据共享内存存储key、value结构 将redis替换session可以解决session共享问题

Day65 代码随想录打卡|回溯算法篇---组合总和II

题目&#xff08;leecode T40&#xff09;&#xff1a; 给定一个候选人编号的集合 candidates 和一个目标数 target &#xff0c;找出 candidates 中所有可以使数字和为 target 的组合。 candidates 中的每个数字在每个组合中只能使用 一次 。 注意&#xff1a;解集不能包含…

第十八节 LLaVA如何按需构建LORA训练(视觉、语言、映射多个组合训练)

文章目录 前言一、基于llava源码构建新的参数1、添加lora_vit参数2、训练命令脚本设置二、修改源码,构建lora训练1、修改源码-lora训练2、LLM模型lora加载3、VIT模型加载4、权重冻结操作5、结果显示三、实验结果前言 如果看了我前面文章,想必你基本对整个代码有了更深认识。…

DevOps实战:使用GitLab+Jenkins+Kubernetes(k8s)建立CI_CD解决方案

一.系统环境 本文主要基于Kubernetes1.21.9和Linux操作系统CentOS7.4。 服务器版本docker软件版本Kubernetes(k8s)集群版本CPU架构CentOS Linux release 7.4.1708 (Core)Docker version 20.10.12v1.21.9x86_64CI/CD解决方案架构图:CI/CD解决方案架构图描述:程序员写好代码之…

学习数据库2

在数据库中创建一个表student&#xff0c;用于存储学生信息 查看建表结果 向student表中添加一条新记录 记录中id字段的值为1&#xff0c;name字段的值为"monkey"&#xff0c;grade字段的值为98.5 并查看结果 向student表中添加多条新记录 2,"bob"…

算法之工程化内容(2)—— Git常用命令

目录 1. git初始化配置 2. 新建仓库 3. 工作区——>暂存区——>本地仓库 4. git reset回退版本 5. 查看差异 git diff 6. 删除文件git rm 7. .gitignore 8. vscode操作git 9. git分支、合并和删除 10. 解决合并冲突 11. 回退和rebase 12. 添加远程仓库 参考链接&#xff…

用于视频生成的扩散模型

学习自https://lilianweng.github.io/posts/2024-04-12-diffusion-video/ 文章目录 3D UNet和DiTVDMImagen VideoSora 调整图像模型生成视频Make-A-Video&#xff08;对视频数据微调&#xff09;Tune-A-VideoGen-1视频 LDMSVD稳定视频扩散 免训练Text2Video-ZeroControlVideo 参…