Bluetooth-соединение;не может отправлять строки правильно - PullRequest
6 голосов
/ 06 января 2012

У меня проблемы с моей программой, когда мне нужно отправить строки из моего серверного bluetooth-сокета в мой клиентский bluetooth-сокет.Все работает хорошо, пока я отправляю только одну строку за раз (например, в чате), но если мне нужно за короткое время написать больше строк (для обмена информацией), строки не будут отделены от клиентского кода,Например, если я отправляю «FirstUser» и сразу после этого «SecondUser», клиент не читает «FirstUser», а затем «SecondUser».Он будет читать «FirstUserSecondUser».Как я могу избежать такого поведения?

Редактировать: Если я оставлю Thread в спящем режиме до того, как он сможет отправить новое сообщение, он прочитает правильные строки, но это решение не работает нормально для моих нужд.

Код сервера: отправка всем клиентам (отредактировано)

   public synchronized void sendToAll(String message)
{   
    try {
        Thread.sleep(100);
    } catch (InterruptedException e1) {
        e1.printStackTrace();
    }

    publishProgress(message);
    for(OutputStream writer:outputList) {
        try {
            writer.write(message.getBytes());
            writer.flush();
        } catch (IOException e) {
            System.out.println("Some-Error-Code");
        }
    }
}

Код сервера: чтение с клиента:

   public void run() {
    String nachricht;
    int numRead;
    byte[] buffer = new byte[1024];
    while (runningFlag) 
    {
        try {
            if((numRead = inputStream.read(buffer)) >= 0) { 
                nachricht = new String(buffer, 0, numRead);
                serverThread.handleMessage(nachricht); 
            }
            }
             catch (IOException e) {
                this.cancel();
                e.printStackTrace();
            }
    } 
} 

Код клиента: чтение с сервера (отредактировано))

@Override
    protected Void doInBackground(Integer... ints) {    
        String nachricht = new String();
        byte[] buffer = new byte[1024];
        int numRead;
        while (runningFlag) 
        {
            try {
                if(((numRead = inputStream.read(buffer)) >= 0)) { 
                    nachricht = new String(buffer, 0, numRead);
                    publishProgress(nachricht);
                }
            }
             catch (IOException e) {
                clientGame.finish();
                e.printStackTrace();
            }                      
        }
        return null;
}

Код клиента: запись на сервер

public synchronized void write(String nachricht)
    {
        try {
            Thread.sleep(100);
        } catch (InterruptedException e1) {
            e1.printStackTrace();
        }

        try {
            outputStream.write(nachricht.getBytes());
            outputStream.flush();
        } catch (IOException e) {
            this.cancel();
            e.printStackTrace();
        }
    }

Я ценю любую маленькую помощь :).

1 Ответ

8 голосов
/ 13 января 2012

Вы должны инкапсулировать свой элемент данных, чтобы избежать конкатенации.Это означает, что вы должны написать и прочитать весь элемент данных, прежде чем продолжить.

У вас должны быть некоторые вспомогательные методы, чтобы сделать это вместо непосредственного использования методов OutputStream и InputStream:

public static void writeItem(OutputStream out, String s) throws IOException
{
    // Get the array of bytes for the string item:
    byte[] bs = s.getBytes(); // as bytes
    // Encapsulate by sending first the total length on 4 bytes :
    //   - bits 7..0 of length
    out.write(bs.length);      // modulo 256 done by write method
    //   - bits 15..8 of length
    out.write(bs.length>>>8);  // modulo 256 done by write method
    //   - bits 23..16 of length
    out.write(bs.length>>>16); // modulo 256 done by write method
    //   - bits 31..24 of length
    out.write(bs.length>>>24); // modulo 256 done by write method
    // Write the array content now:
    out.write(bs); // Send the bytes
    out.flush();
}

public static String readItem(InputStream in) throws IOException
{
    // first, read the total length on 4 bytes
    //  - if first byte is missing, end of stream reached
    int len = in.read(); // 1 byte
    if (len<0) throw new IOException("end of stream");
    //  - the other 3 bytes of length are mandatory
    for(int i=1;i<4;i++) // need 3 more bytes:
    {
        int n = in.read();
        if (n<0) throw new IOException("partial data");
        len |= n << (i<<3); // shift by 8,16,24
    }
    // Create the array to receive len bytes:
    byte[] bs = new byte[len];
    // Read the len bytes into the created array
    int ofs = 0;
    while (len>0) // while there is some byte to read
    {
        int n = in.read(bs, ofs, len); // number of bytes actually read
        if (n<0) throw new IOException("partial data");
        ofs += n; // update offset
        len -= n; // update remaining number of bytes to read
    }
    // Transform bytes into String item:
    return new String(bs);
}

Затем вы используете эти методы как для сервера, так и для клиента, чтобы читать и записывать элементы String.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...