STM32-SPI1控制AD7705(Sigma-Delta-ADC芯片)

这篇具有很好参考价值的文章主要介绍了STM32-SPI1控制AD7705(Sigma-Delta-ADC芯片)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

原理图

stm32通信ad7705,电路应用,STM32开发,stm32,嵌入式硬件,单片机
该芯片需要晶振(Y2)和参考电源电压(U3)

手册说明

AD7705与国产TM7705型号差不多,也就是可以参考国产的手册。

AD7705利用 Σ-Δ 转换技术实现了 16 位无丢失代码性能。选定的输入信号被送到一个基于模拟调制器的增益可编程专用前端。片内数字滤波器处理调制器的输出信号。通过片内控制寄存器可调节滤波器的截止点和输出更新速率,从而对数字滤波器的第一个陷波进行编程。

TM7705 是双通道全差分模拟输入,带有一个差分基准输入。当电源电压为 5V、基准电压为 2.5V 时,该器件都可将输入信号范围从 0~+20mV 到 0~+2.5V 的信号进行处理。还可处理±20mV~±2.5V 的双极性输入信号,对于 TM7705 是以 AIN(-)输入端为参考点。

功能方框图

stm32通信ad7705,电路应用,STM32开发,stm32,嵌入式硬件,单片机

引脚功能

stm32通信ad7705,电路应用,STM32开发,stm32,嵌入式硬件,单片机

stm32通信ad7705,电路应用,STM32开发,stm32,嵌入式硬件,单片机

片内寄存器

stm32通信ad7705,电路应用,STM32开发,stm32,嵌入式硬件,单片机

通信寄存器(RS2、RS1、RS0=0、0、0)

stm32通信ad7705,电路应用,STM32开发,stm32,嵌入式硬件,单片机
stm32通信ad7705,电路应用,STM32开发,stm32,嵌入式硬件,单片机

stm32通信ad7705,电路应用,STM32开发,stm32,嵌入式硬件,单片机
stm32通信ad7705,电路应用,STM32开发,stm32,嵌入式硬件,单片机

设置寄存器

stm32通信ad7705,电路应用,STM32开发,stm32,嵌入式硬件,单片机
stm32通信ad7705,电路应用,STM32开发,stm32,嵌入式硬件,单片机

时钟寄存器

stm32通信ad7705,电路应用,STM32开发,stm32,嵌入式硬件,单片机
stm32通信ad7705,电路应用,STM32开发,stm32,嵌入式硬件,单片机
stm32通信ad7705,电路应用,STM32开发,stm32,嵌入式硬件,单片机

数据寄存器(RS2、RS1、RS0=0、1、1)

stm32通信ad7705,电路应用,STM32开发,stm32,嵌入式硬件,单片机

测试寄存器(RS2、RS1、RS0=1、0、0);上电/复位状态:00Hex

stm32通信ad7705,电路应用,STM32开发,stm32,嵌入式硬件,单片机

零标度校准寄存器(RS2、RS1、RS0=1、1、0);上电/复位状态:1F4000Hex

stm32通信ad7705,电路应用,STM32开发,stm32,嵌入式硬件,单片机

满标度校准寄存器(RS2、RS1、RS0=1、1、1);上电/复位状态:5761ABHex

stm32通信ad7705,电路应用,STM32开发,stm32,嵌入式硬件,单片机

校准过程

stm32通信ad7705,电路应用,STM32开发,stm32,嵌入式硬件,单片机
stm32通信ad7705,电路应用,STM32开发,stm32,嵌入式硬件,单片机

代码部分

以STM32F103和标准库作为底板
main.c

#include "led.h"
#include "delay.h"
#include "key.h"
#include "sys.h"
#include "usart.h"

#include "bsp_spi.h"



#define CS_ADC_LOW()    GPIO_ResetBits(GPIOA,GPIO_Pin_4)   
#define CS_ADC_HIGH()   GPIO_SetBits(GPIOA,GPIO_Pin_4)
/************************************************
 ALIENTEK精英STM32开发板实验4
 串口 实验   
 技术支持:www.openedv.com
 淘宝店铺:http://eboard.taobao.com 
 关注微信公众平台微信号:"正点原子",免费获取STM32资料。
 广州市星翼电子科技有限公司  
 作者:正点原子 @ALIENTEK
************************************************/

 u8 num1[6];
 float l_ncm1;
 u8 num2[6];
 float l_ncm2;


//写数据
void AD7705_WriteByte(u8 Dst_Addr)   
{   
        CS_ADC_LOW();//使能器件  
        delay_us(20);
        Spi1_readwritebyte(Dst_Addr);
        delay_us(100);
        CS_ADC_HIGH();//使能器件  
}
/********AD7705初始化函数***********/

void Init_AD7705(u8 chnanel)
{
    u8 i;
    for(i=0;i<150;i++)/* 多于连续32个 DIN=1 使串口复位 */
    {
        AD7705_WriteByte(0xff);//持续DIN高电平写操作,恢复AD7705接口
    }  
        delay_ms(1);
    switch(chnanel)
    {
      case 1:
             AD7705_WriteByte(0x20); /* 写时钟寄存器选中ch1*/
             AD7705_WriteByte(0x0C); /* 4.9152MHz时钟,250Hz数据更新速率 */
             AD7705_WriteByte(0x10); /*选择设置寄存器,使用chnanel 1*/
             AD7705_WriteByte(0x47); //写设置寄存器 ,设置成双极性、无缓冲、增益为2、滤波器工作、自校准
      break;
			/*有更改,时钟寄存器设为0x0a,4.9152MHz时钟,500Hz数据更新速率,*/
      case 2:
             AD7705_WriteByte(0x21); /* 写时钟寄存器选中ch2 */
             AD7705_WriteByte(0x0f); /* 4.9152MHz时钟,500Hz数据更新速率 */
             AD7705_WriteByte(0x11); /*选择设置寄存器,使用chnane 2*/
             AD7705_WriteByte(0x46); //写设置寄存器,设置成双极性、无缓冲、增益为2、滤波器工作、自校准
       break;
       default:       
                break;
        }
}

/* 读AD7705转换数据 输入通道channel */
u16 GetData7705_CH1(void)
{
	u16 temp1 = 0;
	u16 DataL = 0;
	u16 DataH = 0;
	Init_AD7705(1);                        //初始化通道1
	delay_ms(1);
	AD7705_WriteByte(0x39);   //选中CH1数据寄存器读  
	while(GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_2==0)){}//待数据准备好AdDrdy=0  
	CS_ADC_LOW(); //使能器件  
	delay_us(20);
	DataH = Spi1_readwritebyte(0xff);
	DataL = Spi1_readwritebyte(0xff);
	delay_us(100);
	CS_ADC_HIGH();   //取消片选        
	DataH = DataH << 8;
	temp1  =  DataH | DataL;

	return temp1;
}


/* 读AD7705转换数据 输入通道channel */
u16 GetData7705_CH2(void)
{
	u16 temp2 = 0;
	u16 DataL = 0;
	u16 DataH = 0;
	Init_AD7705(2);                //初始化通道2
	delay_ms(1);
	AD7705_WriteByte(0x38);   //选中CH2数据寄存器读  
	while(GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_2==0));  //待数据准备好AdDrdy=0  
	CS_ADC_LOW(); //使能器件  
	delay_us(20);
	DataH = Spi1_readwritebyte(0xff);
	DataL = Spi1_readwritebyte(0xff);
	delay_us(100);
	CS_ADC_HIGH();   //取消片选        
	DataH = DataH << 8;
	temp2  =  DataH | DataL;

	return temp2;
}
//数据处理


void ADC_7705(void)
{        
	
			u16 RCH1_16bit,RCH2_16bit;              
			RCH1_16bit = GetData7705_CH1();								
			l_ncm1  = (float)(RCH1_16bit*(2.5/65535)); //算出通道1电压
			RCH2_16bit = GetData7705_CH2();
			l_ncm2  = (float)(RCH2_16bit*(2.5/65535)); //算出通道2电压

//			 num1[0]    = l_ncm1/10000+'0';
//			 num1[2]    = (l_ncm1%10000)/1000+'0';
//			 num1[3]    = (l_ncm1%1000)/100+'0';
//			 num1[4]    = (l_ncm1%100)/10+'0';
//			 num1[5]    = l_ncm1%10+'0';

//			num2[0]    = l_ncm2/10000+'0';
//			num2[2]    = (l_ncm2%10000)/1000+'0';
//			num2[3]    = (l_ncm2%1000)/100+'0';
//			num2[4]    = (l_ncm2%100)/10+'0';
//			num2[5]    = l_ncm2%10+'0';

			printf("buff1:%f\n",l_ncm1);
			printf("buff2:%f\n",l_ncm2);
		 if(l_ncm2>8500|l_ncm2<8200)
		 {

			 delay_ms(10);
			 l_ncm2=0;
				 
		 }
		 else
		 {

		 }
		 
}


 int main(void)
 {		

	delay_init();	    	 //延时函数初始化	  
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); //设置NVIC中断分组2:2位抢占优先级,2位响应优先级
	uart_init(115200);	 //串口初始化为115200
 	LED_Init();			     //LED端口初始化
	KEY_Init();          //初始化与按键连接的硬件接口
	 
  Spi1_init();				 //SPI 初始化
	 GPIO_SetBits(GPIOA,GPIO_Pin_0);
	 Init_AD7705(1);
	 delay_ms(250);
	 Init_AD7705(2);
 	while(1)
	{
//		Spi1_readwritebyte(0xAA);
//		printf("buff1:%X\n",GetData7705_CH1());
//		delay_ms(100);
//		printf("buff2:%X\n",GetData7705_CH2());

		
		ADC_7705();		
		
		
		delay_ms(250);
		
	}	 
	
	
	
 }


bsp_spi.c

#include "bsp_spi.h"

 /**
  * 函数功能: SPI 读写一个字节
  * 输入参数: 要写入的字节
  * 返 回 值: 读取到的字节
  * 说    明:无
  */
void Spi1_init(void)
{
 	GPIO_InitTypeDef GPIO_InitStructure;
  SPI_InitTypeDef  SPI_InitStructure;
	
	/* 使能GPIO和SPI时钟 */
	RCC_APB2PeriphClockCmd(	RCC_APB2Periph_GPIOA, ENABLE );//PORTB时钟使能 
	RCC_APB2PeriphClockCmd(	RCC_APB2Periph_SPI1,  ENABLE );//SPI2时钟使能 	

  /* 配置SPI功能引脚:SCK 时钟引脚 */	
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;  			 //复用推挽输出 
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA, &GPIO_InitStructure);         				 //初始化GPIO
	
	 /* 配置SPI功能引脚:MISO 主机输出从机输入引脚 */	
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6;
  GPIO_Init(GPIOA, &GPIO_InitStructure);
	
  /* 配置SPI功能引脚:MOSI 主机输入从机输出引脚 */	
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7;
  GPIO_Init(GPIOA, &GPIO_InitStructure);

  /* SPI外设配置 --NSS 引脚由软件控制以及 MSB 先行模式*/
  SPI_Cmd(SPI1, DISABLE); //失能能SPI外设
  SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;  //设置SPI单向或者双向的数据模式:SPI设置为双线双向全双工
  SPI_InitStructure.SPI_Mode = SPI_Mode_Master;                //设置SPI工作模式:设置为主SPI
  SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;                //设置SPI的数据大小:SPI发送接收8位帧结构
  SPI_InitStructure.SPI_CPOL = SPI_CPOL_High;                //选择了串行时钟的稳态:时钟悬空高
  SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge;        //数据捕获于第二个时钟
  SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;                //NSS信号由硬件(NSS管脚)还是软件(使用SSI位)管理:内部NSS信号有SSI位控制
  SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_256;                //定义波特率预分频的值:波特率预分频值为256
  SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;        //指定数据传输从MSB位还是LSB位开始:数据传输从MSB位开始
  SPI_InitStructure.SPI_CRCPolynomial = 7;        //CRC值计算的多项式
  SPI_Init(SPI1, &SPI_InitStructure);  //根据SPI_InitStruct中指定的参数初始化外设SPIx寄存器
	
  /* 配置SPI功能引脚:CS 串行Flash片选引脚 */	
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;	
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;  			 //推挽输出 
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  GPIO_Init(GPIOA, &GPIO_InitStructure);
	
	/* 配置SPI所用的引脚:默认高电平 */	
	GPIO_SetBits(GPIOA,GPIO_Pin_5|GPIO_Pin_6|GPIO_Pin_7);
	
	SPI_Cmd(SPI1, ENABLE); //使能SPI外设

/* RES */	
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;  			 //复用推挽输出 
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA, &GPIO_InitStructure);         				 //初始化GPIO
/* CS */	
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;  			 //复用推挽输出 
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA, &GPIO_InitStructure);         				 //初始化GPIO
/* DRDY */	
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;  			 //复用推挽输出 
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA, &GPIO_InitStructure);         				 //初始化GPIO
	
		GPIO_SetBits(GPIOA,GPIO_Pin_0|GPIO_Pin_4|GPIO_Pin_2);
}


 /**
  * 函数功能: SPI 速度设置函数
  * 输入参数: 
							SPI_BaudRatePrescaler_2   2分频 
							SPI_BaudRatePrescaler_8   8分频 
							SPI_BaudRatePrescaler_16  16分频  
							SPI_BaudRatePrescaler_256 256分频 

  * 返 回 值: 无
  * 说    明:
  */ 
void Spi1_SetSpeed(u8 Spi_baudrateprescaler)
{
  assert_param(IS_SPI_BAUDRATE_PRESCALER(SPI_BaudRatePrescaler));
	SPI1->CR1&=0XFFC7;
	SPI1->CR1|=Spi_baudrateprescaler;	//设置SPI1速度 
	SPI_Cmd(SPI1,ENABLE); 
} 


 /**
  * 函数功能: SPI 读写一个字节
  * 输入参数: 要写入的字节
  * 返 回 值: 读取到的字节
  * 说    明:无
  */
u8 Spi1_readwritebyte(u8 Txdata)
{		
	u8 retry=0;				 	
	while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_TXE) == RESET) //检查指定的SPI标志位设置与否:发送缓存空标志位
		{
		retry++;
		if(retry>200)return 0;
		}			  
	SPI_I2S_SendData(SPI1, Txdata); //通过外设SPIx发送一个数据
	retry=0;

	while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_RXNE) == RESET) //检查指定的SPI标志位设置与否:接受缓存非空标志位
		{
		retry++;
		if(retry>200)return 0;
		}
	return SPI_I2S_ReceiveData(SPI1); //返回通过SPIx最近接收的数据					    
}


bsp_spi.h文章来源地址https://www.toymoban.com/news/detail-825492.html

#ifndef __BSP_SPI_H__
#define __BSP_SPI_H__

/* 包含头文件 ----------------------------------------------------------------*/
#include <stm32f10x.h>
#include "sys.h"
/* 类型定义 ------------------------------------------------------------------*/

/* 宏定义 --------------------------------------------------------------------*/

/* 

SPI引脚分布

SPI1			挂载时钟 APB1
SCK				PA5
MISO			PA6
MOSI			PA7
CS				PA4

SPI2			挂载时钟 APB2
SCK				PB13
MISO			PB14
MOSI			PB15
CS				PB12

*/



/* SPI配置*/
#define SPIx                        SPI1
#define SPI_CLK                     RCC_APB2Periph_SPI1

#define SPI_SCK_CLK                 RCC_APB2Periph_GPIOA   
#define SPI_SCK_PORT                GPIOA   
#define SPI_SCK_PIN                 GPIO_Pin_5

#define SPI_MISO_CLK                RCC_APB2Periph_GPIOA    
#define SPI_MISO_PORT               GPIOA
#define SPI_MISO_PIN                GPIO_Pin_6

#define SPI_MOSI_CLK                RCC_APB2Periph_GPIOA    
#define SPI_MOSI_PORT               GPIOA 
#define SPI_MOSI_PIN                GPIO_Pin_7

#define SPI_CS_CLK                  RCC_APB2Periph_GPIOA    
#define SPI_CS_PORT                 GPIOA
#define SPI_CS_PIN                  GPIO_Pin_4

#define SPI_CS_ENABLE()             GPIO_ResetBits(SPI_CS_PORT, SPI_CS_PIN)
#define SPI_CS_DISABLE()            GPIO_SetBits(SPI_CS_PORT, SPI_CS_PIN)


#define CS_LOW	GPIO_WriteBit(SPI_CS_PORT, SPI_CS_PIN, Bit_RESET)
#define CS_HIGH	GPIO_WriteBit(SPI_CS_PORT, SPI_CS_PIN, Bit_SET)


/* 扩展变量 ------------------------------------------------------------------*/
/* 函数声明 ------------------------------------------------------------------*/


void Spi1_SetSpeed(u8 SPI_BaudRatePrescaler); 		//SPI 速度设置函数
void Spi1_init(void);															//SPI 初始化
u8 Spi1_readwritebyte(u8 TxData);									//SPI 数据写入或读取



#endif

到了这里,关于STM32-SPI1控制AD7705(Sigma-Delta-ADC芯片)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • sigma-delta ADC原理

    主要是想大致了解Sigma-delta ADC是怎么工作的,写了个乱七八糟的代码来简单看下。很粗略的解释,主要给自己参考。 successive approximation register adc,简单理解为一个采样开关和采样电容。采样开关定时闭合,忽略暂态,则采样电容上的电压等于采样开关闭合时刻的输入电压。

    2023年04月11日
    浏览(27)
  • STM32模拟SPI协议获取24位模数转换(24bit ADC)芯片AD7791电压采样数据

    STM32大部分芯片只有12位的ADC采样性能,如果要实现更高精度的模数转换如24位ADC采样,则需要连接外部ADC实现。AD7791是亚德诺(ADI)半导体一款用于低功耗、24位Σ-Δ型模数转换器(ADC) ,适合低频测量应用,提供50 Hz/60 Hz同步抑制。 这里介绍基于AD7791的24位ADC采样实现。 AD7791的管脚

    2024年02月09日
    浏览(28)
  • 基于FPGA的sigma delta ADC软件无线电设计

    目录 一、理论基础 二、核心程序 三、测试结果         Sigma-Delta ADC是一种目前使用最为普遍的高精度ADC结构,在精度达到20位以上的场合,Sigma-Delta是必选的结构。通过采用过采样、噪声整形以及数字滤波技术,降低对模拟电路的设计要求,实现了其他类型的ADC无法达到的

    2024年02月05日
    浏览(31)
  • 【STM32 CubeMX】SPI层次结构SPI协议与SPI控制器结构

    随着嵌入式系统的迅猛发展,STM32系列微控制器在各种应用中得到广泛应用。在嵌入式系统设计中,串行外设接口(SPI)是一种常见且重要的通信协议。为了更便捷地配置和使用SPI,STMicroelectronics提供了一款强大的工具——STM32 CubeMX。本文将着重介绍STM32 CubeMX中SPI的层次结构

    2024年02月20日
    浏览(15)
  • 【STM32】SPI通讯控制ILI9341显示屏

    ILI9341是一款分辨率为 240x320 分辨率 的a- tft液晶显示单片SOC驱动,由720通道源驱动、320通道门驱动、 172800字节GRAM (240RGBx320点位图形显示数据)和电源电路组成。 ILI9341支持并行8-/9-/16-/18位数据总线MCU接口,6-/16-/18位数据总线RGB接口和3 /4线串行外围接口 SPI通讯 。通过窗口地址函

    2024年01月21日
    浏览(22)
  • 【STM32】BLDC驱动&控制开发笔记 | 07_SPI通信测试 - STM32F407用SPI配置DRV8323驱动芯片

    最近在埋头搞STM32 + 无刷直流电机控制,想实现用自己的STM32F407VGT6芯片板子,外加一块驱动板(目前选用到TI的DRV8302或者DRV8323驱动芯片),搞定电机驱动,最后实现比较好的控制效果。如果不是同一块芯片的同学也不用急着走,大体上都是可借鉴哒~ 本文主要实现使用SPI通信

    2024年02月08日
    浏览(28)
  • Sigma-delta ADC数字抽取滤波器的verilog与MATLAB设计

           模数转换器根据采样率的不同发展为奈奎斯特(Nyquist)型和过采样(Oversampling)型两大类。奈奎斯特型ADC 采用2-3倍信号带宽的采样时钟进行采样。过采样型ADC采用过采样技术和噪声整形技术,以远高于2倍信号带宽的采样时钟进行采样,将信号中的噪声搬移到高频以

    2024年04月26日
    浏览(13)
  • STM32模拟SPI控制NRF24L01发送和接收

    NRF24L01是一款2.4Ghz ISM频段无线收发芯片。NRF24L01模块可视为无需配对和连接型的WIFI或蓝牙模块。NRF24L01可工作于1发6收工作模式。一个NRF24L01模块工作于发送模式时,每次根据设定的接收端地址发送射频信号和数据;一个NRF24L01模块工作于接收模式时,可以接收来自1~6个发送端

    2024年02月08日
    浏览(21)
  • STM32模拟SPI协议控制数字电位器MCP41010电阻值

    MCP41010是单路8位分辨率数字电位器,通过SPI接口可控制电位器阻值分配,相当于PW0端在PA0和PB0之间滑动。如下图所示: MCP41010是10K欧姆规格的数字电位器,即PA0和PB0之间的阻值恒定为10K,PW0与PA0或PW0与PB0之间的阻值互补可配。 另外有相关型号不同阻值规格和集成双路的MCP系列

    2024年02月16日
    浏览(16)
  • STM32F103硬件SPI控制6针/7针0.96寸OLED显示屏

    OLED主要参数 1、高分辨率:128 64(和12864LCD相同分辨率,但该OLED屏的单位面积像素点多)。 2、广可视角度:大于160°。 3、低功耗:正常显示时0.04W。 4、宽供电范围:直流3.3V-5V。 5、工业级:工作温度范围-30°℃~70°℃。 6、体积小:28.8mm 28.5mm。 7、通信方式:lIC、SPI。 8、亮

    2024年02月17日
    浏览(46)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包