<meter id="pryje"><nav id="pryje"><delect id="pryje"></delect></nav></meter>
          <label id="pryje"></label>

          新聞中心

          EEPW首頁 > 嵌入式系統(tǒng) > 設(shè)計(jì)應(yīng)用 > 第61節(jié):組合和非組合BCD碼以及數(shù)值相互轉(zhuǎn)換

          第61節(jié):組合和非組合BCD碼以及數(shù)值相互轉(zhuǎn)換

          作者: 時(shí)間:2016-11-22 來源:網(wǎng)絡(luò) 收藏
          第六十一節(jié):組合BCD碼,非組合BCD碼,以及數(shù)值三者之間的相互轉(zhuǎn)換和關(guān)系。
          開場(chǎng)白:
          本來這一節(jié)打算講大數(shù)據(jù)的加法運(yùn)算的,但是考慮大數(shù)據(jù)運(yùn)算的基礎(chǔ)是非組合BCD碼,所以多增加一節(jié)講BCD碼的內(nèi)容。
          計(jì)算機(jī)中的BCD碼,經(jīng)常使用的有兩種格式,即組合BCD碼,非組合BCD碼。
          組合BCD碼,是將兩位十進(jìn)制數(shù),存放在一個(gè)字節(jié)中,例如:十進(jìn)制數(shù)51的存放格式是0101 0001。
          非組合BCD碼,是將一個(gè)字節(jié)的低四位編碼表示十進(jìn)制數(shù)的一位,而高4位都為0。例如:十進(jìn)制數(shù)51的占用了兩個(gè)字節(jié)的空間,存放格式為:00000101 00000001。
          這一節(jié)要教大家兩個(gè)知識(shí)點(diǎn):
          第一個(gè):如何編寫組合BCD碼,非組合BCD碼,以及數(shù)值三者之間的相互轉(zhuǎn)換函數(shù)。
          第二個(gè):通過轉(zhuǎn)換函數(shù)的編寫,重溫前面幾節(jié)所講到的指針用法。

          具體內(nèi)容,請(qǐng)看源代碼講解。

          (1)硬件平臺(tái):
          基于朱兆祺51單片機(jī)學(xué)習(xí)板。

          (2)實(shí)現(xiàn)功能:
          波特率是:9600 。
          通過電腦串口調(diào)試助手模擬上位機(jī),往單片機(jī)發(fā)送EB 00 55 XX YY YY … YY YY指令,其中EB 00 55是數(shù)據(jù)頭,XX 是指令類型。YY是具體的數(shù)據(jù)。
          指令類型01代表發(fā)送的是數(shù)值,需要轉(zhuǎn)成組合BCD碼和非組合BCD碼,并且返回上位機(jī)顯示。
          指令類型02代表發(fā)送的是組合BCD碼,需要轉(zhuǎn)成數(shù)值和非組合BCD碼,并且返回上位機(jī)顯示。
          指令類型03代表發(fā)送的是非組合BCD碼,需要轉(zhuǎn)成數(shù)值和組合BCD碼,并且返回上位機(jī)顯示。

          返回上位機(jī)的數(shù)據(jù)中,中間3個(gè)數(shù)據(jù)EE EE EE是分割線,為了方便觀察,沒實(shí)際意義。

          例如:十進(jìn)制的數(shù)據(jù)52013140,它的十六進(jìn)制數(shù)據(jù)是03 19 A8 54。
          (a)上位機(jī)發(fā)送數(shù)據(jù):eb 00 55 01 03 19 a8 54
          單片機(jī)返回:52 01 31 40 EE EE EE 05 02 00 01 03 01 04 00
          (b)上位機(jī)發(fā)送組合BCD碼:eb 00 55 02 52 01 31 40
          單片機(jī)返回:03 19 A8 54 EE EE EE 05 02 00 01 03 01 04 00
          (c)發(fā)送非組合BCD碼:eb 00 55 03 05 02 00 01 03 01 04 00
          單片機(jī)返回:03 19 A8 54 EE EE EE 52 01 31 40

          (3)源代碼講解如下:
          1. #include "REG52.H"
          2. #define const_voice_short40 //蜂鳴器短叫的持續(xù)時(shí)間
          3. /* 注釋一:
          4. * 注意,此處的const_rc_size是20,比之前章節(jié)的緩沖區(qū)稍微改大了一點(diǎn)。
          5. */
          6. #define const_rc_size20//接收串口中斷數(shù)據(jù)的緩沖區(qū)數(shù)組大小
          7. #define const_receive_time5//如果超過這個(gè)時(shí)間沒有串口數(shù)據(jù)過來,就認(rèn)為一串?dāng)?shù)據(jù)已經(jīng)全部接收完,這個(gè)時(shí)間根據(jù)實(shí)際情況來調(diào)整大小
          8. void initial_myself(void);
          9. void initial_peripheral(void);
          10. void delay_long(unsigned int uiDelaylong);
          11. void delay_short(unsigned int uiDelayShort);
          12. void T0_time(void);//定時(shí)中斷函數(shù)
          13. void usart_receive(void); //串口接收中斷函數(shù)
          14. void usart_service(void);//串口服務(wù)程序,在main函數(shù)里
          15. void eusart_send(unsigned char ucSendData);
          16. void number_to_BCD4(const unsigned char *p_ucNumber,unsigned char *p_ucBCD_bit4);//把數(shù)值轉(zhuǎn)換成組合BCD碼
          17. void number_to_BCD8(const unsigned char *p_ucNumber,unsigned char *p_ucBCD_bit8);//把數(shù)值轉(zhuǎn)換成非組合BCD碼
          18. void BCD4_to_number(const unsigned char *p_ucBCD_bit4,unsigned char *p_ucNumber); //組合BCD碼轉(zhuǎn)成數(shù)值
          19. void BCD4_to_BCD8(const unsigned char *p_ucBCD_bit4,unsigned char *p_ucBCD_bit8); //組合BCD碼轉(zhuǎn)成非組合BCD碼
          20. void BCD8_to_number(const unsigned char *p_ucBCD_bit8,unsigned char *p_ucNumber); //非組合BCD碼轉(zhuǎn)成數(shù)值
          21. void BCD8_to_BCD4(const unsigned char *p_ucBCD_bit8,unsigned char *p_ucBCD_bit4); //非組合BCD碼轉(zhuǎn)成組合BCD碼
          22. sbit beep_dr=P2^7; //蜂鳴器的驅(qū)動(dòng)IO口
          23. unsigned intuiSendCnt=0; //用來識(shí)別串口是否接收完一串?dāng)?shù)據(jù)的計(jì)時(shí)器
          24. unsigned char ucSendLock=1; //串口服務(wù)程序的自鎖變量,每次接收完一串?dāng)?shù)據(jù)只處理一次
          25. unsigned intuiRcregTotal=0;//代表當(dāng)前緩沖區(qū)已經(jīng)接收了多少個(gè)數(shù)據(jù)
          26. unsigned char ucRcregBuf[const_rc_size]; //接收串口中斷數(shù)據(jù)的緩沖區(qū)數(shù)組
          27. unsigned intuiRcMoveIndex=0;//用來解析數(shù)據(jù)協(xié)議的中間變量
          28. /* 注釋二:
          29. * 注意,本程序規(guī)定數(shù)值的最大范圍是0至99999999
          30. * 數(shù)組中的數(shù)據(jù)。高位在數(shù)組下標(biāo)大的方向,低位在數(shù)組下標(biāo)小的方向。
          31. */
          32. unsigned char ucBufferNumber[4]; //數(shù)值,用4個(gè)字節(jié)表示long類型的數(shù)值
          33. unsigned char ucBufferBCB_bit4[4]; //組合BCD碼
          34. unsigned char ucBufferBCB_bit8[8]; //非組合BCD碼
          35. void main()
          36. {
          37. initial_myself();
          38. delay_long(100);
          39. initial_peripheral();
          40. while(1)
          41. {
          42. usart_service();//串口服務(wù)程序
          43. }
          44. }
          45. void number_to_BCD4(const unsigned char *p_ucNumber,unsigned char *p_ucBCD_bit4)//把數(shù)值轉(zhuǎn)換成組合BCD碼
          46. {
          47. unsigned long ulNumberTemp=0;
          48. unsigned char ucTemp=0;
          49. ulNumberTemp=p_ucNumber[3];//把4個(gè)字節(jié)的數(shù)值合并成一個(gè)long類型數(shù)據(jù)
          50. ulNumberTemp=ulNumberTemp<<8;
          51. ulNumberTemp=ulNumberTemp+p_ucNumber[2];
          52. ulNumberTemp=ulNumberTemp<<8;
          53. ulNumberTemp=ulNumberTemp+p_ucNumber[1];
          54. ulNumberTemp=ulNumberTemp<<8;
          55. ulNumberTemp=ulNumberTemp+p_ucNumber[0];
          56. p_ucBCD_bit4[3]=ulNumberTemp%100000000/10000000;
          57. p_ucBCD_bit4[3]=p_ucBCD_bit4[3]<<4; //前半4位存第8位組合BCD碼
          58. ucTemp=ulNumberTemp%10000000/1000000;
          59. p_ucBCD_bit4[3]=p_ucBCD_bit4[3]+ucTemp; //后半4位存第7位組合BCD碼
          60. p_ucBCD_bit4[2]=ulNumberTemp%1000000/100000;
          61. p_ucBCD_bit4[2]=p_ucBCD_bit4[2]<<4; //前半4位存第6位組合BCD碼
          62. ucTemp=ulNumberTemp%100000/10000;
          63. p_ucBCD_bit4[2]=p_ucBCD_bit4[2]+ucTemp;//后半4位存第5位組合BCD碼
          64. p_ucBCD_bit4[1]=ulNumberTemp%10000/1000;
          65. p_ucBCD_bit4[1]=p_ucBCD_bit4[1]<<4; //前半4位存第4位組合BCD碼
          66. ucTemp=ulNumberTemp%1000/100;
          67. p_ucBCD_bit4[1]=p_ucBCD_bit4[1]+ucTemp;//后半4位存第3位組合BCD碼
          68. p_ucBCD_bit4[0]=ulNumberTemp%100/10;
          69. p_ucBCD_bit4[0]=p_ucBCD_bit4[0]<<4; //前半4位存第2位組合BCD碼
          70. ucTemp=ulNumberTemp%10;
          71. p_ucBCD_bit4[0]=p_ucBCD_bit4[0]+ucTemp;//后半4位存第1位組合BCD碼
          72. }
          73. void number_to_BCD8(const unsigned char *p_ucNumber,unsigned char *p_ucBCD_bit8)//把數(shù)值轉(zhuǎn)換成非組合BCD碼
          74. {
          75. unsigned long ulNumberTemp=0;
          76. ulNumberTemp=p_ucNumber[3];//把4個(gè)字節(jié)的數(shù)值合并成一個(gè)long類型數(shù)據(jù)
          77. ulNumberTemp=ulNumberTemp<<8;
          78. ulNumberTemp=ulNumberTemp+p_ucNumber[2];
          79. ulNumberTemp=ulNumberTemp<<8;
          80. ulNumberTemp=ulNumberTemp+p_ucNumber[1];
          81. ulNumberTemp=ulNumberTemp<<8;
          82. ulNumberTemp=ulNumberTemp+p_ucNumber[0];
          83. p_ucBCD_bit8[7]=ulNumberTemp%100000000/10000000;//一個(gè)字節(jié)8位存儲(chǔ)第8位非組合BCD碼
          84. p_ucBCD_bit8[6]=ulNumberTemp%10000000/1000000;//一個(gè)字節(jié)8位存儲(chǔ)第7位非組合BCD碼
          85. p_ucBCD_bit8[5]=ulNumberTemp%1000000/100000;//一個(gè)字節(jié)8位存儲(chǔ)第6位非組合BCD碼
          86. p_ucBCD_bit8[4]=ulNumberTemp%100000/10000;//一個(gè)字節(jié)8位存儲(chǔ)第5位非組合BCD碼
          87. p_ucBCD_bit8[3]=ulNumberTemp%10000/1000;//一個(gè)字節(jié)8位存儲(chǔ)第4位非組合BCD碼
          88. p_ucBCD_bit8[2]=ulNumberTemp%1000/100;//一個(gè)字節(jié)8位存儲(chǔ)第3位非組合BCD碼
          89. p_ucBCD_bit8[1]=ulNumberTemp%100/10;//一個(gè)字節(jié)8位存儲(chǔ)第2位非組合BCD碼
          90. p_ucBCD_bit8[0]=ulNumberTemp%10;//一個(gè)字節(jié)8位存儲(chǔ)第1位非組合BCD碼
          91. }
          92. void BCD4_to_number(const unsigned char *p_ucBCD_bit4,unsigned char *p_ucNumber) //組合BCD碼轉(zhuǎn)成數(shù)值
          93. {
          94. unsigned long ulTmep;
          95. unsigned long ulSum;
          96. ulSum=0;//累加和數(shù)值清零
          97. ulTmep=0;
          98. ulTmep=p_ucBCD_bit4[3];
          99. ulTmep=ulTmep>>4;//把組合BCD碼第8位分解出來
          100. ulTmep=ulTmep*10000000;
          101. ulSum=ulSum+ulTmep; //累加各位數(shù)值
          102. ulTmep=0;
          103. ulTmep=p_ucBCD_bit4[3];
          104. ulTmep=ulTmep&0x0000000f;//把組合BCD碼第7位分解出來
          105. ulTmep=ulTmep*1000000;
          106. ulSum=ulSum+ulTmep; //累加各位數(shù)值
          107. ulTmep=0;
          108. ulTmep=p_ucBCD_bit4[2];
          109. ulTmep=ulTmep>>4;//把組合BCD碼第6位分解出來
          110. ulTmep=ulTmep*100000;
          111. ulSum=ulSum+ulTmep; //累加各位數(shù)值
          112. ulTmep=0;
          113. ulTmep=p_ucBCD_bit4[2];
          114. ulTmep=ulTmep&0x0000000f;//把組合BCD碼第5位分解出來
          115. ulTmep=ulTmep*10000;
          116. ulSum=ulSum+ulTmep; //累加各位數(shù)值
          117. ulTmep=0;
          118. ulTmep=p_ucBCD_bit4[1];
          119. ulTmep=ulTmep>>4;//把組合BCD碼第4位分解出來
          120. ulTmep=ulTmep*1000;
          121. ulSum=ulSum+ulTmep; //累加各位數(shù)值
          122. ulTmep=0;
          123. ulTmep=p_ucBCD_bit4[1];
          124. ulTmep=ulTmep&0x0000000f;//把組合BCD碼第3位分解出來
          125. ulTmep=ulTmep*100;
          126. ulSum=ulSum+ulTmep; //累加各位數(shù)值
          127. ulTmep=0;
          128. ulTmep=p_ucBCD_bit4[0];
          129. ulTmep=ulTmep>>4;//把組合BCD碼第2位分解出來
          130. ulTmep=ulTmep*10;
          131. ulSum=ulSum+ulTmep; //累加各位數(shù)值
          132. ulTmep=0;
          133. ulTmep=p_ucBCD_bit4[0];
          134. ulTmep=ulTmep&0x0000000f;//把組合BCD碼第1位分解出來
          135. ulTmep=ulTmep*1;
          136. ulSum=ulSum+ulTmep; //累加各位數(shù)值
          137. //以上代碼非常有規(guī)律,有興趣的讀者也可以自己想辦法把它壓縮成一個(gè)for循環(huán)的函數(shù),可以極大節(jié)省容量。
          138. p_ucNumber[3]=ulSum>>24;//把long類型數(shù)據(jù)分解成4個(gè)字節(jié)
          139. p_ucNumber[2]=ulSum>>16;
          140. p_ucNumber[1]=ulSum>>8;
          141. p_ucNumber[0]=ulSum;
          142. }
          143. void BCD4_to_BCD8(const unsigned char *p_ucBCD_bit4,unsigned char *p_ucBCD_bit8) //組合BCD碼轉(zhuǎn)成非組合BCD碼
          144. {
          145. unsigned char ucTmep;
          146. ucTmep=p_ucBCD_bit4[3];
          147. p_ucBCD_bit8[7]=ucTmep>>4; //把組合BCD碼第8位分解出來
          148. p_ucBCD_bit8[6]=ucTmep&0x0f;//把組合BCD碼第7位分解出來
          149. ucTmep=p_ucBCD_bit4[2];
          150. p_ucBCD_bit8[5]=ucTmep>>4; //把組合BCD碼第6位分解出來
          151. p_ucBCD_bit8[4]=ucTmep&0x0f;//把組合BCD碼第5位分解出來
          152. ucTmep=p_ucBCD_bit4[1];
          153. p_ucBCD_bit8[3]=ucTmep>>4; //把組合BCD碼第4位分解出來
          154. p_ucBCD_bit8[2]=ucTmep&0x0f;//把組合BCD碼第3位分解出來
          155. ucTmep=p_ucBCD_bit4[0];
          156. p_ucBCD_bit8[1]=ucTmep>>4; //把組合BCD碼第2位分解出來
          157. p_ucBCD_bit8[0]=ucTmep&0x0f;//把組合BCD碼第1位分解出來
          158. }
          159. void BCD8_to_number(const unsigned char *p_ucBCD_bit8,unsigned char *p_ucNumber) //非組合BCD碼轉(zhuǎn)成數(shù)值
          160. {
          161. unsigned long ulTmep;
          162. unsigned long ulSum;
          163. ulSum=0;//累加和數(shù)值清零
          164. ulTmep=0;
          165. ulTmep=p_ucBCD_bit8[7];
          166. ulTmep=ulTmep*10000000;
          167. ulSum=ulSum+ulTmep; //累加各位數(shù)值
          168. ulTmep=0;
          169. ulTmep=p_ucBCD_bit8[6];
          170. ulTmep=ulTmep*1000000;
          171. ulSum=ulSum+ulTmep; //累加各位數(shù)值
          172. ulTmep=0;
          173. ulTmep=p_ucBCD_bit8[5];
          174. ulTmep=ulTmep*100000;
          175. ulSum=ulSum+ulTmep; //累加各位數(shù)值
          176. ulTmep=0;
          177. ulTmep=p_ucBCD_bit8[4];
          178. ulTmep=ulTmep*10000;
          179. ulSum=ulSum+ulTmep; //累加各位數(shù)值
          180. ulTmep=0;
          181. ulTmep=p_ucBCD_bit8[3];
          182. ulTmep=ulTmep*1000;
          183. ulSum=ulSum+ulTmep; //累加各位數(shù)值
          184. ulTmep=0;
          185. ulTmep=p_ucBCD_bit8[2];
          186. ulTmep=ulTmep*100;
          187. ulSum=ulSum+ulTmep; //累加各位數(shù)值
          188. ulTmep=0;
          189. ulTmep=p_ucBCD_bit8[1];
          190. ulTmep=ulTmep*10;
          191. ulSum=ulSum+ulTmep; //累加各位數(shù)值
          192. ulTmep=0;
          193. ulTmep=p_ucBCD_bit8[0];
          194. ulTmep=ulTmep*1;
          195. ulSum=ulSum+ulTmep; //累加各位數(shù)值
          196. //以上代碼非常有規(guī)律,有興趣的讀者也可以自己想辦法把它壓縮成一個(gè)for循環(huán)的函數(shù),可以極大節(jié)省容量。
          197. p_ucNumber[3]=ulSum>>24;//把long類型數(shù)據(jù)分解成4個(gè)字節(jié)
          198. p_ucNumber[2]=ulSum>>16;
          199. p_ucNumber[1]=ulSum>>8;
          200. p_ucNumber[0]=ulSum;
          201. }
          202. void BCD8_to_BCD4(const unsigned char *p_ucBCD_bit8,unsigned char *p_ucBCD_bit4) //非組合BCD碼轉(zhuǎn)成組合BCD碼
          203. {
          204. unsigned char ucTmep;
          205. ucTmep=p_ucBCD_bit8[7]; //把非組合BCD碼第8位分解出來
          206. p_ucBCD_bit4[3]=ucTmep<<4;
          207. p_ucBCD_bit4[3]=p_ucBCD_bit4[3]+p_ucBCD_bit8[6]; //把非組合BCD碼第7位分解出來
          208. ucTmep=p_ucBCD_bit8[5]; //把非組合BCD碼第6位分解出來
          209. p_ucBCD_bit4[2]=ucTmep<<4;
          210. p_ucBCD_bit4[2]=p_ucBCD_bit4[2]+p_ucBCD_bit8[4]; //把非組合BCD碼第5位分解出來
          211. ucTmep=p_ucBCD_bit8[3]; //把非組合BCD碼第4位分解出來
          212. p_ucBCD_bit4[1]=ucTmep<<4;
          213. p_ucBCD_bit4[1]=p_ucBCD_bit4[1]+p_ucBCD_bit8[2]; //把非組合BCD碼第3位分解出來
          214. ucTmep=p_ucBCD_bit8[1]; //把非組合BCD碼第2位分解出來
          215. p_ucBCD_bit4[0]=ucTmep<<4;
          216. p_ucBCD_bit4[0]=p_ucBCD_bit4[0]+p_ucBCD_bit8[0]; //把非組合BCD碼第1位分解出來
          217. }
          218. void usart_service(void)//串口服務(wù)程序,在main函數(shù)里
          219. {
          220. unsigned char i=0;
          221. if(uiSendCnt>=const_receive_time&&ucSendLock==1) //說明超過了一定的時(shí)間內(nèi),再也沒有新數(shù)據(jù)從串口來
          222. {
          223. ucSendLock=0; //處理一次就鎖起來,不用每次都進(jìn)來,除非有新接收的數(shù)據(jù)
          224. //下面的代碼進(jìn)入數(shù)據(jù)協(xié)議解析和數(shù)據(jù)處理的階段
          225. uiRcMoveIndex=0; //由于是判斷數(shù)據(jù)頭,所以下標(biāo)移動(dòng)變量從數(shù)組的0開始向最尾端移動(dòng)
          226. while(uiRcregTotal>=5&&uiRcMoveIndex<=(uiRcregTotal-5))
          227. {
          228. if(ucRcregBuf[uiRcMoveIndex+0]==0xeb&&ucRcregBuf[uiRcMoveIndex+1]==0x00&&ucRcregBuf[uiRcMoveIndex+2]==0x55)//數(shù)據(jù)頭eb 00 55的判斷
          229. {
          230. switch(ucRcregBuf[uiRcMoveIndex+3])//根據(jù)命令類型來進(jìn)行不同的處理
          231. {
          232. case 1://接收到的是數(shù)值,需要轉(zhuǎn)成組合BCD碼和非組合BCD碼
          233. for(i=0;i<4;i++)
          234. {
          235. ucBufferNumber[3-i]=ucRcregBuf[uiRcMoveIndex+4+i]; //從串口接收到的數(shù)據(jù),注意,高位在數(shù)組下標(biāo)大的方向
          236. }
          237. number_to_BCD4(ucBufferNumber,ucBufferBCB_bit4);//把數(shù)值轉(zhuǎn)換成組合BCD碼
          238. number_to_BCD8(ucBufferNumber,ucBufferBCB_bit8);//把數(shù)值轉(zhuǎn)換成非組合BCD碼
          239. for(i=0;i<4;i++)
          240. {
          241. eusart_send(ucBufferBCB_bit4[3-i]);////把組合BCD碼返回給上位機(jī)觀察,注意,高位在數(shù)組下標(biāo)大的方向
          242. }
          243. eusart_send(0xee);//為了方便上位機(jī)觀察,多發(fā)送3個(gè)字節(jié)ee ee ee作為分割線
          244. eusart_send(0xee);
          245. eusart_send(0xee);
          246. for(i=0;i<8;i++)
          247. {
          248. eusart_send(ucBufferBCB_bit8[7-i]);////把非組合BCD碼返回給上位機(jī)觀察,注意,高位在數(shù)組下標(biāo)大的方向
          249. }
          250. break;
          251. case 2://接收到的是組合BCD碼,需要轉(zhuǎn)成數(shù)值和非組合BCD碼
          252. for(i=0;i<4;i++)
          253. {
          254. ucBufferBCB_bit4[3-i]=ucRcregBuf[uiRcMoveIndex+4+i]; //從串口接收到的組合BCD碼,注意,高位在數(shù)組下標(biāo)大的方向
          255. }
          256. BCD4_to_number(ucBufferBCB_bit4,ucBufferNumber); //組合BCD碼轉(zhuǎn)成數(shù)值
          257. BCD4_to_BCD8(ucBufferBCB_bit4,ucBufferBCB_bit8); //組合BCD碼轉(zhuǎn)成非組合BCD碼
          258. for(i=0;i<4;i++)
          259. {
          260. eusart_send(ucBufferNumber[3-i]);////把數(shù)值返回給上位機(jī)觀察,注意,高位在數(shù)組下標(biāo)大的方向
          261. }
          262. eusart_send(0xee);//為了方便上位機(jī)觀察,多發(fā)送3個(gè)字節(jié)ee ee ee作為分割線
          263. eusart_send(0xee);
          264. eusart_send(0xee);
          265. for(i=0;i<8;i++)
          266. {
          267. eusart_send(ucBufferBCB_bit8[7-i]);////把非組合BCD碼返回給上位機(jī)觀察,注意,高位在數(shù)組下標(biāo)大的方向
          268. }
          269. break;
          270. case 3://接收到的是非組合BCD碼,需要轉(zhuǎn)成數(shù)值和組合BCD碼
          271. for(i=0;i<8;i++)
          272. {
          273. ucBufferBCB_bit8[7-i]=ucRcregBuf[uiRcMoveIndex+4+i]; //從串口接收到的非組合BCD碼,注意,高位在數(shù)組下標(biāo)大的方向
          274. }
          275. BCD8_to_number(ucBufferBCB_bit8,ucBufferNumber); //非組合BCD碼轉(zhuǎn)成數(shù)值
          276. BCD8_to_BCD4(ucBufferBCB_bit8,ucBufferBCB_bit4); //非組合BCD碼轉(zhuǎn)成組合BCD碼
          277. for(i=0;i<4;i++)
          278. {
          279. eusart_send(ucBufferNumber[3-i]);////把數(shù)值返回給上位機(jī)觀察
          280. }
          281. eusart_send(0xee);//為了方便上位機(jī)觀察,多發(fā)送3個(gè)字節(jié)ee ee ee作為分割線,注意,高位在數(shù)組下標(biāo)大的方向
          282. eusart_send(0xee);
          283. eusart_send(0xee);
          284. for(i=0;i<4;i++)
          285. {
          286. eusart_send(ucBufferBCB_bit4[3-i]);////把組合BCD碼返回給上位機(jī)觀察,注意,高位在數(shù)組下標(biāo)大的方向
          287. }
          288. break;
          289. }
          290. break; //退出循環(huán)
          291. }
          292. uiRcMoveIndex++; //因?yàn)槭桥袛鄶?shù)據(jù)頭,游標(biāo)向著數(shù)組最尾端的方向移動(dòng)
          293. }
          294. uiRcregTotal=0;//清空緩沖的下標(biāo),方便下次重新從0下標(biāo)開始接受新數(shù)據(jù)
          295. }
          296. }
          297. void eusart_send(unsigned char ucSendData) //往上位機(jī)發(fā)送一個(gè)字節(jié)的函數(shù)
          298. {
          299. ES = 0; //關(guān)串口中斷
          300. TI = 0; //清零串口發(fā)送完成中斷請(qǐng)求標(biāo)志
          301. SBUF =ucSendData; //發(fā)送一個(gè)字節(jié)
          302. delay_short(400);//每個(gè)字節(jié)之間的延時(shí),這里非常關(guān)鍵,也是最容易出錯(cuò)的地方。延時(shí)的大小請(qǐng)根據(jù)實(shí)際項(xiàng)目來調(diào)整
          303. TI = 0; //清零串口發(fā)送完成中斷請(qǐng)求標(biāo)志
          304. ES = 1; //允許串口中斷
          305. }
          306. void T0_time(void) interrupt 1 //定時(shí)中斷
          307. {
          308. TF0=0;//清除中斷標(biāo)志
          309. TR0=0; //關(guān)中斷
          310. if(uiSendCnt
          311. {
          312. uiSendCnt++; //表面上這個(gè)數(shù)據(jù)不斷累加,但是在串口中斷里,每接收一個(gè)字節(jié)它都會(huì)被清零,除非這個(gè)中間沒有串口數(shù)據(jù)過來
          313. ucSendLock=1; //開自鎖標(biāo)志
          314. }
          315. TH0=0xfe; //重裝初始值(65535-500)=65035=0xfe0b
          316. TL0=0x0b;
          317. TR0=1;//開中斷
          318. }
          319. void usart_receive(void) interrupt 4 //串口接收數(shù)據(jù)中斷
          320. {
          321. if(RI==1)
          322. {
          323. RI = 0;
          324. ++uiRcregTotal;
          325. if(uiRcregTotal>const_rc_size)//超過緩沖區(qū)
          326. {
          327. uiRcregTotal=const_rc_size;
          328. }
          329. ucRcregBuf[uiRcregTotal-1]=SBUF; //將串口接收到的數(shù)據(jù)緩存到接收緩沖區(qū)里
          330. uiSendCnt=0;//及時(shí)喂狗,雖然main函數(shù)那邊不斷在累加,但是只要串口的數(shù)據(jù)還沒發(fā)送完畢,那么它永遠(yuǎn)也長(zhǎng)不大,因?yàn)槊總€(gè)中斷都被清零。
          331. }
          332. else//發(fā)送中斷,及時(shí)把發(fā)送中斷標(biāo)志位清零
          333. {
          334. TI = 0;
          335. }
          336. }
          337. void delay_long(unsigned int uiDelayLong)
          338. {
          339. unsigned int i;
          340. unsigned int j;
          341. for(i=0;i
          342. {
          343. for(j=0;j<500;j++)//內(nèi)嵌循環(huán)的空指令數(shù)量
          344. {
          345. ; //一個(gè)分號(hào)相當(dāng)于執(zhí)行一條空語句
          346. }
          347. }
          348. }
          349. void delay_short(unsigned int uiDelayShort)
          350. {
          351. unsigned int i;
          352. for(i=0;i
          353. {
          354. ; //一個(gè)分號(hào)相當(dāng)于執(zhí)行一條空語句
          355. }
          356. }
          357. void initial_myself(void)//第一區(qū) 初始化單片機(jī)
          358. {
          359. beep_dr=1; //用PNP三極管控制蜂鳴器,輸出高電平時(shí)不叫。
          360. //配置定時(shí)器
          361. TMOD=0x01;//設(shè)置定時(shí)器0為工作方式1
          362. TH0=0xfe; //重裝初始值(65535-500)=65035=0xfe0b
          363. TL0=0x0b;
          364. //配置串口
          365. SCON=0x50;
          366. TMOD=0X21;
          367. TH1=TL1=-(11059200L/12/32/9600);//這段配置代碼具體是什么意思,我也不太清楚,反正是跟串口波特率有關(guān)。
          368. TR1=1;
          369. }
          370. void initial_peripheral(void) //第二區(qū) 初始化外圍
          371. {
          372. EA=1; //開總中斷
          373. ES=1; //允許串口中斷
          374. ET0=1; //允許定時(shí)中斷
          375. TR0=1; //啟動(dòng)定時(shí)中斷
          376. }

          總結(jié)陳詞:
          有了這一節(jié)非組合BCD的基礎(chǔ)知識(shí),下一節(jié)就開始講大數(shù)據(jù)的算法程序。這些算法程序經(jīng)常要用在計(jì)算器,工控,以及高精度的儀器儀表等領(lǐng)域。C語言的語法中不是已經(jīng)提供了+,-,*,/這些運(yùn)算符號(hào)嗎?為什么還要專門寫算法程序?因?yàn)槟切┻\(yùn)算符只能進(jìn)行簡(jiǎn)單的運(yùn)算,一旦數(shù)據(jù)超過了unsigned long(4個(gè)字節(jié))的范圍就會(huì)出錯(cuò)。而這種大數(shù)據(jù)算法的程序是什么樣的?欲知詳情,請(qǐng)聽下回分解----大數(shù)據(jù)的加法運(yùn)算。


          評(píng)論


          技術(shù)專區(qū)

          關(guān)閉
          看屁屁www成人影院,亚洲人妻成人图片,亚洲精品成人午夜在线,日韩在线 欧美成人 (function(){ var bp = document.createElement('script'); var curProtocol = window.location.protocol.split(':')[0]; if (curProtocol === 'https') { bp.src = 'https://zz.bdstatic.com/linksubmit/push.js'; } else { bp.src = 'http://push.zhanzhang.baidu.com/push.js'; } var s = document.getElementsByTagName("script")[0]; s.parentNode.insertBefore(bp, s); })();