使用Chrome通过Websockets发送大数据的问题

时间:2017-10-17 02:31:05

标签: java google-chrome websocket

我一直在尝试从头开始为我的聊天程序编写一个java HTTP服务器,它通过websockets发送数据,一切都进行得很顺利,直到我开始尝试使用谷歌浏览器用我的程序发送大图像。我发现当chrome向我的服务器发送多帧数据时,它从不发送适当长度的数据。第一帧始终很好,但是在初始帧之后的所有帧都发送一个明显小于帧所包含数据的数字。例如,当它实际发送超过100千字节时,它将发送一个表示它包含123字节数据的帧。我测试了其他浏览器,如firefox和safari(在我的手机上),他们似乎不会将大数据分成像chrome这样的帧,所以他们没有同样的问题。以下是我的结果的一些截图: 初始框架: enter image description here

以下框架(这是失败的地方): enter image description here

这是我的服务器端代码,它从输入流中获取数据并将其写入文件:

byte[] headerdata = new byte[12];
input.read(headerdata, 0, 2);
int frame = (headerdata[0] & 0b10000000) & 0xFF;
long length = (headerdata[1] & 0b01111111) & 0xFF;
System.out.format("Before: %d, Frame: %d%n", length, frame);
if(length == 127){
    input.read(headerdata, 0, 12);
    length = ((long)(headerdata[0] << 56) & 0xFFFFFFFFFFFFFFFFL | (long)(headerdata[1] << 48) & 0xFFFFFFFFFFFFFFL | (long)(headerdata[2] << 40) & 0xFFFFFFFFFFFFL | (long)(headerdata[3] << 32) & 0xFFFFFFFFFFL | (headerdata[4] << 24) & 0xFFFFFFFFL | (headerdata[5] << 16) & 0xFFFFFFL | (headerdata[6] << 8) & 0xFFFFL | headerdata[7] & 0xFFL)
} else if(length == 126){
    input.read(headerdata, 6, 6);
    length = ((headerdata[6] << 8) & 0xFFFF | headerdata[7] & 0xFF);
}else{
    input.read(headerdata, 8, 4);
}
System.out.println("After: " + length);

int count;
int total = 0;
byte[] buffer = new byte[16384];
boolean done = false;
while(done == false){
    count = input.read(buffer, 0, buffer.length);
    //System.out.println(count);
    for(int i = 0; i < count; i++){
        buffer[i] = (byte)(buffer[i] ^ headerdata[(total & 3) + 8]);
        total++;
    }
    fileoutput.write(buffer, 0, count);
    System.out.format("Count: %d Left: %d%n", count, (length - total));
    if(total >= length){
        done = true;
    }
}

if(frame == 128) break;

关于为什么我可能无法阅读后续帧的长度的任何解释将非常感激(我也有一些怀疑我自己)。感谢您的时间和精力。

1 个答案:

答案 0 :(得分:-1)

我发现缓冲区最终溢出,导致数据被误读或丢失,这就是我从帧中获取错误长度值的原因。我通过添加一个if语句来修复它,该语句检查读取的字节数是否接近该帧的长度值,然后只读取剩余的内容而不是完整的缓冲区大小以阻止溢出。以下是修改后的代码:

byte[] headerdata = new byte[12];
                input.read(headerdata, 0, 2);
                int frame = (headerdata[0] & 0b10000000) & 0xFF;
                long length = (headerdata[1] & 0b01111111) & 0xFF;
                System.out.format("Before: %d, Frame: %d%n", length, frame);
                if(length == 127){
                    input.read(headerdata, 0, 12);
                    length = ((long)(headerdata[0] << 56) & 0xFFFFFFFFFFFFFFFFL | (long)(headerdata[1] << 48) & 0xFFFFFFFFFFFFFFL | (long)(headerdata[2] << 40) & 0xFFFFFFFFFFFFL | (long)(headerdata[3] << 32) & 0xFFFFFFFFFFL | (headerdata[4] << 24) & 0xFFFFFFFFL | (headerdata[5] << 16) & 0xFFFFFFL | (headerdata[6] << 8) & 0xFFFFL | headerdata[7] & 0xFFL);
                }else if(length == 126){
                    input.read(headerdata, 6, 6);
                    length = ((headerdata[6] << 8) & 0xFFFF | headerdata[7] & 0xFF);
                }else{
                    input.read(headerdata, 8, 4);
                }
                System.out.println("After: " + length);

                int count;
                int total = 0;
                byte[] buffer = new byte[16384];
                boolean done = false;
                while(done == false){
                    if(length - total > buffer.length) count = input.read(buffer, 0, buffer.length);
                    else count = input.read(buffer, 0, (int) (length - total));
                    //System.out.println(count);
                    for(int i = 0; i < count; i++){
                        buffer[i] = (byte)(buffer[i] ^ headerdata[(total & 3) + 8]);
                        total++;
                    }
                    fileoutput.write(buffer, 0, count);
                    //System.out.format("Count: %d Left: %d%n", count, (length - total));
                    if(total >= length){
                        done = true;
                    }
                }

                if(frame == 128) break;

我还研究了从套接字读取字节流的不同方法,发现也可以使用DataInputStream.readFully()