96SEO 2026-02-19 11:18 7
如STM32F103系列#xff0c;内部Flash最多只能达到512KByte涉及大量数据需要储存时例如使用了屏幕作为显示设备常常需要存储图片、动画等数据单靠单片机内部的Flash往往是不够用的。

如STM32F103系列内部Flash最多只能达到512KByte假设要储存240*240分辨率、64K彩色图片只够存储4张左右。
如果使用外置储存器将图片等其他数据放置在外置储存器内部Flash只储存程序就能减小内部Flash的需求降低成本。
Flash。
NAND的串行结构使得其容量很容易做的很大SD卡、U盘、硬盘大都采用该类Flash但是其读取速度却比不上并行结构的NOR
Flash且可靠性要差些一旦出现数据块坏点是不可逆、无法修复的。
由于其数据存储原理Flash在写入新的数据之前都需要将数据地址所在的块擦除NOR
Flash按其数据传输方式的不同可分为并口传输与串口传输。
STM32的并口传输需使用FSMC接口虽然其读写速度很快但对于100PIN脚以下的封装是不带FSMC功能的。
所以使用更多的是串口传输方式。
串口方式一般采用的是SPI通讯。
W25Q系列Flash是Winbond台湾华邦科技生产的SPI
Flash系列是单片机开发中比较常用的外置Flash。
其支持标准四线SPI、Dual
SPI、QPI其时钟频率分别可达到104MHz、208MHz、416MHz。
对于STM32F103系列其主频最高72MHzSPI通信速率最高18Mbps所以标准SPI就已经是足够F103系列单片机使用了。
这里我使用W25Q128FV来讲解Flash的使用。
先来了解其引脚定义上面展示的是SOP8封装还有SOP16封装的功能都是差不多的。
Flash的片选引脚。
当/CS高时Flash的串行数据输出(DO或IO0、IO1、IO2、IO3)引脚处于高阻抗此时设备功耗将处于待机水平除非正在进行内部擦除、程序或写入状态寄存器周期。
当/CS为低电平Flash将被选中功耗将增加到活动水平并且可以向该设备写入指令和从该设备读取数据。
启动后/CS必须从高电平转换到低电平才能接受新的指令。
Out)一般连接到单片机SPI接口的数据输入端即MISO。
IO1是其复用功能当启用SPI四位传输模式时该引脚功能为IO1.
/WP:写保护(WP)引脚。
可以用来防止状态寄存器被写入。
与状态寄存器的块保护(CMP,
BP1和BPO)位和状态寄存器保护(SRP)位一起使用小到4KB扇区或整个内存阵列都可以被硬件保护。
/WP引脚低电平有效。
当状态寄存器2的QE位设置为Quad
I/O时/WP引脚功能不可用因为该引脚用于IO2。
如果不想使用该功能可以直接将该引脚接VCC。
In)一般连接到单片机SPI接口的数据输出端即MOSI。
IO0是其复用功能当启用SPI四位传输模式时该引脚为IO0
/HOLD/RESET:/HOLD能让设备主动暂停数据传输。
当/HOLD低时而/CS低时DO引脚将处于高阻抗Dl和CLK引脚上的信号将被忽略。
当/HOLD调高时设备可以恢复运行。
当多个设备共享相同的SPl信号时/HOLD就能发挥作用。
/RESET引脚用于设备复位。
注意如果在写入数据时复位可能会造成数据丢失。
所以如果不需要给Flash复位该引脚常常直接与VCC相接。
不管何种存储器在进行数据读写时都需要知道数据的地址。
数据存储在寄存器中所以数据的地址即寄存器地址。
我们来看看W25Q128的内部原理图。
存储单元的最小单位为一个寄存器每个寄存器可存储1个字节的数据。
每256个寄存器组成一页(Page)也就是一页能存储256Byte数据
每16页组成一个扇区(Sector)一个扇区能储存16x2564096Byte数据(近似4KB)。
比如扇区0的数据地址范围为000000
每16个扇区又组成一个块(Block)一个块能储存4096x1665536Byte数据(近似64K)。
例如块0的数据地址范围为000000
整个存储单元共256个块所以其总存储容量为256x6553616777216Byte数据近似为16MByte。
数据地址范围为000000
不管何种外设都是通过发送命令与数据来控制的。
Flash也不例外所以需要知道如何使用Flash只需在其技术手册上找到其命令表即可。
1.SPI2初始化。
为了将读出来的数据显示出来这里我使用串口将数据传输到电脑上。
所以对usart1也初始化。
GPIO_InitStructure;RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,
//使能GPIOB的时钟GPIO_InitStructure.GPIO_Pin
//PA13为SCK时钟PA15为MISOGPIO_InitStructure.GPIO_Speed
//速度50MHzGPIO_InitStructure.GPIO_Mode
GPIO_InitStructure);GPIO_InitStructure.GPIO_Pin
//PA14为MISOGPIO_InitStructure.GPIO_Speed
//速度50MHzGPIO_InitStructure.GPIO_Mode
GPIO_InitStructure);GPIO_InitStructure.GPIO_Pin
//PA12为片选GPIO_InitStructure.GPIO_Speed
//速度50MHzGPIO_InitStructure.GPIO_Mode
GPIO_InitStructure);SPI_InitTypeDef
SPI_InitStructure;RCC_APB1PeriphClockCmd(RCC_APB1Periph_SPI2,
//使能SPI时钟SPI_InitStructure.SPI_Direction
SPI_Direction_2Lines_FullDuplex;//设置双向双线全双工SPI_InitStructure.SPI_Mode
//设置为SPI主站SPI_InitStructure.SPI_DataSize
//设置为8位帧结构SPI_InitStructure.SPI_CPOL
//串行时钟的稳态为时钟高SPI_InitStructure.SPI_CPHA
//位捕获的时钟活动沿为第1个时钟沿SPI_InitStructure.SPI_NSS
//指定NSS信号由软件控制SPI_InitStructure.SPI_BaudRatePrescaler
//波特率预分频值SPI_InitStructure.SPI_FirstBit
//数据位从MSB开始SPI_InitStructure.SPI_CRCPolynomial
//使能SPI2GPIO_SetBits(GPIOB,GPIO_Pin_12);
GPIO_InitStructure;RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,
ENABLE);GPIO_InitStructure.GPIO_Pin
//PA9为USART1_TX将这个GPIO初始化GPIO_InitStructure.GPIO_Speed
//速度50MHzGPIO_InitStructure.GPIO_Mode
GPIO_InitStructure);GPIO_InitStructure.GPIO_Pin
//PA10为USART_RX将这个GPIO初始化GPIO_InitStructure.GPIO_Speed
//速度50MHzGPIO_InitStructure.GPIO_Mode
GPIO_InitStructure);USART_InitTypeDef
//定义USART配置结构体RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,
//打开USART1时钟USART_InitStructure.USART_BaudRate
//波特率USART_InitStructure.USART_WordLength
USART_InitStructure.USART_StopBits
//停止位数目USART_InitStructure.USART_Parity
偶SART_Parity_Odd奇USART_InitStructure.USART_HardwareFlowControl
USART_HardwareFlowControl_None;
//硬件流控制模式USART_InitStructure.USART_Mode
USART_ITConfig(USART1,USART_IT_RXNE,
NVIC_InitStructure;NVIC_InitStructure.NVIC_IRQChannel
//中断通道NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority3;
//抢占优先级NVIC_InitStructure.NVIC_IRQChannelSubPriority
//子优先级NVIC_InitStructure.NVIC_IRQChannelCmd
//使能中断NVIC_Init(NVIC_InitStructure);
2.发送命令或者读写数据都是通过最基本的发送数据、接受数据函数来实现。
为了不出现数据丢失每次发送数据前都需要判断上次发送的数据是否已经发送玩这可以通过相关标志位来判断;同样为了不出现数据重复每次接收数据前都要判断接收缓存区是否为空。
//检查指定的SPI标志位设置与否:发送缓存空标志位RESET表示正在发送数据{Wait;//循环计数200计数200此大概20us不管是否标志位为空都退出等待}SPI_I2S_ClearFlag(SPI2,
//清除发送完成标志位SPI_I2S_SendData(SPI2,Data);
Wait0;SPI_I2S_SendData(SPI2,0xff);
//发送0x00产生时钟信号用来接收数据也可以发送其他无响应的命令while
//检查指定的SPI标志位设置与否:接受缓存非空标志位{Wait;
需要注意在接受数据的函数中之所以在SPI_I2S_ReceiveData(SPI2)函数之前要使用SPI_I2S_SendData(SPI2,0xff)函数是为了产生时钟信号。
SPI采用的是主从通信结构时钟信号只能由主设备产生主设备发送数据的过程中会产生时钟信号但是从设备发送数据时并不能自己产生时钟信号所以就无法将数据一位一位发送出去同步通信必须依靠时钟信号保持时序一致那就只能依靠主设备产生时钟信号。
主设备发送的0xFF对从设备来说是无效的数据不会对该数据做出响应但是主设备发送0xFF这个数据的时候产生了时钟信号所以从设备就依靠这段时钟信号将数据发送给了主设备主设备接受会暂存在接收缓存寄存器中等接受到新的数据自动更新缓存器。
具体的通信时序可查阅W25Q128的技术手册W25Q128FV_PDF_数据手册_Datasheet这里就不一一列举了。
3.把W25Q128常用的命令封装成函数只要调用对应的函数就能实现命令的发送与数据的读写
GPIO_SetBits(GPIOB,GPIO_Pin_12)
GPIO_ResetBits(GPIOB,GPIO_Pin_12)
//低电平选中高电平取消选中/*****W25Q128常用命令定义*****/
//CS选中Flash_WriteData8(W25X_ReadStatusReg);
Flash_WriteData8(W25X_WriteStatusReg);
Flash_CS_L();Flash_WriteData8(W25X_WriteEnable);
Flash_WriteData8(W25X_WriteDisable);
Flash_CS_L();Flash_WriteData8(W25X_JedecDeviceID);
//高8位与低8位合并成16位与运算后赋值Flash_CS_H();return
Flash_WriteData8(W25X_ReadData);
Flash_WriteData8((ReadAddr16)0xff);
Flash_WriteData8((ReadAddr8)0xff);
Flash_WriteData8(ReadAddr0xff);
while((Flash_ReadSR()0x01)0x01);
等待BUSY位清空{}Flash_WriteData8(W25X_SectorErase);
Flash_WriteData8((Dst_Addr16)0xff);
Flash_WriteData8((Dst_Addr8)0xff);
Flash_WriteData8(Dst_Addr0xff);
while((Flash_ReadSR()0x01)0x01);
Flash_CS_L();Flash_WriteData8(W25X_PageProgram);
Flash_WriteData8((WriteAddr16)0xff);
Flash_WriteData8((WriteAddr8)0xff);
Flash_WriteData8(WriteAddr0xff);
for(i0;iNumByteToWrite;i){Flash_WriteData8(ARR1[i]);}
Flash_CS_L();Flash_WriteData8(W25X_PageProgram);
Flash_WriteData8((WriteAddr16)0xff);
Flash_WriteData8((WriteAddr8)0xff);
Flash_WriteData8(WriteAddr0xff);
4.主函数。
先读取设备ID然后将数组ARR1的数据写入Flash的扇区0再将扇区0的数据读取出来放在数组ARR2中通过串口将ARR2的数据显示到电脑串口调试助手。
ARR1[10]{0x01,0x02,0x03,0x04,0x05,0x06,0x07,0x08,0x09,0x0A};//随意填入几个元素后面将其元素写入Flash中
{Wait;//循环计数200计数200此大概20us不管是否标志位为空都退出等待}USART_ClearFlag(USART1,USART_FLAG_TC);jASCII[i];USART_SendData(USART1,j);}Wait0;while
{Wait;//循环计数200计数200此大概20us不管是否标志位为空都退出等待}USART_ClearFlag(USART1,USART_FLAG_TC);USART_SendData(USART1,10);
//USART1初始化--控制串口CH340Flash_Write_Enable();
RESET){}USART_SendData(USART1,(Data16)0xFF);while
RESET){}USART_SendData(USART1,(Data8)0xFF);Flash_EraseSector(0x000000);
扇区擦除完成);Flash_WritePage(0x000000,10);
数据写入成功);Flash_ReadSector(0x000000)
//检查指定的SPI标志位设置与否:发送缓存空标志位RESET表示正在发送数据{Wait;//循环计数200计数200此大概20us不管是否标志位为空都退出等待}USART_ClearFlag(USART1,USART_FLAG_TC);jARR2[i];USART_SendData(USART1,j);}
作为专业的SEO优化服务提供商,我们致力于通过科学、系统的搜索引擎优化策略,帮助企业在百度、Google等搜索引擎中获得更高的排名和流量。我们的服务涵盖网站结构优化、内容优化、技术SEO和链接建设等多个维度。
| 服务项目 | 基础套餐 | 标准套餐 | 高级定制 |
|---|---|---|---|
| 关键词优化数量 | 10-20个核心词 | 30-50个核心词+长尾词 | 80-150个全方位覆盖 |
| 内容优化 | 基础页面优化 | 全站内容优化+每月5篇原创 | 个性化内容策略+每月15篇原创 |
| 技术SEO | 基本技术检查 | 全面技术优化+移动适配 | 深度技术重构+性能优化 |
| 外链建设 | 每月5-10条 | 每月20-30条高质量外链 | 每月50+条多渠道外链 |
| 数据报告 | 月度基础报告 | 双周详细报告+分析 | 每周深度报告+策略调整 |
| 效果保障 | 3-6个月见效 | 2-4个月见效 | 1-3个月快速见效 |
我们的SEO优化服务遵循科学严谨的流程,确保每一步都基于数据分析和行业最佳实践:
全面检测网站技术问题、内容质量、竞争对手情况,制定个性化优化方案。
基于用户搜索意图和商业目标,制定全面的关键词矩阵和布局策略。
解决网站技术问题,优化网站结构,提升页面速度和移动端体验。
创作高质量原创内容,优化现有页面,建立内容更新机制。
获取高质量外部链接,建立品牌在线影响力,提升网站权威度。
持续监控排名、流量和转化数据,根据效果调整优化策略。
基于我们服务的客户数据统计,平均优化效果如下:
我们坚信,真正的SEO优化不仅仅是追求排名,而是通过提供优质内容、优化用户体验、建立网站权威,最终实现可持续的业务增长。我们的目标是与客户建立长期合作关系,共同成长。
Demand feedback