单片机C语言实例:19、EEPROM 24C02读写

一、24C02存储一个数据

此程序用于检测EEPROM性能,测试方法如下:

1、写入24c02一个数据,然后在内存中改变这些数据,

2、掉电后主内存将失去这些信息,然后从24c02中调入这些数据。看是否与写入的相同。

3、这里用8个LED演示

4、函数是采用软件延时的方法产生SCL脉冲,固对高晶振频率要作 一定的修改

5、本例是1us机器周期,即晶振频率要小于12MHZ

程序实例1:

#include <reg52.h>          //头文件的包含
#include <intrins.h>

#define  _Nop()  _nop_()        //定义空指令
                                 
sbit SDA=P2^1;            //模拟I2C数据传送位
sbit SCL=P2^0;            //模拟I2C时钟控制位
                        
bit ack;	              //应答标志位
   

void DelayUs2x(unsigned char t);//函数声明 
void DelayMs(unsigned char t);

/*------------------------------------------------
 uS延时函数,含有输入参数 unsigned char t,无返回值
 unsigned char 是定义无符号字符变量,其值的范围是
 0~255 这里使用晶振12M,精确延时请使用汇编,大致延时
 长度如下 T=tx2+5 uS 
------------------------------------------------*/
void DelayUs2x(unsigned char t)
{   
	while(--t);
}

/*------------------------------------------------
 mS延时函数,含有输入参数 unsigned char t,无返回值
 unsigned char 是定义无符号字符变量,其值的范围是
 0~255 这里使用晶振12M,精确延时请使用汇编
------------------------------------------------*/
void DelayMs(unsigned char t)
{     
	while(t--)
	{
		//大致延时1mS
		DelayUs2x(245);
		DelayUs2x(245);
	}
}

/*------------------------------------------------
                    启动总线
------------------------------------------------*/
void Start_I2c()
{
	SDA=1;   //发送起始条件的数据信号
	_Nop();
	SCL=1;
	_Nop();    //起始条件建立时间大于4.7us,延时
	_Nop();
	_Nop();
	_Nop();
	_Nop();    
	SDA=0;     //发送起始信号
	_Nop();    //起始条件锁定时间大于4μ
	_Nop();
	_Nop();
	_Nop();
	_Nop();       
	SCL=0;    //钳住I2C总线,准备发送或接收数据
	_Nop();
	_Nop();
}
/*------------------------------------------------
                    结束总线
------------------------------------------------*/
void Stop_I2c()
{
	SDA=0;    //发送结束条件的数据信号
	_Nop();   //发送结束条件的时钟信号
	SCL=1;    //结束条件建立时间大于4μ
	_Nop();
	_Nop();
	_Nop();
	_Nop();
	_Nop();
	SDA=1;    //发送I2C总线结束信号
	_Nop();
	_Nop();
	_Nop();
	_Nop();
}




/*----------------------------------------------------------------
                 字节数据传送函数               
函数原型: void  SendByte(unsigned char c);
功能:  将数据c发送出去,可以是地址,也可以是数据,发完后等待应答,并对
     此状态位进行操作.(不应答或非应答都使ack=0 假)     
     发送数据正常,ack=1; ack=0表示被控器无应答或损坏。
------------------------------------------------------------------*/
void  SendByte(unsigned char c)
{
	unsigned char BitCnt;
	
	for(BitCnt=0;BitCnt<8;BitCnt++)  //要传送的数据长度为8位
		{
			if((c<<BitCnt)&0x80)SDA=1;   //判断发送位
			else  SDA=0;                
			_Nop();
			SCL=1;               //置时钟线为高,通知被控器开始接收数据位
			_Nop(); 
			_Nop();             //保证时钟高电平周期大于4μ
			_Nop();
			_Nop();
			_Nop();         
			SCL=0; 
		}
		
		_Nop();
		_Nop();
		SDA=1;               //8位发送完后释放数据线,准备接收应答位
		_Nop();
		_Nop();   
		SCL=1;
		_Nop();
		_Nop();
		_Nop();
		if(SDA==1)ack=0;     
		else ack=1;        //判断是否接收到应答信号
		SCL=0;
		_Nop();
		_Nop();
}







/*----------------------------------------------------------------
                 字节数据传送函数               
函数原型: unsigned char  RcvByte();
功能:  用来接收从器件传来的数据,并判断总线错误(不发应答信号),
     发完后请用应答函数。  
------------------------------------------------------------------*/	
unsigned char  RcvByte()
{
	unsigned char retc;
	unsigned char BitCnt;
	
	retc=0; 
	SDA=1;             //置数据线为输入方式
	for(BitCnt=0;BitCnt<8;BitCnt++)
	{
		_Nop();           
		SCL=0;       //置时钟线为低,准备接收数据位
		_Nop();
		_Nop();      //时钟低电平周期大于4.7us
		_Nop();
		_Nop();
		_Nop();
		SCL=1;       //置时钟线为高使数据线上数据有效
		_Nop();
		_Nop();
		retc=retc<<1;
		if(SDA==1)retc=retc+1; //读数据位,接收的数据位放入retc中
		_Nop();
		_Nop(); 
	}
	SCL=0;    
	_Nop();
	_Nop();
	return(retc);
}



/*----------------------------------------------------------------
                     应答子函数
原型:  void Ack_I2c(void);
 
----------------------------------------------------------------*/
void Ack_I2c(void)
{
	SDA=0;     
	_Nop();
	_Nop();
	_Nop();      
	SCL=1;
	_Nop();
	_Nop();              //时钟低电平周期大于4μ
	_Nop();
	_Nop();
	_Nop();  
	SCL=0;               //清时钟线,钳住I2C总线以便继续接收
	_Nop();
	_Nop();    
}
/*----------------------------------------------------------------
                     非应答子函数
原型:  void NoAck_I2c(void);
 
----------------------------------------------------------------*/
void NoAck_I2c(void)
{ 
	SDA=1;
	_Nop();
	_Nop();
	_Nop();      
	SCL=1;
	_Nop();
	_Nop();              //时钟低电平周期大于4μ
	_Nop();
	_Nop();
	_Nop();  
	SCL=0;                //清时钟线,钳住I2C总线以便继续接收
	_Nop();
	_Nop();    
}






/*----------------------------------------------------------------
                    向无子地址器件发送字节数据函数               
函数原型: bit  ISendByte(unsigned char sla,ucahr c);  
功能:     从启动总线到发送地址,数据,结束总线的全过程,从器件地址sla.
           如果返回1表示操作成功,否则操作有误。
注意:    使用前必须已结束总线。
----------------------------------------------------------------*/
/*bit ISendByte(unsigned char sla,unsigned char c)
{
   Start_I2c();               //启动总线
   SendByte(sla);             //发送器件地址
     if(ack==0)return(0);
   SendByte(c);               //发送数据
     if(ack==0)return(0);
  Stop_I2c();                 //结束总线
  return(1);
}
*/

/*----------------------------------------------------------------
                    向有子地址器件发送多字节数据函数               
函数原型: bit  ISendStr(unsigned char sla,unsigned char suba,ucahr *s,unsigned char no);  
功能:     从启动总线到发送地址,子地址,数据,结束总线的全过程,从器件
          地址sla,子地址suba,发送内容是s指向的内容,发送no个字节。
           如果返回1表示操作成功,否则操作有误。
注意:    使用前必须已结束总线。
----------------------------------------------------------------*/
bit ISendStr(unsigned char sla,unsigned char suba,unsigned char *s,unsigned char no)
{
	unsigned char i;
	
	Start_I2c();               //启动总线
	SendByte(sla);             //发送器件地址
	if(ack==0)return(0);
	SendByte(suba);            //发送器件子地址
	if(ack==0)return(0);
	
	for(i=0;i<no;i++)
	{   
		SendByte(*s);            //发送数据
		if(ack==0)return(0);
		s++;
	} 
	Stop_I2c();                  //结束总线
	return(1);
}

/*----------------------------------------------------------------
                    向无子地址器件读字节数据函数               
函数原型: bit  IRcvByte(unsigned char sla,ucahr *c);  
功能:     从启动总线到发送地址,读数据,结束总线的全过程,从器件地
          址sla,返回值在c.
           如果返回1表示操作成功,否则操作有误。
注意:    使用前必须已结束总线。
----------------------------------------------------------------*/
/*bit IRcvByte(unsigned char sla,unsigned char *c)
{
   Start_I2c();                //启动总线
   SendByte(sla+1);            //发送器件地址
     if(ack==0)return(0);
   *c=RcvByte();               //读取数据
     NoAck_I2c();              //发送非就答位
     Stop_I2c();               //结束总线
  return(1);
}

*/
/*----------------------------------------------------------------
                    向有子地址器件读取多字节数据函数               
函数原型: bit  ISendStr(unsigned char sla,unsigned char suba,ucahr *s,unsigned char no);  
功能:     从启动总线到发送地址,子地址,读数据,结束总线的全过程,从器件
          地址sla,子地址suba,读出的内容放入s指向的存储区,读no个字节。
           如果返回1表示操作成功,否则操作有误。
注意:    使用前必须已结束总线。
----------------------------------------------------------------*/
bit IRcvStr(unsigned char sla,unsigned char suba,unsigned char *s,unsigned char no)
{
	unsigned char i;

	Start_I2c();               //启动总线
	SendByte(sla);             //发送器件地址
    if(ack==0)return(0);
	SendByte(suba);            //发送器件子地址
    if(ack==0)return(0);

	Start_I2c();
	SendByte(sla+1);
    if(ack==0)return(0);

	for(i=0;i<no-1;i++)
    {   
		*s=RcvByte();              //发送数据
		Ack_I2c();                //发送就答位 
		s++;
    } 
	*s=RcvByte();
	NoAck_I2c();                 //发送非应位
	Stop_I2c();                    //结束总线
	return(1);
}

/*------------------------------------------------
                    主函数
------------------------------------------------*/
void main()
{
	unsigned char temp;       // 定义临时变量
	unsigned char i;
	 
	IRcvStr(0xae,4,&temp,1);  //调用存储数据
	 
	while(1)
	{  
		P1=temp;             //数值用二进制显示,直接用8个LED表示
        for(i=0;i<5;i++)   
             DelayMs(200);
        temp++;              //1s钟变量自加1,改变值后存储到24c02
                               //下次开机时将显示当前数值
        ISendStr(0xae,4,&temp,1); //写入24c02
	}
}

二、24C02存储多个数据

此程序用于检测EEPROM性能,测试方法如下:写入24c02一些数据,然后在内存中清除这些数据,掉电后主内存将失去这些信息,然后从24c02中调入这些数据。看是否与写入的相同。

程序实例2:

#include <reg52.h>          //头文件的包含
#include <intrins.h>

#define  _Nop()  _nop_()        //定义空指令

// 常,变量定义区
unsigned char code DuanMa[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,
		                  	         0x77,0x7c,0x39,0x5e,0x79,0x71};// 显示段码值0~F
unsigned char code WeiMa[]={0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f};//分别对应相应的数码管点亮,即位码

sbit LATCH1=P2^2;
sbit LATCH2=P2^3;
                                              
sbit SDA=P2^1;            //模拟I2C数据传送位
sbit SCL=P2^0;            //模拟I2C时钟控制位

                          
bit ack;	              //应答标志位
   

void DelayUs2x(unsigned char t);//函数声明 
void DelayMs(unsigned char t);

/*------------------------------------------------
 uS延时函数,含有输入参数 unsigned char t,无返回值
 unsigned char 是定义无符号字符变量,其值的范围是
 0~255 这里使用晶振12M,精确延时请使用汇编,大致延时
 长度如下 T=tx2+5 uS 
------------------------------------------------*/
void DelayUs2x(unsigned char t)
{   
	while(--t);
}

/*------------------------------------------------
 mS延时函数,含有输入参数 unsigned char t,无返回值
 unsigned char 是定义无符号字符变量,其值的范围是
 0~255 这里使用晶振12M,精确延时请使用汇编
------------------------------------------------*/
void DelayMs(unsigned char t)
{
     
	while(t--)
	{
		//大致延时1mS
		DelayUs2x(245);
		DelayUs2x(245);
	}
}

/*------------------------------------------------
                    启动总线
------------------------------------------------*/
void Start_I2c()
{
	SDA=1;   //发送起始条件的数据信号
	_Nop();
	SCL=1;
	_Nop();    //起始条件建立时间大于4.7us,延时
	_Nop();
	_Nop();
	_Nop();
	_Nop();    
	SDA=0;     //发送起始信号
	_Nop();    //起始条件锁定时间大于4μ
	_Nop();
	_Nop();
	_Nop();
	_Nop();       
	SCL=0;    //钳住I2C总线,准备发送或接收数据
	_Nop();
	_Nop();
}

/*------------------------------------------------
                    结束总线
------------------------------------------------*/
void Stop_I2c()
{
	SDA=0;    //发送结束条件的数据信号
	_Nop();   //发送结束条件的时钟信号
	SCL=1;    //结束条件建立时间大于4μ
	_Nop();
	_Nop();
	_Nop();
	_Nop();
	_Nop();
	SDA=1;    //发送I2C总线结束信号
	_Nop();
	_Nop();
	_Nop();
	_Nop();
}




/*----------------------------------------------------------------
                 字节数据传送函数               
函数原型: void  SendByte(unsigned char c);
功能:  将数据c发送出去,可以是地址,也可以是数据,发完后等待应答,并对
     此状态位进行操作.(不应答或非应答都使ack=0 假)     
     发送数据正常,ack=1; ack=0表示被控器无应答或损坏。
------------------------------------------------------------------*/
void  SendByte(unsigned char c)
{
	unsigned char BitCnt;
 
	for(BitCnt=0;BitCnt<8;BitCnt++)  //要传送的数据长度为8位
    {
		if((c<<BitCnt)&0x80)SDA=1;   //判断发送位
		else  SDA=0;                
		_Nop();
		SCL=1;               //置时钟线为高,通知被控器开始接收数据位
		_Nop(); 
		_Nop();             //保证时钟高电平周期大于4μ
		_Nop();
		_Nop();
		_Nop();         
		SCL=0; 
    }
    
    _Nop();
    _Nop();
    SDA=1;               //8位发送完后释放数据线,准备接收应答位
    _Nop();
    _Nop();   
    SCL=1;
    _Nop();
    _Nop();
    _Nop();
    if(SDA==1)ack=0;     
       else ack=1;        //判断是否接收到应答信号
    SCL=0;
    _Nop();
    _Nop();
}







/*----------------------------------------------------------------
                 字节数据传送函数               
函数原型: unsigned char  RcvByte();
功能:  用来接收从器件传来的数据,并判断总线错误(不发应答信号),
     发完后请用应答函数。  
------------------------------------------------------------------*/	
unsigned char  RcvByte()
{
	unsigned char retc;
	unsigned char BitCnt;
	
	retc=0; 
	SDA=1;             //置数据线为输入方式
	for(BitCnt=0;BitCnt<8;BitCnt++)
    {
        _Nop();           
        SCL=0;       //置时钟线为低,准备接收数据位
        _Nop();
        _Nop();      //时钟低电平周期大于4.7us
        _Nop();
        _Nop();
        _Nop();
        SCL=1;       //置时钟线为高使数据线上数据有效
        _Nop();
        _Nop();
        retc=retc<<1;
        if(SDA==1)retc=retc+1; //读数据位,接收的数据位放入retc中
        _Nop();
        _Nop(); 
    }
	SCL=0;    
	_Nop();
	_Nop();
	return(retc);
}



/*----------------------------------------------------------------
                     应答子函数
原型:  void Ack_I2c(void);
 
----------------------------------------------------------------*/
void Ack_I2c(void)
{
	SDA=0;     
	_Nop();
	_Nop();
	_Nop();      
	SCL=1;
	_Nop();
	_Nop();              //时钟低电平周期大于4μ
	_Nop();
	_Nop();
	_Nop();  
	SCL=0;               //清时钟线,钳住I2C总线以便继续接收
	_Nop();
	_Nop();    
}

/*----------------------------------------------------------------
                     非应答子函数
原型:  void NoAck_I2c(void);
 
----------------------------------------------------------------*/
void NoAck_I2c(void)
{
	SDA=1;
	_Nop();
	_Nop();
	_Nop();      
	SCL=1;
	_Nop();
	_Nop();              //时钟低电平周期大于4μ
	_Nop();
	_Nop();
	_Nop();  
	SCL=0;                //清时钟线,钳住I2C总线以便继续接收
	_Nop();
	_Nop();    
}


/*----------------------------------------------------------------
                    向无子地址器件发送字节数据函数               
函数原型: bit  ISendByte(unsigned char sla,ucahr c);  
功能:     从启动总线到发送地址,数据,结束总线的全过程,从器件地址sla.
           如果返回1表示操作成功,否则操作有误。
注意:    使用前必须已结束总线。
----------------------------------------------------------------*/
/*bit ISendByte(unsigned char sla,unsigned char c)
{
   Start_I2c();               //启动总线
   SendByte(sla);             //发送器件地址
     if(ack==0)return(0);
   SendByte(c);               //发送数据
     if(ack==0)return(0);
  Stop_I2c();                 //结束总线
  return(1);
}
*/

/*----------------------------------------------------------------
                    向有子地址器件发送多字节数据函数               
函数原型: bit  ISendStr(unsigned char sla,unsigned char suba,ucahr *s,unsigned char no);  
功能:     从启动总线到发送地址,子地址,数据,结束总线的全过程,从器件
          地址sla,子地址suba,发送内容是s指向的内容,发送no个字节。
           如果返回1表示操作成功,否则操作有误。
注意:    使用前必须已结束总线。
----------------------------------------------------------------*/
bit ISendStr(unsigned char sla,unsigned char suba,unsigned char *s,unsigned char no)
{
	unsigned char i;

	Start_I2c();               //启动总线
	SendByte(sla);             //发送器件地址
    if(ack==0)return(0);
	SendByte(suba);            //发送器件子地址
    if(ack==0)return(0);

	for(i=0;i<no;i++)
    {   
		SendByte(*s);            //发送数据
       if(ack==0)return(0);
		s++;
    } 
	Stop_I2c();                  //结束总线
	return(1);
}

/*----------------------------------------------------------------
                    向无子地址器件读字节数据函数               
函数原型: bit  IRcvByte(unsigned char sla,ucahr *c);  
功能:     从启动总线到发送地址,读数据,结束总线的全过程,从器件地
          址sla,返回值在c.
           如果返回1表示操作成功,否则操作有误。
注意:    使用前必须已结束总线。
----------------------------------------------------------------*/
/*bit IRcvByte(unsigned char sla,unsigned char *c)
{
   Start_I2c();                //启动总线
   SendByte(sla+1);            //发送器件地址
     if(ack==0)return(0);
   *c=RcvByte();               //读取数据
     NoAck_I2c();              //发送非就答位
     Stop_I2c();               //结束总线
  return(1);
}

*/
/*----------------------------------------------------------------
                    向有子地址器件读取多字节数据函数               
函数原型: bit  ISendStr(unsigned char sla,unsigned char suba,ucahr *s,unsigned char no);  
功能:     从启动总线到发送地址,子地址,读数据,结束总线的全过程,从器件
          地址sla,子地址suba,读出的内容放入s指向的存储区,读no个字节。
           如果返回1表示操作成功,否则操作有误。
注意:    使用前必须已结束总线。
----------------------------------------------------------------*/
bit IRcvStr(unsigned char sla,unsigned char suba,unsigned char *s,unsigned char no)
{
	unsigned char i;

	Start_I2c();               //启动总线
	SendByte(sla);             //发送器件地址
    if(ack==0)return(0);
	SendByte(suba);            //发送器件子地址
    if(ack==0)return(0);

	Start_I2c();
	SendByte(sla+1);
    if(ack==0)return(0);

	for(i=0;i<no-1;i++)
    {   
		*s=RcvByte();              //发送数据
		Ack_I2c();                //发送就答位 
		s++;
    } 
	*s=RcvByte();
    NoAck_I2c();                 //发送非应位
	Stop_I2c();                    //结束总线
	return(1);
}
/*------------------------------------------------
                    主函数
------------------------------------------------*/
void main()
{
	unsigned char show[4]={1,2,3,4};          // 显示码值 1234
	unsigned char i;
	
	ISendStr(0xae,4,show,4);                   //写入24c02
	DelayMs(200);
	show[0]=0;                                  //清除当前数据
	show[1]=0;
	show[2]=0;
	show[3]=0;
	IRcvStr(0xae,4,show,4);                  //调用存储数据
	
	while(1)
	{ 
           
		P0=DuanMa[show[i]];//显示存储数据
		LATCH1=1;                  //锁存
		LATCH1=0;
	
		P0=WeiMa[i];  //取位码
		LATCH2=1;          // 锁存
		LATCH2=0;
		DelayMs(200);      //延时用于演示显示数据
		DelayMs(200);
		i++;
		if(i==4)
			i=0;
	}
}

三、24C02存储花样流水灯

程序实例3:

#include <reg52.h>                
#include "i2c.h"
/*------------------------------------------------
                    花样灯表格
------------------------------------------------*/
unsigned char   dat[]={	0x7f,0xbf,0xdf,0xef,
						0xf7,0xfb,0xfd,0xfe,
						0xff,0xff,0x00,0x00,
						0x55,0x55,0xaa,0xaa
						};//ram允许情况可以无限添加
						
/*------------------------------------------------
                   主函数
------------------------------------------------*/
main()
{
	//bit flag;
	unsigned char i;
	ISendStr(0xae,80,dat,16);            //写入24c02
	DelayMs(1);                          //写入后必须延时等待24c02内部
										//烧录数据处理完毕方可继续其他操作
	for(i=0;i<16;i++)                    //清除数据
		dat[i]=0;
	IRcvStr(0xae,80,dat,16);             //从24c02读出数据 
	while(1)
    {

		for(i=0;i<16;i++)//查表
	    {
			DelayMs(200);
			P1=dat[i];
		}
	}
}

四、24C02记录开机次数

#include <reg52.h>                
#include "i2c.h"
#include "delay.h"
#include "display.h"


main()
{
	unsigned char num=0;
	Init_Timer0();
	IRcvStr(0xae,50,&num,1);   //从24c02读出数据                             
	num++;                 
	ISendStr(0xae,50,&num,1);  //写入24c02 
	DelayMs(10);
	TempData[0]	=	DuanMa[num/100];    
	TempData[1]	=	DuanMa[(num%100)/10];
	TempData[2]	=	DuanMa[(num%100)%10];

	while(1)
    {
	}
}

五、24C02存储上次使用中状态

程序实例5:

#include <reg52.h>                
#include "i2c.h"
#include "delay.h"
#include "display.h"

sbit KEY_ADD=P3^3;  //定义按键输入端口
sbit KEY_DEC=P3^4;

/*-----------------------------------------------
                     主函数
------------------------------------------------*/  
main()
{
	unsigned char num=0;
	Init_Timer0();
	
	IRcvStr(0xae,0,&num,1);                                
					
	KEY_ADD=1; //按键输入端口电平置高
	KEY_DEC=1;
	while (1)         //主循环
	{
  
	if(!KEY_ADD)  //如果检测到低电平,说明按键按下
    {
		DelayMs(10); //延时去抖,一般10-20ms
		if(!KEY_ADD)     //再次确认按键是否按下,没有按下则退出
		{
			while(!KEY_ADD);//如果确认按下按键等待按键释放,没有释放则一直等待
			{
				if(num<999)    //加操作
				{
					num++;
					ISendStr(0xae,0,&num,1);  //写入24c02 
					DelayMs(10);
				}

	 		}
		}
	}

	if(!KEY_DEC)  //如果检测到低电平,说明按键按下
    {
		DelayMs(10); //延时去抖,一般10-20ms
		if(!KEY_DEC)     //再次确认按键是否按下,没有按下则退出
		{
			while(!KEY_DEC);//如果确认按下按键等待按键释放,没有释放则一直等待
			{
				if(num>0)  //减操作
				{
					num--;
					ISendStr(0xae,0,&num,1);  //写入24c02
					DelayMs(10);
				}

	 		}
		}
	}
	TempData[0]=DuanMa[num/100];    
	TempData[1]=DuanMa[(num%100)/10];
	TempData[2]=DuanMa[(num%100)%10];	
  }
}

  • 2
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值