標題: 單片機串口程序調(diào)試問題,請大神們幫忙 按鍵讓他發(fā)一條,它發(fā)9條 [打印本頁]

作者: 稻草人008    時間: 2022-1-5 16:56
標題: 單片機串口程序調(diào)試問題,請大神們幫忙 按鍵讓他發(fā)一條,它發(fā)9條
#include <reg51.h>
#include "string.h"

#define uchar unsigned char
#define uint unsigned  int


sbit S1=P1^0;
sbit S2=P1^1;
sbit S3=P1^2;
sbit S4=P1^3;
void uart_init(void)
{
     TMOD = 0x20;
    SCON = 0x50;
    TH1 = 0xFA;
    TL1 = TH1;
    PCON = 0x00;
    EA = 1;
    ES = 1;
    TR1 = 1;

}
//

void uart_tx_byte(uchar dat)
{
   SBUF=dat;
   while(!TI);
   TI=0;



}

void uart_tx_string(uchar *P)
{
    while(*P)
           {
             uart_tx_byte(*P);
         P++;
           }
  }
void main()
{
   uart_init();


   while(1)
   {
      if(S1==0)    //按鍵S1按下
          {
            
                 uart_tx_string("EF 09 73 00 01 02 96 \n\r");
        
        
        
               
          }
          if(S2==0)   //按鍵S2按下
          {
            
                 uart_tx_string("EF 09 73 00 02 01 96 \n\r");

                        

          }
          if(S3==0)   //按鍵S3按下
          {
            
                 uart_tx_string("EF 09 66 03 00 00 83 \n\r");

                 
          }
          if (S4==0)
             uart_tx_string("EF 09 73 01 00 00 94 \n\r");
               
               
                 
          {
               
          }
            
   }

}

[attach]288528[/attach]
給位大神幫我看看哪里錯了,每按一下按鍵讓他發(fā)一條,它發(fā)9條。初學者,別笑話。

0.png (53.06 KB, 下載次數(shù): 152)

0.png

作者: lkc8210    時間: 2022-1-5 17:45
加個消抖延時就可以了
  1. void main()
  2. {
  3.         uart_init();

  4.         while(1)
  5.         {static unsigned char delay;
  6.                 if(S1==0 || S2==0 || S3==0 || S4==0)//有鍵按下
  7.                 {
  8.                         if(delay<0xFF)delay++;//消抖延時
  9.                         if(delay == 250)
  10.                         {
  11.                                 if(S1==0)    //按鍵S1按下
  12.                                 {
  13.                                         uart_tx_string("EF 09 73 00 01 02 96 \n\r");
  14.                                 }
  15.                                 if(S2==0)   //按鍵S2按下
  16.                                 {
  17.                                         uart_tx_string("EF 09 73 00 02 01 96 \n\r");
  18.                                 }
  19.                                 if(S3==0)   //按鍵S3按下
  20.                                 {
  21.                                         uart_tx_string("EF 09 66 03 00 00 83 \n\r");
  22.                                 }
  23.                                 if (S4==0)
  24.                                 {
  25.                                         uart_tx_string("EF 09 73 01 00 00 94 \n\r");
  26.                                 }                               
  27.                         }
  28.                 }else{
  29.                         delay = 0;
  30.                 }
  31.         }
  32. }
復制代碼





作者: 天ノ憶    時間: 2022-1-5 17:51
發(fā)9條多嗎,你這程序不發(fā)幾十條算少的了
你理順一下程序邏輯,看看你按下去按鍵的時候if判斷走了多少次
作者: 稻草人008    時間: 2022-1-5 17:55
天ノ憶 發(fā)表于 2022-1-5 17:51
發(fā)9條多嗎,你這程序不發(fā)幾十條算少的了
你理順一下程序邏輯,看看你按下去按鍵的時候if判斷走了多少次

謝了,跟盆友學寫了個蹩腳程序,望多指教
作者: 稻草人008    時間: 2022-1-5 17:55
lkc8210 發(fā)表于 2022-1-5 17:45
加個消抖延時就可以了

十分感謝,我試試
作者: 稻草人008    時間: 2022-1-5 18:03
lkc8210 發(fā)表于 2022-1-5 17:45
加個消抖延時就可以了

十分感謝,用你的程序好了。
作者: 稻草人008    時間: 2022-1-6 14:30
稻草人008 發(fā)表于 2022-1-5 18:03
十分感謝,用你的程序好了。

大神你好現(xiàn)在又遇到新問題了,我在仿真上數(shù)據(jù)正常,燒到單片機里通過485轉出來在串口助手上顯示的數(shù)據(jù)不對。我改了下頭文件,stc12c5410ad的

IMG_20220106_142618.jpg (1.28 MB, 下載次數(shù): 131)

IMG_20220106_142618.jpg

作者: lkc8210    時間: 2022-1-6 14:45
如果你現(xiàn)實用的是stc12c5410ad
要把定時器設為12分頻
void UartInit(void)                //9600bps@22.1184MHz
{
        PCON &= 0x7F;                //波特率不倍速
        SCON = 0x50;                //8位數(shù)據(jù),可變波特率
        AUXR &= 0xBF;                //定時器時鐘12T模式
        AUXR &= 0xFE;                //串口1選擇定時器1為波特率發(fā)生器
        TMOD &= 0x0F;                //設置定時器模式
        TMOD |= 0x20;                //設置定時器模式
        TL1 = 0xFA;                //設置定時初始值
        TH1 = 0xFA;                //設置定時重載值
        ET1 = 0;                //禁止定時器%d中斷
        TR1 = 1;                //定時器1開始計時
        ES = 1;
        EA = 1;
}

作者: boboxuexi    時間: 2022-1-6 15:36
阻塞試延時是單片機程序設計最大的忌諱,
作者: 稻草人008    時間: 2022-1-6 18:31
boboxuexi 發(fā)表于 2022-1-6 15:36
阻塞試延時是單片機程序設計最大的忌諱,

十分感謝,等忙完試試。
作者: 13205495918    時間: 2022-1-6 19:45
正點原子論壇有個按鍵掃描程序,里面有個 靜態(tài)變量的用法,你可以了解一下,有個對你有幫助。
作者: 稻草人008    時間: 2022-1-6 22:00
13205495918 發(fā)表于 2022-1-6 19:45
正點原子論壇有個按鍵掃描程序,里面有個 靜態(tài)變量的用法,你可以了解一下,有個對你有幫助。

好的謝了,我去看看
作者: 稻草人008    時間: 2022-1-7 15:08
lkc8210 發(fā)表于 2022-1-6 14:45
如果你現(xiàn)實用的是stc12c5410ad
要把定時器設為12分頻
void UartInit(void)                //9600bps@22.1184MHz

高手,你好,現(xiàn)在都調(diào)出來了,
但是我發(fā)的EF 09 73 00 01 02 96這字符在串口上讀出來的是ASCII正確,16進制45 46 20 30 39 20 37 33 20 30 30 20 30 32 20 30 31 20 39 36 20 0A 0D 。但是我要的是16位進制的。是不是需要轉碼
作者: lkc8210    時間: 2022-1-7 15:21
稻草人008 發(fā)表于 2022-1-7 15:08
高手,你好,現(xiàn)在都調(diào)出來了,
但是我發(fā)的EF 09 73 00 01 02 96這字符在串口上讀出來的是ASCII正確,16 ...

如果你要的是0xEF 0x09 0x73...就要轉碼
作者: 稻草人008    時間: 2022-1-7 18:35
lkc8210 發(fā)表于 2022-1-7 15:21
如果你要的是0xEF 0x09 0x73...就要轉碼

我要的是16進制下的EF 09 73 00 01 02 96
作者: lkc8210    時間: 2022-1-7 18:40
稻草人008 發(fā)表于 2022-1-7 18:35
我要的是16進制下的EF 09 73 00 01 02 96

就是45 46 20 30 39 20 37 33 20 30 30 20 30 32 20 30 31 20 39 36 20 0A 0D嗎?
那就不用轉
作者: 稻草人008    時間: 2022-1-7 19:02
lkc8210 發(fā)表于 2022-1-7 18:40
就是45 46 20 30 39 20 37 33 20 30 30 20 30 32 20 30 31 20 39 36 20 0A 0D嗎?
那就不用轉

是不是我發(fā)的被我指定為文本了,然后它在轉了一遍
作者: 稻草人008    時間: 2022-1-7 19:07
稻草人008 發(fā)表于 2022-1-7 19:02
是不是我發(fā)的被我指定為文本了,然后它在轉了一遍

"EF 09 73 00 01 02 96這串數(shù)就是16進制的,在ascll上顯示正常,在16進制下就被轉了一下。是不是這樣。
作者: 稻草人008    時間: 2022-1-7 19:16
lkc8210 發(fā)表于 2022-1-7 18:40
就是45 46 20 30 39 20 37 33 20 30 30 20 30 32 20 30 31 20 39 36 20 0A 0D嗎?
那就不用轉

EF 09 73 00 01 02 96這串數(shù)就是16進制的,在ascll上顯示正常,在16進制下就被轉了一下。是不是這樣。
作者: suncat0504    時間: 2022-1-7 19:49
對于接收方,它不知道來的是什么數(shù)據(jù),是ASCII還是十六進制數(shù)據(jù),它都統(tǒng)一按照16進制收取。所以發(fā)送方即使發(fā)的是ASCII碼,接收方會按照16進制收取ASCII碼對應的16進制數(shù)據(jù)。實際上,無論收發(fā),走在線路上的數(shù)據(jù)都是二進制數(shù)據(jù)0和1的組合而已。所以才有協(xié)議一說,接收方按照發(fā)射方的意圖去解析數(shù)據(jù)。
作者: 稻草人008    時間: 2022-1-7 20:03
suncat0504 發(fā)表于 2022-1-7 19:49
對于接收方,它不知道來的是什么數(shù)據(jù),是ASCII還是十六進制數(shù)據(jù),它都統(tǒng)一按照16進制收取。所以發(fā)送方即使 ...

但是我要的數(shù)據(jù)它在ASCII里傳送正常,在16進制里就被編了一次,設備不認
作者: 稻草人008    時間: 2022-1-8 01:05
lkc8210 發(fā)表于 2022-1-6 14:45
如果你現(xiàn)實用的是stc12c5410ad
要把定時器設為12分頻
void UartInit(void)                //9600bps@22.1184MHz

大神你好,還是要請你幫忙。EF 09 73 00 01 02 96這串數(shù)就是16進制的,在ascll上顯示正常,在16進制下就被從編了一遍。是不是這樣。要怎么改。
作者: lkc8210    時間: 2022-1-8 09:15
稻草人008 發(fā)表于 2022-1-7 19:02
是不是我發(fā)的被我指定為文本了,然后它在轉了一遍

對~
所以你要以16進數(shù)組來發(fā)
char table_s1[] = {0xEF, 0x09, 0x73, 0x00, 0x02, 0x01, 0x96};
char table_s2[] = {0xEF, ...};
char table_s3[] = {0xEF, ...};
char table_s4[] = {0xEF, ...};

                                if(S1==0)    //按鍵S1按下
                                {
                                        uart_tx_string(table_s1);
                                }
                                if(S2==0)    //按鍵S2按下
                                {
                                        uart_tx_string(table_s2);
                                }
作者: 稻草人008    時間: 2022-1-8 10:42
lkc8210 發(fā)表于 2022-1-8 09:15
對~
所以你要以16進數(shù)組來發(fā)
char table_s1[] = {0xEF, 0x09, 0x73, 0x00, 0x02, 0x01, 0x96};

謝謝,我是搞硬件的,對軟件只知道點皮毛。。再次謝謝你的指點。
作者: suncat0504    時間: 2022-1-8 11:09
稻草人008 發(fā)表于 2022-1-7 20:03
但是我要的數(shù)據(jù)它在ASCII里傳送正常,在16進制里就被編了一次,設備不認

樓主,我覺得你應該抽點時間,看看串口通訊在芯片、線路上是怎么實現(xiàn)的。這個是基礎,學明白了,如何解決問題,你會有重新認識。用來實現(xiàn)數(shù)據(jù)傳送的,是芯片中的寄存器,你把數(shù)據(jù)送進去,通過線路送到接收方的寄存器。收發(fā)正常的場合,兩邊寄存器里的內(nèi)容是一致的。當你把可顯示、打印的ASCII和不可顯示、打印的16禁止數(shù)據(jù)混合傳輸?shù)臅r候,比如以下語句(假設printf做個輸出轉向,轉向給串口輸出,假設comout是輸出16進制數(shù)據(jù)給串口):
printf("12345"); // 注:這五個字符,對應的6進制數(shù)據(jù)為:0x31,0x32,0x33,0x34,0x35
unsigned char dat[]={0xF1,0xC1,0x00,0x23};
comout(dat,0,3);  // 像串口發(fā)送dat數(shù)組中從0號元素開始,到3號元素截止的數(shù)據(jù),包含0號元素和3號元素
那么經(jīng)過這些發(fā)送,接收方收到的數(shù)據(jù),以16進制表達:0x31,0x32,0x33,0x34,0x35,0xF1,0xC1,0x00,0x23
當你使用APP接收這些數(shù)據(jù),如果以文本形式表示出來的時候,因為前五個對應的是可顯示打印的數(shù)據(jù),所以顯示“12345”,但后五個是沒法顯示的。
上面的發(fā)送,是可以用以下形式,一樣實現(xiàn):
unsigned char dat[]={0x31,0x32,0x33,0x34,0x35,0xF1,0xC1,0x00,0x23};
comout(dat, 0, 8);
作者: 稻草人008    時間: 2022-1-8 11:30
suncat0504 發(fā)表于 2022-1-8 11:09
樓主,我覺得你應該抽點時間,看看串口通訊在芯片、線路上是怎么實現(xiàn)的。這個是基礎,學明白了,如何解決 ...

謝謝指點,要好好學習學習一下
作者: 稻草人008    時間: 2022-1-8 13:22
lkc8210 發(fā)表于 2022-1-7 15:21
如果你要的是0xEF 0x09 0x73...就要轉碼

//發(fā)送字符串到串口助手
#include <STC12C5410AD.h>
#include "string.h"

#define uchar unsigned char
#define uint unsigned  int



sbit S1=P1^3;
sbit S2=P1^2;
sbit S3=P1^1;
sbit S4=P1^0;


char table_s1[] = {0xEF, 0x09, 0x73, 0x00, 0x01, 0x02, 0x96};
char table_s2[] = {0xEF, 0x09, 0x73, 0x00, 0x02, 0x01, 0x96};
char table_s3[] = {0xEF, 0x09, 0x66, 0x03, 0x00, 0x00, 0x83};
char table_s4[] = {0xEF, 0x09, 0x73, 0x01, 0x00, 0x00, 0x94};


//初始化
void uart_init(void)               //9600bps@22.1184MHz
{
        PCON &= 0x7F;                //波特率不倍速
        SCON = 0x50;                //8位數(shù)據(jù),可變波特率
        AUXR &= 0xBF;                //定時器時鐘12T模式
        AUXR &= 0xFE;                //串口1選擇定時器1為波特率發(fā)生器
        TMOD &= 0x0F;                //設置定時器模式
        TMOD |= 0x20;                //設置定時器模式
        TL1 = 0xFA;                //設置定時初始值
        TH1 = 0xFA;                //設置定時重載值
        ET1 = 0;                //禁止定時器%d中斷
        TR1 = 1;                //定時器1開始計時
        ES = 1;
        EA = 1;
   
}
//

void uart_tx_byte(uchar dat)
{
   SBUF=dat;
   while(!TI);
   TI=0;



}

void uart_tx_string(uchar *P)
{
    while(*P)
           {
             uart_tx_byte(*P);
         P++;
           }
  }
void main()
     
{
        uart_init();

        while(1)

   
        {static unsigned char delay;
                if(S1==0 || S2==0 || S3==0 || S4==0)//有鍵按下
              {
                        if(delay<0xFF)delay++;//消抖延時
                        if(delay == 250)
                        {
                                if(S1==0)    //按鍵S1按下
                                {
                                        uart_tx_string(table_s1);
                                }
                                if(S2==0)   //按鍵S2按下
                                {
                                        uart_tx_string(table_s2);
                                }
                                if(S3==0)   //按鍵S3按下
                                {
                                        uart_tx_string(table_s3);
                                }
                                if (S4==0)           //按鍵S4按下
                                {
                                        uart_tx_string(table_s4);
                                }                              
                        }
                }else{
                 
       delay = 0;
                }
        }
}不全
高手,在幫我看看。怎么數(shù)據(jù)發(fā)不全
發(fā)送1:EF 09 73 00 01 02 96
接收1:EF 09 73
發(fā)送2:EF 09 66 03 00 00 83
接收2:EF 09 66 03







歡迎光臨 (http://www.torrancerestoration.com/bbs/) Powered by Discuz! X3.1