javax.websocketclient:如何从clientendpoint向serverendpoint发送大型二进制数据

时间:2020-05-01 19:28:15

标签: java websocket jetty java-websocket jsr356

我正在尝试使用码头构建服务器客户端应用程序。我已经安装了一个码头服务器并配置了websockets。在客户端和服务器之间发送文本消息可以正常工作。但是如何从客户端端点发送二进制数据作为输入流。我找不到有关websocket客户端的任何摘要。以下是我尝试过的

ServerEndPoint:

   @OnMessage
   public void handleBinaryMessage(InputStream input, Session session) {

       logger.info("onMessage::inputstream");

       try {

        byte[] buffer = new byte[2048];
        try (OutputStream output = session.getBasicRemote().getSendStream())
        {
            int read;
            while ((read = input.read(buffer)) >= 0)
                output.write(buffer, 0, read);
        }

    } catch (IOException e) {
      e.printStackTrace();
    }

ClientEndpoint:

@OnOpen 
public void onOpen(Session s) throws IOException {
  logger.info("Client Connected ... " + s.getId());
  this.session=s;

  session.getBasicRemote().sendText("Ping from client");

  // size of the file 200~500MB
  File source= new File("/tmp/Setup.exe");

  try(InputStream input = new FileInputStream(source)) {


              session.getAsyncRemote().sendObject(input);


            }

}        

感谢您的帮助

编辑:

我修改了clientendpoint和serverendpoint。尝试以块的形式发送数据,但zip文件是部分文件,有时甚至比源文件小。

源大小:1.5gb 使用流从缓冲区写入数据后:20kb

@ClientEndpoint

   private static void sendFileToRemote(File source) throws FileNotFoundException, IOException {
        // TODO Auto-generated method stub
            Session session=null;
            final WsClient wc = new WsClient("ws://localhost:2714/events/","test");

            session=wc.getSession();

             try (
                        InputStream inputStream = new FileInputStream(source);

                    ) {


                    byte[] chunk = new byte[102400];
                    int chunkLen = 0;
                    while ((chunkLen = inputStream.read(chunk)) != -1) {

                        session.getAsyncRemote().sendBinary(ByteBuffer.wrap(chunk, 0, chunkLen));
                    }

                    } catch (IOException ex) {
                        ex.printStackTrace();
                    }

             }

@serverendpoint

@ServerEndpoint("/events/")
public class EventListener {
    static Logger logger = Logger.getLogger(Initservice.class.getName());
   private OutputStream os=null; 
@OnOpen
    public void Init(Session session) throws FileNotFoundException {

        this.user_session = session;
        logger.info("onOpen:: server" + session.getId());     

        this.os = new FileOutputStream(new File("/tmp/silicon_test.zip"));
        logger.info("instantiate zip files");

    }

@OnMessage
    public void onMessage(Session session, ByteBuffer byteBuffer) throws IOException {
        try {

          os.write(byteBuffer.get());

        } catch(Exception e) {
            close();
            logger.log(Level.SEVERE,"Exception occured in onMessage :: ", e);
            throw e;
        }
    }
}

1 个答案:

答案 0 :(得分:1)

ServerEndpoint中的代码看起来应该可以正常工作,但是在ClientEndpoint中,您仅将文本数据发送到ServerEndpoint,并且只能由配置为接收文本消息的服务器onMessage方法读取。

应该使用方法session.getRemoteEndpoint().sendText(...)而不是使用session.getRemoteEndpoint().sendBinary(...)。这样将以二进制帧而不是文本帧发送数据,您将可以通过服务器的handleBinaryMessage方法接收数据。

对于session.getAsyncRemote().sendObject(input),要完成此工作,您还需要提供Encoder.BinaryEncoder.BinaryStream以便将对象作为二进制数据发送。

编辑:

WebSocket是基于消息的协议,您正在通过多个websocket消息从文件中发送数据。您可以使用session.getBasicRemote().sendBinary(ByteBuffer, boolean)发送部分邮件,并在同一封邮件中发送所有数据。

或者您可以尝试像这样的代码,它可能会更简单。

try (InputStream inputStream = new FileInputStream(source))
{
    try (OutputStream sendStream = session.getBasicRemote().getSendStream())
    {
        inputStream.transferTo(sendStream);
    }
}