① 單片機無線通信的校驗和方法怎麼用
這是通信中都必須做的一件事,是一種協議。說簡單點就是我把一包數據發給你,還多幾個位元組的校驗值。sum,CRC,這些都是演算法,演算法是收發兩端事先約定好的。所以你算的結果如果和剛收到的校驗值一樣,就說明那個包全部收對。並告訴對面接著發下一個包包,如果不一致,那就告訴對面,剛才的包重發,如果重發的次數超過你們設定的最大值,那就宣告連接失敗。皆如此類...
② 用單片機控制YX5200語音晶元,出現一段校驗和程序,讀不懂什麼意思,哪位大神幫我解讀一下,萬分感謝!!
這個是為了驗證數據是不是正確做的校驗運算。先把前面的len個位元組變成16進制加在一起,結果是一個16bit的數字,然後將這個16bit的數變成負的,(0-x)得到的就是x的負數,最後將這個16bit的數高8bit放到len為止,而第8bit放到len+1的位置。
③ c語言單片機裡面怎麼替換字元串
可以用sscanf和sprintf配合使用。
首先 定義一個字元串 緩存發信機器哪液咐編號。
char no[3];
sscanf(str_get, "set %s ", no);
執行這李純句之後 no中就是"02"
然埋掘後
char str_rec_buf[20];
這個用來生成命令
sprintf(str_rec_buf, "rec %s %d", no, str_len);
這樣str_rec_buf就是你需要的了。
不過 str_len不是你說的3,而是strlen(str_get) = 9
如果要計算678的長度,那麼同樣需要把這個取出來。
④ proteus模擬電路圖,51單片機按鍵計數
按照你題目,用了2個2位顯示,實際有4位合一起的。
k3:切換計數模式/預置模式。
計數模式:LED顯示計時數字,從0開始計時,直到預置最大值。
預置模式:LED顯示當前預置最大值,按k1,k2可對預置值+-操作,長按k1,k2大約2秒,會進入自動加減預置值。直到再次點擊k1,k2,k3任意一鍵停止自動。
k4:在計數模式下使用,每按下一次顯示的數字加一(會在正常計時同時額外+1)。
當計數達到預置最大值,會停止計數,LEN閃爍(實際就是交替顯示間隔邊長),蜂鳴器響。
按鍵時長、LED動態顯示間隔、閃爍間隔、計數速度,均可直接修改常量,需要自己改,我備注寫的很詳細。
電路基本按照你上圖,略有修改。
#include <reg52.h>
#define uint unsigned int
#define uchar unsigned char
#define an P0
#define on 0
#define off 1
#define SSSPEED 35 //LED交替閃爍間隔時間
#define JSPEED 5000//計數模式,速度默認數值(5000*200us=1S) 值越小計數越快
#define PREESTIME 500//按鈕長按時間判定,預設500(大約2秒),需要自改,值越大,長按時間越長
sbit fm=P3^3;
sbit wei1=P3^4;
sbit wei2=P3^5;
sbit wei3=P3^6;
sbit wei4=P3^7;
sbit k1=P1^4;
sbit k2=P1^5;
sbit k3=P1^6;
sbit k4=P1^7;
uint jsSpd=JSPEED;//計時速度,默認1s一次(5000*200us)
uint ssSpd=SSSPEED;//LED交替閃爍速度
//共陽極
int delay(uint xms);
void init();
void jspause();//計數器開啟/停止
void setnumYS();//設置預設數值
void numJsChange();//計數模式數字改變
void showLED();
int pressWait(uint btn);
uint g=0;
uint s=0;
uint b=0;
uint q=0;
uint count=0;
uint ispause=1;
uint numYS=0;//預設數值
uint numJS=0;//實際計時的數字
uint isMaxJs=0;//標識:計時達最大。 達最大1,否0
uint isk3press=0;//標識:k3按鈕是否被點擊。 點擊1,否0
uint ispress1=0;//標識:k1被長按
uint ispress2=0;//標識:k2被長按
uint isbtn4=0;//標識:k4被按下
uint btnName=0;//按鈕長按計時
void main()
{
init();
while(1)
{
if(ispause==1 && ispress1==1 && numYS<9999) //預置模式下,k1已長按,自動增
{
numYS++;
setnumYS();
}
if(ispause==1 && ispress2==1 && numYS>0) //預置模式下,k2已長按,自動減
{
numYS--;
setnumYS();
}
if(isMaxJs==0 && numJS>=numYS && ispause==0) //計時模式下達最大值
{
fm=on;
ssSpd=1000;//增加LED交替間隔,實現數字閃爍
isMaxJs=1;
EA=0;
setnumYS();
numJS=0;
}
if(k1==0 ||k2==0|| k3==0) //k1k2k3任意一個按鈕被按下,停止預置數自動增長
{
ispress1=0;
ispress2=0;
}
if(k1==0 && ispause==1)//預置模式下+
{
delay(10);
if(k1==0)
{
btnName=1;
if(pressWait(btnName))//判斷連按
{
while(k1==0);
ispress1=1;
}
else if(numYS<9999)
{
numYS++;
setnumYS();
}
}
}
if(k2==0 && ispause==1)//預置模式下-
{
delay(10);
if(k2==0)
{
btnName=2;
if(pressWait(btnName))//判斷連按
{
while(k2==0);
ispress2=1;
}
else if(numYS>0)
{
numYS--;
setnumYS();
}
}
}
if(k3==0)
{
delay(10);
if(k3==0)
{
while(k3==0);
fm=off;
jspause();
}
}
if(k4==0 && ispause==0)//計數模式下按下k4,k4的防抖寫在中斷中
{
delay(10);
if(k4==0)
{
while(k4==0);
isbtn4=1;
}
}
showLED();
}
}
void showLED()
{
uchar nums[10]={0xc0,0xf9,0xa4,0xB0,0x99,0x92,0x82,0xf8,0x80,0x98};
if(g>=0)
{
an=nums[g];
wei4=on;
delay(ssSpd);
wei4=off;
}
if(s>0 || (s==0 && b>0))
{
an=nums[s];
wei3=on;
delay(ssSpd);
wei3=off;
}
if(b>0 || (b==0 && q>0))
{
an=nums[b];
wei2=on;
delay(ssSpd);
wei2=off;
}
if(q>0)
{
an=nums[q];
wei1=on;
delay(ssSpd);
wei1=off;
}
}
void setnumYS()//設置預設數值
{
q=numYS/1000;
b=(numYS%1000)/100;
s=(numYS%100)/10;
g=numYS%10;
}
void jspause()
{
if(ispause==0 || isMaxJs==1)//關閉計時模式 / 啟動預置模式
{
EA=0;
isMaxJs=0;
ispause=1;
ssSpd=SSSPEED;
ispress1=0;
ispress2=0;
setnumYS();
}
else if(ispause==1) //啟動計時模式 / 關閉預置模式
{
ispause=0;
q=b=s=g=0;
numJS=0;
ssSpd=SSSPEED;
EA=1;
}
}
void init()
{
TMOD=0x02; //T0 工作模式2 自動裝填8位 200us
TH0=0x38;
TL0=0x38;
EA=0;
ET0=1;
TR0=1;
wei1=off;
wei2=off;
wei3=off;
wei4=off;
}
void numJsChange()//計數模式數字改變
{
if(g==9)
{
g=0;
if(s==9)
{
s=0;
if(b==9)
{
b=0;
if(q==9)
{
q=0;
}
else
q++;
}
else
b++;
}
else
s++;
}
else
g++;
}
void ct() interrupt 1 //一次中斷200us
{
if(count<jsSpd)
count++;
else
{
count=0;
numJsChange();
numJS++;
}
if(isbtn4==1)
{
isbtn4=0;
numJsChange();
numJS++;
}
}
int pressWait(uint btn)
{
uint i,j;
for(i=PREESTIME;i>0;i--)
for(j=110;j>0;j--)
{
if((k1==1 && btn==1)||(k2==1 && btn==2))
return 0;
}
return 1;
}
int delay(uint xms)
{
uint i,j;
for(i=xms;i>0;i--)
for(j=110;j>0;j--)
{
if(k1==0 || k2==0 ||k3==0)
return 1;
}
return 0;
}
⑤ 51單片機數據類型轉換 高分!
假如你的串口發送函數叫SendData,接收函數叫RecvData,參考如下代碼
發送端:
intSendData(unsignedchar*pSendBuf,unsignedintnSendLen);
intRecvData(unsignedchar*pRecvBuf,unsignedintnBufLen);
voidmain(void)
{
floatf;
inti;
doubled;
SendData((unsignedchar*)&f,sizeof(float));
SendData((unsignedchar*)&i,sizeof(int));
SendData((unsignedchar*)&d,sizeof(double));
}
接收端:
voidmain(void)
{
floatf;
inti;
doubled;
RecvData((unsignedchar*)&f,sizeof(float));
RecvData((unsignedchar*)&i,sizeof(int));
RecvData((unsignedchar*)&d,sizeof(double));
}
宗旨就是,先取其他類型數據的地址,用&符號,然後把地址轉換為unsigned char*型,就可以通過*(unsigned char)的方式或(unsigned char)[]的方式訪問了,就這么簡單,希望可以幫到你.