5036-单片机按键控制PWM占空比(仿真+程序)

本文详细介绍了51单片机的P0、P1、P2和P3端口特性,以及如何在Proteus8.10中使用汇编语法学习和实现PWM控制。此外,还涵盖了中断处理和单片机管脚操作的关键知识点。
摘要由CSDN通过智能技术生成

功能描述

1、51+Proteus8.10;

2、学习汇编语法、PWM控制原理;

仿真设计

单片机管脚说明:

P0端口(P0.0-P0.7):P0口为一个8位漏极开路双向I/O口,每个引脚可吸收8TTL门电流。当P1口的管脚第一次写1时,被定义为高阻输入。P0能够用于外部程序数据存储器,它可以被定义为数据/地址的第八位。在FIASH编程时,P0 口作为原码输入口,当FIASH进行校验时,P0输出原码,此时P0外部必须被拉高。

P1端口(P1.0-P1.7):P1口是一个内部提供上拉电阻的8位双向I/O口,P1口缓冲器能接收输出4TTL门电流。P1口管脚写入1后,被内部上拉为高电平,可用作输入,P1口被外部下拉为低电平时,将输出电流,这是由于内部上拉的缘故。在FLASH编程和校验时,P1口作为第八位地址接收。

P2端口(P2.0-P2.7):P2口为一个内部上拉电阻的8位双向I/O口,P2口缓冲器可接收,输出4个TTL门电流,当P2口被写“1”时,其管脚被内部上拉电阻拉高,且作为输入。并因此作为输入时,P2口的管脚被外部拉低,将输出电流。这是由于内部上拉的缘故。P2口,用于外部程序存储器或16位地址外部数据存储器进行存取时,P2口输出地址的高八位。在给出地址“1”时,它利用内部上拉优势,当对外部八位地址数据存储器进行读写时,P2口输出其特殊功能寄存器的内容。P2口在FLASH编程和校验时接收高八位地址信号和控制信号。

P3端口(P3.0-P3.7):P3口管脚是一个带有内部上拉电阻的8位的双向I/O端口,可接收输出4个TTL门电流。当P3口写入“1”后,它们被内部上拉为高电平,并用作输入。作为输入端时,由于外部下拉为低电平,P3口将输出电流(ILL)。P3口同时为闪烁编程和编程校验接收一些控制信号。

VCC(40):供电电压,其工作电压为5V。
GND(20):接地。

RST(9):复位输入。在振荡器运行时,有两个机器周期(24个振荡周期)以上的高电平出现在此引脚时,将使单片机复位,只要这个引脚保持高电平,51芯片便循环复位。复位后P3.0-P3.7口均置1,引脚表现为高电平,程序计数器和特殊功能寄存器SFR全部清零。当复位脚由高电平变为低电平时,芯片为ROM的00H处开始运行程序。复位操作不会对内部RAM有所影响。

ALE/PROG (30):当访问外部存储器时,地址锁存允许的输出电平用于锁存地址的地低位字节。在FLASH编程期间,此引脚用于输入编程脉冲。在平时,ALE端以不变的频率周期输出正脉冲信号,此频率为振荡器频率的1/6。因此它可用作对外部输出的脉冲或用于定时目的。然而要注意的是:每当用作外部数据存储器时,将跳过一个ALE脉冲。如果想禁止ALE的输出可在SFR8EH地址上置0。此时, ALE只有在执行MOVX,MOVC指令是ALE才起作用。另外,该引脚被略微拉高。如果微处理器在外部执行状态ALE禁止,则置位无效。

PSEN(29):外部程序存储器的选通信号。在由外部程序存储器取指令期间,每个机器周期两次PSEN有效。但在访问外部数据存储器时,这两次有效的PSEN信号将不出现。

XTAL1(19):来自反向振荡放大器的输入及内部时钟工作电路的输入。
XTAL2(18):来自反向振荡器的输出。

EA/VPP(31):当EA保持低电平时,则在此期间外部程序存储器(0000H-FFFFH),不管是否有内部程序存储器。注意加密方式1时,EA将内部锁定为RESET;当EA端保持高电平时,此间内部程序存储器。在FLASH编程期间,此引脚也用于施加12V的编程电源(VPP)。

 

程序设计

PWMH DATA 30H ;高电平脉冲的个数 
PWM DATA 31H ;PWM周期 
COUNTER DATA 32H 
TEMP DATA 33H 

ORG 0000H 
AJMP MAIN 
ORG 000BH 
AJMP INTT0 

ORG 0100H 
MAIN: 
	MOV SP,#60H ;给堆栈指针赋初值 
	MOV PWMH,#02H ; 
	MOV COUNTER,#01H 
	MOV PWM,#15H 
	MOV TMOD,#02H ;定时器0在模式2下工作 
	MOV TL0,#38H ;定时器每200us产生一次溢出 
	MOV TH0,#38H ;自动重装的值 
	SETB ET0 ;使能定时器0中断 
	SETB EA ;使能总中断 
	SETB TR0 ;开始计时 
KSCAN: 
	JNB P1.1,K1CHECK ;扫描KEY1, 
	JNB P1.2,K2CHECK ;扫描KEY2,如果按下KEY2,跳转到KEY2处理程序 
	SJMP KSCAN 

K1CHECK: 
	JB P1.1,K1HANDLE ;去抖动,如果按下KEY1,跳转到KEY1处理程序 
	SJMP K1CHECK 
K1HANDLE: 
	MOV A,PWMH 
	CJNE A,PWM,K1H0 ;判断是否到达上边界 
	SJMP KSCAN ;是,则不进行任何操作 
K1H0: 
	MOV A,PWMH 
	INC A 
	CJNE A,PWM,K1H1 ;如果在加1后到达最大值 
	CLR TR0 ;定时器停止 
	SETB P1.0 ;P1.0为高电平 
	SJMP K1H2 
K1H1: 
	CJNE A,#02H,K1H2 ;如果加1后到达下边界 
	SETB TR0 ;重开定时器 
K1H2: 
	INC PWMH ;增加占空比 
	SJMP KSCAN 

K2CHECK: 
	JB P1.2,K2HANDLE ;去抖动,如果按下KEY2,跳转到KEY2处理程序 
	SJMP K2CHECK 
K2HANDLE: 
	MOV A,PWMH 
	CJNE A,#01H,K2H0 ;判断是否到达下边界 
	SJMP KSCAN ;是,则不进行任何操作 
K2H0: 
	MOV A,PWMH 
	MOV TEMP,PWM 
	DEC A 
	CJNE A,#01H,K2H1 ;如果在减1后到达下边界 
	CLR TR0 ;定时器停止 
	CLR P1.0 ;P1.0为低电平 
	SJMP K2H2 
K2H1: 
	DEC TEMP 
	CJNE A,TEMP,K2H2 ;如果到达上边界 
	SETB TR0 ;启动定时器 
K2H2: 
	DEC PWMH ;降低占空比 
	SJMP KSCAN 


INTT0: 
	PUSH PSW ;现场保护 
	PUSH ACC 
	INC COUNTER ;计数值加1 
	MOV A,COUNTER 
	CJNE A,PWMH,INTT01 ;如果等于高电平脉冲数 
	CLR P1.0 ;P1.0变为低电平 
INTT01: 
	CJNE A,PWM,INTT02 ;如果等于周期数 
	MOV COUNTER,#01H ;计数器复位 
	SETB P1.0 ;P1.0为高电平 
INTT02: 
	POP ACC ;出栈 
	POP PSW 
	RETI 

	END

关注公众号 👇👇👇 获取更多设计资料 

  • 12
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
以下是一个简单的STC89C52RC单片机按键控制PWM输出LED灯亮度的C语言程序: ```c #include <reg52.h> #define FOSC 12000000UL // 定义晶振频率为12MHz #define T1MS (FOSC / 12000) // 定义1ms的计数值 sbit LED = P2^0; // 将P2.0定义为LED输出口 sbit KEY = P3^2; // 将P3.2定义为按键输入口 unsigned int duty = 0; // 定义占空比变量 void Timer1_Init() // 定时器1初始化函数 { TMOD = 0x10; // 定时器1工作在模式1,16位自动重装载 TH1 = 0xFC; // 计数初值为0xFC67,定时1ms TL1 = 0x67; ET1 = 1; // 开启定时器1中断 TR1 = 1; // 启动定时器1 EA = 1; // 开启总中断 } void Delay(unsigned int t) // 延时函数 { unsigned int i, j; for(i = 0; i < t; i++) for(j = 0; j < T1MS; j++); } void main() { Timer1_Init(); // 初始化定时器1 while(1) { if(KEY == 0) // 检测按键是否按下 { Delay(10); // 延时去抖 if(KEY == 0) // 再次检测按键是否按下 { duty += 10; // 占空比加10% if(duty > 100) // 限制占空比最大值为100% duty = 100; Delay(100); // 延时防止多次按下 } } } } void Timer1_ISR() interrupt 3 // 定时器1中断服务函数 { static unsigned int cnt = 0; if(++cnt > 100) // 计数器满100次,PWM周期为100ms cnt = 0; if(cnt < duty) // 占空比控制LED亮度 LED = 1; else LED = 0; } ``` 程序的原理是通过定时器1产生1ms的中断,每100ms计数器增加一次,根据占空比控制LED的亮度。按键按下时,占空比增加10%,最大值为100%。需要注意的是,由于单片机IO口输出的PWM精度有限,因此LED的亮度调节可能存在一定的误差。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

电子开发圈

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值