Sockets 直播(套接字)-如何同步音频和视频?

Sockets 直播(套接字)-如何同步音频和视频?,sockets,streaming,video-streaming,live-streaming,Sockets,Streaming,Video Streaming,Live Streaming,1-哪一个更适合用于流式视频?TCP或UDP套接字,为什么 2-流媒体直播时,音频和视频分别来自服务器,因此如何确保我在设备上显示的视频和播放的音频同步?我会选择UDP。然而,这取决于你想要什么。UDP将丢弃数据包而不是等待(TCP)。取舍是你想要一个稳定的,但有时缓慢和昂贵的,还是一个高效的,但有时可能无法交付。您可以选择如何实现它以及如何使用它。我不久前编写了一个语音聊天应用程序,TCP是不可能的,如果您正在寻找接近实时的数据流,UDP多播确实是唯一的选择。不过,通过UDP进行流式传输存在两

1-哪一个更适合用于流式视频?TCP或UDP套接字,为什么


2-流媒体直播时,音频和视频分别来自服务器,因此如何确保我在设备上显示的视频和播放的音频同步?

我会选择UDP。然而,这取决于你想要什么。UDP将丢弃数据包而不是等待(TCP)。取舍是你想要一个稳定的,但有时缓慢和昂贵的,还是一个高效的,但有时可能无法交付。您可以选择如何实现它以及如何使用它。

我不久前编写了一个语音聊天应用程序,TCP是不可能的,如果您正在寻找接近实时的数据流,UDP多播确实是唯一的选择。不过,通过UDP进行流式传输存在两个主要问题:

  • 丢弃的数据包。就音频而言,这是一个非常简单的修复方法。通常丢弃的数据包不会产生明显的差别(数据包被单独解压缩)。然而,在处理视频时,尤其是在视频被压缩(通常是压缩的)的情况下,至少可以说,找到一个合适的传输协议来确保网络的健壮性是一项艰巨的任务,特别是如果你是从头开始做这件事的话。视频帧被分成不同的数据包。弄清楚当这些数据包丢失时该怎么办是很困难的
  • 音频和视频之间的同步。这是一个非常棘手的问题,我建议您阅读RTSP(实时流协议)等协议。这不是一项容易的任务,但这里有一些介绍性信息:-有时它是通过发送单独的同步数据包(一些协议通过TCP发送这些数据包)来完成的,这些数据包告诉播放器声音应该如何与视频匹配

  • 今天,甚至连youtube上的HTTP流。。。下面是一个nodejs应用程序,它将文件流式传输到浏览器客户端。。。将其用作实时视频流与音频完美同步的起点

    // usage 
    // do following on server side (your laptop running nodejs)
    // node this_file.js
    //
    // then once above is running point your browser at
    //    http://localhost:8888
    //
    // of course your browser could be on your mobile or own custom app
    
    
    
    var http = require('http'),
        fs = require('fs'),
        util = require('util');
    
    var path = "/path/to/audio/or/video/file/local/to/server/cool.mp4"; // put any audio or video file here
    
    var port = 8888;
    var host = "localhost";
    
    http.createServer(function (req, res) {
    
      var stat = fs.statSync(path);
      var total = stat.size;
    
      if (req.headers.range) {   // meaning client (browser) has moved the forward/back slider
                                             // which has sent this request back to this server logic ... cool
        var range = req.headers.range;
        var parts = range.replace(/bytes=/, "").split("-");
        var partialstart = parts[0];
        var partialend = parts[1];
    
        var start = parseInt(partialstart, 10);
        var end = partialend ? parseInt(partialend, 10) : total-1;
        var chunksize = (end-start)+1;
        console.log('RANGE: ' + start + ' - ' + end + ' = ' + chunksize);
    
        var file = fs.createReadStream(path, {start: start, end: end});
        res.writeHead(206, { 'Content-Range': 'bytes ' + start + '-' + end + '/' + total, 'Accept-Ranges': 'bytes', 'Content-Length': chunksize, 'Content-Type': 'video/mp4' });
        file.pipe(res);
    
      } else {
    
        console.log('ALL: ' + total);
        res.writeHead(200, { 'Content-Length': total, 'Content-Type': 'video/mp4' });
        fs.createReadStream(path).pipe(res);
      }
    }).listen(port, host);
    
    console.log("Server running at http://" + host + ":" + port + "/");