本文为 Dennis Gao 原创技术文章,发表于博客园博客,未经作者本人允许禁止任何形式的转载。

许久以前写了篇文章《基于.NET打造IP智能网络视频监控系统》,记录和介绍了自己几年来积累和演练的一个系统。发现几个月过去了,没有任何进展。

目前已经实现了 UDP+RTP 方式在不同物理机之间的媒体流传输。当然,由于没有基于 .NET 的媒体流压缩实现,所以直接传输的裸图 Bitmap。不过要求不高,帧率低一些,机器性能强一些,看着也很流畅。

能在桌面客户端上看到视频图像的功能已经完成了。下面需要考虑,如何通过浏览器来查看视频。

在不考虑使用 Flash、ActiveX 的条件下,貌似只能选择 MJPEG 方式。目前还没有研究在 HTML5 下视频是如何处理的,以后有时间可以探索。

目录

什么是 MJPEG?

看这里:

当然,我主要关注 MJPEG over HTTP 这段。

M-JPEG over HTTP
HTTP streaming separates each image into individual HTTP replies on a specified marker. RTP streaming creates packets of a sequence of JPEG images that can be received by clients such as QuickTime or VLC.
In response to a GET request for a MJPEG file or stream, the server streams the sequence of JPEG frames over HTTP. A special mime-type content type multipart/x-mixed-replace;boundary=<boundary-name> informs the client to expect several parts (frames) as an answer delimited by <boundary-name>. This boundary name is expressly disclosed within the MIME-type declaration itself. The TCP connection is not closed as long as the client wants to receive new frames and the server wants to provide new frames. Two basic implementations of a M-JPEG streaming server are cambozola and MJPG-Streamer. The more robust ffmpeg-server also provides M-JPEG streaming support.

也就是说,建立 HTTP 连接后,服务端在 Response 消息中先发一个数据头 Header 告诉客户端,我后面的都是 JPEG 图片。图片之间使用 boundary-name 来区分,每个图片前都有自己的数据头来描述图片数据长度。

MJPEG数据头定义

     /// <summary>
/// 流头部
/// </summary>
public string StreamHeader
{
get
{
return "HTTP/1.1 200 OK" +
"\r\n" +
"Content-Type: multipart/x-mixed-replace; boundary=" + this.Boundary +
"\r\n";
}
}
     /// <summary>
/// 图片头部
/// </summary>
public string PayloadHeader
{
get
{
return "\r\n" +
this.Boundary +
"\r\n" +
"Content-Type: image/jpeg" +
"\r\n" +
"Content-Length: " + _contentLengthString +
"\r\n\r\n";
}
}

这里的 Boundary 可以是任意字符串,只要你觉得唯一并能区分即可,比如我可以设置为“--dennisgao”。

服务器端实现

Http 服务器其实就是个支持 Tcp 连接的服务器。

 private AsyncTcpServer _server;

 _server = new AsyncTcpServer(Port);
_server.Encoding = Encoding.ASCII;
     public void Start()
{
_server.Start();
_server.ClientConnected += new EventHandler<TcpClientConnectedEventArgs>(OnClientConnected);
_server.ClientDisconnected += new EventHandler<TcpClientDisconnectedEventArgs>(OnClientDisconnected);
} public void Stop()
{
_server.Stop();
_server.ClientConnected -= new EventHandler<TcpClientConnectedEventArgs>(OnClientConnected);
_server.ClientDisconnected -= new EventHandler<TcpClientDisconnectedEventArgs>(OnClientDisconnected);
} private void OnClientConnected(object sender, TcpClientConnectedEventArgs e)
{
_clients.AddOrUpdate(e.TcpClient.Client.RemoteEndPoint.ToString(), e.TcpClient, (n, o) => { return e.TcpClient; });
} private void OnClientDisconnected(object sender, TcpClientDisconnectedEventArgs e)
{
TcpClient clientToBeThrowAway;
_clients.TryRemove(e.TcpClient.Client.RemoteEndPoint.ToString(), out clientToBeThrowAway);
}

这里可以参考两篇文章中的实现。

发送图片数据

首先要保证,对一个HTTP连接只能发一次流头,因为后面是接连不断的图片数据。当然,发点别的数据客户端也不会解码。

     private void WriteStreamHeader()
{
if (_clients.Count > )
{
foreach (var item in _clients)
{
Logger.Debug(string.Format(CultureInfo.InvariantCulture,
"Writing stream header, {0}, {1}{2}", item.Key, Environment.NewLine, StreamHeader)); _server.SyncSend(item.Value, StreamHeader); TcpClient clientToBeThrowAway;
_clients.TryRemove(item.Key, out clientToBeThrowAway);
}
}
}

发送图片数据时,要保证图片的前面是图片头和长度信息,数据尾部要有换行符。

     private void WritePayload(byte[] payload)
{
string payloadHeader = this.PayloadHeader.Replace(_contentLengthString, payload.Length.ToString());
string payloadTail = "\r\n"; Logger.Debug(string.Format(CultureInfo.InvariantCulture,
"Writing payload header, {0}{1}", Environment.NewLine, payloadHeader)); byte[] payloadHeaderBytes = _server.Encoding.GetBytes(payloadHeader);
byte[] payloadTailBytes = _server.Encoding.GetBytes(payloadTail);
byte[] packet = new byte[payloadHeaderBytes.Length + payload.Length + payloadTail.Length];
Buffer.BlockCopy(payloadHeaderBytes, , packet, , payloadHeaderBytes.Length);
Buffer.BlockCopy(payload, , packet, payloadHeaderBytes.Length, payload.Length);
Buffer.BlockCopy(payloadTailBytes, , packet, payloadHeaderBytes.Length + payload.Length, payloadTailBytes.Length); _server.SendToAll(packet);
}

结果演示

在可以成功发送流信息和图片信息后,就可以在浏览器上查看视频了。当然,我用的 Google Chrome 。IE10 好奇葩,它会把流当成文件不停的下载,搞不懂。

远程访问

局域网内的无线设备,只要浏览器支持 MJPEG ,均可以查看视频。我测试了 iPad 上的 Safari 是可以的,但 Chrome 却直接解析成乱码。

当然,如果在路由器上配置转发规则,就可以在外网访问了。

完整代码

   public class MJpegStreamingServer
{
private static string _contentLengthString = "__PayloadHeaderContentLength__";
private AsyncTcpServer _server;
private ConcurrentDictionary<string, TcpClient> _clients; public MJpegStreamingServer(int listenPort)
: this(listenPort, "--dennisgao")
{
} public MJpegStreamingServer(int listenPort, string boundary)
{
Port = listenPort;
Boundary = boundary; _server = new AsyncTcpServer(Port);
_server.Encoding = Encoding.ASCII;
_clients = new ConcurrentDictionary<string, TcpClient>();
} /// <summary>
/// 监听的端口
/// </summary>
public int Port { get; private set; } /// <summary>
/// 分隔符
/// </summary>
public string Boundary { get; private set; } /// <summary>
/// 流头部
/// </summary>
public string StreamHeader
{
get
{
return "HTTP/1.1 200 OK" +
"\r\n" +
"Content-Type: multipart/x-mixed-replace; boundary=" + this.Boundary +
"\r\n";
}
} /// <summary>
/// 图片头部
/// </summary>
public string PayloadHeader
{
get
{
return "\r\n" +
this.Boundary +
"\r\n" +
"Content-Type: image/jpeg" +
"\r\n" +
"Content-Length: " + _contentLengthString +
"\r\n\r\n";
}
} public void Start()
{
_server.Start();
_server.ClientConnected += new EventHandler<TcpClientConnectedEventArgs>(OnClientConnected);
_server.ClientDisconnected += new EventHandler<TcpClientDisconnectedEventArgs>(OnClientDisconnected);
} public void Stop()
{
_server.Stop();
_server.ClientConnected -= new EventHandler<TcpClientConnectedEventArgs>(OnClientConnected);
_server.ClientDisconnected -= new EventHandler<TcpClientDisconnectedEventArgs>(OnClientDisconnected);
} private void OnClientConnected(object sender, TcpClientConnectedEventArgs e)
{
_clients.AddOrUpdate(e.TcpClient.Client.RemoteEndPoint.ToString(), e.TcpClient, (n, o) => { return e.TcpClient; });
} private void OnClientDisconnected(object sender, TcpClientDisconnectedEventArgs e)
{
TcpClient clientToBeThrowAway;
_clients.TryRemove(e.TcpClient.Client.RemoteEndPoint.ToString(), out clientToBeThrowAway);
} public void Write(Image image)
{
if (_server.IsRunning)
{
byte[] payload = BytesOf(image); WriteStreamHeader();
WritePayload(payload);
}
} private void WriteStreamHeader()
{
if (_clients.Count > )
{
foreach (var item in _clients)
{
Logger.Debug(string.Format(CultureInfo.InvariantCulture,
"Writing stream header, {0}, {1}{2}", item.Key, Environment.NewLine, StreamHeader)); _server.SyncSend(item.Value, StreamHeader); TcpClient clientToBeThrowAway;
_clients.TryRemove(item.Key, out clientToBeThrowAway);
}
}
} private void WritePayload(byte[] payload)
{
string payloadHeader = this.PayloadHeader.Replace(_contentLengthString, payload.Length.ToString());
string payloadTail = "\r\n"; Logger.Debug(string.Format(CultureInfo.InvariantCulture,
"Writing payload header, {0}{1}", Environment.NewLine, payloadHeader)); byte[] payloadHeaderBytes = _server.Encoding.GetBytes(payloadHeader);
byte[] payloadTailBytes = _server.Encoding.GetBytes(payloadTail);
byte[] packet = new byte[payloadHeaderBytes.Length + payload.Length + payloadTail.Length];
Buffer.BlockCopy(payloadHeaderBytes, , packet, , payloadHeaderBytes.Length);
Buffer.BlockCopy(payload, , packet, payloadHeaderBytes.Length, payload.Length);
Buffer.BlockCopy(payloadTailBytes, , packet, payloadHeaderBytes.Length + payload.Length, payloadTailBytes.Length); _server.SendToAll(packet);
} private byte[] BytesOf(Image image)
{
MemoryStream ms = new MemoryStream();
image.Save(ms, System.Drawing.Imaging.ImageFormat.Jpeg); byte[] payload = ms.ToArray(); return payload;
}
}

本文为 Dennis Gao 原创技术文章,发表于博客园博客,未经作者本人允许禁止任何形式的转载。

C#开源实现MJPEG流传输的更多相关文章

  1. openwrt,mjpeg流,wifi摄像头与APP联动,拍照、录像

    最近公司好忙,自己主管的产品又忙着上线,好久都没更新博客了. 最近产品在做一款wifi摄像头,摄像头与手机同时连接在一个局域网内,即可实现摄像头图像在手机显示,并且拍照录像等功能 mjpeg是一张一张 ...

  2. RTSP RTSP(Real Time Streaming Protocol),RFC2326,实时流传输协议,是TCP/IP协议体系中的一个应用层协议

    RTSP 编辑 RTSP(Real Time Streaming Protocol),RFC2326,实时流传输协议,是TCP/IP协议体系中的一个应用层协议,由哥伦比亚大学.网景和RealNetwo ...

  3. Kafka 在行动:7步实现从RDBMS到Hadoop的实时流传输

    原文:https://coyee.com/article/11095-kafka-in-action-7-steps-to-real-time-streaming-from-rdbms-to-hado ...

  4. EzHttp 流传输调用代码示例

    EzHttp框架提供的内置接口,用于文件流等传输 流传输调用代码示例 内置接口: public interface IEzStreamHandler { Task<byte[]> GetD ...

  5. [转]开源实时视频码流分析软件:VideoEye

    原文太长了,就直接贴上链接,以便大家学习. 引文链接:开源实时视频码流分析软件:VideoEye

  6. .net WebApi 批量文件进行压缩zip以二进制流传输至前端(Vue)下载

    前言:最近接了个项目,需要进行将服务端生成的文件进行打包压缩供前端下载,百度查了下资料,决定采用SharpZipLib C#开园的压缩解压库进行服务器文件压缩,在实现过程,郁闷的是前端接收下载下来的压 ...

  7. 开源实时视频码流分析软件:VideoEye

    本文介绍一个自己做的码流分析软件:VideoEye.为什么要起这个名字呢?感觉这个软件的主要功能就是对"视频"进行"分析".而分析是要用眼睛来看的,因此取了&q ...

  8. 基于TCP协议的项目架构之Socket流传输的实现

    项目背景  某银行的影像平台由于使用时间长,服务器等配置原因,老影像系统满足不了现在日益增长的数据量的需求,所以急需要升级改造.传统的影像平台使用的是Oracle数据库和简单的架构来存储数据(视频.图 ...

  9. HttpUrlConnection流传输问题(正确传输包含中文的JSON字符串)

    目前在写一个功能,主要是使用 HttpURLConnection 发送http请求调用外部接口.本来一切正常的,可是在发送post请求上传数据给服务端时,服务端返回错误信息:获取的JSON请求是乱码的 ...

随机推荐

  1. Windows server 2012 添加中文语言包(英文转为中文)(离线)

    Windows server 2012 添加中文语言包(英文转为中文)(离线) 相关资料: 公司环境:亚马孙aws虚拟机 英文版Windows2012 中文SQL Server2012安装包,需要安装 ...

  2. Javascript - Promise学习笔记

    最近工作轻松了点,想起了以前总是看到的一个单词promise,于是耐心下来学习了一下.   一:Promise是什么?为什么会有这个东西? 首先说明,Promise是为了解决javascript异步编 ...

  3. Hyper-V2:向VM增加虚拟硬盘

    使用Hyper-V创建VM,在VM成功安装OS之后,发现VM只有一个逻辑盘C,用于存储VM的操作系统.在产品环境中,需要向VM增加虚拟硬盘,便于将数据单独存储在不同的逻辑盘符中.在Hyper-V中,分 ...

  4. SQL:指定名称查不到数据的衍伸~空格 换行符 回车符的批量处理

    异常处理汇总-数据库系列  http://www.cnblogs.com/dunitian/p/4522990.html 先看看啥情况 复制查询到的数据,粘贴一下看看啥情况 那就批量处理一下~ 就这样 ...

  5. 如何进行python性能分析?

    在分析python代码性能瓶颈,但又不想修改源代码的时候,ipython shell以及第三方库提供了很多扩展工具,可以不用在代码里面加上统计性能的装饰器,也能很方便直观的分析代码性能.下面以我自己实 ...

  6. git克隆项目到本地&&全局安装依赖项目&&安装依赖包&&启动服务

     一.安装本地开发环境 1.安装本项目 在需要保存到本地的项目的文件夹,进入到文件夹里点击右键,bash here,出现下图: 2.安装依赖项目  3.安装依赖包(进入到命令行) # 安装依赖包 $ ...

  7. .NET 基础 一步步 一幕幕[面向对象之对象和类]

    对象和类 本篇正式进入面向对象的知识点简述: 何为对象,佛曰:一花一世界,一木一浮生,一草一天堂,一叶一如来,一砂一极乐,一方一净土,一笑一尘缘,一念一清静.可见"万物皆对象". ...

  8. 如何使用SHOW WARNINGS?

    1.show warnings:显示上一个语句的错误.警告以及注意.如图:

  9. mysql 赋予用户权限

    # 赋予权限MySQL> grant 权限参数 on 数据库名称.表名称 to 用户名@用户地址 identified by '用户密码'; # 立即生效权限MySQL> flush pr ...

  10. 如何让我们的PHP在Jexus中跑起来

    最近一段时间,经常看到不少的朋友在问,应该怎么设置才能够让Jexus支持PHP.其实,Jexus在很早之前就已经是可以支持PHP,像Apache或Nginx一样充当PHP的Web服务器的.不过由于没有 ...