第一篇:51藍(lán)牙串口通信給親參考
/**********51藍(lán)牙串口調(diào)試程序**************/
//串口接收的數(shù)據(jù)送至P1口,可接幾個led試一下
#include
void main(void)
{
TMOD=0x20;//T0方式2,作為波特率發(fā)生器
TH1=0xfd;//9600波特率
TL1=0xfd;
SM0=0;//串口方式1
SM1=1;
REN=1;
EA=1;
ES=1;
PCON=0x80;
TR1=1;
while(1);//wait here
}
//******************串口中斷*******************
void isr()interrupt 4
{
ES=0;
if(RI==1)
{
P1=SBUF;//P1口作為驅(qū)動口,根據(jù)需要結(jié)合自己發(fā)送的指令自己調(diào)整哦}
ES=1;
}
第二篇:串口通信實驗報告范文
華南農(nóng)業(yè)大學(xué)
實驗報告
----------目錄----------
1、實驗任務(wù)和目的..............................................................................................................2、實驗準(zhǔn)備..........................................................................................................................3、實驗步驟................................................................................................................................4、實驗分析與總結(jié)....................................................................................................................(1)、分析.............................................................................................................................(2)、總結(jié).............................................................................................................................1、實驗任務(wù)和目的
了解串行通信的背景知識后,通過三線制制作一條串口通信線(PC-PC),并編程實現(xiàn)兩臺PC間通過RS-232C通信。要求兩臺PC機能進(jìn)行實時的字符通信,并了解工業(yè)自動化控制中的基本通信方式。
2、實驗準(zhǔn)備
1、檢查PC是否具有串行通信接口,并按其針腳類準(zhǔn)備一條串口通信線纜。
2、串口包的安裝,下載javacomm20-win32.zip并解壓,將win32com.dll復(fù)制到
3、實驗步驟
1、將實驗所需RS-232纜線準(zhǔn)備好,并將JAVA串口包復(fù)制到相應(yīng)地目錄下。
2、查找有關(guān)串口通信的書籍以及在網(wǎng)上查找相應(yīng)地串口通信代碼。
3、用JAVA編程軟件JCreator編寫代碼。
4、實驗分析與總結(jié)
(1)、分析
(I)、對串口讀寫之前需要先打開一個串口并檢測串口是否被占用: public void open(){//打開串口以及輸入輸出流
recieve=false;
try
{serialPort=(SerialPort)portId.open(“Serial Communication”, 2000);}
catch(PortInUseException e){System.out.println(“端口正被占用!”);}
try
{serialPort.setSerialPortParams(9600,SerialPort.DATABITS_8,SerialPort.STOPBITS_1,SerialPort.PARITY_NONE);}
catch(UnsupportedCommOperationException e){System.out.println(“不支持通信”);}
try
{
outputStream=serialPort.getOutputStream();
inputStream=serialPort.getInputStream();
1-完整運行程序如圖所示:
圖1
(2)、總結(jié)
通過本次串口實驗,我對串口通信的知識了解的更透徹,這是在剛開始對串口通信知識不了解的情況下就編程而造成許多錯誤之后才得到的結(jié)果。在網(wǎng)上查找資料的時候也接觸到了不少其他的編程語言例如VB,delphi,C#等,這也讓我對這些從沒有學(xué)過的語言有所了解,我想這些知識對以后的實驗工作都有幫助。
3--
第三篇:藍(lán)牙通信原理
藍(lán)牙耳機的工作原理:
關(guān)于音頻流的藍(lán)牙傳輸可以通過兩個方式:
1)通過PCM接口來傳送
2)通過模擬UART來傳送
下面分別來講述:
1)通過PCM接口來傳送
通過音頻播放器(eg: Media Player)來打開音頻文件,調(diào)用Audio驅(qū)動,音頻文件通過解碼后,由PCM輸出到Host端藍(lán)牙模塊的PCM輸入端,接著,經(jīng)過藍(lán)牙模塊的處理后,由RF無線模塊發(fā)送給Client 端藍(lán)牙設(shè)備。
Client 端藍(lán)牙設(shè)備經(jīng)由無線接收模塊后,濾波,穩(wěn)壓,經(jīng)微處理芯片處理后,直接由Speaker播放。
2)通過模擬UART來傳送
通過設(shè)置注冊表【HKEY_LOCAL_MACHINEServicesBTAGSVC】IsEnabled =1 使得系統(tǒng)引導(dǎo)時自動加載語音網(wǎng)關(guān)(AG)服務(wù)。
首先,通過手動配置建立Host端藍(lán)牙設(shè)備與Client端藍(lán)牙設(shè)備ACL鏈接(面向無連接的異步鏈路),接著在Applicaiton或Audio Driver中調(diào)用
IOCTL_AG_OPEN_AUDIO,重新建立Host端藍(lán)牙設(shè)備與Client端藍(lán)牙設(shè)備SCO鏈接(面向連接的同步鏈路),接著AG自動發(fā)送
waveOutMessage((HWAVEOUT)i, WODM_BT_SCO_AUDIO_CONTROL, 0, TRUE);從而建立了Audio至藍(lán)牙芯片之間的通道,也即,實現(xiàn)了音頻流到藍(lán)牙模塊的傳送。
然后,經(jīng)由主機端藍(lán)牙模塊將音頻流打包經(jīng)由RF模塊發(fā)送出去。
客戶端藍(lán)牙耳機接收到無線音頻包后,濾波,穩(wěn)壓,經(jīng)微處理芯片處理后,由PCM傳送給音頻編解碼器芯片,最后,由Speaker播放。
第四篇:單片機串口通信方式總結(jié)
IIC總線通信協(xié)議————數(shù)據(jù)傳輸高位在前p233 1,起始和停止條件
開始信號:SCL為高電平,SDA由高電平向低電平跳變,開始傳送數(shù)據(jù)。void start()// 開始位 { SDA = 1;
//SDA初始化為高電平“1”
SCL = 1;
//開始數(shù)據(jù)傳送時,要求SCL為高電平“1”
_nop_();
//等待一個機器周期
_nop_();
//等待一個機器周期
SDA = 0;
//SDA的下降沿被認(rèn)為是開始信號
_nop_();
//等待一個機器周期
_nop_();
//等待一個機器周期
_nop_();
//等待一個機器周期
_nop_();
//等待一個機器周期
SCL = 0;
//SCL為低電平時,SDA上數(shù)據(jù)才允許變化(即允許以后的數(shù)據(jù)傳遞)} 結(jié)束信號:SCL為高電平,SDA由低電平向高電平跳變,結(jié)束傳送數(shù)據(jù)。void stop()// 停止位 { SDA = 0;
//SDA初始化為低電平“0”
_nop_();
//等待一個機器周期
_nop_();
//等待一個機器周期
SCL = 1;
//結(jié)束數(shù)據(jù)傳送時,要求SCL為高電平“1”
_nop_();
//等待一個機器周期
_nop_();
//等待一個機器周期
_nop_();
//等待一個機器周期
_nop_();
//等待一個機器周期
SDA = 1;
//SDA的上升沿被認(rèn)為是結(jié)束信號 }
2,數(shù)據(jù)格式(數(shù)據(jù)輸入)
在IIC總線開始信號后,送出的第一個字節(jié)數(shù)據(jù)是用來選擇器件地址和數(shù)據(jù)方向的,其格式為
從器件收到地址型號后與自己的地址比較,一致則此器件就是主器件要找的器件,并返回ACK(不管是寫數(shù)據(jù)還是地址都會返回)。IIC傳送數(shù)據(jù)時SCL為低電平時SDA可改變高低電平,SCL轉(zhuǎn)跳為高時數(shù)據(jù)輸入(此時SDA不能跳變),發(fā)送數(shù)據(jù):bit WriteCurrent(unsigned char y){ unsigned char i;bit ack_bit;
//儲存應(yīng)答位
for(i = 0;i < 8;i++)// 循環(huán)移入8個位
{
SDA =(bit)(y&0x80);
//通過按位“與”運算將最高位數(shù)據(jù)送到S
//因為傳送時高位在前,低位在后
_nop_();
//等待一個機器周期
SCL = 1;
//在SCL的上升沿將數(shù)據(jù)寫入AT24Cxx
_nop_();
//等待一個機器周期
_nop_();
//等待一個機器周期
SCL = 0;
//將SCL重新置為低電平,以在SCL線形成傳送數(shù)據(jù)所需的8個脈沖
y <<= 1;
//將y中的各二進(jìn)位向左移一位
} SDA = 1;
// 發(fā)送設(shè)備(主機)應(yīng)在時鐘脈沖的高電平期間(SCL=1)釋放SDA線,//以讓SDA線轉(zhuǎn)由接收設(shè)備(AT24Cxx)控制
_nop_();
//等待一個機器周期
_nop_();
//等待一個機器周期
SCL = 1;
//根據(jù)上述規(guī)定,SCL應(yīng)為高電平
_nop_();
//等待一個機器周期
_nop_();
//等待一個機器周期
_nop_();
//等待一個機器周期
_nop_();
//等待一個機器周期
ack_bit = SDA;//接受設(shè)備(AT24Cxx)向SDA送低電平,表示已經(jīng)接收到一個字節(jié)
//若送高電平,表示沒有接收到,傳送異常
SCL = 0;
//SCL為低電平時,SDA上數(shù)據(jù)才允許變化(即允許以后的數(shù)據(jù)傳遞)
return ack_bit;
// 返回AT24Cxx應(yīng)答位 } 讀數(shù)據(jù):unsigned char ReadData()// 從AT24Cxx移入數(shù)據(jù)到MCU { unsigned char i;unsigned char x;
//儲存從AT24Cxx中讀出的數(shù)據(jù)
for(i = 0;i < 8;i++){
SCL = 1;
//SCL置為高電平
x<<=1;
//將x中的各二進(jìn)位向左移一位
x|=(unsigned char)SDA;//將SDA上的數(shù)據(jù)通過按位“或“運算存入x中
SCL = 0;
//在SCL的下降沿讀出數(shù)據(jù)
} return(x);
//將讀取的數(shù)據(jù)返回 } 發(fā)送數(shù)據(jù)步驟:
oid WriteSet(unsigned char add, unsigned char dat)// 在指定地址addr處寫入數(shù)據(jù)WriteCurrent { start();
//開始數(shù)據(jù)傳遞
WriteCurrent(OP_WRITE);//選擇要操作的AT24Cxx芯片,并告知要對其寫入數(shù)據(jù)
WriteCurrent(add);
//寫入指定地址
WriteCurrent(dat);
//向當(dāng)前地址(上面指定的地址)寫入數(shù)據(jù)
stop();
//停止數(shù)據(jù)傳遞
delaynms(4);
//1個字節(jié)的寫入周期為1ms, 最好延時1ms以上 } 讀數(shù)據(jù)步驟:
/*************************************************** 函數(shù)功能:從AT24Cxx中的當(dāng)前地址讀取數(shù)據(jù) 出口參數(shù):x(儲存讀出的數(shù)據(jù))
***************************************************/ unsigned char ReadCurrent(){ unsigned char x;start();
//開始數(shù)據(jù)傳遞
WriteCurrent(OP_READ);
//選擇要操作的AT24Cxx芯片,并告知要讀其數(shù)據(jù)
x=ReadData();
//將讀取的數(shù)據(jù)存入x stop();
//停止數(shù)據(jù)傳遞
return x;
//返回讀取的數(shù)據(jù) } /*************************************************** 函數(shù)功能:從AT24Cxx中的指定地址讀取數(shù)據(jù) 入口參數(shù):set_add 出口參數(shù):x
***************************************************/ unsigned char ReadSet(unsigned char set_add)// 在指定地址讀取 { start();
//開始數(shù)據(jù)傳遞
WriteCurrent(OP_WRITE);
//選擇要操作的AT24Cxx芯片,并告知要對其寫入數(shù)據(jù)
WriteCurrent(set_add);
//寫入指定地址
return(ReadCurrent());
//從指定地址讀出數(shù)據(jù)并返回 }
單總線協(xié)議————數(shù)據(jù)傳輸?shù)臀辉谇啊穑玻常?1,初始化單總線器件
初始化時序程序:
函數(shù)功能:將DS18B20傳感器初始化,讀取應(yīng)答信號 出口參數(shù):flag
***************************************************/ bit Init_DS18B20(void){ bit flag;
//儲存DS18B20是否存在的標(biāo)志,flag=0,表示存在;flag=1,表示不存在
DQ = 1;
//先將數(shù)據(jù)線拉高
for(time=0;time<2;time++)//略微延時約6微秒
;DQ = 0;
//再將數(shù)據(jù)線從高拉低,要求保持480~960us for(time=0;time<200;time++)//略微延時約600微秒
;
//以向DS18B20發(fā)出一持續(xù)480~960us的低電平復(fù)位脈沖
DQ = 1;
//釋放數(shù)據(jù)線(將數(shù)據(jù)線拉高)
for(time=0;time<10;time++)
;//延時約30us(釋放總線后需等待15~60us讓DS18B20輸出存在脈沖)
flag=DQ;
//讓單片機檢測是否輸出了存在脈沖(DQ=0表示存在)
for(time=0;time<200;time++)//延時足夠長時間,等待存在脈沖輸出完畢
;return(flag);
//返回檢測成功標(biāo)志 }
單總線通信協(xié)議中存在兩種寫時隙:寫0寫1。主機采用寫1時隙向從機寫入1,而寫0時隙向從機寫入0。所有寫時隙至少要60us,且在兩次獨立的寫時隙之間至少要1us的恢復(fù)時間。兩種寫時隙均起始于主機拉低數(shù)據(jù)總線。產(chǎn)生1時隙的方式:主機拉低總線后,接著必須在15us之內(nèi)釋放總線,由上拉電阻將總線拉至高電平;產(chǎn)生寫0時隙的方式為在主機拉低后,只需要在整個時隙間保持低電平即可(至少60us)。在寫時隙開始后15~60us期間,單總線器件采樣總電平狀態(tài)。如果在此期間采樣值為高電平,則邏輯1被寫入器件;如果為0,寫入邏輯0。
下圖為寫時隙(包括1和0)時序
上圖中黑色實線代表系統(tǒng)主機拉低總線,黑色虛線代表上拉電阻將總線拉高。下面是代碼:
WriteOneChar(unsigned char dat){ unsigned char i=0;for(i=0;i<8;i++)
{
DQ =1;
// 先將數(shù)據(jù)線拉高
_nop_();
//等待一個機器周期
DQ=0;
//將數(shù)據(jù)線從高拉低時即啟動寫時序
DQ=dat&0x01;
//利用與運算取出要寫的某位二進(jìn)制數(shù)據(jù),//并將其送到數(shù)據(jù)線上等待DS18B20采樣
for(time=0;time<10;time++)
;//延時約30us,DS18B20在拉低后的約15~60us期間從數(shù)據(jù)線上采樣
DQ=1;
//釋放數(shù)據(jù)線
for(time=0;time<1;time++)
;//延時3us,兩個寫時序間至少需要1us的恢復(fù)期
dat>>=1;
//將dat中的各二進(jìn)制位數(shù)據(jù)右移1位
}
for(time=0;time<4;time++)
;//稍作延時,給硬件一點反應(yīng)時間 }
對于讀時隙,單總線器件僅在主機發(fā)出讀時隙時,才向主機傳輸數(shù)據(jù)。所有主機發(fā)出讀數(shù)據(jù)命令后,必須馬上產(chǎn)生讀時隙,以便從機能夠傳輸數(shù)據(jù)。所有讀時隙至少需要60us,且在兩次獨立的讀時隙之間至少需要1us恢復(fù)時間。每個讀時隙都由主機發(fā)起,至少拉低總線1us。在主機發(fā)出讀時隙后,單總線器件才開始在總線上發(fā)送1或0。若從機發(fā)送1,則保持總線為高電平;若發(fā)出0,則拉低總線。
當(dāng)發(fā)送0時,從機在讀時隙結(jié)束后釋放總線,由上拉電阻將總線拉回至空閑高電平狀態(tài)。從機發(fā)出的數(shù)據(jù)在起始時隙之后,保持有效時間15us,因此主機在讀時隙期間必須釋放總線,并且在時隙起始后的15us之內(nèi)采樣總線狀態(tài)。
下圖給出讀時隙(包括0或1)時序
圖中黑色實線代表系統(tǒng)主機拉低總線,灰色實線代表總局拉低總線,而黑色的虛線則代表上拉電阻總線拉高。代碼為:
unsigned char ReadOneChar(void){
unsigned char i=0;
unsigned char dat;//儲存讀出的一個字節(jié)數(shù)據(jù)
for(i=0;i<8;i++)
{
DQ =1;
// 先將數(shù)據(jù)線拉高
_nop_();
//等待一個機器周期
DQ = 0;
//單片機從DS18B20讀書據(jù)時,將數(shù)據(jù)線從高拉低即啟動讀時序
dat>>=1;
_nop_();
//等待一個機器周期
DQ = 1;
//將數(shù)據(jù)線“人為”拉高,為單片機檢測DS18B20的輸出電平作準(zhǔn)備
for(time=0;time<2;time++)
;
//延時約6us,使主機在15us內(nèi)采樣
if(DQ==1)
dat|=0x80;//如果讀到的數(shù)據(jù)是1,則將1存入dat
else
dat|=0x00;//如果讀到的數(shù)據(jù)是0,則將0存入dat
//將單片機檢測到的電平信號DQ存入r[i]
for(time=0;time<8;time++)
;
//延時3us,兩個讀時序之間必須有大于1us的恢復(fù)期
}
return(dat);
//返回讀出的十進(jìn)制數(shù)據(jù) }
每個單總線器件內(nèi)部都光刻了一個全球唯一的64位二進(jìn)制序列碼,用于該單總線器件的識別
SPI總線協(xié)議
SPI總線有四種工作方式(SP0, SP1, SP2, SP3),其中使用的最為廣泛的是SPI0和SPI3方式。
SPI是一個環(huán)形總線結(jié)構(gòu),由ss(cs)、sck、sdi、sdo構(gòu)成,其時序其實很簡單,主要是在sck的控制下,兩個雙向移位寄存器進(jìn)行數(shù)據(jù)交換。
上升沿發(fā)送、下降沿接收、高位先發(fā)送。
上升沿到來的時候,sdo上的電平將被發(fā)送到從設(shè)備的寄存器中。
下降沿到來的時候,sdi上的電平將被接收到主設(shè)備的寄存器中。讀代碼:
unsigned char ReadCurrent(void){
unsigned char i;unsigned char x=0x00;
//儲存從X5045中讀出的數(shù)據(jù)
SCK=1;
//將SCK置于已知的高電平狀態(tài)
for(i = 0;i < 8;i++){
SCK=1;
//拉高SCK
SCK=0;
//在SCK的下降沿輸出數(shù)據(jù)
x<<=1;//將x中的各二進(jìn)位向左移一位,因為首先讀出的是字節(jié)的最高位數(shù)據(jù)
x|=(unsigned char)SO;//將SO上的數(shù)據(jù)通過按位“或“運算存入 x
} return(x);
//將讀取的數(shù)據(jù)返回
} 寫代碼:
void WriteCurrent(unsigned char dat){
unsigned char i;SCK=0;
//將SCK置于已知的低電平狀態(tài)
for(i = 0;i < 8;i++)// 循環(huán)移入8個位
{
SI=(bit)(dat&0x80);
//通過按位“與”運算將最高位數(shù)據(jù)送到S
//因為傳送時高位在前,低位在后
SCK=0;
SCK=1;
//在SCK上升沿寫入數(shù)據(jù)
dat<<=1;
//將y中的各二進(jìn)位向左移一位,因為首先寫入的是字節(jié)的最高位
} } RS232通訊協(xié)議 串行通訊方式3 RS485通訊協(xié)議 串行通訊方式1
第五篇:串口通信實驗報告
實驗三
雙機通信實驗
一、實驗?zāi)康?/p>
UART 串行通信接口技術(shù)應(yīng)用
二、實驗實現(xiàn)的功能
用兩片核心板之間實現(xiàn)串行通信,將按鍵信息互發(fā)到對方數(shù)碼管顯示。
三、系統(tǒng)硬件設(shè)計
(1)單片機的最小系統(tǒng)部分
(2)電源部分
(3)人機界面部分
數(shù)碼管部分
按鍵部分
(4)串口通信部分
四、系統(tǒng)軟件設(shè)計
#include
sbit H1=P3^6;sbit H2=P3^7;sbit L1=P0^5;sbit L2=P0^6;sbit L3=P0^7;
uint m=0,i=0,j;uchar temp,prt;/***y延時函數(shù)***/ void delay(uint k){ uint i,j;
}
/***鍵盤掃描***/ char scan_key(){ H1=0;H2=0;
L1=1;L2=1;L3=1;if(L1==0){ delay(5);if(L1==0){ L1=0;H1=1;H2=1;if(H1==0)} //定義局部變量ij
//外層循環(huán) for(i=0;i { m=1;return(m);} if(H2==0){ m=4;return(m);} } } //KEY1鍵按下 //KEY4鍵按下 if(L2==0){ delay(5);if(L2==0){ L2=0;H1=1;H2=1;if(H1==0) { m=2;return(m);} if(H2==0){ m=5;return(m);} } } //KEY5鍵按下 //KEY2鍵按下 if(L3==0){ delay(5);if(L3==0){ L3=0;H1=1;H2=1;if(H1==0){ m=3; //KEY3鍵按下 } return(m);} if(H2==0){ m=6;return(m);} } } return(0); // KEY6鍵按下 /***主函數(shù)***/ main(){ P1M1=0x00;P1M0=0xff; SCON=0x50;//設(shè)定串行口工作方式1 TMOD=0x20;//定時器1,自動重載,產(chǎn)生數(shù)據(jù)傳輸速率 TH1=0xfd;//數(shù)據(jù)傳輸率為9600 TR1=1;//啟動定時器1 P0&=0xf0;while(1){ //如果有按鍵按下 if(scan_key()){ SBUF=scan_key();//發(fā)送數(shù)據(jù) while(!TI);TI=0;} if(RI){ RI=0;} // // 等待數(shù)據(jù)傳送 清除數(shù)據(jù)傳送標(biāo)志 //是否有數(shù)據(jù)到來 // 清除數(shù)據(jù)傳送標(biāo)志 temp=SBUF; // 將接收到的數(shù)據(jù)暫存在temp中 P1=code0[temp];// 數(shù)據(jù)傳送到P1口輸出 delay(500);} } //延時500ms 五、實驗中遇到的問題及解決方法 (1)串行口和定時器的工作方式設(shè)定是關(guān)鍵,本次是按需傳輸?shù)氖莾晌皇M(jìn)制數(shù),串行口為工作方式1,定時器為8位自動重載;(2)采用P0&=0xf0語句使4個數(shù)碼管靜態(tài)點亮; (3)在發(fā)送和接受過程中,用標(biāo)識位TI和RI來檢測發(fā)送和接受是否完成;(4)在用電腦和單片機進(jìn)行串口通信測試時,電腦的傳世速率一定要和單片機的傳輸速率相等,否則顯示會出現(xiàn)錯誤。 指導(dǎo)老師簽字: 日期: