java - Android:将相机流式传输为 mjpeg

标签 java android streaming mjpeg

在 SO 和 google 搜索了几天后,我开始放弃了,所以我想我不妨在这里发帖。

我正在创建一个应该提供某种视频聊天的安卓应用。由于这应该尽可能接近实时,我阅读了各种协议(protocol)并决定尝试 MJPEG 作为初学者(暂时不关心音频)。

现在流式传输数据让我抓狂。连接建立,应用程序开始将相机预览帧写入流,但 VLC 和 mplayer 都没有开始播放视频。监控连接显示数据正在到达。

正在连接 此代码由异步任务执行,成功通知监听器:

try
    {
        ServerSocket server = new ServerSocket(8080);

        socket = server.accept();

        server.close();

        Log.i(TAG, "New connection to :" + socket.getInetAddress());

        stream = new DataOutputStream(socket.getOutputStream());
        prepared = true;
    }
    catch (IOException e)
    {
        Log.e(TAG, e.getMessage();
    }

在我的 PC 上,我执行“mplayer http://tabletIP:8080”并且平板电脑注册了一个连接(从而启动了我的流媒体和相机预览)。这也适用于 VLC。

流式传输这会将 header 写入流:

if (stream != null)
{
    try
    {
        // send the header
        stream.write(("HTTP/1.0 200 OK\r\n" +
                      "Server: iRecon\r\n" +
                      "Connection: close\r\n" +
                      "Max-Age: 0\r\n" +
                      "Expires: 0\r\n" +
                      "Cache-Control: no-cache, private\r\n" + 
                      "Pragma: no-cache\r\n" + 
                      "Content-Type: multipart/x-mixed-replace; " +
                      "boundary=--" + boundary +
                      "\r\n\r\n").getBytes());

        stream.flush();

        streaming = true;
    }
    catch (IOException e)
    {
        notifyOnEncoderError(this, "Error while writing header: " + e.getMessage());
        stop();
    }
}

之后通过 Camera.onPreviewFrame() 回调触发流式传输:

@Override
public void onPreviewFrame(byte[] data, Camera camera)
{
    frame = data;

    if (streaming)
        mHandler.post(this);
}

@Override
public void run()
{
    // TODO: cache not filling?
    try
    {
                    // buffer is a ByteArrayOutputStream
        buffer.reset();

        switch (imageFormat)
        {
            case ImageFormat.JPEG:
                // nothing to do, leave it that way
                buffer.write(frame);
                break;

            case ImageFormat.NV16:
            case ImageFormat.NV21:
            case ImageFormat.YUY2:
            case ImageFormat.YV12:
                new YuvImage(frame, imageFormat, w, h, null).compressToJpeg(area, 100, buffer);
                break;

            default:
                throw new IOException("Error while encoding: unsupported image format");
        }

        buffer.flush();

        // write the content header
        stream.write(("--" + boundary + "\r\n" + 
                      "Content-type: image/jpg\r\n" + 
                      "Content-Length: " + buffer.size() + 
                      "\r\n\r\n").getBytes());

        // Should omit the array copy
        buffer.writeTo(stream);

        stream.write("\r\n\r\n".getBytes());
        stream.flush();
    }
    catch (IOException e)
    {
        stop();
        notifyOnEncoderError(this, e.getMessage());
    }
}

没有抛出异常。 mHandler 在它自己的 HandlerThread 中运行。只是为了确保我尝试使用 AsyncTask,但无济于事(顺便说一句,这更好吗?)。

编码帧在android端很好,我将它们保存为jpg文件并且可以打开它们。

我的猜测是我必须以某种方式对数据进行聚类,或者必须为套接字或其他东西设置一些选项,但是....好吧,我被卡住了。

tl;dr: VLC 没有播放流,mplayer 说'缓存没有填充',问题可能在最后一个代码段,需要帮助~:)

谢谢!

最佳答案

我明白了。好像我的 http-/content-headers 搞砸了。正确的标题应该是:

stream.write(("HTTP/1.0 200 OK\r\n" +
                          "Server: iRecon\r\n" +
                          "Connection: close\r\n" +
                          "Max-Age: 0\r\n" +
                          "Expires: 0\r\n" +
                          "Cache-Control: no-store, no-cache, must-revalidate, pre-check=0, post-check=0, max-age=0\r\n" +
                          "Pragma: no-cache\r\n" + 
                          "Content-Type: multipart/x-mixed-replace; " +
                          "boundary=" + boundary + "\r\n" +
                          "\r\n" +
                          "--" + boundary + "\r\n").getBytes());

stream.write(("Content-type: image/jpeg\r\n" +
                      "Content-Length: " + buffer.size() + "\r\n" +
                      "X-Timestamp:" + timestamp + "\r\n" +
                      "\r\n").getBytes());

buffer.writeTo(stream);
stream.write(("\r\n--" + boundary + "\r\n").getBytes());

当然,在哪里设置边界是你自己的选择。也可能有一些字段是可选的(例如大多数在 Cache-Control 中),但这是有效的,直到现在我都懒得把它们去掉。重要的部分是记住换行符(\r\n 东西)...

关于java - Android:将相机流式传输为 mjpeg,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14815103/

相关文章:

java - 在 JAVA 中使用 Kafka JSON 输入格式进行 Spark 结构化流式传输

android - Eclipse 声称我缺少 </application> 标记;不知道出了什么问题

android - Android中的多个动画

groovy - 使用Groovy在Hadoop流中包含jar文件

c# - 如何安全地读取 ASP.NET 中的流?

java - 使用 key=String 和 value=ArrayList<Double> 创建一个映射

java - 要求输入的密码必须包含大写字母和数字?

java - 从另一个 jar 文件调用方法

android - Mockito/电源 Mockito : unable to get expected output when mocking method of LayoutParams in android

iphone - 使用缓冲/流媒体 block 播放远程 mp3 文件