打印
[单片机资料]

单片机编程80实例

[复制链接]
1108|3
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
罗小群|  楼主 | 2023-11-21 11:56 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
【建议收藏】单片机编程实例80例大全
*实例1:

使用P3口流水点亮8位LED

#include<reg51.h> //包含单片机寄存器的头文件 / 函数功能:延时一段时间

void delay(void)

{

unsigned char i,j;

for(i=0;i<250;i++)

for(j=0;j<250;j++)

;

}

/ 函数功能:主函数

void main(void)

{

while(1)

{

P3=0xfe; //第一个灯亮

delay; //调用延时函数

P3=0xfd; //第二个灯亮

delay; //调用延时函数

P3=0xfb; //第三个灯亮

delay; //调用延时函数

P3=0xf7; //第四个灯亮

delay; //调用延时函数

P3=0xef; //第五个灯亮

delay; //调用延时函数

P3=0xdf; //第六个灯亮

delay; //调用延时函数

P3=0xbf; //第七个灯亮

delay; //调用延时函数

P3=0x7f; //第八个灯亮

delay; //调用延时函数

} / /

}

*实例2:

通过对P3口地址的操作流水点亮8位LED

#include<reg51.h> //包含单片机寄存器的头文件

sfr x=0xb0; //P3口在存储器中的地址是b0H, 通过sfr可定义8051内核单片机

//的所有内部8位特殊功能寄存器,对地址x的操作也就是对P1口的操作

/

函数功能:延时一段时间

/

void delay(void)

{

unsigned char i,j;

for(i=0;i<250;i++)

for(j=0;j<250;j++)

; //利用循环等待若干机器周期,从而延时一段时间

}

/

函数功能:主函数

/

void main(void)

{

while(1)

{

x=0xfe; //第一个灯亮

delay; //调用延时函数

x=0xfd; //第二个灯亮

delay; //调用延时函数

x=0xfb; //第三个灯亮

delay; //调用延时函数

x=0xf7; //第四个灯亮

delay; //调用延时函数

x=0xef; //第五个灯亮

delay; //调用延时函数

x=0xdf; //第六个灯亮

delay; //调用延时函数

x=0xbf; //第七个灯亮

delay; //调用延时函数

x=0x7f; //第八个灯亮

delay; //调用延时函数

}

}

*实例3:用不同数据类型控制灯闪烁时间

#include<reg51.h> //包含单片机寄存器的头文件

/

函数功能:用整形数据延时一段时间

/

void int_delay(void) //延时一段较长的时间

{

unsigned int m; //定义无符号整形变量,双字节数据,值域为0~65535 for(m=0;m<36000;m++)

; //空操作

}

/

函数功能:用字符型数据延时一段时间

/

void char_delay(void) //延时一段较短的时间

{

unsigned char i,j; //定义无符号字符型变量,单字节数据,值域0~255 for(i=0;i<200;i++)

for(j=0;j<180;j++)

; //空操作

}

/

函数功能:主函数

/

void main(void)

{

unsigned char i;

while(1)

{

for(i=0;i<3;i++)

{

P1=0xfe; //P1.0口的灯点亮

int_delay; //延时一段较长的时间

P1=0xff; //熄灭

int_delay; //延时一段较长的时间

}

for(i=0;i<3;i++)

{

P1=0xef; //P1.4口的灯点亮

char_delay; //延时一段较长的时间

} P1=0xff; //熄灭 char_delay; //延时一段较长的时间 } }

*实例4:用单片机控制第一个灯亮

#include<reg51.h> //包含51单片机寄存器定义的头文件

void main(void)

{

P1=0xfe; //P1=1111 1110B,即P1.0输出低电平

}

*实例5:用单片机控制一个灯闪烁:认识单片机的工作频率 #include<reg51.h> //包含单片机寄存器的头文件

/

函数功能:延时一段时间

/

void delay(void) //两个void意思分别为无需返回值,没有参数传递 {

unsigned int i; //定义无符号整数,最大取值范围65535 for(i=0;i<20000;i++) //做20000次空循环

; //什么也不做,等待一个机器周期

}

/ 函数功能:主函数 (C语言规定必须有也只能有1个主函数) / void main(void)

{

while(1) //无限循环

{

P1=0xfe; //P1=1111 1110B, P1.0输出低电平

delay; //延时一段时间

P1=0xff; //P1=1111 1111B, P1.0输出高电平

delay; //延时一段时间

}

}

*实例6:将 P1口状态分别送入P0、P2、P3口:认识I/O口

的引脚功能

#include<reg51.h> //包含单片机寄存器的头文件

/ 函数功能:主函数 (C语言规定必须有也只能有1个主函数) /

void main(void)

{

while(1) //无限循环

{

P1=0xff; // P1=1111 1111B,熄灭LED

P0=P1; // 将 P1口状态送入P0口

P2=P1; // 将 P1口状态送入P2口

P3=P1; // 将 P1口状态送入P3口

}

}

*实例7:用P0口、P1 口分别显示加法和减法运算结果 #include<reg51.h>

void main(void)

{

unsigned char m,n;

m=43; //即十进制数2x16+11=43

n=60; //即十进制数3x16+12=60

P1=m+n; //P1=103=0110 0111B,结果P1.3、P1.4、P1.7 口的灯被点亮 P0=n-m; //P0=17=0001 0001B,结果P0.0、P0.4的灯被熄灭 }

*实例8:用P0、P1口显示乘法运算结果

#include<reg51.h> //包含单片机寄存器的头文件

void main(void)

{

unsigned char m,n;

unsigned int s;

m=64;

n=71;

s=m n; //s=64 71=4544,需要16位二进制数表示,高8位送P1口,低8位送P0口

//由于4544=17 256+192=H3 16 16 16+H2 16 16+H1 16+H0

//两边同除以256,可得17+192/256=H3 16+H2+(H1 16+H0)/256

//因此,高8位16进制数H3 16+H2必然等于17,即4544除以256的商

//低8位16进制数H1 16+H0必然等于192,即4544除以256的余数

P1=s/256; //高8位送P1口 ,P1=17=11H=0001 0001B, P1.0和P1.4口灭,其余亮

P0=s%256; //低8位送P0口 , P3=192=c0H=1100 0000B,P3.1,P3.6,P3.7口灭,其余亮

}

*实例9:用P1、P0口显示除法运算结果

#include<reg51.h> //包含单片机寄存器的头文件

void main(void)

{

P1=36/5; //求整数

P0=((36%5) 10)/5; //求小数

while(1)

; //无限循环防止程序“跑飞”

}

*实例10:用自增运算控制P0口8位LED流水花样

#include<reg51.h> //包含单片机寄存器的头文件

/

函数功能:延时一段时间

/

void delay(void)

{

unsigned int i;

for(i=0;i<20000;i++)

;

}

/ 函数功能?:主函数

/ void main(void)

{

unsigned char i;

for(i=0;i<255;i++) //注意i的值不能超过255

{

P0=i; //将i的值送P0口

delay; //调用延时函数

}

}

*实例11:用P0口显示逻辑"与"运算结果 #include<reg51.h> //包含单片机寄存器的头文件 void main(void)

{

P0=(4>0)&&(9>0xab);//将逻辑运算结果送P0口

while(1)

; //设置无限循环,防止程序“跑飞”

}

*实例12:用P0口显示条件运算结果

#include<reg51.h> //包含单片机寄存器的头文件 void main(void)

{

P0=(8>4)?8:4;//将条件运算结果送P0口,P0=8=0000 1000B while(1)

; //设置无限循环,防止程序“跑飞”

}

*实例13:用P0口显示按位"异或"运算结果 #include<reg51.h> //包含单片机寄存器的头文件 void main(void)

{

P0=0xa2^0x3c;//将条件运算结果送P0口,P0=8=0000 1000B while(1)

; //设置无限循环,防止程序“跑飞”

}

*实例16:用P0显示左移运算结果

#include<reg51.h> //包含单片机寄存器的头文件 void main(void)

{

P0=0x3b<<2;//将左移运算结果送P0口,P0=1110 1100B=0xec while(1)

; //无限循环,防止程序“跑飞”

}

*实例17:"万能逻辑电路"实验

#include<reg51.h> //包含单片机寄存器的头文件

sbit F=P1^4; //将F位定义为 P1.4

sbit X=P1^5; //将X位定义为 P1.5

sbit Y=P1^6; //将Y位定义为 P1.6

sbit Z=P1^7; //将Z位定义为 P1.7

void main(void)

{

while(1)

{

F=((~X)&Y)|Z; //将逻辑运算结果赋给F

;

}

}

*实例18:用右移运算流水点亮P1口8位LED #include<reg51.h> //包含单片机寄存器的头文件

/

函数功能:延时一段时间

/

void delay(void)

{

unsigned int n;

for(n=0;n<30000;n++)

;

}

/

函数功能:主函数

/

void main(void)

{

unsigned char i;

while(1)

{

P1=0xff;

delay;

for(i=0;i<8;i++)//设置循环次数为8

{

P1=P1>>1; //每次循环P1的各二进位右移1位,高位补0 delay; //调用延时函数

}

}

}

*实例19:用if语句控制P0口8位LED的流水方向 #include<reg51.h> //包含单片机寄存器的头文件

sbit S1=P1^4; //将S1位定义为P1.4

sbit S2=P1^5; //将S2位定义为P1.5

/

函数功能:主函数

/

void main(void)

{

while(1)

}

{ if(S1==0) //如果按键S1按下 P0=0x0f; //P0口高四位LED点亮 if(S2==0) //如果按键S2按下 P0=0xf0; //P0口低四位LED点亮 }

*实例20:用swtich语句的控制P0口8位LED的点亮状态 #include<reg51.h> //包含单片机寄存器的头文件

sbit S1=P1^4; //将S1位定义为P1.4

/

函数功能:延时一段时间

/

void delay(void)

{

unsigned int n;

for(n=0;n<10000;n++)

;

}

/

函数功能:主函数

/

void main(void)

{

unsigned char i;

i=0; //将i初始化为0

while(1)

{

if(S1==0) //如果S1键按下

{

delay; //延时一段时间

if(S1==0) //如果再次检测到S1键按下

i++; //i自增1

if(i==9) //如果i=9,重新将其置为1

i=1;

}

switch(i) //使用多分支选择语句

{

}

} case 1: P0=0xfe; //第一个LED亮 break; case 2: P0=0xfd; //第二个LED亮 break; case 3:P0=0xfb; //第三个LED亮 break; case 4:P0=0xf7; //第四个LED亮 break; case 5:P0=0xef; //第五个LED亮 break; case 6:P0=0xdf; //第六个LED亮 break; case 7:P0=0xbf; //第七个LED亮 break; case 8:P0=0x7f; //第八个LED亮 break; default: //缺省值,关闭所有LED P0=0xff; }

*实例21:用for语句控制蜂鸣器鸣笛次数

#include<reg51.h> //包含单片机寄存器的头文件 sbit sound=P3^7; //将sound位定义为P3.7 / 函数功能:延时形成1600Hz音频

/ void delay1600(void)

{

unsigned char n;

for(n=0;n<100;n++)

;

}

/ 函数功能:延时形成800Hz音频

/ void delay800(void)

{

unsigned char n;

for(n=0;n<200;n++)

;

}

/ 函数功能:主函数

/ void main(void)

{

unsigned int i;

while(1)

{

for(i=0;i<830;i++)

{

sound=0; //P3.7输出低电平 delay1600;

sound=1; //P3.7输出高电平 delay1600;

}

for(i=0;i<200;i++)

{

sound=0; //P3.7输出低电平 delay800;

sound=1; //P3.7输出高电平 delay800;

}

}

}

*实例22:用while语句控制LED

#include<reg51.h> //包含单片机寄存器的头文件 / 函数功能:延时约60ms (3 100 200=60000μs) / void delay60ms(void)

{

unsigned char m,n;

for(m=0;m<100;m++)

for(n=0;n<200;n++)

;

}

/

函数功能:主函数

/

void main(void)

{

unsigned char i;

while(1) //无限循环

{

i=0; //将i初始化为0

while(i<0xff) //当i小于0xff(255)时执行循环体 {

P0=i; //将i送P0口显示

delay60ms; //延时

i++; //i自增1

}

}

}

*实例23:用do-while语句控制P0口8位LED流水点亮 #include<reg51.h> //包含单片机寄存器的头文件

/

函数功能:延时约60ms (3 100 200=60000μs)

/

void delay60ms(void)

{

unsigned char m,n;

for(m=0;m<100;m++)

for(n=0;n<200;n++)

;

}

/

函数功能:主函数

/

void main(void)

{

do

{

P0=0xfe; //第一个LED亮

delay60ms;

}

P0=0xfd; //第二个LED亮 delay60ms; P0=0xfb; //第三个LED亮 delay60ms; P0=0xf7; //第四个LED亮 delay60ms; P0=0xef; //第五个LED亮 delay60ms; P0=0xdf; //第六个LED亮 delay60ms; delay60ms; P0=0xbf; //第七个LED亮 delay60ms; P0=0x7f; //第八个LED亮 delay60ms; }while(1); //无限循环,使8位LED循环流水点亮

*实例24:用字符型数组控制P0口8位LED流水点亮 #include<reg51.h> //包含单片机寄存器的头文件

/

函数功能:延时约60ms (3 100 200=60000μs)

/

void delay60ms(void)

{

unsigned char m,n;

for(m=0;m<100;m++)

for(n=0;n<200;n++)

;

}

/

函数功能:主函数

/

void main(void)

{

unsigned char i;

unsigned char code Tab[ ]={0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f}; //定义无符号字符型数组

while(1)

{

for(i=0;i<8;i++)

{

P0=Tab[i];//依次引用数组元素,并将其送P0口显示

delay60ms;//调用延时函数

}

}

}

*实例25:用P0口显示字符串常量

#include<reg51.h> //包含单片机寄存器的头文件

/

函数功能:延时约150ms (3 200 250=150 000μs=150ms

/

void delay150ms(void)

{

unsigned char m,n;

for(m=0;m<200;m++)

for(n=0;n<250;n++)

;

}

/

函数功能:主函数

/

void main(void)

{

unsigned char str[]={"Now,Temperature is :"}; //将字符串赋给字符型全部元素赋值

unsigned char i;

while(1)

{

i=0; //将i初始化为0,从第一个元素开始显示

while(str[i]!='\0') //只要没有显示到结束标志'\0'

{

P0=str[i]; //将第i个字符送到P0口显示

delay150ms; //调用150ms延时函数

i++; //指向下一个待显字符

}

}

}

*实例26:用P0 口显示指针运算结果

#include<reg51.h>

void main(void)

{

unsigned char p1, p2; //定义无符号字符型指针变量p1,p2 unsigned char i,j; //定义无符号字符型数据

i=25; //给i赋初值25

j=15;

p1=&i; //使指针变量指向i ,对指针初始化

p2=&j; //使指针变量指向j ,对指针初始化

P0= p1+ p2; // p1+ p2相当于i+j,所以P0=25+15=40=0x28

//则P0=0010 1000B,结果P0.3、P0.5引脚LED熄灭,其余点亮 while(1)

; //无限循环,防止程序“跑飞”

}

*实例27:用指针数组控制P0口8位LED流水点亮 #include<reg51.h>

/

函数功能:延时约150ms (3 200 250=150 000μs=150ms

/

void delay150ms(void)

{

unsigned char m,n;

for(m=0;m<200;m++)

for(n=0;n<250;n++)

;

}

/

函数功能:主函数

/

void main(void)

{

unsigned char code Tab[]={0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f};

unsigned char p[ ]={&Tab[0],&Tab[1],&Tab[2],&Tab[3],&Tab[4],&Tab[5], &Tab[6],&Tab[7]};

unsigned char i; //定义无符号字符型数据

while(1)

{

for(i=0;i<8;i++)

{

P0= p[i];

delay150ms;

}

}

}

*实例28:用数组的指针控制P0 口8 位LED流水点亮 #include<reg51.h>

/

函数功能:延时约150ms (3 200 250=150 000μs=150ms

/

void delay150ms(void)

{

unsigned char m,n;

for(m=0;m<200;m++)

for(n=0;n<250;n++)

;

}

/

函数功能:主函数

/

void main(void)

{

unsigned char i;

unsigned char Tab[ ]={0xFF,0xFE,0xFD,0xFB,0xF7,0xEF,0xDF,0xBF, 0x7F,0xBF,0xDF,0xEF,0xF7,0xFB,0xFD,0xFE, 0xFE,0xFC,0xFB,0xF0,0xE0,0xC0,0x80,0x00, 0xE7,0xDB,0xBD,0x7E,0x3C,0x18,0x00,0x81, 0xC3,0xE7,0x7E,0xBD,0xDB,0xE7,0xBD,0xDB}; //流水灯控制码

unsigned char p; //定义无符号字符型指针

p=Tab; //将数组首地址存入指针p

while(1)

{

for(i=0;i<32;i++) //共32个流水灯控制码

{

P0= (p+i); // (p+i)的值等于a[i]

}

delay150ms; //调用150ms延时函数 } }

*实例29:用P0 、P1口显示整型函数返回值 #include<reg51.h>

/ 函数功能:计算两个无符号整数的和

/ unsigned int sum(int a,int b)

{

unsigned int s;

s=a+b;

return (s);

}

/ 函数功能:主函数

/ void main(void)

{

unsigned z;

z=sum(2008,2009);

P1=z/256; //取得z的高8位

P0=z%256; //取得z的低8位

while(1)

;

}

*实例30:用有参函数控制P0口8位LED流水速度 #include<reg51.h>

/ 函数功能:延时一段时间

/ void delay(unsigned char x)

{

unsigned char m,n;

for(m=0;m<x;m++)

for(n=0;n<200;n++)

;

}

/

函数功能:主函数

/

void main(void)

{

unsigned char i;

unsigned char code Tab[ ]={0xFE,0xFD,0xFB,0xF7,0xEF,0xDF,0xBF,0x7F}; //流水灯控制码

while(1)

{

//快速流水点亮LED

for(i=0;i<8;i++) //共8个流水灯控制码

{

P0=Tab[i];

delay(100); //延时约60ms, (3 100 200=60 000μs) }

//慢速流水点亮LED

for(i=0;i<8;i++) //共8个流水灯控制码

{

P0=Tab[i];

delay(250); //延时约150ms, (3 250 200=150 000μs) }

}

}

*实例31:用数组作函数参数控制流水花样

#include<reg51.h>

/

函数功能:延时约150ms

/

void delay(void)

{

unsigned char m,n;

for(m=0;m<200;m++)

for(n=0;n<250;n++)

;

}

/

函数功能:流水点亮P0口8位LED

/

void led_flow(unsigned char a[8])

{

unsigned char i;

for(i=0;i<8;i++)

{

P0=a[i];

delay;

}

}

/

函数功能:主函数

/

void main(void)

{

unsigned char code Tab[ ]={0xFE,0xFD,0xFB,0xF7,0xEF,0xDF,0xBF,0x7F}; //流水灯控制码

led_flow(Tab);

}

*实例32:用指针作函数参数控制P0口8位LED流水点亮 #include<reg51.h>

/

函数功能:延时约150ms

/

void delay(void)

{

unsigned char m,n;

for(m=0;m<200;m++)

for(n=0;n<250;n++)

;

}

/

函数功能:流水点亮P0口8位LED

/

void led_flow(unsigned char p) //形参为无符号字符型指针

{

unsigned char i;

while(1)

{

i=0; //将i置为0,指向数组第一个元素

while( (p+i)!='\0') //只要没有指向数组的结束标志

{

P0= (p+i);// 取的指针所指变量(数组元素)的值,送P0口 delay; //调用延时函数

i++; //指向下一个数组元素

}

}

}

/

函数功能:主函数

/

void main(void)

{

unsigned char code Tab[ ]={0xFE,0xFD,0xFB,0xF7,0xEF,0xDF,0xBF,0x7F, 0x7F,0xBF,0xDF,0xEF,0xF7,0xFB,0xFD,0xFE,

0xFF,0xFE,0xFC,0xFB,0xF0,0xE0,0xC0,0x80,

0x00,0xE7,0xDB,0xBD,0x7E,0xFF,0xFF,0x3C,

0x18,0x0,0x81,0xC3,0xE7,0xFF, 0xFF,0x7E};

//流水灯控制码

unsigned char pointer;

pointer=Tab;

led_flow(pointer);

}

*实例33:用函数型指针控制P1口灯花样

#include<reg51.h> //包含51单片机寄存器定义的头文件

unsigned char code Tab[]={0xFE,0xFD,0xFB,0xF7,0xEF,0xDF,0xBF,0x7F}; //流水灯控制码,该数组被定义为全局变量

/

函数功能:延时约150ms

/ void delay(void)

{

unsigned char m,n;

for(m=0;m<200;m++)

for(n=0;n<250;n++)

;

}

/ 函数功能:流水灯左移

/ void led_flow(void)

{

unsigned char i;

for(i=0;i<8;i++) //8位控制码

{

P0=Tab[i];

delay;

}

}

/ 函数功能:主函数

/ void main(void)

{

void ( p)(void); //定义函数型指针,所指函数无参数,无返回值 p=led_flow; //将函数的入口地址赋给函数型指针p

while(1)

( p); //通过函数的指针p调用函数led_flow

}

*实例34:用指针数组作为函数的参数显示多个字符串 #include<reg51.h> //包含51单片机寄存器定义的头文件

unsigned char code str1[ ]="Temperature is tested by DS18B20";//C语言中,字符串是作为字符数组来处理的

unsigned char code str2[ ]="Now temperature is:"; //所以,字符串的名字就是字符串的首地址

unsigned char code str3[ ]="The Systerm is designed by Zhang San";

unsigned char code str4[ ]="The date is 2008-9-30";

unsigned char p[ ]={str1,str2,str3,str4}; //定义p[4]为指向4个字符串的字符型指针数组

/ 函数功能:延时约150ms

/ void delay(void)

{

unsigned char m,n;

for(m=0;m<200;m++)

for(n=0;n<250;n++)

;

}

/ 函数功能:流水点亮P0口8位LED

/ void led_display(unsigned char x[ ]) //形参必须为指针数组

{

unsigned char i,j;

for(i=0;i<4;i++) //有4个字符串要显示

{

j=0; //指向待显字符串的第0号元素

while( (x[i]+j)!='\0') //只要第i个字符串的第j号元素不是结束标志 {

P0= (x[i]+j); //取得该元素值送到P0口显示

delay; //调用延时函数

j++; //指向下一个元素

}

}

}

/ 函数功能:主函数

/ void main(void)

{

unsigned char i;

while(1)

{

for(i=0;i<4;i++)

led_display(p); //将指针数组名作实际参数传递

}

}

*实例35:字符函数ctype.h应用举例

#include<reg51.h> //包含51单片机寄存器定义的头文件 #include<ctype.h>

void main(void)

{

while(1)

{

P3=isalpha('_')?0xf0:0x0f;//条件运算,若'_'是英文字母,P3=0xf0 }

}

*实例36:内部函数intrins.h应用举例

#include<reg51.h> //包含51单片机寄存器定义的头文件 #include<intrins.h> //包含函数isalpha声明的头文件 / 函数功能:延时约150ms

/ void delay(void)

{

unsigned char m,n;

for(m=0;m<200;m++)

for(n=0;n<250;n++)

;

}

/ 函数功能:主函数

/ void main(void)

{

P3=0xfe; //P3=1111 1110B

while(1)

{

P3=_crol_(P3,1);// 将P3的二进制位循环左移1位后再赋给P3 delay; //调用延时函数

}

}

*实例37:标准函数stdlib.h应用举例

#include<reg51.h> //包含51单片机寄存器定义的头文件 #include<stdlib.h> //包含函数isalpha声明的头文件 / 函数功能:延时约150ms

/ void delay(void)

{

unsigned char m,n;

for(m=0;m<200;m++)

for(n=0;n<250;n++)

;

}

/ 函数功能:主函数

/ void main(void)

{

unsigned char i;

while(1)

{

for(i=0;i<10;i++) //产生10个随机数

{

P3=rand/160; //将产生的随机数缩小160倍后送P3显示 delay;

}

}

}

*实例38:字符串函数string.h应用举例

#include<reg51.h> //包含51单片机寄存器定义的头文件 #include<string.h> //包含函数isalpha声明的头文件 void main(void)

{

unsigned char str1[ ]="Now, The temperature is :";

unsigned char str2[ ]="Now, The temperature is 36 Centgrade:"; unsigned char i;

i=strcmp(str1,str2); //比较两个字符串,并将结果存入i

if(i==0) //str1=str2

P3=0x00;

else

if(i<0) //str1<str2

P3=0xf0;

else //str1>str2

P3=0x0f;

while(1)

; //防止程序“跑飞”

}

*实例39:宏定义应用举例2

#include<reg51.h> //包含51单片机寄存器定义的头文件

# define F(a,b) (a)+(a) (b)/256+(b) //带参数的宏定义,a和b为形参 void main(void)

{

unsigned char i,j,k;

i=40;

j=30;

k=20;

P3=F(i,j+k); //i和j+k分别为实参,宏展开时,实参将替代宏定义中的形参

while(1)

;

}

*实例40:宏定义应用举例2

#include<AT89X51.h>

#include<ctype.h>

void main(void)

{

P3_0=0; //将P3.0引脚置低电平,LED点亮

P3_1=0; //将P3.0引脚置低电平,LED点亮

P3_2=0; //将P3.0引脚置低电平,LED点亮

P3_3=0; //将P3.0引脚置低电平,LED点亮

P3_4=1; //将P3.4引脚置高电平,LED熄灭

}

P3_5=1; //将P3.5引脚置高电平,LED熄灭 P3_6=1; //将P3.7引脚置高电平,LED熄灭 P3_7=1; //将P3.7引脚置高电平,LED熄灭 while(1) ;

*实例41:宏定义应用举例3

#include<reg51.h> //包含51单片机寄存器定义的头文件

#define MAX 100 //将MAX宏定义为字符串100

void main(void)

{

#if MAX>80 //如果字符串100大于80

P3=0xf0; //P3口低四位LED点亮

#else

P3=0x0f; //否则,P3口高四位LED点亮

#endif //结束本次编译

}

/ 中断、定时器 中断、定时器

中断、定时器 中断、定时器 /

*实例42:用定时器T0查询方式P2口8位控制LED闪烁

#include<reg51.h> // 包含51单片机寄存器定义的头文件

/

函数功能:主函数

/ void main(void)

{

// EA=1; //开总中断

// ET0=1; //定时器T0中断允许 TMOD=0x01; //使用定时器T0的模式1

TH0=(65536-46083)/256; //定时器T0的高8位赋初值

TL0=(65536-46083)%256; //定时器T0的高8位赋初值

TR0=1; //启动定时器T0

TF0=0;

P2=0xff;

while(1)//无限循环等待查询

{

while(TF0==0)

;

TF0=0;

P2=~P2;

TH0=(65536-46083)/256; //定时器T0的高8位赋初值 TL0=(65536-46083)%256; //定时器T0的高8位赋初值 }

}

*实例43:用定时器T1查询方式控制单片机发出1KHz音频 #include<reg51.h> // 包含51单片机寄存器定义的头文件 sbit sound=P3^7; //将sound位定义为P3.7引脚

/ 函数功能:主函数

/ void main(void)

{

// EA=1; //开总中断

// ET0=1; //定时器T0中断允许 TMOD=0x10; //使用定时器T1的模式1

TH1=(65536-921)/256; //定时器T1的高8位赋初值

TL1=(65536-921)%256; //定时器T1的高8位赋初值

TR1=1; //启动定时器T1

TF1=0;

while(1)//无限循环等待查询

{

while(TF1==0)

;

TF1=0;

sound=~sound; //将P3.7引脚输出电平取反

TH1=(65536-921)/256; //定时器T0的高8位赋初值 TL1=(65536-921)%256; //定时器T0的高8位赋初值 }

}

*实例44:将计数器T0计数的结果送P1口8位LED显示 #include<reg51.h> // 包含51单片机寄存器定义的头文件 sbit S=P3^4; //将S位定义为P3.4引脚

/ 函数功能:主函数

/ void main(void)

{

// EA=1; //开总中断

// ET0=1; //定时器T0中断允许 TMOD=0x02; //使用定时器T0的模式2

TH0=256-156; //定时器T0的高8位赋初值

TL0=256-156; //定时器T0的高8位赋初值

TR0=1; //启动定时器T0

while(1)//无限循环等待查询

{

while(TF0==0) //如果未计满就等待

{

if(S==0) //按键S按下接地,电平为0

P1=TL0; //计数器TL0加1后送P1口显示

}

TF0=0; //计数器溢出后,将TF0清0

}

}

*实例45:用定时器T0的中断控制1位LED闪烁 #include<reg51.h> // 包含51单片机寄存器定义的头文件

sbit D1=P2^0; //将D1位定义为P2.0引脚

/ 函数功能:主函数

/ void main(void)

{

EA=1; //开总中断

ET0=1; //定时器T0中断允许

TMOD=0x01; //使用定时器T0的模式2

TH0=(65536-46083)/256; //定时器T0的高8位赋初值

TL0=(65536-46083)%256; //定时器T0的高8位赋初值

TR0=1; //启动定时器T0

while(1)//无限循环等待中断

;

}

/ 函数功能:定时器T0的中断服务程序

/ void Time0(void) interrupt 1 using 0 //“interrupt”声明函数为中断服务函数 //其后的1为定时器T0的中断编号;0表示使用第0组工作寄存器

{

D1=~D1; //按位取反操作,将P2.0引脚输出电平取反

TH0=(65536-46083)/256; //定时器T0的高8位重新赋初值

TL0=(65536-46083)%256; //定时器T0的高8位重新赋初值

}

*实例46:用定时器T0的中断实现长时间定时

#include<reg51.h> // 包含51单片机寄存器定义的头文件

sbit D1=P2^0; //将D1位定义为P2.0引脚

unsigned char Countor; //设置全局变量,储存定时器T0中断次数

/ 函数功能:主函数

/ void main(void)

{

EA=1; //开总中断

ET0=1; //定时器T0中断允许

TMOD=0x01; //使用定时器T0的模式2

TH0=(65536-46083)/256; //定时器T0的高8位赋初值

TL0=(65536-46083)%256; //定时器T0的高8位赋初值

TR0=1; //启动定时器T0

Countor=0; //从0开始累计中断次数

while(1)//无限循环等待中断

;

}

/ 函数功能:定时器T0的中断服务程序

/ void Time0(void) interrupt 1 using 0 //“interrupt”声明函数为中断服务函数

//其后的1为定时器T0的中断编号;0表示使用第0组工作寄存器

{

Countor++; //中断次数自加1

if(Countor==20) //若累计满20次,即计时满1s

{

D1=~D1; //按位取反操作,将P2.0引脚输出电平取反

Countor=0; //将Countor清0,重新从0开始计数

}

TH0=(65536-46083)/256; //定时器T0的高8位重新赋初值

TL0=(65536-46083)%256; //定时器T0的高8位重新赋初值

}

*实例47:用定时器T1中断控制两个LED以不同周期闪烁 #include<reg51.h> // 包含51单片机寄存器定义的头文件

sbit D1=P2^0; //将D1位定义为P2.0引脚

sbit D2=P2^1; //将D2位定义为P2.1引脚

unsigned char Countor1; //设置全局变量,储存定时器T1中断次数

unsigned char Countor2; //设置全局变量,储存定时器T1中断次数

/ 函数功能:主函数

/ void main(void)

{

EA=1; //开总中断

ET1=1; //定时器T1中断允许

TMOD=0x10; //使用定时器T1的模式1

TH1=(65536-46083)/256; //定时器T1的高8位赋初值

TL1=(65536-46083)%256; //定时器T1的高8位赋初值

TR1=1; //启动定时器T1

Countor1=0; //从0开始累计中断次数

Countor2=0; //从0开始累计中断次数

while(1)//无限循环等待中断

;

}

/ 函数功能:定时器T1的中断服务程序

/ void Time1(void) interrupt 3 using 0 //“interrupt”声明函数为中断服务函数 //其后的3为定时器T1的中断编号;0表示使用第0组工作寄存器

{

Countor1++; //Countor1自加1

Countor2++; //Countor2自加1

if(Countor1==2) //若累计满2次,即计时满100ms

{

D1=~D1; //按位取反操作,将P2.0引脚输出电平取反 Countor1=0; //将Countor1清0,重新从0开始计数

}

if(Countor2==8) //若累计满8次,即计时满400ms

{

D2=~D2; //按位取反操作,将P2.1引脚输出电平取反 Countor2=0; //将Countor1清0,重新从0开始计数

}

TH1=(65536-46083)/256; //定时器T1的高8位重新赋初值

TL1=(65536-46083)%256; //定时器T1的高8位重新赋初值

}

*实例48:用计数器T1的中断控制蜂鸣器发出1KHz音频 #include<reg51.h> // 包含51单片机寄存器定义的头文件

sbit sound=P3^7; //将sound位定义为P3.7引脚

/ 函数功能:主函数

/ void main(void)

{

EA=1; //开总中断

ET1=1; //定时器T1中断允许

TMOD=0x10; //TMOD=0001 000B,使用定时器T1的模式1 TH1=(65536-921)/256; //定时器T1的高8位赋初值

TL1=(65536-921)%256; //定时器T1的高8位赋初值

TR1=1; //启动定时器T1

while(1)//无限循环等待中断

;

}

/ 函数功能:定时器T1的中断服务程序

/ void Time1(void) interrupt 3 using 0 //“interrupt”声明函数为中断服务函数 {

sound=~sound;

TH1=(65536-921)/256; //定时器T1的高8位重新赋初值

TL1=(65536-921)%256; //定时器T1的高8位重新赋初值

}

*实例49:用定时器T0的中断实现"渴望"主题曲的播放 #include<reg51.h> //包含51单片机寄存器定义的头文件

sbit sound=P3^7; //将sound位定义为P3.7

unsigned int C; //储存定时器的定时常数

//以下是C调低音的音频宏定义

#define l_dao 262 //将“l_dao”宏定义为低音“1”的频率262Hz #define l_re 286 //将“l_re”宏定义为低音“2”的频率286Hz #define l_mi 311 //将“l_mi”宏定义为低音“3”的频率311Hz #define l_fa 349 //将“l_fa”宏定义为低音“4”的频率349Hz #define l_sao 392 //将“l_sao”宏定义为低音“5”的频率392Hz #define l_la 440 //将“l_a”宏定义为低音“6”的频率440Hz #define l_xi 494 //将“l_xi”宏定义为低音“7”的频率494Hz //以下是C调中音的音频宏定义

#define dao 523 //将“dao”宏定义为中音“1”的频率523Hz #define re 587 //将“re”宏定义为中音“2”的频率587Hz #define mi 659 //将“mi”宏定义为中音“3”的频率659Hz #define fa 698 //将“fa”宏定义为中音“4”的频率698Hz #define sao 784 //将“sao”宏定义为中音“5”的频率784Hz #define la 880 //将“la”宏定义为中音“6”的频率880Hz

#define xi 987 //将“xi”宏定义为中音“7”的频率523H //以下是C调高音的音频宏定义

#define h_dao 1046 //将“h_dao”宏定义为高音“1”的频率1046Hz #define h_re 1174 //将“h_re”宏定义为高音“2”的频率1174Hz #define h_mi 1318 //将“h_mi”宏定义为高音“3”的频率1318Hz #define h_fa 1396 //将“h_fa”宏定义为高音“4”的频率1396Hz

#define h_sao 1567 //将“h_sao”宏定义为高音“5”的频率1567Hz #define h_la 1760 //将“h_la”宏定义为高音“6”的频率1760Hz #define h_xi 1975 //将“h_xi”宏定义为高音“7”的频率1975Hz /

函数功能:1个延时单位,延时200ms

/

void delay

{

unsigned char i,j;

for(i=0;i<250;i++)

for(j=0;j<250;j++)

;

}

/

函数功能:主函数

/

void main(void)

{

unsigned char i,j;

//以下是《渴望》片头曲的一段简谱

unsigned int code f[]={re,mi,re,dao,l_la,dao,l_la, //每行对应一小节音符 l_sao,l_mi,l_sao,l_la,dao,

l_la,dao,sao,la,mi,sao,

re,

mi,re,mi,sao,mi,

l_sao,l_mi,l_sao,l_la,dao,

l_la,l_la,dao,l_la,l_sao,l_re,l_mi,

l_sao,

re,re,sao,la,sao,

fa,mi,sao,mi,

la,sao,mi,re,mi,l_la,dao,

re,

mi,re,mi,sao,mi,

l_sao,l_mi,l_sao,l_la,dao,

l_la,dao,re,l_la,dao,re,mi,

re,

l_la,dao,re,l_la,dao,re,mi,

re,

0xff}; //以0xff作为音符的结束标志

//以下是简谱中每个音符的节拍

//"4"对应4个延时单位,"2"对应2个延时单位,"1"对应1个延时单位 unsigned char code JP[ ]={4,1,1,4,1,1,2,

2,2,2,2,8,

4,2,3,1,2,2,

10,

4,2,2,4,4,

2,2,2,2,4,

2,2,2,2,2,2,2,

10,

4,4,4,2,2,

4,2,4,4,

4,2,2,2,2,2,2,

10,

4,2,2,4,4,

2,2,2,2,6,

4,2,2,4,1,1,4,

10,

4,2,2,4,1,1,4,

10

};

EA=1; //开总中断

ET0=1; //定时器T0中断允许

TMOD=0x00; // 使用定时器T0的模式1(13位计数器)

while(1) //无限循环

{

i=0; //从第1个音符f[0]开始播放

while(f[i]!=0xff) //只要没有读到结束标志就继续播放 {

C=460830/f[i];

TH0=(8192-C)/32; //可证明这是13位计数器TH0高8位的赋初值方法

TL0=(8192-C)%32; //可证明这是13位计数器TL0低5位的赋初值方法

TR0=1; //启动定时器T0

for(j=0;j<JP[i];j++) //控制节拍数

delay; //延时1个节拍单位

TR0=0; //关闭定时器T0

i++; //播放下一个音符

}

}

}

/ 函数功能:定时器T0的中断服务子程序,使P3.7引脚输出音频的方波 / void Time0(void ) interrupt 1 using 1

{

sound=!sound; //将P3.7引脚输出电平取反,形成方波

TH0=(8192-C)/32; //可证明这是13位计数器TH0高8位的赋初值方法 TL0=(8192-C)%32; //可证明这是13位计数器TL0低5位的赋初值方法 }

*实例50-1:输出50个矩形脉冲

#include<reg51.h> //包含51单片机寄存器定义的头文件

sbit u=P1^4; //将u位定义为P1.4

/

函数功能:延时约30ms (3 100 100=30 000μs =30m

/

void delay30ms(void)

{

unsigned char m,n;

for(m=0;m<100;m++)

for(n=0;n<100;n++)

;

}

/

函数功能:主函数

/

void main(void)

{

unsigned char i;

u=1; //初始化输出高电平

for(i=0;i<50;i++) //输出50个矩形脉冲

{

u=1;

delay30ms;

u=0;

delay30ms;

}

while(1)

; //无限循环,防止程序“跑飞”

}

*实例50-2:计数器T0统计外部脉冲数

#include<reg51.h> //包含51单片机寄存器定义的头文件

/

函数功能:主函数

/

void main(void)

{

TMOD=0x06; // TMOD=0000 0110B,使用计数器T0的模式2 EA=1; //开总中断

ET0=0; //不使用定时器T0的中断

TR0=1; //启动T0

TH0=0; //计数器T0高8位赋初值

TL0=0; //计数器T0低8位赋初值

while(1) //无限循环,不停地将TL0计数结果送P1口

P1=TL0;

}

*实例51-2:定时器T0的模式2测量正脉冲宽度

#include<reg51.h> //包含51单片机寄存器定义的头文件

sbit ui=P3^2; //将ui位定义为P3.0(INT0)引脚,表示输入电压

/

函数功能:主函数

/

void main(void)

{

TMOD=0x0a; // TMOD=0000 1010B,使用定时器T0的模式2,GATE置1 EA=1; //开总中断

ET0=0; //不使用定时器T0的中断

TR0=1; //启动T0

TH0=0; //计数器T0高8位赋初值

TL0=0; //计数器T0低8位赋初值

while(1) //无限循环,不停地将TL0计数结果送P1口 {

while(ui==0) //INT0为低电平,T0不能启动

;

TL0=0; //INT0为高电平,启动T0计时,所以将TL0清0 while(ui==1) //在INT0高电平期间,等待,计时

;

P1=TL0; //将计时结果送P1口显示

}

}

*实例52:用定时器T0控制输出高低宽度不同的矩形波 #include<reg51.h> //包含51单片机寄存器定义的头文件

sbit u=P3^0; //将u位定义为P3.0,从该引脚输出矩形脉冲 unsigned char Countor; //设置全局变量,储存负跳变累计数 / 函数功能:延时约30ms (3 100 100=30 000μs =30ms) / void delay30ms(void)

{

unsigned char m,n;

for(m=0;m<100;m++)

for(n=0;n<100;n++)

;

}

/

函数功能:主函数

/ void main(void)

{

unsigned char i;

EA=1; //开放总中断

EX0=1; //允许使用外中断

IT0=1; //选择负跳变来触发外中断

Countor=0;

for(i=0;i<100;i++) //输出100个负跳变

{

u=1;

delay30ms;

u=0;

delay30ms;

}

while(1)

; //无限循环, 防止程序跑飞

}

/ 函数功能:外中断T0的中断服务程序

void int0(void) interrupt 0 using 0 //外中断0的中断编号为0 /

{

Countor++;

P1=Countor;

}

*实例53:用外中断0的中断方式进行数据采集 #include<reg51.h> //包含51单片机寄存器定义的头文件

sbit S=P3^2; //将S位定义为P3.2,

/

函数功能:主函数

/

void main(void)

{

EA=1; //开放总中断

EX0=1; //允许使用外中断

IT0=1; //选择负跳变来触发外中断

P1=0xff;

while(1)

; //无限循环, 防止程序跑飞

}

/ 函数功能:外中断T0的中断服务程序

/ void int0(void) interrupt 0 using 0 //外中断0的中断编号为0

{

P1=~P1; //每产生一次中断请求,P1取反一次。

}

*实例54-1:输出负脉宽为200微秒的方波

#include<reg51.h> //包含51单片机寄存器定义的头文件

sbit u=P1^4; //将u位定义为P1.4

/

函数功能:主函数

/

void main(void)

{

TMOD=0x02; //TMOD=0000 0010B,使用定时器T0的模式2 EA=1; //开总中断

ET0=1; //定时器T0中断允许

TH0=256-200; //定时器T0的高8位赋初值

TL0=256-200; //定时器T0的高8位赋初值

TR0=1; //启动定时器T0

while(1) //无限循环,等待中断

;

}

/ 函数功能:定时器T0的中断服务程序

/ void Time0(void) interrupt 1 using 0 //"interrupt"声明函数为中断服务函数 {

u=~u; //将P1.4引脚输出电平取反,产生方波

}

*实例54-2:测量负脉冲宽度

#include<reg51.h> //包含51单片机寄存器定义的头文件

sbit u=P3^2; //将u位定义为P3.2

/

函数功能:主函数

/

void main(void)

{

TMOD=0x02; //TMOD=0000 0010B,使用定时器T0的模式2

EA=1; //开放总中断

EX0=1; //允许使用外中断

IT0=1; //选择负跳变来触发外中断

ET0=1; //允许定时器T0中断

TH0=0; //定时器T0赋初值0

TL0=0; //定时器T0赋初值0

TR0=0; //先关闭T0

while(1)

; //无限循环, 不停检测输入负脉冲宽度

}

/ 函数功能:外中断0的中断服务程序

/ void int0(void) interrupt 0 using 0 //外中断0的中断编号为0

{

TR0=1; //外中断一到来,即启动T0计时

TL0=0; //从0开始计时

while(u==0) //低电平时,等待T0计时

;

P1=TL0; //将结果送P1口显示

TR0=0; //关闭T0

}

*实例55:方式0控制流水灯循环点亮

#include<reg51.h> //包含51单片机寄存器定义的头文件

#include<intrins.h> //包含函数_nop_定义的头文件

unsigned char code Tab[]={0xFE,0xFD,0xFB,0xF7,0xEF,0xDF,0xBF,0x7F};//流水灯控制码,该数组被定义为全局变量

sbit P17=P1^7;

/ 函数功能:延时约150ms

/ void delay(void)

{

unsigned char m,n;

for(m=0;m<200;m++)

for(n=0;n<250;n++)

;

}

/ 函数功能:发送一个字节的数据

/ void Send(unsigned char dat)

{

P17=0; //P1.7引脚输出清0信号,对74LS164清0

_nop_; //延时一个机器周期

_nop_; //延时一个机器周期,保证清0完成

P17=1; //结束对74LS164的清0

SBUF=dat; //将数据写入发送缓冲器,启动发送

while(TI==0) //若没有发送完毕,等待

;

TI=0; //发送完毕,TI被置“1”,需将其清0

}

/

函数功能:主函数

/

void main(void)

{

unsigned char i;

SCON=0x00; //SCON=0000 0000B,使串行口工作于方式0 while(1)

{

for(i=0;i<8;i++)

{

Send(Tab[i]); //发送数据

delay; //延时

}

}

}

*实例56-1:数据发送程序

#include<reg51.h> //包含单片机寄存器的头文件

unsigned char code Tab[ ]={0xFE,0xFD,0xFB,0xF7,0xEF,0xDF,0xBF,0x7F}; //流水灯控制码,该数组被定义为全局变量

/ 函数功能:向PC发送一个字节数据

/ void Send(unsigned char dat)

{

SBUF=dat;

while(TI==0)

;

TI=0;

}

/ 函数功能:延时约150ms

void delay(void)

{ /

unsigned char m,n;

for(m=0;m<200;m++)

for(n=0;n<250;n++)

;

}

/ 函数功能:主函数

/ void main(void)

{

unsigned char i;

TMOD=0x20; //TMOD=0010 0000B,定时器T1工作于方式2 SCON=0x40; //SCON=0100 0000B,串口工作方式1 PCON=0x00; //PCON=0000 0000B,波特率9600

TH1=0xfd; //根据规定给定时器T1赋初值

TL1=0xfd; //根据规定给定时器T1赋初值

TR1=1; //启动定时器T1

while(1)

{

for(i=0;i<8;i++) //模拟检测数据

{

Send(Tab[i]); //发送数据i

delay; //50ms发送一次检测数据 }

}

}

*实例56-2:数据接收程序

#include<reg51.h> //包含单片机寄存器的头文件

/ 函数功能:接收一个字节数据

/ unsigned char Receive(void)

{

unsigned char dat;

while(RI==0) //只要接收中断标志位RI没有被置“1” ; //等待,直至接收完毕(RI=1)

RI=0; //为了接收下一帧数据,需将RI清0 dat=SBUF; //将接收缓冲器中的数据存于dat

return dat;

}

/

函数功能:主函数

/

void main(void)

{

TMOD=0x20; //定时器T1工作于方式2

SCON=0x50; //SCON=0101 0000B,串口工作方式1,允许接收(REN=1) PCON=0x00; //PCON=0000 0000B,波特率9600

TH1=0xfd; //根据规定给定时器T1赋初值

TL1=0xfd; //根据规定给定时器T1赋初值

TR1=1; //启动定时器T1

REN=1; //允许接收

while(1)

{

P1=Receive; //将接收到的数据送P1口显示

}

}

*实例57-1:数据发送程序

#include<reg51.h> //包含单片机寄存器的头文件

sbit p=PSW^0;

unsigned char code Tab[ ]={0xFE,0xFD,0xFB,0xF7,0xEF,0xDF,0xBF,0x7F}; //流水灯控制码,该数组被定义为全局变量

/

函数功能:向PC发送一个字节数据

/

void Send(unsigned char dat)

{

ACC=dat;

TB8=p;

SBUF=dat;

while(TI==0)

;

TI=0;

}

/ 函数功能:延时约150ms

/

void delay(void)

{

unsigned char m,n;

for(m=0;m<200;m++)

for(n=0;n<250;n++)

;

}

/ 函数功能:主函数

/ void main(void)

{

unsigned char i;

TMOD=0x20; //TMOD=0010 0000B,定时器T1工作于方式2 SCON=0xc0; //SCON=1100 0000B,串口工作方式3, //SM2置0,不使用多机通信,TB8置0 PCON=0x00; //PCON=0000 0000B,波特率9600

TH1=0xfd; //根据规定给定时器T1赋初值

TL1=0xfd; //根据规定给定时器T1赋初值

TR1=1; //启动定时器T1

while(1)

{

for(i=0;i<8;i++) //模拟检测数据

{

Send(Tab[i]); //发送数据i

delay; //50ms发送一次检测数据 }

}

}

*实例57-2:数据接收程序

#include<reg51.h> //包含单片机寄存器的头文件 sbit p=PSW^0;

/ 函数功能:接收一个字节数据

/ unsigned char Receive(void)

{

unsigned char dat;

while(RI==0) //只要接收中断标志位RI没有被置"1"

; //等待,直至接收完毕(RI=1)

RI=0; //为了接收下一帧数据,需将RI清0

ACC=SBUF; //将接收缓冲器中的数据存于dat

if(RB8==p)

{

dat=ACC;

return dat;

}

}

/

函数功能:主函数

/

void main(void)

{

TMOD=0x20; //定时器T1工作于方式2

SCON=0xd0; //SCON=1101 0000B,串口工作方式1,允许接收(REN=1) PCON=0x00; //PCON=0000 0000B,波特率9600

TH1=0xfd; //根据规定给定时器T1赋初值

TL1=0xfd; //根据规定给定时器T1赋初值

TR1=1; //启动定时器T1

REN=1; //允许接收

while(1)

{

P1=Receive; //将接收到的数据送P1口显示

}

}

*实例58:单片机向PC发送数据

#include<reg51.h> //包含单片机寄存器的头文件

unsigned char code Tab[ ]={0xFE,0xFD,0xFB,0xF7,0xEF,0xDF,0xBF,0x7F}; //流水灯控制码,该数组被定义为全局变量

/

函数功能:向PC发送一个字节数据

/

void Send(unsigned char dat)

{

SBUF=dat;

while(TI==0)

;

TI=0;

}

/ 函数功能:延时约150ms

/ void delay(void)

{

unsigned char m,n;

for(m=0;m<200;m++)

for(n=0;n<250;n++)

;

}

/ 函数功能:主函数

/

void main(void)

{

unsigned char i;

TMOD=0x20; //TMOD=0010 0000B,定时器T1工作于方式2 SCON=0x40; //SCON=0100 0000B,串口工作方式1

PCON=0x00; //PCON=0000 0000B,波特率9600

TH1=0xfd; //根据规定给定时器T1赋初值

TL1=0xfd; //根据规定给定时器T1赋初值

TR1=1; //启动定时器T1

while(1)

{

for(i=0;i<8;i++) //模拟检测数据

{

Send(Tab[i]); //发送数据i

delay; //150ms发送一次数据

}

}

}

*实例59:单片机接收PC发出的数据

#include<reg51.h> //包含单片机寄存器的头文件

/ 函数功能:接收一个字节数据

/

unsigned char Receive(void)

{

unsigned char dat;

while(RI==0) //只要接收中断标志位RI没有被置“1”

; //等待,直至接收完毕(RI=1)

RI=0; //为了接收下一帧数据,需将RI清0

dat=SBUF; //将接收缓冲器中的数据存于dat

return dat;

}

/

函数功能:主函数

/

void main(void)

{

TMOD=0x20; //定时器T1工作于方式2

SCON=0x50; //SCON=0101 0000B,串口工作方式1,允许接收(REN=1)

PCON=0x00; //PCON=0000 0000B,波特率9600

TH1=0xfd; //根据规定给定时器T1赋初值

TL1=0xfd; //根据规定给定时器T1赋初值

TR1=1; //启动定时器T1

REN=1; //允许接收

while(1)

{

P1=Receive; //将接收到的数据送P1口显示

}

}

/ 数码管显示 数码管显示 数

码管显示 数码管显示 /

*实例60:用LED数码显示数字5

#include<reg51.h> // 包含51单片机寄存器定义的头文件

void main(void)

{

P2=0xfe; //P2.0引脚输出低电平,数码显示器接通电源准备点亮

P0=0x92; //让P0口输出数字"5"的段码92H

}

*实例61:用LED数码显示器循环显示数字0~9

#include<reg51.h> // 包含51单片机寄存器定义的头文件

/

函数功能:延时函数,延时一段时间

/

void delay(void)

{

unsigned char i,j;

for(i=0;i<255;i++)

for(j=0;j<255;j++)

;

}

/

函数功能:主函数

/

void main(void)

{

unsigned char i;

unsigned char code

Tab[10]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90};

//数码管显示0~9的段码表,程序运行中当数组值不发生变化

时,

//前面加关键字code ,可以大大节约单片机的存储空间 P2=0xfe; //P2.0引脚输出低电平,数码显示器DS0接通电源工作

while(1) //无限循环

{

for(i=0;i<10;i++)

{

P0=Tab[i]; //让P0口输出数字的段码92H

delay; //调用延时函数

}

}

}

*实例62:用数码管慢速动态扫描显示数字"1234" #include<reg51.h> // 包含51单片机寄存器定义的头文件 void delay(void) //延时函数,延时一段时间

{

unsigned char i,j;

for(i=0;i<250;i++)

for(j=0;j<250;j++)

;

}

void main(void)

{

while(1) //无限循环

{

P2=0xfe; //P2.0引脚输出低电平,DS0点亮

P0=0xf9; //数字1的段码

delay;

P2=0xfd ; //P2.1引脚输出低电平,DS1点亮

P0=0xa4; //数字2的段码

delay;

P2=0xfb; //P2.2引脚输出低电平,DS2点亮

P0=0xb0; //数字3的段码

delay;

P2=0xf7; //P2.3引脚输出低电平,DS3点亮

P0=0x99; //数字4的段码

delay;

P2=0xff;

}

}

*实例63:用LED数码显示器伪静态显示数字1234 #include<reg51.h> // 包含51单片机寄存器定义的头文件 void delay(void) //延时函数,延时约0.6毫秒

{

unsigned char i;

for(i=0;i<200;i++)

;

}

void main(void)

{

while(1) //无限循环

{

P2=0xfe; //P2.0引脚输出低电平,DS0点亮

P0=0xf9; //数字1的段码

delay;

P2=0xfd ; //P2.1引脚输出低电平,DS1点亮

P0=0xa4; //数字2的段码

delay;

P2=0xfb; //P2.2引脚输出低电平,DS2点亮

P0=0xb0; //数字3的段码

delay;

P2=0xf7; //P2.3引脚输出低电平,DS3点亮

P0=0x99; //数字4的段码

delay;

P2=0xff;

}

}

*实例64:用数码管显示动态检测结果

#include<reg51.h> // 包含51单片机寄存器定义的头文件

#include<stdlib.h> //包含随机函数rand的定义文件

unsigned char i; //记录中断次数

unsigned int x; //随机检测的数据

unsigned char code Tab[10]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90}; //数码管显示0~9的段码表

/ 函数功能:快速动态扫描延时,延时约0.9毫秒

/

void delay(void)

{

unsigned int i;

for(i=0;i<300;i++)

;

}

/

函数功能:4位数的数码显示器显示

入口参数:k

出口参数:无

/

void display(unsigned int k)

{

P2=0xfe; //即P2=1111 1110B,P2.0引脚输出低电平,数码显示器DS0接通电源

P0=Tab[k/1000]; //显示千位

delay;

P2=0xfd ; //即P2=1111 1101B,P2.1引脚输出低电平,数码显示器DS1接通电源

P0=Tab[(k%1000)/100]; //显示百位

delay;

P2=0xfb; //即P2=1111 1011B,P2.2引脚输出低电平,数码显示器DS2接通电源

P0=Tab[(k%100)/10]; //显示十位

delay;

P2=0xf7; //即P2=1111 0111B ,P2.3引脚输出低电平,数码显示器DS3接通电源

P0=Tab[k%10];//显示个位

delay;

P2=0xff; //关闭所有显示器

}

void main(void) //主函数

{

TMOD=0x01; //使用定时器T0

TH0=(65536-46083)/256; //将定时器计时时间设定为46083×1.085微秒=50000微秒=50毫秒

TL0=(65536-46083)%256;

EA=1; //开启总中断

ET0=1; //定时器T0中断允许

TR0=1; //启动定时器T0开始运行

while(1)

{

display(x); //调用检测结果的显示程序

}

}

/

函数功能:定时器T0的中断服务程序

/

void Time0(void) interrupt 1 using 1

{

TR0=0; //关闭定时器T0

i++; //每来一次中断,i自加1

if(i==20) //够20次中断,即1秒钟进行一次检测结果采样

{

x=rand/10; //随机产生一个从0到32767的整数,再将其除以10,获得一个随机4位数,模拟检测结果

i=0; //将i清0,重新统计中断次数

}

TH0=(65536-46083)/256; //重新给计数器T0赋初值

TL0=(65536-46083)%256;

TR0=1; //启动定时器T0

}

*实例65:数码秒表设计

#include<reg51.h> // 包含51单片机寄存器定义的头文件

unsigned char code Tab[10]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90}; //数码管显示0~9的段码表

unsigned char int_time; //记录中断次数

unsigned char second; //储存秒

/ 函数功能:快速动态扫描延时,延时约0.6毫秒

/

void delay(void)

{

unsigned char i;

for(i=0;i<200;i++)

;

}

/ 函数功能:显示秒

入口参数:k

出口参数:无

/

void DisplaySecond(unsigned char k)

{

P2=0xfb; //P2.6引脚输出低电平, DS6点亮

P0=Tab[k/10]; //显示十位

delay;

P2=0xf7; //P2.7引脚输出低电平, DS7点亮

P0=Tab[k%10]; //显示个位

delay;

P2=0xff; //关闭所有数码管

}

void main(void) //主函数

{

TMOD=0x01; //使用定时器T0

TH0=(65536-46083)/256; //将定时器计时时间设定为46083×1.085微秒

//=50000微秒=50毫秒

TL0=(65536-46083)%256;

EA=1; //开启总中断

ET0=1; //定时器T0中断允许

TR0=1; //启动定时器T0开始运行

int_time=0; //中断次数初始化

second=0; //秒初始化

while(1)

{

DisplaySecond(second); //调用秒的显示子程序

}

}

//

//函数功能:定时器T0的中断服务程序

//

void interserve(void ) interrupt 1 using 1

{

TR0=0; //关闭定时器T0

int_time ++; //每来一次中断,中断次数int_time自加1

if(int_time==20) //够20次中断,即1秒钟进行一次检测结果采样 {

int_time=0; //中断次数清0

second++; //秒加1

if(second==60)

second =0; //秒等于60就返回0

}

TH0=(65536-46083)/256; //重新给计数器T0赋初值

TL0=(65536-46083)%256;

TR0=1; //启动定时器T0

}

*实例66:数码时钟设计

#include<reg51.h> // 包含51单片机寄存器定义的头文件

unsigned char Tab[ ]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90}; //control shape

unsigned char port[8]={0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f}; unsigned char int_time ; //中断次数计数变量

unsigned char second; //秒计数变量

unsigned char minute; //分钟计数变量

unsigned char hour; //小时计数变量

/////////////////////////////////////////////////////

void delay(void) //延时函数,延时约0.6ms

{

unsigned char j;

for(j=0;j<200;j++)

;

}

/ 函数功能:显示秒的子程序

入口参数:s

void DisplaySecond(unsigned char s)

{

P2=0xbf; //P2.6引脚输出低电平, DS6点亮

P0=Tab[s/10]; //显示十位

delay;

P2=0x7f; //P2.7引脚输出低电平, DS7点亮

P0=Tab[s%10]; //显示个位 /

delay;

P2=0xff; //关闭所有数码管

}

/ 函数功能:显示分钟的子程序

入口参数:m

/ void DisplayMinute(unsigned char m)

{

P2=0xf7; // P2.3引脚输出低电平, DS3点亮

P0=Tab[m/10];//显示个位

delay;

P2=0xef; // P2.4引脚输出低电平, DS4点亮

P0=Tab[m%10];

delay;

P2=0xdf; //P2.5引脚输出低电平, DS5点亮

P0=0xbf; //分隔符“-”的段码

delay;

P2=0xff; //关闭所有数码管

}

/ 函数功能:显示小时的子程序

入口参数:h

void DisplayHour(unsigned char h)

{

P2=0xfe; //P2.0引脚输出低电平, DS0点亮

P0=Tab[h/10]; //显示十位

delay;

P2=0xfd; //P2.1引脚输出低电平, DS1点亮

P0=Tab[h%10]; //显示个位

delay;

P2=0xfb; //P2.2引脚输出低电平, DS2点亮

P0=0xbf; //分隔符“-”的段码

delay;

P2=0xff; //关闭所有数码管

/

}

/ 函数功能:主函数

/

void main(void)

{

TMOD=0x01; //使用定时器T0

EA=1; //开中断总允许

ET0=1; //允许T0中断

TH0=(65536-46083)/256; //定时器高八位赋初值

TL0=(65536-46083)%256; //定时器低八位赋初值

TR0=1;

int_time=0; //中断计数变量初始化

second=0; //秒计数变量初始化

minute=0; //分钟计数变量初始化

hour=0; //小时计数变量初始化

while(1)

{

DisplaySecond(second); //调用秒显示子程序 delay;

DisplayMinute(minute); //调用分钟显示子程序 delay;

DisplayHour(hour);

delay;

}

}

/ 函数功能:定时器T0的中断服务子程序

void interserve(void ) interrupt 1 using 1 //using Time0

{

int_time++;

if(int_time==20)

{

int_time=0; //中断计数变量清0

second++; //秒计数变量加1

}

if(second==60)

{

second=0; //如果秒计满60,将秒计数变量清0 /

清0

清0

}

} minute++; //分钟计数变量加1 if(minute==60) { minute=0; //如果分钟计满60,将分钟计数变量 hour++; //小时计数变量加1 } if(hour==24) { hour=0; //如果小时计满24,将小时计数变量 } TH0=(65536-46083)/256; //定时器重新赋初值 TL0=(65536-46083)%256;

*实例67:用LED数码管显示计数器T0的计数值

#include<reg51.h> //包含51单片机寄存器定义的头文件

sbit S=P3^2 ; //将S位定义为P3.2引脚

unsigned char Tab[ ]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90}; //段码表

unsigned char x;

/ 函数功能: 延时约0.6ms

/ void delay(void)

{

unsigned char j;

for(j=0;j<200;j++)

;

}

/ 函数功能:显示计数次数的子程序

入口参数:x

/ void Display(unsigned char x)

{

P2=0xf7; //P2.6引脚输出低电平,DS6点亮

P0=Tab[x/10]; //显示十位

delay;

P2=0xfb; //P2.7引脚输出低电平,DS7点亮

P0=Tab[x%10]; //显示个位

delay;

}

/

函数功能:主函数

/

void main(void)

{

EA=1; //开放总中断

EX0=1; //允许使用外中断

IT0=1; //选择负跳变来触发外中断

x=0;

while(1)

Display(x);

}

/ 函数功能:外中断T0的中断服务程序

void int0(void) interrupt 0 using 0 //外中断0的中断编号为0

{

x++;

if(x==100)

x=0;

}

*实例68:静态显示数字“59”

#include<reg51.h> //包含51单片机寄存器定义的头文件

/ /

函数功能:主函数

/

void main(void)

{

P0=0x92; //将数字5的段码送P0口

P1=0x90; //将数字9的段码送P1口

while(1) //无限循环,防止程序跑飞

;

}

*实例69:无软件消抖的独立式键盘输入实验 #include<reg51.h> // 包含51单片机寄存器定义的头文件 sbit S1=P1^4; //将S1位定义为P1.4引脚

sbit LED0=P3^0; //将LED0位定义为P3.0引脚 void main(void) //主函数

{

LED0=0; //P3.0引脚输出低电平

while(1)

{

if(S1==0) //P1.4引脚输出低电平,按键S1被按下 LED0=!LED0; //P3.0引脚取反

}

}

*实例70:软件消抖的独立式键盘输入实验

#include<reg51.h> // 包含51单片机寄存器定义的头文件 sbit S1=P1^4; //将S1位定义为P1.4引脚

sbit LED0=P3^0; //将LED0位定义为P3.0引脚

/ 函数功能:延时约30ms

/ void delay(void)

{

unsigned char i,j;

for(i=0;i<100;i++)

for(j=0;j<100;j++)

;

}

/ 函数功能:主函数

/ void main(void) //主函数

{

LED0=0; //P3.0引脚输出低电平

while(1)

{

if(S1==0) //P1.4引脚输出低电平,按键S1被按下 {

delay; //延时一段时间再次检测

if(S1==0) // 按键S1的确被按下

LED0=!LED0; //P3.0引脚取反

}

}

}

*实例71:CPU控制的独立式键盘扫描实验

#include<reg51.h> //包含51单片机寄存器定义的头文件 sbit S1=P1^4; //将S1位定义为P1.4引脚 sbit S2=P1^5; //将S2位定义为P1.5引脚 sbit S3=P1^6; //将S3位定义为P1.6引脚 sbit S4=P1^7; //将S4位定义为P1.7引脚 unsigned char keyval; //储存按键值

/ 函数功能:流水灯延时

/ void led_delay(void)

{

unsigned char i,j;

for(i=0;i<250;i++)

for(j=0;j<250;j++)

;

}

/ 函数功能:软件消抖延时

/

void delay30ms(void)

{

unsigned char i,j;

for(i=0;i<100;i++)

for(j=0;j<100;j++)

;

}

/ 函数功能:正向流水点亮LED

void forward(void)

{

P3=0xfe; //第一个灯亮

led_delay;

P3=0xfd; //第二个灯亮

led_delay;

P3=0xfb; //第三个灯亮 led_delay;

P3=0xf7; //第四个灯亮 led_delay;

P3=0xef; //第五个灯亮

led_delay;

P3=0xdf; //第六个灯亮

led_delay;

P3=0xbf; //第七个灯亮

led_delay;

P3=0x7f; //第八个灯亮

led_delay;

P3=0xff;

P3=0xfe; //第一个灯亮

led_delay;

}

/ 函数功能:反向流水点亮LED

void backward(void)

{

P3=0x7f; //第八个灯亮

led_delay;

P3=0xbf; //第七个灯亮

led_delay;

P3=0xdf; //第六个灯亮

led_delay;

P3=0xef; //第五个灯亮 / /

led_delay;

P3=0xf7; //第四个灯亮

led_delay;

P3=0xfb; //第三个灯亮

led_delay;

P3=0xfd; //第二个灯亮

led_delay;

P3=0xfe; //第一个灯亮

led_delay;

}

/ 函数功能:关闭所有LED

/ void stop(void)

{

P3=0xff;

}

/ 函数功能:闪烁点亮LED

/ void flash(void)

{

P3=0xff;

led_delay;

P3=0x00;

led_delay;

}

/ 函数功能:键盘扫描子程序

/ void key_scan(void)

{

if((P1&0xf0)!=0xf0) //第一次检测到有键按下 {

delay30ms; //延时20ms再去检测 if(S1==0) //按键S1被按下 keyval=1;

if(S2==0) //按键S2被按下 keyval=2;

if(S3==0) //按键S3被按下 keyval=3;

if(S4==0) //按键S4被按下 keyval=4;

}

}

/ 函数功能:主函数

/ void main(void) //主函数

{

keyval=0; //按键值初始化为0,什么也不做 while(1)

{

key_scan;

switch(keyval)

{

case 1:forward;

break;

case 2:backward;

break;

case 3:stop;

break;

case 4: flash;

break;

}

}

}

*实例72:定时器中断控制的独立式键盘扫描实验 #include<reg51.h> //包含51单片机寄存器定义的头文件 sbit S1=P1^4; //将S1位定义为P1.4引脚

sbit S2=P1^5; //将S2位定义为P1.5引脚

sbit S3=P1^6; //将S3位定义为P1.6引脚

sbit S4=P1^7; //将S4位定义为P1.7引脚

unsigned char keyval; //储存按键值

/ 函数功能:流水灯延时

/ void led_delay(void)

{

unsigned char i,j;

for(i=0;i<250;i++)

for(j=0;j<250;j++)

;

}

/ 函数功能:软件消抖延时

/ void delay20ms(void)

{

unsigned char i,j;

for(i=0;i<100;i++)

for(j=0;j<60;j++)

;

}

/ 函数功能:正向流水点亮LED

void forward(void)

{

P3=0xfe; //第一个灯亮

led_delay;

P3=0xfd; //第二个灯亮

led_delay;

P3=0xfb; //第三个灯亮

led_delay;

P3=0xf7; //第四个灯亮

led_delay;

P3=0xef; //第五个灯亮

led_delay;

P3=0xdf; //第六个灯亮

led_delay;

P3=0xbf; //第七个灯亮

led_delay;

P3=0x7f; //第八个灯亮

led_delay;

P3=0xff;

P3=0xfe; //第一个灯亮

led_delay;

}

/ 函数功能:反向流水点亮LED

void backward(void) / /

{

P3=0x7f; //第八个灯亮 led_delay;

P3=0xbf; //第七个灯亮 led_delay;

P3=0xdf; //第六个灯亮 led_delay;

P3=0xef; //第五个灯亮 led_delay;

P3=0xf7; //第四个灯亮 led_delay;

P3=0xfb; //第三个灯亮 led_delay;

P3=0xfd; //第二个灯亮 led_delay;

P3=0xfe; //第一个灯亮 led_delay;

}

/ 函数功能:关闭所有LED

void stop(void)

{

P3=0xff; //关闭8个LED }

/ 函数功能:闪烁点亮LED

void flash(void)

{

P3=0xff; //关闭8个LED

led_delay;

P3=0x00; //点亮8个LED

led_delay;

}

/ 函数功能:主函数

void main(void) //主函数

{

TMOD=0x01; //使用定时器T0的模式1 / / /

EA=1; //开总中断

ET0=1; //定时器T0中断允许

TR0=1; //启动定时器T0

TH0=(65536-1000)/256; //定时器T0赋初值,每计数200次(217微秒)发送一次中断请求

TL0=(65536-1000)%256; //定时器T0赋初值

keyval=0; //按键值初始化为0,什么也不做

while(1)

{

switch(keyval)

{

case 1:forward;

break;

case 2:backward;

break;

case 3:stop;

break;

case 4: flash;

break;

}

}

}

/

函数功能:定时器T0的中断服务子程序

/

void Time0_serve(void) interrupt 1 using 1

{

if((P1&0xf0)!=0xf0) //第一次检测到有键按下

{

delay20ms; //延时20ms再去检测

if(S1==0) //按键S1被按下

keyval=1;

if(S2==0) //按键S2被按下

keyval=2;

if(S3==0) //按键S3被按下

keyval=3;

if(S4==0) //按键S4被按下

keyval=4;

}

TH0=(65536-1000)/256;

TL0=(65536-1000)%256;

}

*实例73:独立式键盘控制的4级变速流水灯 #include<reg51.h> // 包含51单片机寄存器定义的头文件 unsigned char speed; //储存流水灯的流动速度

sbit S1=P1^4; //位定义S1为P1.4

sbit S2=P1^5; //位定义S2为P1.5

sbit S3=P1^6; //位定义S3为P1.6

sbit S4=P1^7; //位定义S4为P1.7

/ 函数功能:延时20ms的子程序

void delay20ms(void) //3 i j+2 i=3 100 60+2 100=20000μs=20ms; {

unsigned char i,j;

for(i=0;i<100;i++)

for(j=0;j<60;j++)

;

}

/ 函数功能:延时可调子程序

入口参数:x

void delay(unsigned char x)

{

unsigned char k;

for(k=0;k<x;k++)

delay20ms;

}

/ 函数功能:主函数

void main(void)

{

TMOD=0x02; //使用定时器T0的模式2

EA=1; //开总中断

ET0=1; //定时器T0中断允许

TR0=1; //定时器T0开始运行

TH0=256-200; //定时器T0赋初值,每200微妙来1次中断请求 TL0=256-200;

speed=3; //默认流水灯流水点亮延时20ms×3=60ms / / /

while(1)

{

P3=0xfe; //第一个灯亮

delay(speed); //调用延时可调子程序

P3=0xfd; //第二个灯亮

delay(speed);

P3=0xfb; //第三个灯亮

delay(speed);

P3=0xf7; //第四个灯亮

delay(speed);

P3=0xef; //第五个灯亮

delay(speed);

P3=0xdf; //第六个灯亮

delay(speed);

P3=0xbf; //第七个灯亮

delay(speed);

P3=0x7f; //第八个灯亮

delay(speed);

P3=0xff;

}

}

/ 函数功能:定时器T0的中断服务子程序,进行键盘扫描

/ void intersev(void) interrupt 1 using 1

{

TR0=0; //关闭定时器T0/

P1=0xff; //将P1口的均置高电平"1"

if((P1&0xf0)!=0xf0) //如果有键按下

{

delay20ms; //延时20ms,软件消抖

if((P1&0xf0)!=0xf0) //确实有键按下

{

if(S1==0) //如果是按键S1按下

speed=5; //流水灯流水点亮延时20ms×5=100ms if(S2==0) //如果是按键S2按下

speed=10; //流水灯流水点亮延时20ms×10=200ms if(S3==0) //如果是按键S3按下

speed=25; //流水灯流水点亮延时20ms×25=500ms if(S4==0) //如果是按键S4按下

speed=50; //流水灯流水点亮延时20ms×50=1000ms }

}

TR0=1; //启动定时器T0

}

*实例74:独立式键盘的按键功能扩展:"以一当四" #include<reg51.h> // 包含51单片机寄存器定义的头文件 unsigned char ID; //储存流水灯的流动速度

sbit S1=P1^4; //位定义S1为P1.4

/ 函数功能:延时子程序

/ void delay(void) //因为仅对一个按键扫描,所以延时时间较长约200ms {

unsigned char i,j;

for(i=0;i<200;i++)

for(j=0;j<100;j++)

;

}

/ 函数功能:主函数

/ void main(void)

{

TMOD=0x02; //使用定时器T0的模式2

EA=1; //开总中断

ET0=1; //定时器T0中断允许

TR0=1; //定时器T0开始运行

TH0=256-200; //定时器T0赋初值,每200微妙来1次中断请求 TL0=256-200;

ID=0;

while(1)

{

switch(ID)

{

case 0: P3=0xfe;

break;

case 1: P3=0xfd;

break;

case 2: P3=0xfb;

break;

case 3: P3=0xf7;

break;

}

}

}

/ 函数功能:定时器T0的中断服务子程序,进行键盘扫描

/ void intersev(void) interrupt 1 using 1

{

TR0=0; //关闭定时器T0

P1=0xff;

if(S1==0) //如果是按键S1按下

{

delay; //延时20ms,软件消抖

if(S1==0) //如果是按键S1按下

ID=ID+1;

}

if(ID==4)

ID=0;

TR0=1; //启动定时器T0

}

*实例75:独立式键盘调时的数码时钟实验

#include<reg51.h> // 包含51单片机寄存器定义的头文件

unsigned char code Tab[ ]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90}; //数字0~9的段码

unsigned char int_time ; //中断次数计数变量

unsigned char second; //秒计数变量

unsigned char minute; //分钟计数变量

unsigned char hour; //小时计数变量

sbit S1=P1^4; //将S1位定义为P1.4

sbit S2=P1^5; //将S2位定义为P1.5

sbit S3=P1^6; //将S3位定义为P1.6

sbit S4=P1^7; //将S4位定义为P1.7

/ 函数功能:数码管扫描延时

/ void delay(void)

{

unsigned char j;

for(j=0;j<200;j++)

;

}

/ 函数功能:键盘扫描延时

void delay60ms(void)

{

unsigned char i,j;

for(i=0;i<200;i++)

for(j=0;j<100;j++)

;

}

/ 函数功能:显示秒

入口参数:s

void DisplaySecond(unsigned char s)

{

P2=0xbf; //P2.6引脚输出低电平, DS6点亮

P0=Tab[s/10]; //显示十位

delay;

P2=0x7f; //P2.7引脚输出低电平, DS7点亮

P0=Tab[s%10]; //显示个位

delay;

P2=0xff; //关闭所有数码管

}

/ 函数功能:显示分钟

入口参数:m

void DisplayMinute(unsigned char m)

{ / / /

P2=0xf7; // P2.3引脚输出低电平, DS3点亮 P0=Tab[m/10];//显示个位

delay;

P2=0xef; // P2.4引脚输出低电平, DS4点亮 P0=Tab[m%10];

delay;

P2=0xdf; //P2.5引脚输出低电平, DS5点亮 P0=0xbf; //分隔符“-”的段码

delay;

P2=0xff; //关闭所有数码管

}

/ 函数功能:显示小时的子程序

入口参数:h

void DisplayHour(unsigned char h)

{

P2=0xfe; //P2.0引脚输出低电平, DS0点亮 P0=Tab[h/10]; //显示十位

delay;

P2=0xfd; //P2.1引脚输出低电平, DS1点亮 P0=Tab[h%10]; //显示个位

delay;

P2=0xfb; //P2.2引脚输出低电平, DS2点亮 P0=0xbf; //分隔符“-”的段码

delay;

P2=0xff; //关闭所有数码管

}

/ 函数功能:键盘扫描

void key_scan(void)

{

P1=0xf0; //将P1口高4位置高电平“1”

if((P1&0xf0)!=0xf0) //有键按下

{

delay60ms; //延时60ms再检测

if((P1&0xf0)!=0xf0) //确实有键按下 {

if(S1==0) //如果是S1键按下 second++; //秒加1 / /

if(S2==0) //如果是S2键按下 minute++; //分钟加1

if(S3==0) //如果是S3键按下 hour++; //小时加1

if(S4==0) //如果是S4键按下 {

second=0; //秒清0

minute=0; //分钟清0

hour=0; //小时清0

}

}

}

}

/ 函数功能:主函数

void main(void)

{

TMOD=0x01; //使用定时器T0 EA=1; //开中断总允许

ET0=1; //允许T0中断

TH0=(65536-46083)/256; //定时器高八位赋初值 TL0=(65536-46083)%256; //定时器低八位赋初值 TR0=1; //启动定时器T0

int_time=0; //中断计数变量初始化

second=0; //秒计数变量初始化

minute=0; //分钟计数变量初始化

hour=0; //小时计数变量初始化

while(1)

{

DisplaySecond(second); //调用秒显示子程序 DisplayMinute(minute); //调用分钟显示子程序 DisplayHour(hour); //调用小时显示子程序

}

}

/ 函数功能:定时器T0的中断服务子程序

/ /

void interserve(void ) interrupt 1 using 1 //using Time0

{

TR0=0; //关闭定时器T0

int_time++; //中断次数加1

if(int_time==20) //如果中断次数满20

{

int_time=0; //中断计数变量清0

second++; //秒计数变量加1

}

if(second==60) //如果秒计满60

{

second=0; //如果秒计满60,将秒计数变量清0 minute++; //分钟计数变量加1

}

if(minute==60) //如果分钟计满60

{

minute=0; //如果分钟计满60,将分钟计数变量清0 hour++; //小时计数变量加1

}

if(hour==24) //如果小时计满24

{

hour=0; //如果小时计满24,将小时计数变量清0 }

key_scan; //执行键盘扫描

TH0=(65536-46083)/256; //定时器T0高四位赋值

TL0=(65536-46083)%256; //定时器T0低四位赋值

TR0=1; //启动定时器T0

}

*实例76:独立式键盘控制步进电机实验

#include<reg51.h> //包含51单片机寄存器定义的头文件

sbit S1=P1^4; //将S1位定义为P1.4引脚

sbit S2=P1^5; //将S2位定义为P1.5引脚

sbit S3=P1^6; //将S3位定义为P1.6引脚

unsigned char keyval; //储存按键值

unsigned char ID; //储存功能标号

/

函数功能:软件消抖延时(约50ms)

/

void delay(void)

{

unsigned char i,j;

for(i=0;i<150;i++)

for(j=0;j<100;j++)

;

}

/ 函数功能:步进电机转动延时,延时越长,转速越慢

void motor_delay(void)

{

unsigned int i;

for(i=0;i<2000;i++)

;

}

/

函数功能:步进电机正转

void forward( )

{

P0=0xfc;

motor_delay;

P0=0xf6;

motor_delay;

P0=0xf3;

motor_delay;

P0=0xf9;

motor_delay;

}

/

函数功能:步进电机反转

void backward

{

P0=0xfc;

motor_delay;

P0=0xf9;

motor_delay;

P0=0xf3;

motor_delay;

P0=0xf6;

motor_delay;

} //P0口低四位脉冲1100 //P0口低四位脉冲0110 //P0口低四位脉冲0011 //P0口低四位脉冲1001 //P0口低四位脉冲1100 //P0口低四位脉冲1001 //P0口低四位脉冲0011 //P0口低四位脉冲0110 / / /

/

函数功能:步进电机停转

/

void stop(void)

{

P0=0xff ; //停止输出脉冲

}

/

函数功能:主函数

/

void main(void)

{

TMOD=0x01; //使用定时器T0的模式1

EA=1; //开总中断

ET0=1; //定时器T0中断允许

TR0=1; //启动定时器T0

TH0=(65536-500)/256; //定时器T0赋初值,每计数200次(217微秒)发送一次中断请求

TL0=(65536-500)%256; //定时器T0赋初值

keyval=0; //按键值初始化为0,什么也不做

ID=0;

while(1)

{

switch(keyval) //根据按键值keyval选择待执行的功能 {

case 1:forward; //按键S1按下,正转

break;

case 2:backward; //按键S2按下 ,反转

break;

case 3:stop; //按键S3按下,停转

break;

}

}

}

/

函数功能:定时器T0的中断服务子程序

/

void Time0_serve(void) interrupt 1 using 1

{

TR0=0; //关闭定时器T0

if((P1&0xf0)!=0xf0) //第一次检测到有键按下

{

delay; //延时一段时间再去检测

if((P1&0xf0)!=0xf0) //确实有键按下

{

if(S1==0) //按键S1被按下

keyval=1;

if(S2==0) //按键S2被按下

keyval=2;

if(S3==0) //按键S3被按下

keyval=3;

}

}

TH0=(65536-200)/256; //定时器T0的高8位赋初值

TL0=(65536-200)%256; //定时器T0的低8位赋初值

TR0=1; //启动定时器T0

}

*实例77:矩阵式键盘按键值的数码管显示实验

#include<reg51.h> //包含51单片机寄存器定义的头文件

sbit P14=P1^4; //将P14位定义为P1.4引脚

sbit P15=P1^5; //将P15位定义为P1.5引脚

sbit P16=P1^6; //将P16位定义为P1.6引脚

sbit P17=P1^7; //将P17位定义为P1.7引脚

unsigned char code Tab[ ]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90}; //数字0~9的段码

unsigned char keyval; //定义变量储存按键值

/ 函数功能:数码管动态扫描延时

/ void led_delay(void)

{

unsigned char j;

for(j=0;j<200;j++)

;

}

/ 函数功能:按键值的数码管显示子程序

/ void display(unsigned char k)

{

P2=0xbf; //点亮数码管DS6

P0=Tab[k/10]; //显示十位

led_delay; //动态扫描延时

P2=0x7f; //点亮数码管DS7

P0=Tab[k%10]; //显示个位

led_delay; //动态扫描延时

}

/ 函数功能:软件延时子程序

/ void delay20ms(void)

{

unsigned char i,j;

for(i=0;i<100;i++)

for(j=0;j<60;j++)

;

}

/ 函数功能:主函数

/ void main(void)

{

EA=1; //开总中断

ET0=1; //定时器T0中断允许

TMOD=0x01; //使用定时器T0的模式1

TH0=(65536-500)/256; //定时器T0的高8位赋初值

TL0=(65536-500)%256; //定时器T0的高8位赋初值

TR0=1; //启动定时器T0

keyval=0x00; //按键值初始化为0

while(1) //无限循环

{

display(keyval); //调用按键值的数码管显示子程序

}

}

/ 函数功能:定时器0的中断服务子程序,进行键盘扫描,判断键位

/

void time0_interserve(void) interrupt 1 using 1 //定时器T0的中断编号为1,使用第一组寄存器

{

TR0=0; //关闭定时器T0

P1=0xf0; //所有行线置为低电平“0”,所有列线置为高电平“1”

if((P1&0xf0)!=0xf0) //列线中有一位为低电平“0”,说明有键按下 delay20ms; //延时一段时间、软件消抖

if((P1&0xf0)!=0xf0) //确实有键按下

{

P1=0xfe; //第一行置为低电平“0”(P1.0输出低电平“0”)

if(P14==0) //如果检测到接P1.4引脚的列线为低电平“0” keyval=1; //可判断是S1键被按下

if(P15==0) //如果检测到接P1.5引脚的列线为低电平“0”

keyval=2; //可判断是S2键被按下

if(P16==0)

“0”

keyval=3;

if(P17==0)

“0”

keyval=4;

P1=0xfd;

“0”)

if(P14==0)

keyval=5;

if(P15==0)

“0”

keyval=6;

if(P16==0)

“0”

keyval=7;

if(P17==0)

“0”

keyval=8;

P1=0xfb;

“0”)

if(P14==0)

keyval=9;

if(P15==0)

keyval=10;

if(P16==0)

keyval=11;

if(P17==0)

keyval=12;

P1=0xf7;

“0”)

if(P14==0) //如果检测到接P1.6引脚的列线为低电平 //可判断是S3键被按下 //如果检测到接P1.7引脚的列线为低电平 //可判断是S4键被按下 //第二行置为低电平“0”(P1.1输出低电平 //如果检测到接P1.4引脚的列线为低电平“0” //可判断是S5键被按下 //如果检测到接P1.5引脚的列线为低电平 //可判断是S6键被按下 //如果检测到接P1.6引脚的列线为低电平 //可判断是S7键被按下 //如果检测到接P1.7引脚的列线为低电平 //可判断是S8键被按下 //第三行置为低电平“0”(P1.2输出低电平//如果检测到接P1.4引脚的列线为低电平“0” //可判断是S9键被按下 //如果检测到接P1.5引脚的列线为低电平“0” //可判断是S10键被按下 //如果检测到接P1.6引脚的列线为低电平“0” //可判断是S11键被按下 //如果检测到接P1.7引脚的列线为低电平“0” //可判断是S12键被按下 //第四行置为低电平“0”(P1.3输出低电平//如果检测到接P1.4引脚的列线为低电平“0”

keyval=13; //可判断是S13键被按下

if(P15==0) //如果检测到接P1.5引脚的列线为低电平“0” keyval=14; //可判断是S14键被按下

if(P16==0) //如果检测到接P1.6引脚的列线为低电平“0” keyval=15; //可判断是S15键被按下

if(P17==0) //如果检测到接P1.7引脚的列线为低电平“0” keyval=16; //可判断是S16键被按下 }

TR0=1; //开启定时器T0

TH0=(65536-500)/256; //定时器T0的高8位赋初值 TL0=(65536-500)%256; //定时器T0的高8位赋初值 }

*实例78:矩阵式键盘按键音

#include<reg51.h> //包含51单片机寄存器定义的头文件 sbit sound=P3^7; //将sound位定义为P3.7

/ 函数功能:蜂鸣器发声延时约120ms

void delay_sound(void) {

unsigned char i; for(i=0;i<250;i++) ; }

/ 函数功能:软件延时子程序约20ms

void delay20ms(void) {

unsigned char i,j; for(i=0;i<100;i++) for(j=0;j<60;j++) ; }

/ 函数功能:主函数

void main(void)

/ / /

{

EA=1; //开总中断

ET0=1; //定时器T0中断允许

TMOD=0x01; //使用定时器T0的模式1

TH0=(65536-500)/256; //定时器T0的高8位赋初值

TL0=(65536-500)%256; //定时器T0的高8位赋初值

TR0=1; //启动定时器T0

while(1) //无限循环,等待键盘按下

;

}

/ 函数功能:定时器0的中断服务子程序,进行键盘扫描,判断键位

/

void time0_interserve(void) interrupt 1 using 1 //定时器T0的中断编号为1,使用第一组寄存器

{

unsigned char i;

TR0=0; //关闭定时器T0

P1=0xf0; //所有行线置为低电平“0”,所有列线置为高电平“1”

if((P1&0xf0)!=0xf0) //列线中有一位为低电平“0”,说明有键按下 delay20ms; //延时一段时间、软件消抖

if((P1&0xf0)!=0xf0) //确实有键按下

{

for(i=0;i<200;i++) //让P3.7引脚电平不断取反输出音频

{

sound=0;

delay_sound;

sound=1;

delay_sound;

}

}

TR0=1; //开启定时器T0

TH0=(65536-500)/256; //定时器T0的高8位赋初值

TL0=(65536-500)%256; //定时器T0的高8位赋初值 }

*实例79:简易电子琴

#include<reg51.h> //包含51单片机寄存器定义的头文件

sbit P14=P1^4; //将P14位定义为P1.4引脚

sbit P15=P1^5; //将P15位定义为P1.5引脚

sbit P16=P1^6; //将P16位定义为P1.6引脚

sbit P17=P1^7; //将P17位定义为P1.7引脚

unsigned char keyval; //定义变量储存按键值

sbit sound=P3^7; //将sound位定义为P3.7

unsigned int C; //全局变量,储存定时器的定时常数

unsigned int f; //全局变量,储存音阶的频率

//以下是C调低音的音频宏定义

#define l_dao 262 //将“l_dao”宏定义为低音“1”的频率262Hz #define l_re 286 //将“l_re”宏定义为低音“2”的频率286Hz #define l_mi 311 //将“l_mi”宏定义为低音“3”的频率311Hz #define l_fa 349 //将“l_fa”宏定义为低音“4”的频率349Hz #define l_sao 392 //将“l_sao”宏定义为低音“5”的频率392Hz #define l_la 440 //将“l_a”宏定义为低音“6”的频率440Hz #define l_xi 494 //将“l_xi”宏定义为低音“7”的频率494Hz

//以下是C调中音的音频宏定义

#define dao 523 //将“dao”宏定义为中音“1”的频率523Hz #define re 587 //将“re”宏定义为中音“2”的频率587Hz #define mi 659 //将“mi”宏定义为中音“3”的频率659Hz #define fa 698 //将“fa”宏定义为中音“4”的频率698Hz #define sao 784 //将“sao”宏定义为中音“5”的频率784Hz #define la 880 //将“la”宏定义为中音“6”的频率880Hz #define xi 987 //将“xi”宏定义为中音“7”的频率53

//以下是C调高音的音频宏定义

#define h_dao 1046 //将“h_dao”宏定义为高音“1”的频率1046Hz #define h_re 1174 //将“h_re”宏定义为高音“2”的频率1174Hz #define h_mi 1318 //将“h_mi”宏定义为高音“3”的频率1318Hz #define h_fa 1396 //将“h_fa”宏定义为高音“4”的频率1396Hz #define h_sao 1567 //将“h_sao”宏定义为高音“5”的频率1567Hz #define h_la 1760 //将“h_la”宏定义为高音“6”的频率1760Hz #define h_xi 1975 //将“h_xi”宏定义为高音“7”的频率1975Hz / 函数功能:软件延时子程序

/ void delay20ms(void)

{

unsigned char i,j;

for(i=0;i<100;i++)

for(j=0;j<60;j++)

;

}

/

函数功能:节拍的延时的基本单位,延时200ms

/

void delay

{

unsigned char i,j;

for(i=0;i<250;i++)

for(j=0;j<250;j++)

;

}

/

函数功能:输出音频

入口参数:F

/

void Output_Sound(void)

{

C=(46083/f) 10; //计算定时常数

TH0=(8192-C)/32; //可证明这是13位计数器TH0高8位的赋初值方法 TL0=(8192-C)%32; //可证明这是13位计数器TL0低5位的赋初值方法 TR0=1; //开定时T0

delay; //延时200ms,播放音频

TR0=0; //关闭定时器

sound=1; //关闭蜂鸣器

keyval=0xff; //播放按键音频后,将按键值更改,停止播放 }

/

函数功能:主函数

/

void main(void)

{

EA=1; //开总中断

ET0=1; //定时器T0中断允许

ET1=1; //定时器T1中断允许

TR1=1; //定时器T1启动,开始键盘扫描

TMOD=0x10; //分别使用定时器T1的模式1,T0的模式0 TH1=(65536-500)/256; //定时器T1的高8位赋初值

TL1=(65536-500)%256; //定时器T1的高8位赋初值

while(1) //无限循环

{

switch(keyval)

{

case 1:f=dao; //如果第1个键按下,将中音1的频率赋给f

Output_Sound; //转去计算定时常数 break;

case 2:f=l_xi; //如果第2个键按下,将低音7的频率赋给f

Output_Sound; //转去计算定时常数 break;

case 3:f=l_la; //如果第3个键按下,将低音6的频率赋给f

Output_Sound; //转去计算定时常数 break;

case 4:f=l_sao; //如果第4个键按下,将低音5的频率赋给f

Output_Sound; //转去计算定时常数 break;

case 5:f=sao; //如果第5个键按下,将中音5的频率赋给f

Output_Sound; //转去计算定时常数 break;

case 6:f=fa; //如果第6个键按下,将中音4的频率赋给f

Output_Sound; //转去计算定时常数 break;

case 7:f=mi; //如果第7个键按下,将中音3的频率赋给f

Output_Sound; //转去计算定时常数 break;

case 8:f=re; //如果第8个键按下,将中音2的频率赋给f

Output_Sound; //转去计算定时常数 break;

case 9:f=h_re; //如果第9个键按下,将高音2的频率赋给f

Output_Sound; //转去计算定时常数 break;

case 10:f=h_dao; //如果第10个键按下,将高音1的频率赋给f

Output_Sound; //转去计算定时常数 break;

case 11:f=xi; //如果第11个键按下,将中音7的频率赋给f

Output_Sound; //转去计算定时常数 break;

case 12:f=la; //如果第12个键按下,将中音6的频率赋给f

Output_Sound; //转去计算定时常数 break;

case 13:f=h_la; //如果第13个键按下,将高音6的频率赋给f

Output_Sound; //转去计算定时常数 break;

case 14:f=h_sao; //如果第14个键按下,将高音5的频率赋给f

Output_Sound; //转去计算定时常数 break;

case 15:f=h_fa; //如果第15个键按下,将高音4的频率赋给f

Output_Sound; //转去计算定时常数 break;

case 16:f=h_mi; //如果第16个键按下,将高音3的频率赋给f

Output_Sound; //转去计算定时常数 break;

}

}

}

/ 函数功能:定时器T0的中断服务子程序,使P3.7引脚输出音频方波

/

void Time0_serve(void ) interrupt 1 using 1

{

TH0=(8192-C)/32; //可证明这是13位计数器TH0高8位的赋初值方法

TL0=(8192-C)%32; //可证明这是13位计数器TL0低5位的赋初值方法

sound=!sound; //将P3.7引脚取反,输出音频方波

}

/ 函数功能:定时器T1的中断服务子程序,进行键盘扫描,判断键位

/

void time1_serve(void) interrupt 3 using 2 //定时器T1的中断编号为3,使用第2组寄存器 {

TR1=0; //关闭定时器T0

P1=0xf0; //所有行线置为低电平“0”,所有列线置为高电平“1” if((P1&0xf0)!=0xf0) //列线中有一位为低电平“0”,说明有键按下 { delay20ms; //延时一段时间、软件消抖 if((P1&0xf0)!=0xf0) { P1=0xfe; 输出低电平“0”) if(P14==0) 为低电平“0” keyval=1; if(P15==0) 线为低电平“0” keyval=2; if(P16==0) 线为低电平“0” keyval=3; if(P17==0) 线为低电平“0” keyval=4; P1=0xfd; 输出低电平“0”) if(P14==0) 低电平“0” keyval=5; if(P15==0) 线为低电平“0” keyval=6; if(P16==0) 线为低电平“0” keyval=7; if(P17==0) 线为低电平“0” keyval=8; P1=0xfb; 输出低电平“0”)

//确实有键按下 //第一行置为低电平“0”(P1.0 //如果检测到接P1.4引脚的列线 //可判断是S1键被按下 //如果检测到接P1.5引脚的列 //可判断是S2键被按下 //如果检测到接P1.6引脚的列 //可判断是S3键被按下 //如果检测到接P1.7引脚的列 //可判断是S4键被按下 //第二行置为低电平“0”(P1.1 //如果检测到接P1.4引脚的列线为 //可判断是S5键被按下 //如果检测到接P1.5引脚的列 //可判断是S6键被按下 //如果检测到接P1.6引脚的列 //可判断是S7键被按下 //如果检测到接P1.7引脚的列 //可判断是S8键被按下 //第三行置为低电平“0”(P1.2

if(P14==0) //如果检测到接P1.4引脚的列线为低电平“0”

keyval=9; //可判断是S9键被按下

if(P15==0) //如果检测到接P1.5引脚的列线为低电平“0”

keyval=10; //可判断是S10键被按下

if(P16==0) //如果检测到接P1.6引脚的列线为低电平“0”

keyval=11; //可判断是S11键被按下

if(P17==0) //如果检测到接P1.7引脚的列线为低电平“0”

keyval=12; //可判断是S12键被按下

P1=0xf7; //第四行置为低电平“0”(P1.3输出低电平“0”)

if(P14==0) //如果检测到接P1.4引脚的列线为低电平“0”

keyval=13; //可判断是S13键被按下

if(P15==0) //如果检测到接P1.5引脚的列线为低电平“0”

keyval=14; //可判断是S14键被按下

if(P16==0) //如果检测到接P1.6引脚的列线为低电平“0”

keyval=15; //可判断是S15键被按下

if(P17==0) //如果检测到接P1.7引脚的列线为低电平“0”

keyval=16; //可判断是S16键被按下 }

}

TR1=1; //开启定时器T1

TH1=(65536-500)/256; //定时器T1的高8位赋初值

TL1=(65536-500)%256; //定时器T1的高8位赋初值

}

*实例80:矩阵式键盘实现的电子密码锁

#include<reg51.h> //包含51单片机寄存器定义的头文件

sbit P14=P1^4; //将P14位定义为P1.4引脚

sbit P15=P1^5; //将P15位定义为P1.5引脚

sbit P16=P1^6; //将P16位定义为P1.6引脚

sbit P17=P1^7; //将P17位定义为P1.7引脚

sbit sound=P3^7; //将sound位定义为P3.7

unsigned char keyval; //储存按键值

/ 函数功能:延时输出音频

/ void delay(void)

{

unsigned char i;

for(i=0;i<200;i++)

;

}

/ 函数功能:软件延时子程序

void delay20ms(void)

{

unsigned char i,j;

for(i=0;i<100;i++)

for(j=0;j<60;j++)

;

}

/ 函数功能:主函数

void main(void)

{

unsigned char D[ ]={0,8,0,8,7,4,11}; //设定密码

EA=1; //开总中断

ET0=1; //定时器T0中断允许 TMOD=0x01; //使用定时器T0的模式1

TH0=(65536-500)/256; //定时器T0的高8位赋初值

TL0=(65536-500)%256; //定时器T0的高8位赋初值

TR0=1; //启动定时器T0

keyval=0xff; //按键值初始化

while(keyval!=D[0]) //第一位密码输入不正确,等待

;

while(keyval!=D[1]) //第二位密码输入不正确,等待

;

while(keyval!=D[2]) //第三位密码输入不正确,等待

;

while(keyval!=D[3]) //第四位密码输入不正确,等待 / /

;

while(keyval!=D[4]) //第五位密码输入不正确,等待

;

while(keyval!=D[5]) //第六位密码输入不正确,等待

;

while(keyval!=D[6]) //没有输入“OK”,等待

;

P3=0xfe; //P3.0引脚输出低电平,点亮LED

}

/ 函数功能:定时器0的中断服务子程序,进行键盘扫描,判断键位

/

void time0_interserve(void) interrupt 1 using 1 //定时器T0的中断编号为1,使用第一组寄存器

{

unsigned char i;

TR0=0; //关闭定时器T0

P1=0xf0; //所有行线置为低电平“0”,所有列线置为高电平“1”

if((P1&0xf0)!=0xf0) //列线中有一位为低电平“0”,说明有键按下 delay20ms; //延时一段时间、软件消抖

if((P1&0xf0)!=0xf0) //确实有键按下

{

P1=0xfe; //第一行置为低电平“0”(P1.0输出低电平“0”)

if(P14==0) //如果检测到接P1.4引脚的列线为低电平“0” keyval=1; //可判断是S1键被按下

if(P15==0) //如果检测到接P1.5引脚的列线为低电平“0”

keyval=2; //可判断是S2键被按下

if(P16==0) //如果检测到接P1.6引脚的列线为低电平“0”

keyval=3; //可判断是S3键被按下

if(P17==0) //如果检测到接P1.7引脚的列线为低电平“0”

keyval=4; //可判断是S4键被按下

P1=0xfd; //第二行置为低电平“0”(P1.1输出低电平“0”)

if(P14==0) //如果检测到接P1.4引脚的列线为低电平“0” keyval=5; //可判断是S5键被按下

if(P15==0) //如果检测到接P1.5引脚的列线为低电平“0”

keyval=6; //可判断是S6键被按下

if(P16==0) //如果检测到接P1.6引脚的列线为低电平“0”

keyval=7; //可判断是S7键被按下

if(P17==0) //如果检测到接P1.7引脚的列线为低电平“0”

keyval=8; //可判断是S8键被按下

P1=0xfb; //第三行置为低电平“0”(P1.2输出低电平“0”)

if(P14==0) //如果检测到接P1.4引脚的列线为低电平“0” keyval=9; //可判断是S9键被按下

if(P15==0) //如果检测到接P1.5引脚的列线为低电平“0” keyval=0; //可判断是S10键被按下

if(P16==0) //如果检测到接P1.6引脚的列线为低电平“0” keyval=11; //可判断是S11键被按下

if(P17==0) //如果检测到接P1.7引脚的列线为低电平“0” keyval=12; //可判断是S12键被按下

P1=0xf7; //第四行置为低电平“0”(P1.3输出低电平“0”)

if(P14==0) //如果检测到接P1.4引脚的列线为低电平“0”

keyval=13; //可判断是S13键被按下

if(P15==0) //如果检测到接P1.5引脚的列线为低电平“0”

keyval=14; //可判断是S14键被按下

if(P16==0) //如果检测到接P1.6引脚的列线为低电平“0”

keyval=15; //可判断是S15键被按下

if(P17==0) //如果检测到接P1.7引脚的列线为低电平“0”

keyval=16; //可判断是S16键被按下 for(i=0;i<200;i++) //让P3.7引脚电平不断取反输出音频

{

sound=0;

delay;

sound=1;

delay;

}

}

TR0=1; //开启定时器T0

TH0=(65536-500)/256; //定时器T0的高8位赋初值

TL0=(65536-500)%256; //定时器T0的高8位赋初值

免责声明:本文转自网络,版权归原作者所有,如涉及作品版权问题,请及时与我们联系删除,谢谢!

使用特权

评论回复
沙发
闻则123| | 2023-12-12 14:35 | 只看该作者
学习了,感谢楼主分享

使用特权

评论回复
板凳
zwsam| | 2023-12-19 09:06 | 只看该作者

使用特权

评论回复
地板
AProgrammer| | 2023-12-20 09:09 | 只看该作者
不错!

使用特权

评论回复
发新帖 我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

39

主题

59

帖子

0

粉丝