InputStream Java уничтожает информацию, почему? - PullRequest
1 голос
/ 26 февраля 2012

У меня проблема с сервером, который я создаю.Я сократил проблему до простой пары сервер-клиент в Java.Я посылаю байты от 0 до 255. Проблема в том, что InputStream получает значения от 128 (0x80) до 159 (0x9F) как 63 (0x3F).Это убивает мои бинарные переводы.Почему это может происходить?

Это вывод, который я получаю, посмотрим, как после значения 127 появляется куча 63:

0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29
30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56
 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 8
3 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 10
7 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 12
7 63 63 63 63 63 63 63 63 63 63 63 63 63 63 63 63 63 63 63 63 63 63 63 63 63 63
63 63 63 63 63 63 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 17
5 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 19
5 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 21
5 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 23
5 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 25
5

Это мой код сервера, основанный на примере KnockKnockServer:

import java.net.*;
import java.io.*;

public class Server {
        public static void main(String[] args) throws IOException {

                ServerSocket serverSocket = null;
                try {
                        serverSocket = new ServerSocket(4444);
                } catch (IOException e) {
                        System.err.println("Could not listen on port: 4444.");
                        System.exit(1);
                }

                Socket clientSocket = null;
                try {
                        clientSocket = serverSocket.accept();
                } catch (IOException e) {
                        System.err.println("Accept failed.");
                        System.exit(1);
                }

                OutputStream out = clientSocket.getOutputStream();
                InputStream in = clientSocket.getInputStream();

                while (true)
                {
                    int val = in.read();
                    if (val < 0)
                        continue;

                    System.out.print(val);
                    System.out.print(' ');
                    if (val == 255)
                        break;
                }

                out.close();
                in.close();
                clientSocket.close();
                serverSocket.close();
        }
}

И это код моего клиента, основанный на примере EchoClient:

import java.io.*;
import java.net.*;

public class Client {
        public static void main(String[] args) throws IOException {

                Socket echoSocket = null;
                PrintWriter out = null;
                BufferedReader in = null;

                try {
                        echoSocket = new Socket("localhost", 4444);
                        out = new PrintWriter(echoSocket.getOutputStream(), true);
                        in = new BufferedReader(new InputStreamReader(
                                                                                echoSocket.getInputStream()));
                } catch (UnknownHostException e) {
                        System.err.println("Don't know about host: taranis.");
                        System.exit(1);
                } catch (IOException e) {
                        System.err.println("Couldn't get I/O for "
                                                             + "the connection to: taranis.");
                        System.exit(1);
                }

        for (int i = 0; i < 256; i++)
            out.print((char) i);
        out.flush();

        out.close();
        in.close();
        echoSocket.close();
        }
}

Запустите это и сообщите мне, если у вас возникла та же странная проблема.Пожалуйста, помогите мне.Спасибо!

Ответы [ 4 ]

4 голосов
/ 26 февраля 2012

Из документации PrintWriter на http://docs.oracle.com/javase/1.4.2/docs/api/java/io/PrintWriter.html

Печать отформатированных представлений объектов в поток вывода текста. Этот класс реализует все методы печати, найденные в PrintStream. Он не содержит методов для записи сырых байтов, для которых программа следует использовать не кодированные потоки байтов.

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

1 голос
/ 26 февраля 2012

Причина, по которой вы видите 63s, заключается в том, что ваш PrintWriter пытается преобразовать Unicode в ISO-8859-1, но символы от 128 до 159 не являются допустимыми символами Unicode, поэтому они преобразуются в символ вопросительного знака.

1 голос
/ 26 февраля 2012

Не используйте char для отправки байтов.Java не является C ++, поэтому тип char не является 8-битным.Когда вы исчерпали базовый набор ASCII (127), вы начинаете отправлять Unicode.

Drop PrintWriter и вместо этого используете следующий цикл в вашем клиенте:

    for (int i = 0; i < 256; i++) {
        echoSocket.getOutputStream().write(i);
    }
0 голосов
/ 26 февраля 2012

Вы читаете байты и отправляете символы.В Java они не одинакового размера.Символы 2 байта, и байты хорошо, что они одиночные.Если вы хотите отправлять просто байты, вам придется использовать OutputStream, а не PrintWriter.

Также помните, что байты подписаны от -128 до 128, поэтому вы не можете ожидать, что числа выше 128 будут положительныминомера.Вам придется перевернуть подписанную математику.

...