16 bit float 存储_说一说学51时候躺过的坑——bit和sbit的区别

6bb8cbcc10ef64f69e813afd5e0030cc.png

什么是bit和sbit?他们有什么区别?

在刚学51单片机的时候刚看到这个两个东西的时候就是一脸懵逼,脑子里面第一反应就是,这是什么鬼啊,C语言里面没说过啊,于是去B站看了51单片机视频教程,终于知道了这是一种数据类型,类似于int、float一样,但是某天自己写代码使用到的时候就一脸懵逼了,如下面函数返回值的类型应该怎么决定?为什么必须要这样?

函数返回值是应该这样写呢?

c85b6aceed68ecf6edee280d657cc352.png

还是应该这样写?

040c9c38e9688579940cce838aaab26a.png

于是,好奇的我两种情况都试了试,但是事实却是试试就逝世

试第二种的时候编译器报错了

d96afdf02fdad0b4e372a4ed174c3444.png

这又是什么情况?难度两个不一样吗?

于是查了一下资料


bit和sbit都是C51扩展的变量类型。

bit

bit编译时分配空间

sfr(特殊功能寄存器)的bit。SFR是系统指定的内存地址。

bit 动态分配的,有编译器来指定内存地址。

bit 可以在外部或内部定义。

bit和int char之类的差不多,只不过char=8位, bit="1位而已"。都是变量,编译器在编译过程中分配地址。除非你指定,否则这个地址是随机的。这个地址是整个可寻址空间,RAM+FLASH+扩展空间。bit只有0和1两种值,意义有点像Windows下VC中的BOOL。

bit是编译器在的可寻址区分配的一个位变量,是不定的,不是绝对地址目标。

sbit

sbit 只能在外部定义全局变量。

sbit 要在最外面定义,就是说必须定义成外部变量.sbit定义的是SFR(特殊功能寄存器)的

bit

sbit更像是类型定义,不像是变量定义。

sbit: 指示说明性说明

所有可位寻址的位都可由sbit指定,这包括可位寻址区和SFR中的位。

sbit是对应可位寻址空间的一个位,可位寻址区:20H~2FH。一旦用了sbi xxx = REGE^6这样的定义,这个sbit量就确定地址了。sbit大部分是用在寄存器中的,方便对寄存器的某位进行操作的。

sbit位寄存器是可位寻址的绝对地址目标,定义后编译器是不会改变位置的。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是使用DS1302实时时钟模块、LCD1602液晶屏、DS18B20温度传感器和按键,写成带闹钟的可调时钟代码,适用于51单片机。代码注释中有详细的明,希望能够帮到你! ``` #include <reg52.h> //引用51单片机头文件 #include <intrins.h> //引用延时函数头文件 //定义液晶屏引脚 sbit LCD_RS = P2^6; sbit LCD_RW = P2^5; sbit LCD_EN = P2^7; sbit LCD_D4 = P0^4; sbit LCD_D5 = P0^5; sbit LCD_D6 = P0^6; sbit LCD_D7 = P0^7; //定义DS1302实时时钟模块引脚 sbit DS1302_CE = P1^0; sbit DS1302_CLK = P1^1; sbit DS1302_DAT = P1^2; //定义温度传感器引脚 sbit DS18B20_IO = P3^4; //定义闹钟时间 unsigned char alarmHour = 8; unsigned char alarmMinute = 0; //定义闹钟标志 bit alarmSet = 0; //函数声明 void InitLCD(); void WriteCommand(unsigned char Command); void WriteData(unsigned char Data); void LCD1602_Display(unsigned char X,unsigned char Y,unsigned char *Text); void DS1302_WriteByte(unsigned char dat); unsigned char DS1302_ReadByte(); void DS1302_Write(unsigned char address,unsigned char dat); unsigned char DS1302_Read(unsigned char address); void DS1302_SetTime(unsigned char year,unsigned char month,unsigned char day,unsigned char hour,unsigned char minute,unsigned char second); void DS1302_GetTime(unsigned char *year,unsigned char *month,unsigned char *day,unsigned char *hour,unsigned char *minute,unsigned char *second); void DS18B20_Init(); float DS18B20_GetTemp(); void main() { //初始化LCD1602液晶屏 InitLCD(); //初始化DS1302实时时钟模块 DS1302_SetTime(21, 8, 15, 8, 0, 0); //设置初始时间为2021年8月15日8时0分0秒 //初始化DS18B20温度传感器 DS18B20_Init(); while (1) { //获取当前时间和温度 unsigned char year, month, day, hour, minute, second; DS1302_GetTime(&year, &month, &day, &hour, &minute, &second); float temperature = DS18B20_GetTemp(); //显示时间和温度 unsigned char text[16]; sprintf(text, "Time: %02d:%02d", hour, minute); LCD1602_Display(0, 0, text); sprintf(text, "Temp: %.1fC", temperature); LCD1602_Display(0, 1, text); //检查是否到达闹钟时间 if (hour == alarmHour && minute == alarmMinute && !alarmSet) { //闹钟时间到,显示闹钟提示 LCD1602_Display(0, 0, "Alarm!"); LCD1602_Display(0, 1, "Press reset"); alarmSet = 1; //设置闹钟标志 } //检查是否按下重置按钮 if (!P3_0 && alarmSet) { //重置闹钟 alarmSet = 0; } //检查是否按下设置按钮 if (!P3_1) { //进入设置模式 unsigned char hourSetting = hour; unsigned char minuteSetting = minute; while (!P3_1) { //等待设置完成 if (!P3_2) { //按下确认按钮,保存设置 DS1302_SetTime(year, month, day, hourSetting, minuteSetting, 0); break; } //调整时间 if (!P3_3) { //按下小时调整按钮 hourSetting++; if (hourSetting > 23) { hourSetting = 0; } sprintf(text, "Set Time: %02d:%02d", hourSetting, minuteSetting); LCD1602_Display(0, 0, text); } if (!P3_4) { //按下分钟调整按钮 minuteSetting++; if (minuteSetting > 59) { minuteSetting = 0; } sprintf(text, "Set Time: %02d:%02d", hourSetting, minuteSetting); LCD1602_Display(0, 0, text); } _nop_(); } } } } //LCD1602初始化函数 void InitLCD() { WriteCommand(0x38); //设置16*2显示,5*7点阵,8位数据 WriteCommand(0x0c); //显示器开,光标不显示 WriteCommand(0x06); //光标右移,字符不移动 WriteCommand(0x01); //清屏 } //LCD1602发送命令函数 void WriteCommand(unsigned char Command) { LCD_RS = 0; LCD_RW = 0; LCD_D7 = Command & 0x80; LCD_D6 = Command & 0x40; LCD_D5 = Command & 0x20; LCD_D4 = Command & 0x10; LCD_EN = 1; _nop_(); _nop_(); LCD_EN = 0; _nop_(); _nop_(); LCD_D7 = Command & 0x08; LCD_D6 = Command & 0x04; LCD_D5 = Command & 0x02; LCD_D4 = Command & 0x01; LCD_EN = 1; _nop_(); _nop_(); LCD_EN = 0; _nop_(); _nop_(); _nop_(); } //LCD1602发送数据函数 void WriteData(unsigned char Data) { LCD_RS = 1; LCD_RW = 0; LCD_D7 = Data & 0x80; LCD_D6 = Data & 0x40; LCD_D5 = Data & 0x20; LCD_D4 = Data & 0x10; LCD_EN = 1; _nop_(); _nop_(); LCD_EN = 0; _nop_(); _nop_(); LCD_D7 = Data & 0x08; LCD_D6 = Data & 0x04; LCD_D5 = Data & 0x02; LCD_D4 = Data & 0x01; LCD_EN = 1; _nop_(); _nop_(); LCD_EN = 0; _nop_(); _nop_(); _nop_(); } //LCD1602显示函数 void LCD1602_Display(unsigned char X,unsigned char Y,unsigned char *Text) { unsigned char i; if (X > 15) { X = 0; Y++; } if (Y > 1) { Y = 0; } WriteCommand(0x80 + Y * 0x40 + X); for (i = 0; Text[i] != '\0'; i++) { WriteData(Text[i]); } } //DS1302写入一个字节函数 void DS1302_WriteByte(unsigned char dat) { unsigned char i; for (i = 0; i < 8; i++) { if (dat & 0x01) { DS1302_DAT = 1; } else { DS1302_DAT = 0; } DS1302_CLK = 1; _nop_(); _nop_(); dat >>= 1; DS1302_CLK = 0; _nop_(); _nop_(); } } //DS1302读取一个字节函数 unsigned char DS1302_ReadByte() { unsigned char i; unsigned char dat = 0; for (i = 0; i < 8; i++) { dat >>= 1; if (DS1302_DAT) { dat |= 0x80; } DS1302_CLK = 1; _nop_(); _nop_(); DS1302_CLK = 0; _nop_(); _nop_(); } return dat; } //DS1302写入一个寄存器函数 void DS1302_Write(unsigned char address,unsigned char dat) { DS1302_CE = 0; _nop_(); DS1302_CLK = 0; _nop_(); DS1302_CE = 1; _nop_(); DS1302_WriteByte(address); DS1302_WriteByte(dat); DS1302_CE = 0; } //DS1302读取一个寄存器函数 unsigned char DS1302_Read(unsigned char address) { unsigned char dat; DS1302_CE = 0; _nop_(); DS1302_CLK = 0; _nop_(); DS1302_CE = 1; _nop_(); DS1302_WriteByte(address); dat = DS1302_ReadByte(); DS1302_CE = 0; return dat; } //DS1302设置时间函数 void DS1302_SetTime(unsigned char year,unsigned char month,unsigned char day,unsigned char hour,unsigned char minute,unsigned char second) { DS1302_Write(0x8e,

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值