普冉(PUYA)单片机开发笔记(9): FLASH 读写

概述

单片机的 ROM 容量虽然不大,PY32F003 有 64K 字节的 ROM,但实际应用中会在 MCU 中存储持久化的数据,例如:在物联网应用中,需要把物模型持久化,作为非易失性数据,掉电了也要保存。这就要用到在 FLASH 保存这些数据。

PY32F003 支持 FLASH 读写。

PY32F003 的 FLASH 写入支持“按页写入”、“按扇区写入”和“全部写入”三种方式,实用中常会用到前两种方式。PY32F003 的 FLASH 页(Page)是一块 128 字节的 ROM 区域,可以直接寻址;扇区(Sector)是一块 4096 字节的 ROM 区域,可以直接寻址。

数据手册上列出了 FLASH 的可寻址空间和访问限制(后面有数据手册的截图)。

好啦,干货奉上 ;)

实现代码

老样子,以下几个步骤,就能搞好。

在 main.h 中声明 FLASH 读写的函数

/** ----------------------------------------------------------------------------
* @name   : uint32_t* Flash_Read(void);
* @brief  : 从预设的FLASH地址读取1页(128 Byte)的数据,保存在 data 中
* @param  : None.
* @retval : NULL 或者数据的首地址
* @remark : 
*** ----------------------------------------------------------------------------
*/
uint32_t* Flash_Read(uint16_t* buf_size);

/** ----------------------------------------------------------------------------
* @name   : HAL_StatusTypeDef Flash_Write(uint32_t *data);
* @brief  : 从预设的FLASH地址写入1页(128Byte)的数据,data是要写入的数据
* @param  : [in] data: 要写入的数据指针.
* @param  : [in] data_size: 要写入的数据长度
* @retval : HAL_OK: 写入成功; 其它: 错误
* @remark : 上级函数必须检查操作返回值, 只有 HAL_OK 时才是有效数据
*** ----------------------------------------------------------------------------
*/
HAL_StatusTypeDef Flash_Write(uint32_t* data, const uint16_t data_size);

在 app_flash.c 中实现 FLASH 读写的函数

/**
 ******************************************************************************
 * @file    app_flash.c
 * @brief   Application level Flash read/write/erase codes.
 ******************************************************************************
 * @attention
 *
 * Copyright (c) 2023 CuteModem Intelligence.
 * All rights reserved.
 *
 * This software is licensed under terms that can be found in the LICENSE file
 * in the root directory of this software component.
 * If no LICENSE file comes with this software, it is provided AS-IS.
 *
 ******************************************************************************
 */
 
#include "main.h"

/* 定义用户数据在 FLASH 中存储的首地址, 这里指定了扇区号8, 页号256的首地址 */
#define FLASH_USER_START_ADDR 0x08007000
#define FLASH_USER_BUFF_SIZE  32

/* 定义用户数据: 32*4=128 Bytes, 1 page */
uint32_t WD_BUF[FLASH_USER_BUFF_SIZE] = { 0 };
uint32_t RD_BUF[FLASH_USER_BUFF_SIZE] = { 0 };

/** ----------------------------------------------------------------------------
* @name   : HAL_StatusTypeDef app_Flash_Erase(void);
* @brief  : 擦除从预设的FLASH地址的1页(128Byte)FLASH
* @param  : None.
* @retval : HAL_StatusTypeDef. 操作成功返回 HAL_OK, 错误返回错误码
* @remark : 上级函数必须检查操作返回值, 只有 HAL_OK 时才能继续操作
*** ----------------------------------------------------------------------------
*/
static HAL_StatusTypeDef app_Flash_Erase(void)
{
    uint32_t PAGEError = 0;
    FLASH_EraseInitTypeDef EraseInitStruct;

    EraseInitStruct.TypeErase   = FLASH_TYPEERASE_PAGEERASE;        // 擦写类型: 按页擦
    EraseInitStruct.PageAddress = FLASH_USER_START_ADDR;            // 擦写起始地址
    EraseInitStruct.NbPages  = sizeof(WD_BUF) / FLASH_PAGE_SIZE;    // 需要擦写的页数量
    if (HAL_FLASHEx_Erase(&EraseInitStruct, &PAGEError) != HAL_OK)  // 执行page擦除,PAGEError返回擦写错误的page,
        return HAL_ERROR;                                           // 返回0xFFFFFFFF, 表示擦写成功

    return HAL_OK;
}

/** ----------------------------------------------------------------------------
* @name   : HAL_StatusTypeDef Local_Flash_Check_Blank(void);
* @brief  : 查空 FLASH 中已被擦除的地址
* @param  : None.
* @retval : HAL_StatusTypeDef. 操作成功返回 HAL_OK, 错误返回错误码
* @remark : 上级函数必须检查操作返回值, 只有 HAL_OK 时才能继续操作
*** ----------------------------------------------------------------------------
*/
static HAL_StatusTypeDef Local_Flash_Check_Blank(void)
{
    uint32_t addr = 0;

    while (addr < sizeof(WD_BUF))
    {
        if (0xFFFFFFFF != HW32_REG(FLASH_USER_START_ADDR + addr))
            return HAL_ERROR;
        addr += 4;
    }
    
    return HAL_OK;
}

/** ----------------------------------------------------------------------------
* @name   : HAL_StatusTypeDef Local_Flash_Program(void);
* @brief  : 写预设地址的 FLASH
* @param  : None.
* @retval : HAL_StatusTypeDef. 操作成功返回 HAL_OK, 错误返回错误码
* @remark : 上级函数必须检查操作返回值, 只有 HAL_OK 时才能继续操作
*** ----------------------------------------------------------------------------
*/
static HAL_StatusTypeDef Local_Flash_Program(void)
{
    uint32_t flash_program_start = FLASH_USER_START_ADDR ;                 // flash写起始地址
    uint32_t flash_program_end = (FLASH_USER_START_ADDR + sizeof(WD_BUF)); // flash写结束地址
    uint32_t *src = (uint32_t *)WD_BUF;                                    // 数组指针

    while (flash_program_start < flash_program_end)
    {
        if (HAL_FLASH_Program(FLASH_TYPEPROGRAM_PAGE, 
                              flash_program_start, 
                              src) != HAL_OK)
        {
            return HAL_ERROR;
        }

        flash_program_start += FLASH_PAGE_SIZE; //flash 起始指针指向下一个 page
        src += FLASH_PAGE_SIZE / 4;             //更新数据指针
    }
    
    return HAL_OK;
}

/** ----------------------------------------------------------------------------
* @name   : HAL_StatusTypeDef Flash_Read(void);
* @brief  : 校验预设的FLASH地址开始的1页(128Byte)的数据
* @param  : None.
* @retval : HAL_StatusTypeDef. 操作成功返回 HAL_OK, 错误返回错误码。
* @remark : 上级函数必须检查操作返回值, 只有 HAL_OK 时才是有效数据
*** ----------------------------------------------------------------------------
*/
static HAL_StatusTypeDef Local_Flash_Verify(void)
{
    uint32_t addr = 0;

    while (addr < sizeof(WD_BUF))
    {
        if (WD_BUF[addr / 4] != HW32_REG(FLASH_USER_START_ADDR + addr))
        {
          return HAL_ERROR;
        }
        addr += 4;
    }
    
    return HAL_OK;
}

/** ----------------------------------------------------------------------------
* @name   : Local_Flash_Print(const uint32_t* data, const uint16_t data_size);
* @brief  : 打印 data_size 个 data
* @param  : None.
* @retval : NULL 或者数据的首地址
* @remark : 
*** ----------------------------------------------------------------------------
*/
static void Local_Flash_Print(const uint32_t* data, const uint16_t data_size)
{
    printf("%d bytes of data in FLASH:\r\n", data_size*sizeof(uint32_t));
    for(int i = 0; i < data_size; i++)
    {
        printf("0x%08X ", data[i]);
        if((i + 1) % 4 == 0) printf("\r\n");
    }
    printf("\r\n");
}

/** ----------------------------------------------------------------------------
* @name   : uint32_t* Flash_Read(void);
* @brief  : 从预设的FLASH地址读取1页(128Byte)的数据,保存在 data 中
* @param  : None.
* @retval : NULL 或者数据的首地址
* @remark : 
*** ----------------------------------------------------------------------------
*/
uint32_t* Flash_Read(uint16_t* buf_size)
{
    uint32_t addr = 0;
    
    while (addr < sizeof(RD_BUF))
    {
        RD_BUF[addr / 4] = HW32_REG(FLASH_USER_START_ADDR + addr);
        addr += 4;
    }
    
    *buf_size = FLASH_USER_BUFF_SIZE;
    
    Local_Flash_Print((uint32_t*)(&RD_BUF[0]), *buf_size);
    
    return (uint32_t*)(&RD_BUF[0]);
}

/** ----------------------------------------------------------------------------
* @name   : Flash_Write(uint32_t* data, const uint16_t data_size);
* @brief  : 从预设的FLASH地址写入1页(128Byte)的数据,data是要写入的数据
* @param  : [in] data, 一个 uint32_t 的数据缓冲区的首地址, 128字节
* @retval : HAL_HandleTypeDef. 操作成功返回 HAL_OK, 错误则返回错误码。
* @remark : 上级函数必须检查操作返回值, 只有 HAL_OK 时才是正确的操作
*** ----------------------------------------------------------------------------
*/
HAL_StatusTypeDef Flash_Write(uint32_t* data, const uint16_t data_size)
{
    HAL_StatusTypeDef res = HAL_BUSY;
    int i = 0;
    uint16_t w_size = data_size;
    
    if(data_size > FLASH_USER_BUFF_SIZE) w_size = FLASH_USER_BUFF_SIZE;
    
    for(i = 0; i < w_size; i++) WD_BUF[i] = data[i];
    for(i = w_size; i< FLASH_USER_BUFF_SIZE; i++) WD_BUF[i] = 0U;
    
    res = HAL_FLASH_Unlock(); // 解锁 FLASH
    if(res != HAL_OK ) return res;
    
    res = app_Flash_Erase();
    if(res != HAL_OK ) return res;
    
    res = Local_Flash_Check_Blank();
    if(res != HAL_OK ) return res;
    
    res = Local_Flash_Program();
    if(res != HAL_OK ) return res;
    
    res = Local_Flash_Verify();
    if(res != HAL_OK ) return res;
    
    res = HAL_FLASH_Lock();   // 锁定 FLASH
    if(res != HAL_OK ) return res;
    
    return HAL_OK;
}

说明:

  1. 参考厂家的数据手册,选择 FLASH 地址空间的最后一个扇区( Sector 8)的第一页作为存储区域。通过 #define FLASH_USER_START_ADDR 0x08007000 进行定义,同时定义 FLASH_USER_BUFF_SIZE 为 32,把这个页占满。
  2. 定义写数据缓冲区(WD_BUF )和读数据缓冲区(RD_BUF)变量,缓冲区长度设置为 128 字节:32 个 uint32_4 类型整数,占用 FLASH 容量等于 32 X 4 = 128。
  3. 完成“擦写”、“查空”、“编程(写入)”和“校验” 4 个函数的编码。
  4. 写了一个测试用的格式化打印函数。以上这五个函数是 FLASH 操作的底层函数,对上级调用 FLASH 读写功能的函数不必可见,因此都定义成了 static 类型。
  5. 完成 Flash_Read 函数。直接访问预定义的 FLASH 内存空间首地址开始的一页数据,存放在缓冲区 RD_BUF 中,Flash_Read 函数返回 RD_BUF 的首地址。
  6. 完成 Flash_Write 函数。FLASH 写的顺序是“解锁 --> 擦除 --> 查空 --> 写入 --> 校验 --> 锁定”,本文采取了全部检查的方式进行写操作,只要中间任何一步出错就返回。如果“信得过” MCU 的话,“查空”这一步可以省区。

在 main.c 中完成调用

/**
 ******************************************************************************
 * @file    main.c
 * @brief   Main program entry.
 ******************************************************************************
 * @copyright
 *
 * Copyright (c) 2023 CuteModem Intelligence.
 * All rights reserved.
 *
 * This software is licensed under terms that can be found in the LICENSE file
 * in the root directory of this software component.
 * If no LICENSE file comes with this software, it is provided AS-IS.
 *
 ******************************************************************************
 */

/* Private includes*/
#include "main.h"
#include <stdio.h>
/* Private define ------------------------------------------------------------*/
/* Private variables ---------------------------------------------------------*/
struct Student_t {
    uint32_t pri_id;
    char stu_id[13];
    char name[25];
    uint8_t grade;
    uint8_t class;
    char performance[17];
};
    
/* Private function prototypes -----------------------------------------------*/
/* Private user code ---------------------------------------------------------*/
/* Private macro -------------------------------------------------------------*/
/* Private function prototypes -----------------------------------------------*/

/**
* -------------------------------------------------------------------------
* @file   : int main(void)
* @brief  : main函数
* @param  : 无
* @retval : 无限循环,无返回值
* @remark : 
* -------------------------------------------------------------------------
*/
int main(void)
{
    HAL_Init();             // systick初始化
    SystemClock_Config();   // 配置系统时钟
    GPIO_Config();
    
    if(USART_Config() != HAL_OK) Error_Handler();         
    printf("[SYS_INIT] Debug port initilaized.\r\n");

    printf("\r\n+---------------------------------------+"
           "\r\n|        PY32F003 MCU is ready.         |"
           "\r\n+---------------------------------------+"
           "\r\n         10 digits sent to you!          "
           "\r\n+---------------------------------------+"
           "\r\n");
    HAL_Delay(0);
    if (DBG_UART_Start() != HAL_OK) Error_Handler();
    HAL_Delay(0);
    
    struct Student_t stu1, stu2;
    uint16_t stu_rec_size = sizeof(stu1);
    uint16_t data_size = 0;
    uint32_t* flash_data = Flash_Read(&data_size);
    stu2 = *((struct Student_t *)flash_data);
    
    printf("Previous saved student profile: %d bytes \r\n"
           "  Primary ID = %d\r\n"
           "  Student ID = '%s'\r\n"
           "        Name = '%s'\r\n"
           "       Grade = %d\r\n"
           "       Class = %d\r\n"
           " Performance = '%s'\r\n", 
        data_size * 4,
        stu2.pri_id, 
        stu2.stu_id, 
        stu2.name, 
        stu2.grade, 
        stu2.class,
        stu2.performance);

    memset(&stu1, 0, sizeof(stu1));
    stu1.pri_id = 24506;
    strcpy(stu1.stu_id, "CSTU-2020-02");
    strcpy(stu1.name, "Zhang Lao 4");
    stu1.grade = 2;
    stu1.class = 11;
    strcpy(stu1.performance,"A+A-B+B++");

    printf("Current student profile: %d bytes.\r\n"
           " Primary ID = %d\r\n"
           " Student ID = '%s'\r\n"
           "       Name = '%s'\r\n"
           "      Grade = %d\r\n"
           "      Class = %d\r\n"
           " Performance = '%s'\r\n", 
        stu_rec_size,
        stu1.pri_id, 
        stu1.stu_id, 
        stu1.name, 
        stu1.grade, 
        stu1.class,
        stu1.performance);
        
    if(Flash_Write((uint32_t*)(&stu1), stu_rec_size) != HAL_OK ) 
    {
        printf("Data updating error.\r\n");
        Error_Handler();
    }
    
    while (1)
    { 
        BSP_LED_Toggle(LED3);
        HAL_Delay(500);
    }
}

/**
* -------------------------------------------------------------------------
* @brief  : void Error_Handler(void)
* @detail : 错误陷阱函数,提示错误,然后死循环
* @param  : 无
* @retval : 无
* @remark : 
* -------------------------------------------------------------------------
*/
void Error_Handler(void)
{
    Debug_Info("[__ERROR_] System halt.");
    while (1) {}
}


#ifdef USE_FULL_ASSERT
/**
 * @brief  Reports the name of the source file and the source line number
 *         where the assert_param error has occurred.
 * @param  file: pointer to the source file name
 * @param  line: assert_param error line source number
 * @retval None
 */
void assert_failed(uint8_t *file, uint32_t line)
{
    /* User can add his own implementation to report the file name and line number,
       tex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) */
}
#endif /* USE_FULL_ASSERT */

说明:

  1. 首先定义了一个简单的“学生档案”的结构体,结构体的尺寸不能超过128字节。考虑到可能的“四字节”对齐,这个结构体的尺寸不一定等于各个元素尺寸的算术和,需要使用 sizeof() 运算符事先打印一下这个结构体的尺寸。
  2. 然后是和以前的实验相同的时钟选择、LED 和串口的初始化。
  3. 从 FLASH 中读取上一次写入的数据(uint32_t*),使用强制类型转换,得到学生档案结构体的数据,并打印出来。
  4. 更新这个结构体一些元素的数值,使用强制类型转换,重新写入 FLASH。

实验结果

利用主程序的顺序,用硬代码方式修改第3步的数据,观察打印结果。如果修改了数据,在下一次编译运行时可以看到打印信息的相应变化。连续两次运行,然后 RESET 一次,运行结果如下所示:第二次编译前,将该学生的评分从“A+A-B+B+”修改成了“A+A-B+B++”(多了一个+号),可以看出数据已经更新。

[SYS_INIT] Debug port initilaized.

+---------------------------------------+
|        PY32F003 MCU is ready.         |
+---------------------------------------+
         10 digits sent to you!          
+---------------------------------------+
1234567890
128 bytes of data in FLASH:
0x00005FBA 0x55545343 0x3230322D 0x32302D30 
0x61685A00 0x4C20676E 0x34206F61 0x00000000 
0x00000000 0x00000000 0x0B020000 0x2D412B41 
0x2B422B42 0x00000000 0x00000000 0x00000000 
0x0800331C 0x00000001 0x0800331C 0x08000469 
0x08002852 0x21000000 0x2000032C 0x2000032C 
0x00000040 0x00000000 0x0000015C 0x2000032C 
0x200007F0 0x200007F1 0x00000001 0x08001FE5 

Previous saved student profile: 128 bytes 
  Primary ID = 24506
  Student ID = 'CSTU-2020-02'
        Name = 'Zhang Lao 4'
       Grade = 2
       Class = 11
 Performance = 'A+A-B+B+'
Current student profile: 64 bytes.
 Primary ID = 24506
 Student ID = 'CSTU-2020-02'
       Name = 'Zhang Lao 4'
      Grade = 2
      Class = 11
 Performance = 'A+A-B+B+'
[SYS_INIT] Debug port initilaized.

+---------------------------------------+
|        PY32F003 MCU is ready.         |
+---------------------------------------+
         10 digits sent to you!          
+---------------------------------------+
1234567890
128 bytes of data in FLASH:
0x00005FBA 0x55545343 0x3230322D 0x32302D30 
0x61685A00 0x4C20676E 0x34206F61 0x00000000 
0x00000000 0x00000000 0x0B020000 0x2D412B41 
0x2B422B42 0x00000000 0x00000000 0x00000000 
0x0800331C 0x00000001 0x0800331C 0x08000469 
0x08002852 0x21000000 0x2000032C 0x2000032C 
0x00000040 0x00000000 0x0000015C 0x2000032C 
0x200007F0 0x200007F1 0x00000001 0x08001FE5 

Previous saved student profile: 128 bytes 
  Primary ID = 24506
  Student ID = 'CSTU-2020-02'
        Name = 'Zhang Lao 4'
       Grade = 2
       Class = 11
 Performance = 'A+A-B+B+'
Current student profile: 64 bytes.
 Primary ID = 24506
 Student ID = 'CSTU-2020-02'
       Name = 'Zhang Lao 4'
      Grade = 2
      Class = 11
 Performance = 'A+A-B+B++'
[SYS_INIT] Debug port initilaized.

+---------------------------------------+
|        PY32F003 MCU is ready.         |
+---------------------------------------+
         10 digits sent to you!          
+---------------------------------------+
1234567890
128 bytes of data in FLASH:
0x00005FBA 0x55545343 0x3230322D 0x32302D30 
0x61685A00 0x4C20676E 0x34206F61 0x00000000 
0x00000000 0x00000000 0x0B020000 0x2D412B41 
0x2B422B42 0x0000002B 0x00000000 0x00000000 
0x0800331C 0x00000001 0x0800331C 0x08000469 
0x08002852 0x21000000 0x2000032C 0x2000032C 
0x00000040 0x00000000 0x0000015C 0x2000032C 
0x200007F0 0x200007F1 0x00000001 0x08001FE5 

Previous saved student profile: 128 bytes 
  Primary ID = 24506
  Student ID = 'CSTU-2020-02'
        Name = 'Zhang Lao 4'
       Grade = 2
       Class = 11
 Performance = 'A+A-B+B++'
Current student profile: 64 bytes.
 Primary ID = 24506
 Student ID = 'CSTU-2020-02'
       Name = 'Zhang Lao 4'
      Grade = 2
      Class = 11
 Performance = 'A+A-B+B++'

总结

  • 读写 PY32F003 的 FLASH 很简单,利用 HAL 库,按照规定的顺序(解锁-擦除-查空-写入-校验-锁定)操作即可完成 FLASH 的写入。FLASH 解锁要顺序地使用既定的两个标志字。
// Defined in py32f003x8.h
#define FLASH_KEY1_Msk                    (0x45670123UL << FLASH_KEY1_Pos)     /*!< 0x45670123 */
#define FLASH_KEY1                        FLASH_KEY1_Msk                       /*!< Flash program erase key1 */
#define FLASH_KEY2_Pos                    (0U)
#define FLASH_KEY2_Msk                    (0xCDEF89ABUL << FLASH_KEY2_Pos)     /*!< 0xCDEF89AB */
#define FLASH_KEY2                        FLASH_KEY2_Msk                       /*!< Flash program erase key2: used with FLASH_PEKEY1

// Defined in py32f0xx_hal_flash.c
/**
  * @brief  Unlock the FLASH control register access.
  * @retval HAL Status
  */
HAL_StatusTypeDef HAL_FLASH_Unlock(void)
{
  HAL_StatusTypeDef status = HAL_OK;

  if (READ_BIT(FLASH->CR, FLASH_CR_LOCK) != 0x00U)
  {
    /* Authorize the FLASH Registers access */
    WRITE_REG(FLASH->KEYR, FLASH_KEY1);
    WRITE_REG(FLASH->KEYR, FLASH_KEY2);

    /* verify Flash is unlock */
    if (READ_BIT(FLASH->CR, FLASH_CR_LOCK) != 0x00U)
    {
      status = HAL_ERROR;
    }
  }

  return status;
}
  • FLASH 读操作时不需要 Unlock/Lock,可以当作“普通”的内存访问读取 FLASH 相应地址的数据。
#define HW32_REG(ADDRESS) (*((volatile unsigned int *)(ADDRESS)))
  • FLASH 写必须被视为一个完整的过程,五个步骤要完整地、顺序地执行,不能中断,要不的话会导致系统卡死。在主程序的循环中运行 FLASH 写操作时,如果还有其它定时器中断服务程序也要执行 FLASH 写操作的话,要用类似互斥量的方式防止某一个正在进行的 FLASH 写操作的过程被中断(FLASH 读写过程中卡死的情况还是不少的,有兴趣的可以多看一点数据手册的第4章中的内容)。

探索性的开发,差错难免。谬误之处,欢迎在评论区批评指正。

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

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

相关文章

图文教程:stable-diffusion的基本使用教程 txt2img(多图)

之前我介绍了SD的安装过程&#xff0c;那么这篇将介绍怎么使用SD 使用模型 SD安装好之后&#xff0c;我们只有一个默认的模型。这个模型很难满足我们的绘图需求&#xff0c;那么有2种方法。 1是自己训练一个模型&#xff08;有门槛&#xff09;2是去网站上找一个别人练好的模…

宇视科技视频监控 main-cgi 文件信息泄露漏洞复现

0x01 产品简介 宇视(Uniview)高清网络摄像机是一种高性能的网络摄像机,它可以通过网络进行视频传输和监控。该摄像机采用先进的视频技术,具有高清晰度、低照度、宽动态等特点,能够提供高质量的视频图像。 0x02 漏洞概述 宇视(Uniview)高清网络摄像机存在信息泄露漏洞…

【桌面应用开发】Tauri是什么?基于Rust的桌面应用

自我介绍 做一个简单介绍&#xff0c;酒架年近48 &#xff0c;有20多年IT工作经历&#xff0c;目前在一家500强做企业架构&#xff0e;因为工作需要&#xff0c;另外也因为兴趣涉猎比较广&#xff0c;为了自己学习建立了三个博客&#xff0c;分别是【全球IT瞭望】&#xff0c;【…

Edge浏览器版本更新后Copilot按钮消失的解决应对方式

需求背景 今天突然发现Edge浏览器右上角的Copilot按钮不见了&#xff0c;排查了一下&#xff0c;发现可能是浏览器自动升级到120版本后&#xff0c;关闭了右上角的Copilot按钮。案发现场如下&#xff1a; 原因如下&#xff1a; 说实话&#xff0c;人家这个插件还是很好用的&…

OpenCV-opencv下载安装和基本操作

文章目录 一、实验目的二、实验内容三、实验过程OpenCV-python的安装与配置python下载和环境配置PIP镜像安装Numpy安装openCV-python检验opencv安装是否成功 openCV-python的基本操作图像输入和展示以及写出openCV界面编程单窗口显示多图片鼠标事件键盘事件滑动条事件 四、实验…

持续集成交付CICD:GitLabCI上传Nexus制品

目录 一、实验 1.GitLabCI上传Nexus制品 2.优化GitLabCI&#xff08;引用系统变量&#xff09; 3.添加if条件判断项目类型 4.优化GitLabCI&#xff08;模板类&#xff09; 二、问题 1.GitLabCI获取jar文件失败 2. GitLabCI获取流水线项目命名空间失败 3.GItLab Packag…

【产品经理】产品的实现,需要做好战略规划

产品的实现需要做好产品规划&#xff0c;而产品的规划决定了产品的方向。本文从战略规划的重要性、产品定位、设计产品架构图三个方向&#xff0c;详细地为大家梳理了产品实现的前期准备。 我们知晓了如何去发掘问题&#xff0c;并找到解决方案。 可对于问题的处理&#xff0c…

【Jeecg Boot 3 - 第二天】1.1、后端 docker-compose 部署 JEECGBOOT3

一、场景 二、实战 ▶ 2.1 修改配置文件 &#xff1e; 目的一&#xff1a;将 dev 变更为生产环境 prod &#xff1e; 目的二&#xff1a;方便spring项目调用docker同个network下的redis和mysql ▶ 2.2 编写dockerfile ▶ 2.3 编写docker-compose.yaml ▶ 2.4 打…

裸机单片机适用的软件架构

单片机通常分为三种工作模式&#xff0c;分别是 1、前后台顺序执行法 2、操作系统 3、时间片轮询法 1、前后台顺序执行法 利用单片机的中断进行前后台切换&#xff0c;然后进行任务顺序执行&#xff0c;但其实在…

[排序篇] 冒泡排序

目录 一、概念 二、冒泡排序 2.1 冒泡降序(从大到小排序) 2.2 冒泡升序(从小到大排序) 三、冒泡排序应用 总结 一、概念 冒泡排序核心思想&#xff1a;每次比较两个相邻的元素&#xff0c;如果它们不符合排序规则&#xff08;升序或降序&#xff09;则把它们交换过来。…

Web自动化测试工具起到哪些重要作用

随着互联网的迅猛发展&#xff0c;Web应用程序已经成为企业不可或缺的一部分。为了确保Web应用的质量和可靠性&#xff0c;Web自动化测试工具变得至关重要。以下是Web自动化测试工具在软件开发生命周期中发挥的作用&#xff1a; 1. 提高测试效率和速度 Web自动化测试工具可以快…

【媒体开发】利用FFMPEG进行推拉流

1. 下载并启动媒体服务 MediaMTX&#xff0c;也即之前的rtsp-simple-server&#xff0c;是一个即用型、零依赖的实时媒体服务器和媒体代理&#xff0c;允许发布、读取、代理和记录视频和音频流。 从Releases bluenviron/mediamtx GitHub找到最新版&#xff0c;下载对应平台…

Web UI 自动化 元素定位利器

序 元素定位&#xff0c;对于 Web UI 自动化而言&#xff0c;绝对是大家成长道路上的一道绊脚石。 很多初学者&#xff0c;都“死”在了元素定位上&#xff0c;从而失去了学习的兴趣。导致职业规划不得不半途而废~ 那么&#xff0c;今天&#xff0c;我们就使用 Katalon Stu…

python实战演练之迎接冬至的第一场雪

写在前面 WINTER IS COMING Python实现大雪纷飞的效果&#xff0c;完整代码在文末哦~ 准备开始 WINTER IS COMING Python是一种高级编程语言&#xff0c;Turtle是Python的一个图形化模块&#xff0c;它可以帮助学习者更好地理解编程概念&#xff0c;同时可以进行图形化编程。 …

DS二叉排序树之删除

Description 给出一个数据序列&#xff0c;建立二叉排序树&#xff0c;并实现删除功能 对二叉排序树进行中序遍历&#xff0c;可以得到有序的数据序列 Input 第一行输入t&#xff0c;表示有t个数据序列 第二行输入n&#xff0c;表示首个序列包含n个数据 第三行输入n个数据…

修改汽车的控制系统实现自动驾驶,基于一个开源的汽车驾驶辅助系统实现全自动驾驶

修改汽车的控制系统实现自动驾驶,基于一个开源的汽车驾驶辅助系统实现全自动驾驶。 自动驾驶汽车依靠人工智能、视觉计算、雷达、监控装置和全球定位系统协同合作,让电脑可以在没有任何人类主动的操作下,自动安全地操作机动车辆。 演示视频: Openpilot :一个开源的汽车驾…

音乐制作工具 Ableton Live 12中文最新 for Mac

Ableton Live 12 Mac具有直观的界面和强大的功能&#xff0c;使得音乐制作变得更加简单和高效。它支持实时录制、编辑和混音&#xff0c;用户可以在创作过程中随时进行修改和调整。此外&#xff0c;该软件还提供了各种音频效果、虚拟乐器和采样器&#xff0c;使用户可以创建出更…

el-table的复选框占满全格

el-table的复选框格子很小每次点击都点不到&#xff0c;又不想设置行点击&#xff0c;因为每次复制内容都会选中&#xff0c;实现效果是点击el-table的复选框单元格就可以选中 <template><div style"width: 60vw; margin: 10px;"><el-table :data&quo…

1 CPU实现的基本框图

汇编语言 && 指令格式 CPU设计的框架&#xff1a;三级流水线 ROM存放指令和数据&#xff0c;大端模式&小端模式&#xff0c;地址对齐 取指 译码&#xff1a; 执行&#xff1a; 汇编语言 & 指令格式 流水线实现工作机制 模块功能划分&接口信号 参考…

新工科:数据科学与大数据技术实验中心解决方案,赋能高校新工科数智人才培养

随着数字经济蓬勃发展&#xff0c;数字化产业和产业数字化成为就业增长新动能。据人瑞人才与德勤调研显示&#xff0c;未来3年&#xff0c;数字产业化企业最需要运营人员和开发人员&#xff08;包括大数据开发工程师、数据建模开发工程师等&#xff09;&#xff0c;其次是数据分…