如何采用51单片机连接24C02-C语言测试代码

2020-05-23来源: eefocus关键字:51单片机  24C02  测试代码

我总结3点需要注意的地方

1.关闭非IIC通信器件,比如我的开发板SDA和SCL也连接了DS1302,造成干扰会没有结果。


2.IIC通信的应答,发送端在SCL为0时将SDA置1,等待接收端拉低SDA;接收端在拉低SDA持续一个周期后,应将SDA置1释放总线。


主机作为发送端等待应答

如何采用51单片机连接24C02-C语言测试代码

SDA=1;

SCL=1;

while(SDA);

SCL=0;

主机作为接收端,进行应答或不应答

SDA=0; //不应答则为1

SCL=1;

SCL=0;

SDA=1;


3.24C02在写入周期完成后还有有内部写入时间Twr,所以可以查手册加入适当时间的延时函数,或者使用查询应答的方式进行延时等待。


总之,关键是注意参考数据手册及IIC总线规范,尤其是芯片的特别说明以及时序的控制,再结合代码就能弄懂了。


下面是我写的代码,用上了所有的功能,包括 Current Read 对应的函数 readnext() 。多字节写入时不用考虑分页问题,函数自动解决,直接给数组即可。使用数码管显示结果。


1 #include

2

3 sbit RST=P2^4; //用来关闭ds1302

4

5

6 sbit SDA=P2^0;

7 sbit SCL=P2^1;

8 //内部使用的函数

9 bit start(unsigned char dev);

10 void stop(void);

11 bit write_byte(unsigned char dat);

12 unsigned char read_byte(bit ack);

13 void waitack(void);

14 //外部使用

15 void writes(unsigned char address,unsigned char dat[],unsigned char num);

16 void reads(unsigned char address,unsigned char dat[],unsigned char num);

17 unsigned char read(unsigned char address);

18 void write(unsigned char address,unsigned char dat);

19 unsigned char readnext(void);

20

21 /*数码管部分*/

22 sbit du=P2^6;

23 sbit we=P2^7;

24 void display(void);

25 void delayms(unsigned time);

26 unsigned char code table[]={

27 0x3f,0x06,0x5b,0x4f,

28 0x66,0x6d,0x7d,0x07,

29 0x7f,0x6f,0x77,0x7c,

30 0x39,0x5e,0x79,0x71};

31 unsigned char num[6]={0};

32

33

34 void main(void)

35 {

36 //测试用数据

37 unsigned char a[9]={2,3,5,7,13,15,11,4,8};

38 unsigned char b[13]={0};

39

40 RST=0; //关闭ds1302消除影响

41

42 write(11,6);

43 write(12,1);

44 writes(2,a,9);

45 b[9]=readnext(); //b[9]=6

46 b[10]=readnext(); //b[10]=1

47 reads(2,b,9); //b[0-8]=a[0-8]

48 b[11]=readnext(); //b[11]=6

49 b[12]=read(12); //b[12]=1

50

51 num[5]=b[3];

52 num[4]=b[6];

53 num[3]=b[9];

54 num[2]=b[10];

55 num[1]=b[11];

56 num[0]=b[12];

57

58 while(1)

59 {

60 display();

61 }

62 }

63

64 //传入设备地址,返回设备是否应答

65 bit start(unsigned char dev)

66 {

67 SDA=1;

68 SCL=1;

69 SDA=0;

70 SCL=0;

71 return write_byte(dev);

72 }

73

74 void stop(void)

75 {

76 SDA=0;

77 SCL=1;

78 SDA=1;

79 }

80

81 //传入要写入的字节,返回设备是否应答

82 bit write_byte(unsigned char dat)

83 {

84 unsigned char i=8;

85 bit ack;

86 while(i--)

87 {

88 dat《《=1;

89 SDA=CY;

90 SCL=1;

91 SCL=0;

92 }

93 SDA=1; //接收设备应答

94 SCL=1;

95 ack=~SDA;

96 SCL=0;

97 return ack;

98 }

99

100 //在写入后等待24c02完成内部写入

101 //恢复响应的时间为手册中的twr

102 void waitack(void)

103 {

104 while(!start(0xa0));

105 stop();

106 }

107

108 //传入是否应答设备,返回读取的字节

109 unsigned char read_byte(bit ack)

110 {

111 unsigned char i=8,ret;

112

113 while(i--)

114 {

115 ret《《=1;

116 SCL=1;

117 ret|=SDA;

118 SCL=0;

119 }

120 SDA=~ack; //应答或不应答设备

121 SCL=1;

122 SCL=0;

123 SDA=1; //应答时要注意的时序

124 return ret;

125 }

126

127 //传入写入地址,数组,写入字节数

128 //函数自动进行分页写入

129 void writes(unsigned char address,unsigned char dat[],unsigned char num)

130 {

131 unsigned char i;

132

133 for(i=0;i

134 {

135 start(0xa0);

136 write_byte(address);

137 do

138 write_byte(dat[i++]);

139 while(++address&0x07 && i

140 stop();

141 waitack(); //延时等待以完成内部写入

142 }

143 }

144

145 //传入读取地址,接收用的数组,读取字节数

146 void reads(unsigned char address,unsigned char dat[],unsigned char num)

147 {

148 unsigned char i;

149

150 start(0xa0);

151 write_byte(address);

152

153 start(0xa1);

154

155 for(i=0;i

156 dat[i]=read_byte(1);

157

158 dat[i]=read_byte(0);

159 stop();

160 }

161

162 void write(unsigned char address,unsigned char dat)

163 {

164 writes(address,&dat,1);

165 }

166

167 unsigned char read(unsigned char address)

168 {

169 unsigned char ret;

170 reads(address,&ret,1);

171 return ret;

172 }

173

174 //对应手册中的Current Read

175 unsigned char readnext(void)

176 {

177 unsigned char ret;

178 start(0xa1);

179 ret=read_byte(0);

180 stop();

181 return ret;

182 }

183

184 void delayms(unsigned TIme)

185 {

186 unsigned i,j;

187

188 for(i=TIme;i》0;i--)

189 for(j=110;j》0;j--)

190 ;

191 }

192

193 void display(void)

194 {

195 unsigned char i;

196

197 for(i=0;i《6;i++)

198 {

199 P0=0;

200 du=1;

201 du=0;

202

203 P0=~(0x20》》i);

204 we=1;

205 we=0;

206

207 P0=table[num[i]];

208 du=1;

209 du=0;

210

211 delayms(1);

212 }

213 }

关键字:51单片机  24C02  测试代码 编辑:什么鱼 引用地址:http://news.eeworld.com.cn/mcu/ic498006.html 本网站转载的所有的文章、图片、音频视频文件等资料的版权归版权所有人所有,本站采用的非本站原创文章及图片等内容无法一一联系确认版权者。如果本网所选内容的文章作者及编辑认为其作品不宜公开自由传播,或不应无偿使用,请及时通过电子邮件或电话通知我们,以迅速采取适当措施,避免给双方造成不必要的经济损失。

上一篇:基于51单片机可修改错误功能的计算器工作原理解析
下一篇:基于51单片机对1602液晶板的并行操作

关注eeworld公众号 快捷获取更多信息
关注eeworld公众号
快捷获取更多信息
关注eeworld服务号 享受更多官方福利
关注eeworld服务号
享受更多官方福利

推荐阅读

51单片机实验8:led点阵(1):点亮一个点
开发板led点阵模块电路图如下:74HC595:74HC595是一个8位串行输入、并行输出的位移缓存器。芯片第11角为数据输入时钟线,上升沿有效。芯片第12脚为输出存储器锁存时钟线,上升沿有效。芯片第13脚为输出有效(低电平)。芯片第14脚为串行数据输入。为表示出输入74HC595的8位二进制数,开发板加入了led模块(图一中绿色所示)。若要使led发光,则需将JP595接vcc。OE为输出有效控制端,低电平有效,所以务必将JOE短接片短接到GND端。_nop_();函数为延时一个机器周期,所对应头文件为intrins.h#include<reg52.h>#include<intrins.h>#define
发表于 2020-05-08
51单片机实验8:led点阵(1):点亮一个点
51单片机 4个独立按键控制LED灯 (protues仿真)(C语言版)
}}void main(){while(1){key();}}四、运行程序后成功后,打开proteus右键单击单片机AT89C51,找到编辑,然后添加hex文件
发表于 2020-05-08
51单片机 4个独立按键控制LED灯 (protues仿真)(C语言版)
51单片机课程设计——led点阵广告牌程序设计
上学期期末的课设题目是led点阵广告牌,当时的要求如下:(1)能够显示不同字符的LED点阵广告牌;(2)按键切换不同的显示效果(如闪烁,静止,平移等);(3)按键切换不同的显示内容;(4)能够显示图形或自定义字符;(5)其他功能(创新部分);(6)系统调试、分析、总结与功能实现。当时用的是我用是的普中科技的STC90C51RD+的单片机,不过只要是51单片机,换哪个51内核的芯片都可以,只要电路和程序匹配就可以。我用的那款普中科技的51单片机可以直接连线决定线路的布置,不需要自己去重新焊一个电路板。然后我根据任务要求连接了电路,写了对应的程序,还有proteus仿真。另外,觉得有趣可以点个赞;有什么有趣的想法可以评论一下,我感兴趣
发表于 2020-05-08
51单片机课程设计——led点阵广告牌程序设计
51单片机 基于LED点阵的9~0倒计数
#include <reg52.h>sbit ADDR0 = P1^0;sbit ADDR1 = P1^1;sbit ADDR2 = P1^2;sbit ADDR3 = P1^3;sbit ENLED = P1^4;unsigned char code image[11][8] = {{0xC3, 0x81, 0x99, 0x99, 0x99, 0x99, 0x81, 0xC3}, //数字0{0xEF, 0xE7, 0xE3, 0xE7, 0xE7, 0xE7, 0xE7, 0xC3}, //数字1{0xC3, 0x81, 0x9D, 0x87, 0xC3, 0xF9, 0xC1, 0x81}, //数字2{0xC3,
发表于 2020-05-07
基于51单片机智能红外温控风扇设计
1、设计需求及目标本产品采用单片机+最小系统+数码管显示模块+数码管驱动模块+温度采集模块+人体感应模块+风扇模块+按键模块等构成;功能描述:(1)采用DS18B20温度传感器测温,人体感应模块检测是否有人。(2)共3个按键:1键切换/设置、2键加、3键减。(3)本设计共三种模式:自动模式、手动模式和自然风模式。(4)自动模式,按一下1键可以设置温度上限,再按下设置温度下限,均可以按键加减调整。数码管第一位不显示,后三位显示温度值。人体感应模块检测有人时,温度小于下限风扇不转,温度在上下限之间50%转动,大于上限时,风扇全速转动。人离开后,延迟几秒风扇停止转动,起到节能环保的作用。(5)手动模式,数码管第一位显示风扇档位,后三位
发表于 2020-05-07
基于51单片机智能红外温控风扇设计
C51单片机数码管动态显示
数码管作为最廉价的输出设备,在各种自动化设备中有很大的应用,最简单普通的显示方式为动态刷新显示,称为假动态显示,即通过分时扫描每一位,利于人眼的视觉停留现象,造成一种静态显示的效果,如下图所示:C51单片机由于运行速度很慢,在高刷新频率下,单片机的资源耗费很厉害,这样单片机就不可以再进行大量的计算工作,实际上,单片机在刷新时,只需要周期性的改变GPIO口的状态就可以了,剩下的时间其实都是在空转的状态下,我们能不能将这个空转的状态拿来用呢?当然是可以的啦,这里,我们利用单片机的定时器周期地产能中断,在中断内进行数码管的刷新工作,就可以将等待中断的这个CPU时间拿来做别的事情了。硬件电路:代码贴过来:主函数#include "
发表于 2020-05-07
C51单片机数码管动态显示
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

电子工程世界版权所有 京ICP证060456号 京ICP备10001474号 电信业务审批[2006]字第258号函 京公海网安备110108001534 Copyright © 2005-2020 EEWORLD.com.cn, Inc. All rights reserved