文章目錄
前言:
在做手機定位應用軟體,有兩種方法可以實行定位:1.利用基站定位,通過手機擷取基站資訊,再根據該資訊擷取經緯度;2.利用GPS定位,直接擷取經緯度資訊。利弊: 1.基站定位精確度低,誤差範圍估計500米以內(這種情況還是在大城市下的,如果是小城市,估計誤差更大),但是基站定位可以實現處處定位,即不會因為建築物等原因而無法擷取基站資訊。2.GPS定位精確度高,誤差範圍估計10米以內,但是需要在空曠的地方,如果周圍建築物多,或者在室內,沒法擷取GPS資訊。通常用定位這兩種方法都採用,也會增加第三種WIFI定位或者小區定位,這是後話。在windows mobile下獲得CELLID、LAC的途徑有兩條:利用串口發送AT指令或是利用RIL來擷取。RIL(Radio Interface Layer)是微軟自己開發的一個庫,它的程式有固有的特點,在擷取CELLID上,它其實是對第一種方法的封裝,兩者本質是一樣的。但要注意是:串口一旦開啟,就難以關閉,除非重啟機器(可能涉及到底層的中斷),另外並不是所有的裝置都可以取到CELLID。本文章是利用COM口來擷取CELLID,並不保證所有的裝置都支援。測試平台: VS2005 + WM 6.0開發語言: C++定義基站資訊結構體:
typedef struct { char CountryCode[12]; char AreaCode[4]; char NetworkCode[4]; char CellID[4]; } TCREG_DATA;
擷取基站資訊:
void Get_Cellid(void) { char m_sTemp[12] = {0}; strcat(m_sTemp,"COM"); for(int i = 9; i > 0; -- i) { char ch1; _itoa(i,&ch1,10); strcat(m_sTemp,&ch1); strcat(m_sTemp,":"); TCREG_DATA* pData = (TCREG_DATA*)GetCREG(m_sTemp); if(!pData) continue; char szNum1[8] = {0}; char szNum2[8] = {0}; strcpy(szNum1,pData->AreaCode); strcpy(szNum2,pData->CellID); int iLac = (int)strtol(szNum1,NULL,16); int iId = (int)strtol(szNum2,NULL,16); if (iLac && iId) { sprintf(m_sCell.LAC,"%06d", iLac ); sprintf(m_sCell.ID,"%06d", iId ); break; } } }
擷取串口:
char* GetCREG( char * comPort ) { HANDLE hCom; int bufpos; DCB dcb; COMMTIMEOUTS to; DWORD nWritten; DWORD event; DWORD nRead; static char outbuf[20], buf[256]; BYTE comdevcmd[2]= {0x84, 0x00}; WCHAR m_sCom[12] = {0}; mbstowcs(m_sCom,comPort,strlen(comPort)); hCom= CreateFile( m_sCom ,GENERIC_READ|GENERIC_WRITE,0,0,OPEN_EXISTING,0,0); if (hCom==NULL || hCom==INVALID_HANDLE_VALUE) { hCom= NULL; return NULL; } if (!GetCommState(hCom, &dcb)) { return "ERROR:GetCommState Failed"; } dcb.BaudRate= CBR_115200; dcb.ByteSize= 8; dcb.fParity= false; dcb.StopBits= ONESTOPBIT; if (!SetCommState(hCom, &dcb)) { return "ERROR:SetCommState Failed"; } EscapeCommFunction(hCom, SETDTR); EscapeCommFunction(hCom, SETRTS); GetCommTimeouts(hCom, &to); to.ReadIntervalTimeout= 0; to.ReadTotalTimeoutConstant= 200; to.ReadTotalTimeoutMultiplier= 0; to.WriteTotalTimeoutConstant= 20000; to.WriteTotalTimeoutMultiplier= 0; SetCommTimeouts(hCom, &to); if (!SetCommMask(hCom, EV_RXCHAR)) { return "-8"; } DWORD rildevresult=0,nReturned=0; if (!DeviceIoControl (hCom,0xAAAA5679L, comdevcmd, sizeof(comdevcmd),0,0,0,0)) { return "-9"; } bufpos = 0; strcpy(outbuf,"AT+creg=2\r"); if (!WriteFile(hCom, outbuf, 10, &nWritten, NULL)) { return "-10"; } if (nWritten != 10) { return "-11"; } if (!WaitCommEvent(hCom, &event, NULL)) { return "-12"; } while(1) { if (!ReadFile(hCom, buf+bufpos, 256 - bufpos, &nRead, NULL)) { return "-13"; } if (nRead == 0) break; bufpos += nRead; if (bufpos >= 256) break; } strcpy(outbuf,"AT+creg?\r"); if (!WriteFile(hCom, outbuf, 9, &nWritten, NULL)) { return "-14"; } if (nWritten != 9) { return "-15"; } if (!WaitCommEvent(hCom, &event, NULL)) { return "-16"; } while(1) { if (!ReadFile(hCom, buf+bufpos, 256 - bufpos, &nRead, NULL)) { return "-17"; } if (nRead == 0) break; bufpos += nRead; if (bufpos >= 256) break; } puts(buf); rildevresult = 0; if (!EscapeCommFunction(hCom, CLRDTR)) { return "-4"; } if (hCom!=NULL) { CloseHandle(hCom); hCom= NULL; } char* cregResponse = strpbrk( buf, "CREG\0" ); return cregResponse; }
總結: 通過擷取基站資訊,手機可以實現定位,這對於沒有GPS裝置的手機來說是一個不錯的想法。