STM32用flash保存参数实现平衡擦写的一种方法
#FLASH平衡擦写#
一、概述
简易示意图如下:
写参数前要擦除对应的扇区 全为0XFFFFFFFF操作的最小单位为32位 uint32_t; 当一块扇区写完时,将所有有用参数复制到第二块扇区,开始写新的参数,如果所有参数写完,又重第一块参数开始写,这样就能实现平衡写的目的,所以要实现这个功能,至少需要分配2个扇区实现均衡擦写。
/* 储存扇区信息的结构体 */
struct SSCT_HDR
{uint32_t st; // 状态uint32_t cnt; // 标号uint32_t version; // 版本
}; // 扇区HEAD结构typedef struct
{uint16_t len:16; uint16_t alen:16; /* data */
}VARLEN;
struct VAR_ST //Flash数据存储结构
{uint32_t status; //数据当前状态uint32_t key; //数据keyunion {uint32_t len; //数据长度 len +alen 数据实际长度+所在内存长度 内存长度必须是4的整数倍VARLEN len_b;/* data */};};
| 扇区1 |
|
二、源码
#include <stdbool.h>
#include <stdint.h> //引用框架配置文件#define MAX_CVAR_NUM (200) //数据存储最大个数typedef struct
{uint16_t len:16; uint16_t alen:16; /* data */
}VARLEN;
struct VAR_ST //Flash数据存储结构
{uint32_t status; //数据当前状态uint32_t key; //数据keyunion {uint32_t len; //数据长度 len +alen 数据实际长度+所在内存长度 内存长度必须是4的整数倍VARLEN len_b;/* data */};};struct SFVAR_POINT
{uint32_t key; //数据keyuint8_t* flashAddr; //数据地址
};typedef struct
{uint8_t* sectorBaseAdr;uint32_t sectorSize;uint8_t sectorNum;void (*FlashInit_Cbk)(void); // Flash初始化函数bool (*FlashErase_Cbk)(uint8_t* addr, uint32_t size); // Flash擦除函数uint32_t (*FlashWrite_Cbk)(void* addr, const void* buf, uint32_t size); // Flash写入函数uint32_t (*FlashRead_Cbk)(void* addr, void* buf, uint32_t size); /// flash读函数// privatestruct{uint32_t makeTime;uint8_t* sectorUseBaseAdr; // flash 参数存储区基地址uint8_t sectorUseCnt; // Flash 所有的序号uint32_t varNum; // Flash存储数据个数uint32_t tail; // Flash当前地址uint32_t head; // 有效头部位置uint8_t swSctFlag; //扇区切换flaguint8_t* rmAdr; //需要删除的 地址struct VAR_ST pCVar; //单个参数的头部结构struct SFVAR_POINT varList[MAX_CVAR_NUM]; //数据Z指针数组} pri;}FlashPar_Prop;typedef struct
{void (* const Create)(FlashPar_Prop* self); // FlashVarvoid (*Init)(FlashPar_Prop* self,uint32_t makeTime,uint8_t* sectorBaseAdr, // FLASH基地址uint32_t sectorSize, // flash大小uint32_t sectorNum, // flash块的个数void (*FlashInit_Cbk)(void), // flash初始化函数bool (*FlashErase_Cbk)(uint8_t* addr, uint32_t size), // flash擦除函数uint32_t (*FlashWrite_Cbk)(void* addr, const void* buf, uint32_t size), // flash写入函数uint32_t (*FlashRead_Cbk)(void* addr, void* buf, uint32_t size) // flash读函数);// APIuint32_t (*RdPar)(FlashPar_Prop* self, uint32_t key, uint8_t* pRdBuf, uint32_t bufLen); ///数据读取函数MOBJ_BOOL (*WtPar)(FlashPar_Prop* self, uint32_t key, uint8_t* pWtDat, uint32_t datLen); //数据写入函数MOBJ_BOOL (*DelPar)(FlashPar_Prop* self, uint32_t key); //数据删除函数}FlashPar_Func;extern const FlashPar_Func FlashPar;
#include "MFlashVar.h"
#include "string.h"
#include "MTime.h"
/* 扇区使用情况 表示各个扇区状态*/#define SSCT_UNUSE (0xFFFFFFFF) // 未使用
#define SSCT_USE (0xBBBBBBBB) // 使用中
#define SSCT_DEL (0x00000000) // 删除状状态/* 某区域保存参数的状态 */#define SCVAR_UNUSE (0xFFFFFFFF) // 未使用#define SCVAR_USE (0xAAAAAAAA) // 使用中#define SCVAR_DEL (0x00000000) // 删除状状态// 表示各个数据状态/* 储存扇区信息的结构体 */
struct SSCT_HDR
{uint32_t st; // 状态uint32_t cnt; // 标号uint32_t version; // 版本}; // 扇区HEAD结构static int32_t FindVarAddr(FlashPar_Prop *self, uint32_t key);
static uint32_t AllocVar(FlashPar_Prop *self, uint32_t len, uint32_t key);
static void DelVar(FlashPar_Prop *self, uint8_t *addr);
static void PrgVar(FlashPar_Prop *self, void *flashAddr, uint32_t key, uint8_t *pWtDat, uint32_t dataLen);
static void LoadSector(FlashPar_Prop *self);
static void LoadFVar(FlashPar_Prop *self);
static void SwitchSct(FlashPar_Prop *self);/*** @brief FlashPar**/
static void FlashPar_Init(FlashPar_Prop *self,uint32_t makeTime,uint8_t *sectorBaseAdr,uint32_t sectorSize,uint32_t sectorNum,void (*FlashInit_Cbk)(void),bool (*FlashErase_Cbk)(uint8_t *addr, uint32_t size),uint32_t (*FlashWrite_Cbk)(void *addr, const void *buf, uint32_t size),uint32_t (*FlashRead_Cbk)(void *addr, void *buf, uint32_t size) // flash读函数
)
{self->pri.varNum = 0;self->sectorBaseAdr = sectorBaseAdr;self->sectorSize = sectorSize;self->sectorNum = sectorNum;self->pri.makeTime = makeTime;self->FlashInit_Cbk = FlashInit_Cbk;self->FlashErase_Cbk = FlashErase_Cbk;self->FlashWrite_Cbk = FlashWrite_Cbk;self->FlashRead_Cbk = FlashRead_Cbk;// step1: load useing sectorLoadSector(self);// step2 : load flash variableLoadFVar(self);}/*** @brief 申请地址并检查剩余地址是否足够**/
static uint32_t AllocVar(FlashPar_Prop *self, uint32_t len, uint32_t key)
{uint32_t pFVarAddress;uint8_t tmp, actLen;uint16_t index;/******step1 :Caculate the actual space***/tmp = len % 4;if (tmp != 0)actLen = sizeof(struct VAR_ST) + len + (4 - tmp);elseactLen = sizeof(struct VAR_ST) + len;/*step2: check current sector has enough sapace*/if (self->pri.tail + actLen >= self->sectorSize){SwitchSct(self);index = FindVarAddr(self, key);self->pri.rmAdr = self->pri.varList[index].flashAddr;}else {}/*step3: current sector has enough sapace*/if (self->pri.tail + actLen < self->sectorSize){pFVarAddress = (uint32_t)(self->pri.sectorUseBaseAdr + self->pri.tail);self->pri.tail += actLen;}else{pFVarAddress = 0;}return pFVarAddress;
}
/*** @brief 删除原有变量函数**/
static void DelVar(FlashPar_Prop *self, uint8_t *addr)
{uint32_t st;st = SCVAR_DEL;self->FlashWrite_Cbk(addr, &st, sizeof(st));
}
/*** @brief 写入参数**/
static void PrgVar(FlashPar_Prop *self, void *flashAddr, uint32_t key, uint8_t *pWtDat, uint32_t dataLen)
{struct VAR_ST tmpVar;uint32_t tmp;uint32_t dtActLen;uint8_t *pHead = (uint8_t *)flashAddr;uint8_t *pData = (uint8_t *)flashAddr + sizeof(struct VAR_ST);tmp = dataLen % 4;if (tmp != 0)dtActLen = dataLen + (4 - tmp);elsedtActLen = dataLen;tmpVar.status = SCVAR_USE;tmpVar.key = key;tmpVar.len_b.len = dataLen;tmpVar.len_b.alen = dtActLen;self->FlashWrite_Cbk(pHead, (uint8_t *)&tmpVar, sizeof(struct VAR_ST));self->FlashWrite_Cbk(pData, pWtDat, dtActLen);}/*** @brief 根据关键字查询变量**/
static uint32_t FlashPar_RdPar(FlashPar_Prop *self, uint32_t key, uint8_t *pRdBuf, uint32_t bufLen)
{struct VAR_ST pFVar;uint32_t ret = 0;uint8_t *pData;uint32_t len;int32_t index;// find var in ramindex = FindVarAddr(self, key);if (index < 0){return ret;}self->FlashRead_Cbk(self->pri.varList[index].flashAddr, &pFVar, sizeof(struct VAR_ST));len = pFVar.len_b.len;if (bufLen < len){len = bufLen;}pData = self->pri.varList[index].flashAddr + sizeof(struct VAR_ST);self->FlashRead_Cbk(pData, pRdBuf, len);ret = len;return ret;
}
/* 根据KEY 删除一个参数 */
static MOBJ_BOOL FlashPar_DelPar(FlashPar_Prop *self, uint32_t key)
{uint8_t *pFVar;int32_t index;uint32_t MvDataNum;// find var in ramindex = FindVarAddr(self, key);if (index < 0){return NOT;}pFVar = self->pri.varList[index].flashAddr;MvDataNum = self->pri.varNum - (index + 1);while (MvDataNum--){self->pri.varList[index].flashAddr = self->pri.varList[index + 1].flashAddr;self->pri.varList[index].key = self->pri.varList[index + 1].key;index++;}self->pri.varList[index].flashAddr = 0;self->pri.varList[index].key = 0;self->pri.varNum--;DelVar(self, pFVar);return YES;
}/*** @brief 根据key保存一个参数**/
static MOBJ_BOOL FlashPar_WtPar(FlashPar_Prop *self, uint32_t key, uint8_t *pWtDat, uint32_t datLen)
{uint8_t *pNewVar;uint8_t tempdata[258] = {0};struct VAR_ST pOldVar;int32_t index;MOBJ_BOOL ret;if (datLen > 256){}/******step1 :find old var ***/index = FindVarAddr(self, key);/******step2 :wite new var*/if (index < 0) // step2.1 old var not exist{if (self->pri.varNum >= MAX_CVAR_NUM) // check number{ret = NOT;}else if (0 == (pNewVar = (uint8_t *)AllocVar(self, datLen, key))) // alloc space{ret = NOT;}else{PrgVar(self, pNewVar, key, pWtDat, datLen);self->pri.varList[self->pri.varNum].key = key;self->pri.varList[self->pri.varNum].flashAddr = pNewVar;self->pri.varNum++;ret = YES;}}else // step2.2 old var exist{self->pri.rmAdr = self->pri.varList[index].flashAddr;self->FlashRead_Cbk(self->pri.rmAdr, &pOldVar, sizeof(struct VAR_ST));if (pOldVar.key == key){self->FlashRead_Cbk(self->pri.rmAdr + sizeof(struct VAR_ST), tempdata, pOldVar.len_b.len);if ((pOldVar.len_b.len== datLen) && (0 == memcmp(tempdata, pWtDat, datLen))){ret = YES;}else{pNewVar = (uint8_t *)AllocVar(self, datLen, key);if (0 == pNewVar) // alloc space{// EINT;ret = NOT;}else{PrgVar(self, pNewVar, key, pWtDat, datLen); // write new varDelVar(self, self->pri.rmAdr); // 完全删除ret = YES;}self->pri.varList[index].flashAddr = pNewVar;}}}return ret;
}
static int32_t FindVarAddr(FlashPar_Prop *self, uint32_t key)
{int32_t i;for (i = 0; i < self->pri.varNum; i++){if (self->pri.varList[i].key == key)return i;}return -1;
}/*** @brief 加载各个扇区的状态信息**/static void LoadSector(FlashPar_Prop *self)
{int32_t i;int32_t maxSctCnt = 0;uint8_t *useadd = 0;struct SSCT_HDR pSctHdr, newSctHdr;// step1 : find using sectorfor (i = 0; i < self->sectorNum; i++){useadd = self->sectorBaseAdr + i * self->sectorSize;self->FlashRead_Cbk(useadd, &pSctHdr, sizeof(struct SSCT_HDR));// check the version,if ((pSctHdr.version != 0xFFFFFFFF) && (pSctHdr.version != self->pri.makeTime)){self->FlashErase_Cbk(useadd, self->sectorSize);}else{switch (pSctHdr.st){case SSCT_UNUSE: {break;}case SSCT_USE: {if (pSctHdr.cnt >= maxSctCnt){self->pri.sectorUseBaseAdr = useadd;self->pri.sectorUseCnt = i;maxSctCnt = pSctHdr.cnt;}break;}case SSCT_DEL: {break;}}}}// step2 : if don't find using sector them set sector0 is usedif (maxSctCnt == 0){self->pri.sectorUseBaseAdr = self->sectorBaseAdr;self->pri.sectorUseCnt = 0;self->FlashErase_Cbk(self->pri.sectorUseBaseAdr, self->sectorSize); // 擦除 实际地址需newSctHdr.st = SSCT_USE;newSctHdr.cnt = 1;newSctHdr.version = self->pri.makeTime;self->FlashWrite_Cbk(self->pri.sectorUseBaseAdr, (uint8_t *)&newSctHdr, sizeof(struct SSCT_HDR));}self->pri.tail = sizeof(struct SSCT_HDR);
}
/*** @brief 加载flash 区的参数信息**/
static void LoadFVar(FlashPar_Prop *self)
{uint8_t rFlag = 1;struct VAR_ST *pErrVar = 0;struct VAR_ST nowVar;uint32_t errNo = 0;uint8_t *pFVarAddress = 0;// uint8_t* pFVarAddress = 0;while ((self->pri.tail < self->sectorSize) && rFlag){pFVarAddress = self->pri.sectorUseBaseAdr + self->pri.tail;self->FlashRead_Cbk(pFVarAddress, &nowVar, sizeof(struct VAR_ST));switch (nowVar.status){// if the data was unused than over build processcase SCVAR_UNUSE: {rFlag = 0; // stop researchbreak;}case SCVAR_USE: {if ((pErrVar != 0) && (nowVar.key == pErrVar->key)){self->pri.tail += sizeof(struct VAR_ST) + nowVar.len_b.alen;DelVar(self, self->pri.varList[errNo].flashAddr); // 删除原有错误数据self->pri.varList[errNo].flashAddr = pFVarAddress;}else // nomal{self->pri.tail += sizeof(struct VAR_ST) + nowVar.len_b.alen;self->pri.varList[self->pri.varNum].key = nowVar.key;self->pri.varList[self->pri.varNum].flashAddr = pFVarAddress;self->pri.varNum++;}break;}// if deleted than jumpcase SCVAR_DEL: {self->pri.tail += sizeof(struct VAR_ST) + nowVar.len_b.alen;}default: // 参数报错{// self->pri.tail += sizeof(struct VAR_ST) + dtActLen;}}} // end while
}/*** @brief Flash扇区切换 (暂时未确认是否ok)**/
static void SwitchSct(FlashPar_Prop *self)
{uint8_t data[256];struct VAR_ST pOldVar;struct SSCT_HDR pOldSctHD;uint8_t *oldSct = self->pri.sectorUseBaseAdr;self->FlashRead_Cbk(oldSct, &pOldSctHD, sizeof(struct SSCT_HDR));/********step1: find next sector******/self->pri.sectorUseCnt++;if (self->pri.sectorUseCnt >= self->sectorNum){self->pri.sectorUseCnt = 0;}uint8_t *newSct = self->sectorBaseAdr + self->pri.sectorUseCnt * self->sectorSize;self->FlashErase_Cbk(newSct, self->sectorSize); // 擦除即将切换到的扇区 擦除地址开始后的1个扇区/********step2 : produce sector header*******/struct SSCT_HDR newSctHD = {.st = SSCT_USE,.cnt = pOldSctHD.cnt + 1,.version = self->pri.makeTime,};self->pri.sectorUseBaseAdr = (uint8_t *)newSct;self->pri.tail = sizeof(struct SSCT_HDR);// sector write preself->FlashWrite_Cbk(newSct, (uint8_t *)&newSctHD, sizeof(struct SSCT_HDR));/********step3 :将原有参数移动到新区*************/int32_t i;uint8_t *newvaraddr;for (i = 0; i < self->pri.varNum; i++){self->FlashRead_Cbk(self->pri.varList[i].flashAddr, &pOldVar, sizeof(struct VAR_ST));newvaraddr = (uint8_t *)((uint32_t)self->pri.sectorUseBaseAdr + self->pri.tail);self->FlashRead_Cbk(self->pri.varList[i].flashAddr + sizeof(struct VAR_ST), data, pOldVar.len_b.len); // 读数据PrgVar(self, newvaraddr, pOldVar.key, data, pOldVar.len_b.len); // 保存数据self->pri.varList[i].flashAddr = newvaraddr;self->pri.tail += sizeof(struct VAR_ST) + pOldVar.len_b.alen;if (self->pri.tail > self->sectorSize){//扇区溢出return;}}pOldSctHD.st = SSCT_DEL;// sector write preself->FlashWrite_Cbk(oldSct, (uint8_t *)&pOldSctHD, sizeof(struct SSCT_HDR));
}
/*** @brief FlashPar**/
void FlashPar_Create(FlashPar_Prop *self) { memset(self, 0, sizeof(FlashPar_Prop)); }const FlashPar_Func FlashPar = {.Create = FlashPar_Create,.Init = FlashPar_Init,.RdPar = FlashPar_RdPar,.WtPar = FlashPar_WtPar,.DelPar = FlashPar_DelPar};
平衡擦写的流程主要涉及到扇区的选择和切换,以及变量的写入和删除。
1. 首先,在`FlashPar_Init`函数中,通过调用`LoadSector`函数加载扇区的状态信息。该函数会遍历所有扇区,检查状态并选择使用中的扇区作为当前扇区。
2. 接下来,在`LoadFVar`函数中,会加载当前扇区中的flash变量。函数会遍历当前扇区中的每个flash变量,将其存储到`pri.varList`数组中。同时,将当前扇区中的`tail`指针指向下一个可用的地址。
3. 当需要写入一个新的flash变量时,首先通过`AllocVar`函数申请一个地址,并检查当前扇区是否有足够的空间。如果空间不足,则需要切换到下一个扇区,并重新分配地址。然后,通过`PrgVar`函数将变量写入到flash中,并更新`pri.varList`数组和`pri.varNum`变量。如果找到了相同关键字的旧变量,则会先删除旧变量。
4. 当需要删除一个flash变量时,通过`FlashPar_DelPar`函数根据关键字找到变量的地址,并调用`DelVar`函数删除变量。
5. 当需要读取一个flash变量时,通过`FlashPar_RdPar`函数根据关键字找到变量的地址,并将变量的数据读取出来。
在上述流程中,如果当前扇区的空间不足以容纳新的flash变量,就会触发扇区切换。在`SwitchSct`函数中,先找到下一个可用的扇区,并将其标记为使用状态。然后,将当前扇区中的变量逐个移动到新的扇区,并更新变量的地址。同时,将当前扇区的状态设置为删除状态,并将新的扇区的状态设置为使用状态。
通过这样的流程,可以实现对flash的平衡擦写,避免频繁的擦写操作,延长flash的使用寿命。同时,可以方便地存储和读取变量,提供了一种简单有效的持久化存储解决方案。
相关文章:
STM32用flash保存参数实现平衡擦写的一种方法
#FLASH平衡擦写# 一、概述 简易示意图如下: 写参数前要擦除对应的扇区 全为0XFFFFFFFF操作的最小单位为32位 uint32_t; 当一块扇区写完时,将所有有用参数复制到第二块扇区,开始写新的参数,如果所有参数写完,又重第…...
Aho Corasick Algorithm
文章目录 前言介绍实现参考 前言 Aho Corasick Algorithm又叫AC自动机,该算法是一个匹配算法,用来匹配文本Text中多个patterns分别出现的次数; 我们定义n为patterns的总长度;m为Text的长度; 问题:在ahis…...
用户管理 --汇总
一、第一节课 1.1 本人写的 前端: 鱼皮 --> 用户中心 第1节课-CSDN博客 中期: 一、用户管理 第1节课中间-CSDN博客 后端: 一、用户管理-CSDN博客 其他的链接 亿图脑图MindMaster 1.2 优秀球友,推荐 Docs 另…...
Flutter视频播放器在iOS端和Android端都能实现全屏播放
Flutter开发过程中,对于视频播放的三方组件有很多,在Android端适配都挺好,但是在适配iPhone手机的时候,如果设置了UIInterfaceOrientationLandscapeLeft和UIInterfaceOrientationLandscapeRight都为false的情况下,无法…...
面试遇到的一些问题(二)
1、v-if v-show 区别,他们的生命周期区别 v-show: (类似于display:none/black 的切换)不管初始值是true 或false 都会进行渲染,状态改变也不会销毁和重新生成。不会影响生命周期 v-if : 是根据条件,dom进行删除插入操作。 依附于普通元素时:会触发父组件的beforeUpdate和u…...
JDK8新特性:Lambda表达式规则及用法,方法引用
目录 Lambda表达式是JDK8新增的一种语法格式 1.作用 2.用法规则: 3.方法引用 Lambda表达式是JDK8新增的一种语法格式 1.作用 简化匿名内部类的代码写法 Lambad用法前提:只能简化函数式接口(一般加有Funcationallnterface)&a…...
【GIS】JDK版本升级到17后,GeoServer的图层无法通过openLayer预览
JDK版本升级到17后,图层无法通过openLayer预览 1. 错误图示 终端输出的错误 网页端无法显示图层,并且输出错误提示 2.原因猜测 估计可能是由于java17的模块化,Java被分成了多个独立部署和运行的模块,这使得Java应用能够更快…...
vue 批量下载文件,不走后端接口的方法
今天ld提了一个需求,说页面的列表里面有要下载的地址,然后点击批量下载。我思索片刻,给出了代码 1.这个是列表页面的代码 <!-- 这个是列表页面的代码 --> <el-table :data"userListShow" align"center"border highlight-…...
科技云报道:AI+PaaS,中国云计算市场迎来新“变量”?
科技云报道原创。 没有小的市场,只有还没有被发现的大生意。 随着企业数字化转型的逐级深入,市场需求进一步向PaaS和SaaS层进发,使之成为公有云服务市场增长的主要动力。 根据IDC最新发布的报告显示,2022-2027五年间中国公有云…...
Windows Service Name重复问题
Windows Service Name重复问题 1,问题 2,打开命令提示符,管理员身份运行 3,输入命令:sc delete MYSQL57 4,验证一下,可以看见已经没有感叹号啦 ,可以看见已经没有感叹号啦...
BBS项目
一.BBS项目介绍 1.项目开发流程 项目立项 ------> 公司高层决定需求调研和分析 ------> 市场人员,技术人员参与 -需求文档说明开发部门开会 ------> 确定项目架构,技术选型,数据库设计UI,UD团队(产品经…...
Java基础——对象类型转换(向上、向下转型)
非继承关系的类之间对象类型不可以互相类型转换,只有继承关系才可以互相转换。 简单说,对象类型转换的前提要是继承关系。 对象类型转换分为:向上转型和向下转型。多态就是一种自动向上转型。 向上转型:子类对象用父类类型接收…...
期末速成数据库极简版【查询】(2)
目录 select数据查询----表 【1】筛选列 【2】where简单查询 【3】top-n/distinct/排序的查询 【4】常用内置函数 常用日期函数 常用的字符串函数 【5】模糊查询 【6】表数据操作——增/删/改 插入 更新 删除 【7】数据汇总 聚合 分类 🙂&#…...
2023年终总结-轻舟已过万重山
自我介绍 高考大省的读书人 白,陇西布衣,流落楚、汉。-与韩荆州书 我来自孔孟故里山东济宁,也许是小学时的某一天,我第一次接触到了电脑,从此对它产生了强烈的兴趣,高中我有一个愿望:成为一名计…...
手机号,邮箱,密码,验证码正则表达式[Java]
Util类: public abstract class RegexPatterns {/*** 手机号正则*/public static final String PHONE_REGEX "^1([38][0-9]|4[579]|5[0-3,5-9]|6[6]|7[0135678]|9[89])\\d{8}$";/*** 邮箱正则*/public static final String EMAIL_REGEX "^[a-zA-Z…...
普冉(PUYA)单片机开发笔记(7): ADC-轮询式多路采样
概述 应用中经常会有使用单片机进行模数转换的需求。PY32F003 具有 1 个 12 位的模拟数字转换器(ADC),今天我们一起来使用一下这个 ADC。 数据手册中对 ADC 简介如下。 SAR ADC:逐次逼近式 ADC,原理参见“参考链接&a…...
uniapp切换页面时报错问题
我们来看如下错误: 该错误的意思是不能切换到 tabbar 页面。tabbar页面通常是公共页面或者底部导航栏,如果我们用 navigateTo 或者 redirectTo 都不能实现页面切换。 我们有两种方式: 第一种是用 switchTab 来进行切换,但注意切…...
Nginx 简单入门操作
前言:之前的文章有些过就不罗嗦了。 Nginx 基础内容 是什么? Nginx 是一个轻量级的 HTTP 服务器,采用事件驱动、异步非阻塞处理方式的服务器,它具有极好的 IO 性能,常用于 HTTP服务器(包含动静分离)、正向代理、反向代理、负载均衡 等等. Nginx 和 Node.js 在很多方…...
ChatGPT是科学还是艺术?
OpenAI最近谈到GPT4变懒的问题,说“它更像是多人共同参与的艺术创作”,那到底大模型是科学还是艺术?...
线程及实现方式
一、线程 线程是一个基本的CPU执行单元,也是程序执行流的最小单位。引入线程之后,不仅是进程之间可以并发,进程内的各线程之间也可以并发,从而进一步提升了系统的并发度,使得一个进程内也可以并发处理各种任务&#x…...
XML Group端口详解
在XML数据映射过程中,经常需要对数据进行分组聚合操作。例如,当处理包含多个物料明细的XML文件时,可能需要将相同物料号的明细归为一组,或对相同物料号的数量进行求和计算。传统实现方式通常需要编写脚本代码,增加了开…...
大数据学习栈记——Neo4j的安装与使用
本文介绍图数据库Neofj的安装与使用,操作系统:Ubuntu24.04,Neofj版本:2025.04.0。 Apt安装 Neofj可以进行官网安装:Neo4j Deployment Center - Graph Database & Analytics 我这里安装是添加软件源的方法 最新版…...
STM32F4基本定时器使用和原理详解
STM32F4基本定时器使用和原理详解 前言如何确定定时器挂载在哪条时钟线上配置及使用方法参数配置PrescalerCounter ModeCounter Periodauto-reload preloadTrigger Event Selection 中断配置生成的代码及使用方法初始化代码基本定时器触发DCA或者ADC的代码讲解中断代码定时启动…...
Android15默认授权浮窗权限
我们经常有那种需求,客户需要定制的apk集成在ROM中,并且默认授予其【显示在其他应用的上层】权限,也就是我们常说的浮窗权限,那么我们就可以通过以下方法在wms、ams等系统服务的systemReady()方法中调用即可实现预置应用默认授权浮…...
全面解析各类VPN技术:GRE、IPsec、L2TP、SSL与MPLS VPN对比
目录 引言 VPN技术概述 GRE VPN 3.1 GRE封装结构 3.2 GRE的应用场景 GRE over IPsec 4.1 GRE over IPsec封装结构 4.2 为什么使用GRE over IPsec? IPsec VPN 5.1 IPsec传输模式(Transport Mode) 5.2 IPsec隧道模式(Tunne…...
音视频——I2S 协议详解
I2S 协议详解 I2S (Inter-IC Sound) 协议是一种串行总线协议,专门用于在数字音频设备之间传输数字音频数据。它由飞利浦(Philips)公司开发,以其简单、高效和广泛的兼容性而闻名。 1. 信号线 I2S 协议通常使用三根或四根信号线&a…...
【网络安全】开源系统getshell漏洞挖掘
审计过程: 在入口文件admin/index.php中: 用户可以通过m,c,a等参数控制加载的文件和方法,在app/system/entrance.php中存在重点代码: 当M_TYPE system并且M_MODULE include时,会设置常量PATH_OWN_FILE为PATH_APP.M_T…...
python爬虫——气象数据爬取
一、导入库与全局配置 python 运行 import json import datetime import time import requests from sqlalchemy import create_engine import csv import pandas as pd作用: 引入数据解析、网络请求、时间处理、数据库操作等所需库。requests:发送 …...
学习一下用鸿蒙DevEco Studio HarmonyOS5实现百度地图
在鸿蒙(HarmonyOS5)中集成百度地图,可以通过以下步骤和技术方案实现。结合鸿蒙的分布式能力和百度地图的API,可以构建跨设备的定位、导航和地图展示功能。 1. 鸿蒙环境准备 开发工具:下载安装 De…...
Axure 下拉框联动
实现选省、选完省之后选对应省份下的市区...
