最近单片机有一个实验挺有意思的,使用51单片机以总线的方式读取adc0809的数据

先补充点关于单片机总线的预备知识,我一开始不了解总线的时候做这个实验也是很懵逼的。

单片机的三总线结构

51单片机有三条总线:数据总线、地址总线、控制总线

从图中可以看出,8位数据总线由P0组成,16位地址总线由P0和P2组成,控制总线由P3和相关引脚组成

采用总线的方式可以简化编程,节省I/O口,便于外设扩展

但是数据口和地址口在P0是怎么复用的呢,这就需要看到时序了

从图中可以看出,P0口是数据/地址分时复用的,这是P0口内部的复用结构完成的

实操练习

51单片机与adc0809接线原理图如下

解释电路

P2.7口用作adc0809的选择线

P0.0~P0.2所接的A B C是adc0809的IN0通道选择线

接下来就是计算adc0809的地址了

P2 P0

0xxx xxxx xxxx x000

因此地址为0x7ff8

遇到的问题

本来应该显示5v的位置只显示1.144v,而且在电阻增大的过程中,显示的值先减小后增大又减小,具体情况如图

实在没有办法的情况下,借别的同学的代码来看,没发现自己的程序在时序、地址上的错误。

琢磨了单片机的数值变换的现象后,突然觉得是不是保存ad转换数值的变量溢出了,然后就发现我的变量类型是int,而别人的变量类型是long int

在将保存ad转换的变量类型修改过后,程序就运行正常了

程序代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
#include <reg51.h>
#include <absacc.h>

typedef unsigned char uchar;
typedef unsigned int uint8;
typedef unsigned long int uint16;

uchar led_mod[] = { 0x3f,0x06,0x5b,0x4f, //!< 数码管编码
0x66,0x6d,0x7d,0x07,
0x7f,0x6f,0x77,0x7c,
0x39,0x5e,0x79,0x71};

#define AD_IN0 XBYTE [0x7ff8]

sbit EOC = P3^5;
sbit CLK = P3^3;

sbit seg1 = P2^0;
sbit seg2 = P2^1;
sbit seg3 = P2^2;
sbit seg4 = P2^3;

uint16 adc_data = 0; //> 保存ad转换结果

/**
* @brief 延迟函数
*
*/
void delay_ms(uint8 time)
{
uint8 j;
for (; time>0; time--)
{
for(j=114; j>0; j--);
}
}

/**
* @brief 定时器初始化
*
*/
void timer_init(void)
{
TMOD |= 0x02;
TH0 = 200; //> 产生方波周期2us
TL0 = 200;

ET0 = 1;
TR0 = 1;
}

/**
* @brief 数码管动态显示函数
*
*/
void display(void)
{
adc_data = adc_data*1000/51; //> 分辨率为5/256约为1/51
P1 = 0x00;
P1 = led_mod[adc_data/1000]|0x80;
seg1 = 0;
delay_ms(2);
seg1 = 1;
P1 = 0x00;
P1 = led_mod[(adc_data%1000)/100];
seg2 = 0;
delay_ms(2);
seg2 = 1;
P1 = 0x00;
P1 = led_mod[(adc_data%100)/10];
seg3 = 0;
delay_ms(2);
seg3 = 1;
P1 = 0x00;
P1 = led_mod[adc_data%10];
seg4 = 0;
delay_ms(2);
seg4 = 1;
}

void main()
{
timer_init();
EA = 1;
while(1)
{
AD_IN0 = 0;
while(EOC == 0);
adc_data = AD_IN0;
display();
}
}

/**
* @brief 产生时钟周期
*
*/
void timer0() interrupt 1
{
CLK = ~CLK;
}