wuzhengmin
发表于 2025-8-13 12:03:49
取反就是反转每一位:
wuzhengmin
发表于 2025-8-13 12:08:36
左移一位,空出来的添加0
右移一位,空出来的添加0
都好理解。
其他还有很多运算符,有点难搞........
wuzhengmin
发表于 2025-8-13 12:22:13
今天基本就这样结束啦:
总结一下:
%d%u%s 要重点掌握哦
\t\r\n 要知道什么意思
ASCII码表,2进制,10进制,16进制要熟练
变量的类型,溢出要特别注意
要能回答为什么 20*10 不等于 2000?
8位存储是不能保留 0111 1101 0000 ,结果只剩下:1101 0000=208所以就产生了溢出错误
所以计算结果要放在16位的长整型int里才行!
算数运算符:逻辑运算:
wuzhengmin
发表于 2025-8-14 08:58:26
第九次:《8051U深度入门到32位51大型实战教学视频》第六集1/0输入输出作业
课后任务1:按一下P32按钮灯亮,按一下P33按钮灯灭;课后任务2:按一下亮一颗灯,在按一下亮两颗灯,直到全亮(变量+加法和乘法)
//课后任务1:按一下P32按钮灯亮,按一下P33按钮灯灭;
if( P32== 0 ) //判断P32按钮是否按下,0为按下
{
Delay20ms();//延时20毫秒消抖
if( P32== 0 ) //判断P32按钮是否确实按下
{
state = !state; //变里取反010101
P00 = 0;//P00端口输出0V,共阳极LED点亮
printf("state: %d\r\n", (int)state);
while(P32 == 0); //等待P32松开
}
}
if( P33== 0 ) //判断P33按钮是否按下,0为按下
{
Delay20ms();//延时20毫秒消抖
if( P33== 0 ) //判断P33按钮是否确实按下
{
state = !state; //变里取反010101
P00 = 1;//P00端口输出3.3V,共阳极LED熄灭
printf("state: %d\r\n", (int)state);
while(P33 == 0); //等待P33松开
}
}
下载执行结果:
wuzhengmin
发表于 2025-8-14 10:03:41
课后任务2:按一下亮一颗灯,在按一下亮两颗灯,直到全亮(变量+加法和乘法)
因为无法上传文件,我直接把main.c复制过来:
# include "ai8051u.h" //调用头文件
#include "stc32_stc8_usb.h" //调用头文件
#include "intrins.h" //调用头文件
#define u8 unsigned char//8位无符号变量(0-255)
#define ul6 unsigned int//16位无符号变里(0-65535)
u8 state = 0;//初始状态
char *USER_DEVICEDESC = NULL;
char *USER_PRODUCTDESC = NULL;
char *USER_STCISPCMD = "@STCISP#";
void Delay20ms(void);
void Delay20ms(void) //@24.000MHz Delay20ms(20);
{
unsigned long edata i;
_nop_();
_nop_();
i = 119998UL;
while (i) i--;
}
void main(void) //主函数
{
u8 LED_rksj = 0XFF;//送人P0口数据
u8 num = 0 ; //P32按键次数,
WTST = 0; //设置程序指令延时参数,
EAXFR =1; //扩展寄存器(XFR)访问使能
CKCON = 0; //提高访问XRAM速度
P0M1 = 0x00; P0M0 = 0x00; //端口配置直接用STC的ISP,很方便的
P1M1 = 0x00; P1M0 = 0x00; //端口配置直接用STC的ISP,很方便的
P2M1 = 0x00; P2M0 = 0x00; //端口配置直接用STC的ISP,很方便的
P3M1 = 0x00; P3M0 = 0x00; //端口配置直接用STC的ISP,很方便的
P4M1 = 0x00; P4M0 = 0x00; //端口配置直接用STC的ISP,很方便的
P5M1 = 0x00; P5M0 = 0x00; //端口配置直接用STC的ISP,很方便的
P6M1 = 0x00; P6M0 = 0x00; //端口配置直接用STC的ISP,很方便的
P7M1 = 0x00; P7M0 = 0x00; //端口配置直接用STC的ISP,很方便的
usb_init();
IE2 |= 0x80; //使能USB中断
EA = 1; //IE |= 0X80;
while (DeviceState != DEVSTATE_CONFIGURED); //等待USB完成配置
//注意:擎天柱的LED端口在P2,且没有三极管的电源控制,所以只要控制P2端口即可
P40 = 0; //拉低三极管基极,相当于打开总开关
//这里和冲哥的不同,主要是秦天柱的LED在P2
//试验箱的LED在P0,而且试验箱P40上还有个三极管
while(1)
{
if (bUsbOutReady)//如果接受到了数据
{
//USB_SendData(UsbOutBuffer,OutNumber); //发送数据缓冲区,长度(接收数据原样返回, 用于测试)
//printf("STC YYDS\r\n");
usb_OUT_done();//清除当前的接收区
}
//P00 = 0;//P00端口输出0V,共阳极LED点亮
//P02 = 0;//P02端口输出0V,共阳极LED点亮
//P04 = 0;//P04端口输出0V,共阳极LED点亮
//P06 = 0;//P06端口输出0V,共阳极LED点亮
//P07 = 0;//P07端口输出0V,共阳极LED点亮
//任务1:按下P32按钮灯亮,松开P32按钮灯灭;
// if( P32==0 ) //判断P32按钮是否按下
// {
// P00 = 0;//P00端口输出0V,共阳极LED点亮
// }
// else
// {
// P00 = 1;//P00端口输出3.3V,共阳极LED熄灭
// }
//*******************************************
//任务2:按下P32按钮灯灭,松开P32按钮灯亮;
// if( P32== 1 ) //判断P32按钮是否按下
// {
// P00 = 0;//P00端口输出0V,共阳极LED点亮
// }
// else
// {
// P00 = 1;//P00端口输出3.3V,共阳极LED熄灭
// }
//任务3:按一下灯亮,按一下灯灭;
//
//课后任务2:按一下亮一颗灯,在按一下亮两颗灯,直到全亮(变量+加法和乘法)
if( P32== 0 ) //判断P32按钮是否按下,0为按下
{
Delay20ms();//延时20毫秒消抖
if( P32== 0 ) //判断P32按钮是否确实按下
{
state = !state; //变里取反010101
//LED_rksj = _crol_(LED_rksj,1);//左移一位,循环移入每次往前点亮一位
LED_rksj = LED_rksj<<1; //左移一位,右边补0,每次往前多点亮一位
P0 = LED_rksj;//P00端口左移一位输出0V,共阳极LED点亮
num = num +1 ;
if(num>=8)
{
LED_rksj = 0XFF;//送人P0口数据
num = 0 ; //P32按键次数,
}
printf("state: num: %d %d\r\n", (int)state,num);
while(P32 == 0); //等待P32松开
}
}
if( P33== 0 ) //判断P33按钮是否按下,0为按下
{
Delay20ms();//延时20毫秒消抖
if( P33== 0 ) //判断P33按钮是否确实按下
{
state = !state; //变里取反010101
P0 = 0XFF;//P00端口输出3.3V,8位共阳极LED熄灭
printf("state: %d\r\n", (int)state);
while(P33 == 0); //等待P33松开
}
}
}
}
wuzhengmin
发表于 2025-8-14 10:29:52
这样就完成了冲哥哥布置的作业!
接下来,我们继续学习第七集 定时器中断
P40 = 0;
while(DeviceState!= DEVSTATE_CONFIGURED);//等待USB完成配置
while(1)
{
if(bUsboutReady) //如果接收到了数据
{
//USB_SendData(UsbOutBuffer,OutNumber):
//发送数据缓冲区,长度(接收数据原样返回,用于测试
usb_OUT_done();
}
Delay3000ms():
state=!state://变量取反 0101 0 1
P00= state;
//任务3:按一下灯亮,按一下灯灭
//判断P32按钮是否按下
if(P32 == 0) //判断P32按钮是否按下
{
Delay20ms()://延时20ms消抖
if(P32 == 0)
{
while(P32 == 0) //等待P32松开
printf("按键按下:%d\r\n”,count++)
{
}
这样单核心单任务,就无法相应COM的打印任务,CUP只是在nop(),干等待时间流逝,没人干活啦
wuzhengmin
发表于 2025-8-14 10:55:42
根本原因是停在这里,Delay3000ms ():
什么都不能干了,我们要用定时器解决这个问题!
先把最终程序写下来:
#include "ai8051u.h" //调用头文件
#include "stc32_stc8_usb.h" //调用头文件
#include "intrins.h" //d调用头文件
#define u8unsigned char //8位无符号变量(0-255)
#define u16 unsigned int //16位无符号变量(0-65535)
u8 state = 0; //初始状态
u8 Run_State = 0; //运行状态
char *USER_DEVICEDESC = NULL;
char *USER_PRODUCTDESC = NULL;
char *USER_STCISPCMD = "@STCISP#";
void Delay20ms(void) //@24.000MHzDelay20ms();
{
unsigned long edata i;
_nop_();
_nop_();
i = 119998UL;
while (i) i--;
}
void Timer0_Init(void); //3秒@24.000MHz //函数声明
void main(void)
{
int count=1; //按键计数变量
WTST = 0; //设置程序指令延时参数,赋值为0可将CPU执行指令的速度设置为最快
EAXFR = 1; //扩展寄存器(XFR)访问使能
CKCON = 0; //提高访问XRAM速度
P0M1 = 0x00; P0M0 = 0x00;
P1M1 = 0x00; P1M0 = 0x00;
P2M1 = 0x00; P2M0 = 0x00;
P3M1 = 0x00; P3M0 = 0x00;
P4M1 = 0x00; P4M0 = 0x00;
P5M1 = 0x00; P5M0 = 0x00;
P6M1 = 0x00; P6M0 = 0x00;
P7M1 = 0x00; P7M0 = 0x00;
usb_init(); //USB CDC 接口配置
IE2 |= 0x80; //使能USB中断
// Timer0_Init(); //定时器初始化
EA = 1; //IE |= 0X80;
P40 = 0;
while (DeviceState != DEVSTATE_CONFIGURED); //等待USB完成配置
while(1)
{
if (bUsbOutReady) //如果接收到了数据
{
//USB_SendData(UsbOutBuffer,OutNumber); //发送数据缓冲区,长度(接收数据原样返回, 用于测试)
usb_OUT_done(); //
}
// //任务1:
// if( P32 == 0 ) //判断P32按钮是否按下
// {
// Delay20ms(); //延时20ms消抖
// if( P32 == 0 )
// {
// printf("按键按下次数\xfd:%d 次\r\n",(int)count);
// count++;
//
// while( P32 == 0 ); //等待P32松开
//
// }
// }
// //任务2:灯按一下点亮三秒后熄灭。
// if( P32 == 0 ) //判断P32按钮是否按下
// {
// Delay20ms(); //延时20ms消抖
// if( P32 == 0 )
// {
//// printf("按键按下次数\xfd:%d 次\r\n",(int)count);
//// count++;
// P00 = 0;
// Timer0_Init();
// while( P32 == 0 ); //等待P32松开
//
// }
// }
//
//任务3:救护车灯控制器,按下报警按钮,红蓝交替闪烁(LED1和LED2 表示红和蓝灯),再按一下报警按钮,红蓝灯停止。
if( P32 == 0 ) //判断P32按钮是否按下
{
Delay20ms(); //延时20ms消抖
if( P32 == 0 )
{
Run_State = !Run_State; //运行状态取反
//printf("按键按下次数\xfd:%d 次\r\n",(int)count);
printf("按键按下次数\xfd:%d 次状态标志\%d \r\n",(int)count,Run_State);
count++;
if( Run_State==1 ) //运行
{
Timer0_Init();
}
else
{
TR0 = 0; //关闭定时器
P00 = 1;
P01 = 1;
}
// P00 = 0;
// Timer0_Init();
while( P32 == 0 ); //等待P32松开
}
}
}
}
//void Timer0_Init(void) //3秒@24.000MHz 函数定义
//{
// TM0PS = 0x5B; //设置定时器时钟预分频 ( 注意:并非所有系列都有此寄存器,详情请查看数据手册 )
// AUXR &= 0x7F; //定时器时钟12T模式
// TMOD &= 0xF0; //设置定时器模式
// TL0 = 0x3F; //设置定时初始值
// TH0 = 0x01; //设置定时初始值
// TF0 = 0; //清除TF0标志
// TR0 = 1; //定时器0开始计时
// ET0 = 1; //使能定时器0中断
//
// //TM0PS = 91
// //12T /12
// // THO-TL0 = 319
//
//}
void Timer0_Init(void) //500毫秒@24.000MHz
{
TM0PS = 0x0F; //设置定时器时钟预分频 ( 注意:并非所有系列都有此寄存器,详情请查看数据手册 )
AUXR &= 0x7F; //定时器时钟12T模式
TMOD &= 0xF0; //设置定时器模式
TL0 = 0xDC; //设置定时初始值
TH0 = 0x0B; //设置定时初始值
TF0 = 0; //清除TF0标志
TR0 = 1; //定时器0开始计时
ET0 = 1; //使能定时器0中断
}
void Timer0_Isr(void) interrupt 1 //3秒执行一次
{
state = !state;
P00 = state;
P01 = !state;
}
wuzhengmin
发表于 2025-8-14 11:33:33
定时器的介绍
问题:.LED三秒闪烁一下,这三秒按下按键但是没有反应了?这怎么办呢?
答:因为MCU单核的,同一时间只能执行一个事情,没有特殊情况不能被打断。
所以这里就要引入一个特殊的情况,定时器中断。
定时器作用:(1)用于计时系统,可实现软件计时,或者使程序每隔一固定时间完成一项操作
(2)替代长时间的Delay;提高程序的运行效率和处理速度(可以打断主循环)
任务1:LED灯三秒取反一次,这期间任意时刻按下按钮,串口打印按键次数。
printf("按键按下次数\xfd:%d 次状态标志 %d \r\n",(int)count,(int)Run_State);
注意这里的\xfd 是解决打印乱码的问题...................
用STC的ISP软件可以自动生成定时器和对应的中断函数,很方便的,认真跟着冲哥走,不要跳步,凡是觉得太简单的时候,就是快出问题的时候啦
wuzhengmin
发表于 2025-8-14 11:39:30
定时器中断函数自动的每隔3秒执行一次
void Timer0_Isr(void)interrupt 1//3秒执行一次
{
state =!state;
P00 = state;
}
刚才讲的打印乱码的问题:
printf("按键按下次数\xfd:%d 次状态标志 %d \r\n",(int)count,(int)Run_State);
好好看看手册,我截图如下:
wuzhengmin
发表于 2025-8-14 11:41:56
要注意的汉字如下:
页:
1
2
3
4
5
6
7
8
9
[10]
11
12
13
14
15
16
17