当前位置:文档之家› 51单片机实验报告

51单片机实验报告

51单片机实验报告实验一点亮流水灯实验现象Led灯交替亮,间隔大约10ms。

实验代码#include <reg51.h>void Delay10ms(unsigned int c); void main(){while(1){P0 = 0x00;Delay10ms(50);P0 = 0xff;Delay10ms(50);}}void Delay10ms(unsigned int c){unsigned char a, b;for (;c>0;c--){for (b=38;b>0;b--){for (a=130;a>0;a--);}}}实验原理While(1)表示一直循环。

循环体首先将P0的所有位都置于零,然后延时约50*10=500ms,接着P0位全置于1,于是LED全亮了。

接着循环,直至关掉电源。

延迟函数是通过多个for循环实现的。

实验2 流水灯(不运用库函数)实验现象起初led只有最右面的那一个不亮,半秒之后从右数第二个led也不亮了,直到最后一个也熄灭,然后led除最后一个都亮,接着上述过程#include <reg52.h>#include <intrins.h>void Delay10ms(unsigned int c);main(){unsigned char LED;LED = 0xfe;while (1){P0 = LED;Delay10ms(50);LED = LED << 1;if (P0 == 0x00){LED = 0xfe;}}}void Delay10ms(unsigned int c){unsigned char a, b;for (;c>0;c--){for (b=38;b>0;b--){for (a=130;a>0;a--);}}}实验原理这里运用了C语言中的位运算符,位运算符左移,初始值的二进制为1111 1110,之后左移一次变成1111 1100,当变成0000 0000时通过if语句重置1111 11110.延迟函数在第一个报告已经说出了,不再多说。

实验3流水灯(库函数版)实验现象最开始还是最右边的一个不亮,然后不亮的灯转移到最右边的第二个,此时第一个恢复亮度,这样依次循环。

实验代码#include <reg51.h>#include <intrins.h>void Delay10ms(unsigned int c); void main(void){unsigned char LED;LED = 0xFE;while(1){P0 = LED;Delay10ms(50);LED = _crol_(LED,1);}}void Delay10ms(unsigned int c) {unsigned char a, b;for (;c>0;c--){for (b=38;b>0;b--){for (a=130;a>0;a--);}}}实验原理利用头文件中的函数,_crol_( , ),可以比位操作符更方便的进行2进制的移位操作,比位操作符优越的是,该函数空位补全时都是用那个移位移除的数据,由此比前一个例子不需要if语句重置操作。

数码管实验实验现象单个数码管按顺序显示0-9和A-F。

#include<reg51.h>void Delay10ms(unsigned int c);unsigned char code DIG_CODE[16]={0x3F, 0x06, 0x5B, 0x4F, 0x66, 0x6D, 0x7D, 0x07,0x7F, 0x6F, 0x77, 0x7C, 0x39, 0x5E, 0x79, 0x71};void main(void){unsigned char i = 0;while(1){P0= ~DIG_CODE[i];i++;if(i == 16){i = 0;}Delay10ms(50);}}void Delay10ms(unsigned int c) //Îó²î 0us{unsigned char a, b;for (;c>0;c--){for (b=38;b>0;b--){for (a=130;a>0;a--);}}}实验原理根据数码管的点亮原理,依次找到代表0-9,A-F的位码,用循环和延迟函数就可以达到要求了。

实验动态数码管#include<reg51.h>#define GPIO_DIG P0#define GPIO_PLACE P1unsigned char code DIG_PLACE[8] = { 0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f}; unsigned char code DIG_CODE[17] = { 0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07, 0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71}; unsigned char DisplayData[8];void DigDisplay();void main(void){unsigned char i;for(i=0; i<8; i++){DisplayData[i] = DIG_CODE[i];}while(1){DigDisplay();}}void DigDisplay(){unsigned char i;unsigned int j;for(i=0; i<8; i++){GPIO_PLACE = DIG_PLACE[i];GPIO_DIG = DisplayData[i];j = 10;while(j--);GPIO_DIG = 0x00; }}实验原理依然找到相应数字和字母的编码,由于必须通过快速扫描利用视觉暂留来实现数码管的显示,分段码和位码,不断扫描。

最后如果更换数字的话,需要消隐操作,防止数码管重复显示所带来的不清楚。

实验外部中断实验现象每按一下独立按键,就会在数码管显示屏上+1。

#include < reg51.h >#include <intrins.h>sbit LS138A=P2^2;sbit LS138B=P2^3;sbit LS138C=P2^4;unsigned int LedNumVal_1,LedNumVal_2,LedOut[8]; Unsigned char code Disp_Tab[]= {0x3f ,0x06 , 0x5b ,0x4f, 0x66 , 0x6d , 0x7d , 0x07 , 0x7f , 0x6f , 0x40};void delay(unsigned int i){char j;for(i; i > 0; i--)for(j = 200; j > 0; j--);}void main(void){ unsigned char i;P0=0xff;P1=0xff;P2=0xff;IT0=1;EX0=1;IT1=1;EX1=1;EA=1;while(1){LedOut[0]=Disp_Tab[LedNumVal_1%10000/1000]; LedOut[1]=Disp_Tab[LedNumVal_1%1000/100]|0x80;LedOut[2]=Disp_Tab[LedNumVal_1%100/10]; LedOut[3]=Disp_Tab[LedNumVal_1%10];LedOut[4]=Disp_Tab[LedNumVal_2%10000/1000]; LedOut[5]=Disp_Tab[LedNumVal_2%1000/100]; LedOut[6]=Disp_Tab[LedNumVal_2%100/10]; LedOut[7]=Disp_Tab[LedNumVal_2%10];for( i=0; i<8; i++){ P0 = LedOut[i];switch(i){case 0:LS138A=0; LS138B=0; LS138C=0; break;case 1:LS138A=1; LS138B=0; LS138C=0; break;case 2:LS138A=0; LS138B=1; LS138C=0; break;case 3:LS138A=1; LS138B=1; LS138C=0; break;case 4:LS138A=0; LS138B=0; LS138C=1; break;case 5:LS138A=1; LS138B=0; LS138C=1; break;case 6:LS138A=0; LS138B=1; LS138C=1; break;case 7:LS138A=1; LS138B=1; LS138C=1; break; }delay(150);}}}void counter0(void) interrupt 0 using 1{EX0=0;LedNumVal_1++;EX0=1;}void counter1(void) interrupt 2 using 2{EX1=0;LedNumVal_2++;EX1=1;}实验原理对于数码管的显示采用138译码器,通过switch语句与数字一一对应,通过P3.2 P3.3外部中断接口使数码管成功计数。

外部中断函数为INT0与INT1。

相关主题