首页 > 网页制作 > html5

Html5之webcoekt播放JPEG图片流

admin html5 2022-02-05 15:04:44 Html5   webcoekt   JPEG图片流"

一、简介

既然webcoekt是基于tcp连接的,理论上讲所有的浏览器是可以私有协议处理二进制的,如果我们需要播放视频,我们可以将视频数据在后端解码后直接将图片推送到webcoekt前端,然后前端通过websocket接收图片然后将图片显示到img或canvas中即可,当然这个是我自己设想的,也是应该可以做的到了,做到如下需要以下技术支持:

  • 后端直接ffmpeg转码为jpeg图片流
  • 后端定制播放协议包括基本指令如play、stop、pause、faster、slower
  • 后端需要提供websocket支持发送图片流到前端
  • 前端需要接受图片流并显示出来

后端ffmpeg解码这里就不说明了,我有很多库,需要的单独联系我购买,前端的显示jpg流,这里要借助前端显示图片放的做法,使用图片base64数据!前端HTML显示二进制jpeg图片:图片流=>二进制转image的base64编码=>设置到img的src中,如前端代码


    

二进制通过arraybuffer转base64

function arrayBufferToBase64(buffer) {
        var binary = '';
        var bytes = new Uint8Array(buffer);
        var len = bytes.byteLength;
        for (var i = 0; i < len; i++) {
            binary += String.fromCharCode(bytes[i]);
        }
        return window.btoa(binary);
    }

最后显示:

var player = document.getElementById('player');
      var url= arrayBufferToBase64(data);
      player.src='data:image/jpeg;base64,'+url;

当然,还有其他的方式:

var%20wsCtrl%20=%20new%20WebSocket("ws://127.0.0.1/ctrl/"); //Establish%20channel%20code .... var%20wsVideo%20=%20new%20WebSocket("ws://127.0.0.1/video/channel1"); wsVideo.onmessage%20=%20function(evt) { %20%20%20//Method%201 %20%20%20document.getElementById("img1").src%20=%20URL.createObjectURL(evt.data); %20%20%20 %20%20%20//Method%202 %20%20%20var%20read%20=%20new%20FileReader(); %20%20%20read.onload%20=%20function(e) %20%20%20{ %20%20%20%20%20%20document.getElementById("img2").src%20=%20e.target.result; %20%20%20} %20%20%20read.readAsDataURL(evt.data); } ws.onmessage%20=%20function(evt)%20{ %20%20%20%20if(typeof(evt.data)=="string"){ %20%20%20%20%20%20%20%20//textHandler(JSON.parse(evt.data)); %20%20%20%20}else{ %20%20%20%20%20%20%20%20var%20reader%20=%20new%20FileReader(); %20%20%20%20%20%20%20%20reader.onload%20=%20function(evt){ %20%20%20%20%20%20%20%20%20%20%20%20if(evt.target.readyState%20==%20FileReader.DONE){ %20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20var%20url%20=%20evt.target.result; %20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20alert(url); %20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20var%20img%20=%20document.getElementById("imgDiv"); %20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20img.innerHTML%20=%20""; %20%20%20%20%20%20%20%20%20%20%20%20} %20%20%20%20%20%20%20%20} %20%20%20%20%20%20%20%20reader.readAsDataURL(evt.data); %20%20%20%20} };

关于c++的websocket开源工程:websocketpp、QWebSocketServer

二、websocket播放图片流

多说无益,还不如痛痛快快的来一把,为了减低复杂度,我用java的websocket来实现图片流的发送(当然c++的库我也一个实战项目中用过的名为WebSocket的封装的dll工程项目,需要的自行私下购买源码),前端使用一个img标签展示图片,这里说明一下,后台模拟发送图片(这里仅仅是图片,不是流,如果是流直接连续不断发送即可需要ffmpeg转码)

首先前端的代码如下所示:

%20%20%20%20 %20%20%20%20%20%20%20%20 %20%20%20%20%20%20%20%20websocket显示二进制图片流 %20%20%20%20%20%20%20%20 %20%20%20%20%20%20%20%20 %20%20%20%20 %20%20%20%20 %20%20%20%20 %20%20%20%20 %20%20%20%20%20%20%20%20

WebSocket播放图片

%20%20%20%20%20%20%20%20
%20%20%20%20%20%20%20%20请求图片 %20%20%20%20 %20%20%20%20 %20%20%20%20%20%20%20%20var%20websocket%20=%20{ %20%20%20%20%20%20%20%20%20%20%20%20send:%20function%20(str)%20{ %20%20%20%20%20%20%20%20%20%20%20%20} %20%20%20%20%20%20%20%20}; %20%20%20%20%20%20%20%20window.onload%20=%20function%20()%20{ %20%20%20%20%20%20%20%20%20%20%20%20if%20(!'WebSocket'%20in%20window)%20return; %20%20%20%20%20%20%20%20%20%20%20%20webSocketInit(); %20%20%20%20%20%20%20%20}; %20%20%20%20%20%20%20%20function%20webSocketInit()%20{ %20%20%20%20%20%20%20%20%20%20%20%20//%20连接到服务端端点 %20%20%20%20%20%20%20%20%20%20%20%20websocket%20=%20new%20WebSocket("ws://127.0.0.1:8080/image/show"); %20%20%20%20%20%20%20%20%20%20%20%20//%20成功建立连接 %20%20%20%20%20%20%20%20%20%20%20%20websocket.onopen%20=%20function%20()%20{ %20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20console.log("成功连接到服务器"); %20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20websocket.send("成功连接到服务器"); %20%20%20%20%20%20%20%20%20%20%20%20}; %20%20%20%20%20%20%20%20%20%20%20%20//%20接收到消息 %20%20%20%20%20%20%20%20%20%20%20%20websocket.onmessage%20=%20function%20(event)%20{ %20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20//%20文本数据包 %20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20if(typeof(event.data)=="string"){ %20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20//%20JSON.parse(evt.data) %20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20console.log("收到服务端发送的消息:"%20+%20event.data); %20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20//%20图片数据包Blob %20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20}else{ %20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20var%20reader%20=%20new%20FileReader(); %20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20reader.onload%20=%20function(evt){ %20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20if(evt.target.readyState%20==%20FileReader.DONE){ %20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20//%20base64数据 %20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20var%20url%20=%20evt.target.result; %20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20document.getElementById("player").src%20=%20url; %20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20} %20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20} %20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20reader.readAsDataURL(event.data); %20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20} %20%20%20%20%20%20%20%20%20%20%20%20}; %20%20%20%20%20%20%20%20%20%20%20%20//%20连接发生错误 %20%20%20%20%20%20%20%20%20%20%20%20websocket.onerror%20=%20function%20()%20{ %20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20console.log("WebSocket连接发生错误"); %20%20%20%20%20%20%20%20%20%20%20%20}; %20%20%20%20%20%20%20%20%20%20%20%20//%20连接关闭 %20%20%20%20%20%20%20%20%20%20%20%20websocket.onclose%20=%20function%20()%20{ %20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20console.log("WebSocket连接关闭"); %20%20%20%20%20%20%20%20%20%20%20%20}; %20%20%20%20%20%20%20%20%20%20%20%20//%20监听窗口关闭事件,当窗口关闭时,主动关闭websocket连接 %20%20%20%20%20%20%20%20%20%20%20%20window.onbeforeunload%20=%20function%20()%20{ %20%20%20%20%20%20%20%20%20%20%20%20%20%20%20%20websocket.close() %20%20%20%20%20%20%20%20%20%20%20%20}; %20%20%20%20%20%20%20%20} %20%20%20%20

每次点击发送的时候就向后台请求一张图,后台将改图发送出去(我简单的使用websocket群发,可以使用websocket的可变参数将websocket和http关联起来,这个应该很容易我这里不再赘述,不了解的进群讨论)

package%20com.easystudy.controller; import%20java.io.File; import%20java.io.FileInputStream; import%20java.io.IOException; import%20java.io.InputStream; import%20java.net.URL; import%20java.util.Random; import%20org.springframework.web.bind.annotation.GetMapping; import%20org.springframework.web.bind.annotation.RequestMapping; import%20org.springframework.web.bind.annotation.RequestParam; import%20org.springframework.web.bind.annotation.RestController; import%20com.easystudy.websocket.ImgEndPoint; /** %20*%20@文件名称:%20TestController.java %20*%20@功能描述:%20图片流请求请求发送接口(websocket发送图片到web端) %20*%20@版权信息:%20%20www.easystudy.com %20*%20@技术交流:%20%20961179337(QQ群) %20*%20@编写作者:%20%20lixx2048@163.com %20*%20@联系方式:%20%20941415509(QQ) %20*%20@开发日期:%20%202020年9月21日 %20*%20@历史版本:%20V1.0%20 %20*%20@备注信息: %20*/ @RestController @RequestMapping("/test") public%20class%20TestController%20{ %20%20%20%20 %20%20%20%20/** %20%20%20%20%20*%20@功能描述:%20发送请求接口 %20%20%20%20%20*%20@版权信息:%20%20www.easystudy.com %20%20%20%20%20*%20@编写作者:%20%20lixx2048@163.com %20%20%20%20%20*%20@开发日期:%20%202020年9月21日 %20%20%20%20%20*%20@备注信息: %20%20%20%20%20*/ %20%20%20%20@SuppressWarnings("unused") %20%20%20%20@GetMapping("/send") %20%20%20%20public%20String%20reponseMsgToClient(@RequestParam(name="content",%20required%20=%20true)String%20content)%20throws%20Exception{ %20%20%20%20%20%20%20%20System.out.println("开始发送图片数据"); %20%20%20%20%20%20%20%20 %20%20%20%20%20%20%20%20//%20随机选择一张图片发送 %20%20%20%20%20%20%20%20int%20index%20=%20new%20Random().nextInt(4)%20+%201; %20%20%20%20%20%20%20%20String%20imgName%20=%20index%20+%20".jpg"; // 判断图片是否存在 URL url = getClass().getClassLoader().getResource(imgName); File file = new File(url.getFile()); if (!file.exists()) { return "未找到图片!"; } // 创建输入图片流 InputStream in = new FileInputStream(file); if (null == in) { return "打开文件失败!"; } // 读取图片数据 int size = (int)file.length(); byte[] buffer = new byte[ size]; int count = in.read(buffer, 0, size); System.out.println("文件长度:" + size + ", 读取长度:" + count); // 发送图片数据(理论上讲应该只发对端连接的) ImgEndPoint.fanoutMessage(buffer); // 关闭文件流 try { in.close(); } catch (IOException e) { e.printStackTrace(); } // 接口响应 return "消息【" +content+ "】发送成功!"; } }

我这里多一句嘴,如果是音视频应用的录像播放,这里可以使用websocket传输图片流,然后通过计算发送定点的图片流数据到前端来实现自定义的播放器功能(海康萤石云使用的就是websocket播放录像流的,做法类似)

播放效果如下:

到此这篇关于Html5之webcoekt播放JPEG图片流的文章就介绍到这了,更多相关Html5播放JPEG图片流内容请搜索潘少俊衡以前的文章或继续浏览下面的相关文章,希望大家以后多多支持潘少俊衡!

版权声明

本文仅代表作者观点,不代表本站立场。
本文系作者授权发表,未经许可,不得转载。
本文地址:/web/html5/72963.html

留言与评论(共有 0 条评论)
   
验证码:

潘少俊衡

| 桂ICP备2023010378号-4

Powered By EmpireCMS

爱享小站

中德益农

谷姐神农

环亚肥料

使用手机软件扫描微信二维码

关注我们可获取更多热点资讯

感谢潘少俊衡友情技术支持