java socket接收保證能讀完數據的實例
socket接收保證能讀完數據
// private static byte[] readData(InputStream in,byte[] bData) throws IOException{ // int readLength = in.read(bData); // if(readLength!=bData.length){ // byte[] temp2 = readData(in,new byte[bData.length-readLength]); // System.arraycopy(temp2, 0, bData, readLength, temp2.length); // return bData; // }else{ // return bData; // } // } // private static void readData(InputStream in,byte[] bData) throws IOException{ // readData(in,bData,0,bData.length); // } // private static void readData(InputStream in,byte[] bData,int off,int length) throws IOException{ // int readLength = in.read(bData, off, length); // if(readLength!=length){ // readData(in,bData,readLength+off,length-readLength); // } // } // private static void readData(InputStream in,byte[] bData,int off,int length) throws IOException{ // // while(true){ // int readLength = in.read(bData, off, length); // if(readLength!=length){ // off = readLength+off; // length = length-readLength; // }else{ // break; // } // } // } // private static void readData(InputStream in,byte[] bData,int off,int length) throws IOException{ // int readLength = 0; // do{ // off = readLength+off; // length = length-readLength; // readLength = in.read(bData, off, length); // }while(readLength!=length); // } /** * 最終使用此方法 * @param in 輸入流 * @param bData 讀取數據 * @throws IOException */ private static void readData(InputStream in,byte[] bData) throws IOException{ int off = 0; int length = bData.length; int readLength = 0; do{ off = readLength+off; length = length-readLength; readLength = in.read(bData, off, length); }while(readLength!=length); }
socket接收硬件字節數據並解析
第一次接觸這種類型的項目,在處理數據過程中,發現瞭許多問題,記錄一下,加深記憶。
硬件將數據寫在一個buffer中,傳輸過來的是字節。
一開始我們想到的是按照字節流來接收,但是,C語言中,byte類型沒有符號位,最大值位255,java中byte類型帶有符號位,最大值為127,問題就出現瞭,當接收到的字節數據超過127時,會取第一位為符號位,後幾位補碼,取反再加一變成負數。(處理方法後面有寫到)
後來想偷懶不處理數據的基礎上,考慮用char數組接收。char一共十六位,絕對是可以接收下硬件發來的八位數據的。但是再接收數據的時候,還是出現瞭問題。在對字節流轉變為字符流並保存到char數組中的時候,char類型會自動對數據進行處理。在char類型中,字符所對應的最大十六進制是7F,但硬件傳輸來的數據存在如0X80,0X8D的情況。當char類型接收到大於7F的數據時,無法處理,字符會變成亂碼的格式,數據相對應的也會發生改變。在接收數據的時候就無法正確存儲,更別提後期對數據進行正確處理和校驗瞭。放棄。
最終還是要回到byte接收的方向上。和同事討論瞭下,對於超過java byte類型的數據,進行相應處理後,存放在Int中,即可保證數據正確性。
處理方法:
對byte數組中的數據進行判斷,當為負數時,與0xff相與,並存放在Int數組中,可以保證數據正常
ServerSocket serverSocket; try { serverSocket = new ServerSocket(9090); System.out.println("***等待客戶端連接***"); Socket socket = serverSocket.accept(); InputStream is = socket.getInputStream(); byte[] datas = new byte[500]; int count = is.read(datas); int[] dataFormat=new int[500]; for(int i=0;i<datas.length;i++){ if(datas[i]<0){ dataFormat[i]=datas[i]&0xff; }else{ dataFormat[i]=datas[i]; } } } catch (IOException e) { e.printStackTrace(); }
以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。
推薦閱讀:
- Java進階核心之InputStream流深入講解
- Java Socket上的Read操作阻塞問題詳解
- Java之網絡編程案例講解
- 基於Java實現Socket編程入門
- 淺談java socket的正確關閉姿勢