#include <reg52.h> #include <stdio.h> #define LCD1602_RS P2_0 // LCD1602?RS?? #define LCD1602_RW P2_1 // LCD1602?RW?? #define LCD1602_EN P2_2 // LCD1602?EN?? #define LCD1602_DATAPINS P0 // LCD1602????? sbit UART_RXD = P3^0; // ?????? sbit UART_TXD = P3^1; // ?????? void init_uart() // ????? { TMOD |= 0x20; // ?????1???2 TH1 = 0xfd; // ??????9600 TL1 = 0xfd; TR1 = 1; // ?????1 SCON = 0x50; // ???????1 ES = 1; // ?????? EA = 1; // ????? } void init_lcd() // ???LCD { LCD1602_RS = 0; LCD1602_RW = 0; LCD1602_EN = 0; delay_ms(15); lcd_write_cmd(0x38); // ??LCD?16x2????? delay_ms(5); lcd_write_cmd(0x0c); // ??LCD?? delay_ms(5); lcd_clear(); // ?? lcd_write_cmd(0x06); // ???????? } void lcd_write_cmd(unsigned char cmd) // ????LCD { LCD1602_RS = 0; LCD1602_DATAPINS = cmd; LCD1602_EN = 1; delay_us(2); LCD1602_EN = 0; delay_ms(1); } void lcd_write_data(unsigned char dat) // ????LCD { LCD1602_RS = 1; LCD1602_DATAPINS = dat; LCD1602_EN = 1; delay_us(2); LCD1602_EN = 0; delay_ms(1); } void lcd_clear() // ?? { lcd_write_cmd(0x01); } void lcd_set_cursor(unsigned char x, unsigned char y) // ?????? { unsigned char addr; if (y == 0) addr = 0x80 + x; else addr = 0xc0 + x; lcd_write_cmd(addr); } void lcd_puts(unsigned char x, unsigned char y, unsigned char *str) // ?????????? { lcd_set_cursor(x, y); while (*str != '\0') { lcd_write_data(*str); str++; } } void uart_isr() interrupt 4 // ???????? { if (RI) { RI = 0; lcd_write_data(SBUF); // ?????????LCD? } } void main() { init_uart(); init_lcd(); while (1); }
时间: 2023-07-24 08:16:46 浏览: 230
这段代码实现了一个基于单片机的串口通信并将接收到的数据显示在LCD1602液晶屏上。其中,P2_1和P2_2是定义了LCD1602的控制引脚,P0是定义了LCD1602数据引脚。同时,还定义了一个uart_isr()函数,用于处理串口接收中断。在main()函数中,通过调用init_uart()和init_lcd()函数初始化串口和LCD1602,然后进入一个死循环中。
需要注意的是,如果你在编译这段代码时出现了P2_1和P2_2未定义的错误提示,可能是因为你没有正确定义这些引脚或没有包含相应的头文件。你需要检查一下你的代码,确定是否有定义这些引脚,或者添加对应的头文件。
相关问题
#include <reg52.h> #include <intrins.h> #include <stdio.h> #include "lcd1602.h" #define uchar unsigned char #define uint unsigned int uint wendu; //温度数据 uchar HT=40; //温度上限 uchar LT=15; //温度下限 uchar WenduData[6]; //实时温度 uchar HTem[3],LTem[3];//温度上下限 sbit p33 = P3^3; // RS485使能控制线 #define RS485_FA p33=1;//发送 #define RS485_SO p33=0;//接收 sbit BUTTON = P2^5; // 按钮 /* uchar ReceiveData(void) { uchar dat; if (RI) //判断是否有数据接收 { dat = SBUF; //读取数据 RI = 0; //清除接收标志位 return dat; } else return 0; } */ void Disp_Temperature() { LTem[0] = LT/10+'0'; LTem[1] = LT%10+'0'; LTem[2] = '\0'; HTem[0] = HT/10+'0'; HTem[1] = HT%10+'0'; HTem[2] = '\0'; gotoxy(1,1); LCD_display("LT:"); gotoxy(4,1); LCD_display(LTem); gotoxy(1,2); LCD_display("HT:"); gotoxy(4,2); LCD_display(HTem); gotoxy(8,1); } void main() { uchar buf[16]; uint dat; TMOD = 0x20; //定时器0工作模式2,自动重装8位计数器 TH1 = 0xfd; TL1 = 0xfd;//定时器溢出时,会自动将高8位中的值赋值给低8位.比特率9600 TR1 = 1; SM0 = 0; SM1 = 1; REN = 1; EA = 1; ES = 1; //打开总中断 RS485_SO; LCD_init(); while(1) { Disp_Temperature(); delay_lcd(1000); /* dat=ReceiveData(); */ dat=wendu; sprintf(buf,"T=%d",dat); LCD_display(buf); gotoxy(10,2); } } void UART_interrupt() interrupt 4 { uint str; RI=0; if(RI=0) { str=SBUF; RI=0; wendu=str; } }
根据代码分析,这是一个带温度传感器的系统,通过串口与其他设备通信获取温度数据,并在LCD1602液晶屏上显示实时温度和温度上下限。
具体实现:
1. 定义了温度上下限`HT`和`LT`,并使用`LTem`和`HTem`数组存储字符串类型的温度上下限值,用于在液晶屏上显示。
2. 使用串口通信获取温度数据,并将数据存储在`wendu`变量中。其中,串口通信的波特率为9600,使用中断方式接收数据。
3. 使用`sprintf`函数将`wendu`变量格式化为字符串,存储在`buf`数组中。
4. 调用`LCD_display`函数将温度字符串显示在液晶屏上。同时,使用`Disp_Temperature`函数将温度上下限值显示在液晶屏上。
5. 使用`delay_lcd`函数控制温度显示的更新频率。
6. 程序主要的循环在`while(1)`中,不断更新温度值并显示在液晶屏上。
总的来说,这个代码是一个简单的嵌入式系统示例,通过串口通信获取传感器数据,并在液晶屏上显示数据。
请帮我优化这段代码include <reg52.h> #include <stdio.h> #include <string.h> #define LCD_DATA P0 #define LCD_RS P2_0 #define LCD_RW P2_1 #define LCD_EN P2_2 #define LED_PIN P1_0 #define BUZZER_PIN P1_1 void delay(unsigned int ms); void LCD_init(); void LCD_command(unsigned char cmd); void LCD_data(unsigned char dat); void LCD_string(char *str); void LCD_clear(); void UART_init(); void UART_sendChar(unsigned char ch); void UART_sendString(char *str); unsigned char UART_receiveChar(); void executeCommand(char *command); void main() { char command[20]; UART_init(); LCD_init(); while (1) { if (UART_receiveChar() == ':') { UART_receiveChar(); // Ignore space after ':' fgets(command, sizeof(command), stdin); executeCommand(command); UART_sendString(command); // Send back the received command } } } void delay(unsigned int ms) { unsigned int i, j; for (i = 0; i < ms; i++) for (j = 0; j < 110; j++); } void LCD_init() { LCD_command(0x38); // 2 lines, 5x7 matrix LCD_command(0x0C); // Display on, cursor off LCD_command(0x06); // Increment cursor LCD_command(0x01); // Clear display delay(2); } void LCD_command(unsigned char cmd) { LCD_RS = 0; LCD_RW = 0; LCD_EN = 1; LCD_DATA = cmd; delay(2); LCD_EN = 0; } void LCD_data(unsigned char dat) { LCD_RS = 1; LCD_RW = 0; LCD_EN = 1; LCD_DATA = dat; delay(2); LCD_EN = 0; } void LCD_string(char *str) { while (*str) { LCD_data(*str++); } } void LCD_clear() { LCD_command(0x01); // Clear display delay(2); } void UART_init() { TMOD = 0x20; // Timer1 mode 2: 8-bit auto-reload TH1 = 0xFD; // 9600 baud rate SCON = 0x50; // Serial mode 1: 8-bit data, 1 stop bit, receive enabled TR1 = 1; // Start Timer1 } void UART_sendChar(unsigned char ch) { SBUF = ch; while (TI == 0); // Wait for transmission to complete TI = 0; // Clear transmission flag } void UART_sendString(char *str) { while (*str) { UART_sendChar(*str++); } } unsigned char UART_receiveChar() { while (RI == 0); // Wait for reception to complete RI = 0; // Clear reception flag return SBUF; } void executeCommand(char *command) { if (strncmp(command, "LED on", 6) == 0) { LED_PIN = 1; } else if (strncmp(command, "buzzer on", 9) == 0) { BUZZER_PIN = 1; } else if (strncmp(command, "showstr", 7) == 0) { char *str = command + 8; // Get the string after "showstr" LCD_clear(); LCD_command(0x80); // Move cursor to the beginning of the first line LCD_string(str); } }
这段代码可以进行如下的优化:
1. 将LCD的RS、RW、EN引脚的控制函数封装成一个公共函数,避免代码重复。
2. 可以使用宏定义或枚举类型来定义LED_PIN和BUZZER_PIN,提高代码的可读性。
3. 在executeCommand函数中,可以使用switch-case语句来代替if-else语句,提高代码的可读性。
4. 在UART_receiveChar函数中,可以使用do-while循环来代替while循环,提高代码的可读性。
5. 在executeCommand函数中,可以使用strtok函数来解析命令字符串,提高代码的可读性。
6. 在executeCommand函数中,可以使用strcmp函数来比较字符串,避免使用strncmp函数时出现错误。
7. 在执行LED_PIN = 1或BUZZER_PIN = 1之前,可以先检查这些引脚是否已经被初始化为输出模式,避免出现意外错误。
8. 在executeCommand函数中,可以添加一些错误处理代码,以防止出现错误的命令。
9. 可以添加注释,提高代码的可读性。
下面是优化后的代码:
阅读全文