使用javacv对摄像头视频转码并实现播放

2024-05-31 22:04

本文主要是介绍使用javacv对摄像头视频转码并实现播放,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

要实现Java接受RTSP流解码,并推送给前端实现播放实时流,可以使用一些流媒体处理库,比如JavaCV或者FFmpeg等。以下是一个简单的示例代码:

1.控制层方面的

根据视频rtsp流链接打开转换,通过响应写出流到前台使用flvjs播放视频

 一个播放器销毁时,将对应转换器线程暂停

@RestController
@RequestMapping("flv")
public class FlvVideoController {@Autowiredprivate IFLVService iflvService;/*** 根据视频rtsp流链接打开转换,通过响应写出流到前台使用flvjs播放视频* @param url 视频链接* @param httpServletResponse 响应请求* @author xufeng*/@RequestMapping(method = RequestMethod.GET, value = "/open/{param}")public void open(@PathVariable(value = "param") String url, HttpServletResponse httpServletResponse) {try {System.out.println("==url="+url);if(StringUtils.isBlank(url)) {url="";}BASE64Decoder base64Decoder = new BASE64Decoder();//获取当前登录用户主键String userId = "1";//String userId = UserContext.getCurrentUser().getId();//为保持url长度,需要先对前端传来的url进行base64解码,再调用flvService接口iflvService.open(new String(base64Decoder.decodeBuffer(url)), userId, httpServletResponse);} catch (Exception e) {e.printStackTrace();}}/*** 一个播放器销毁时,将对应转换器线程暂停* @author xufeng* @param videoUrl 视频流链接* @return EosDataTransferObject*/@ResponseBody@RequestMapping(method = RequestMethod.GET, value = "/closeTransThread")public JsonResult closeTransThread(/*@RequestParam(value = "videoUrl") String videoUrl*/) {try {String videoUrl="rtsp://admin:xxx:554/cam/realmonitor?channel=1&subtype=0";//视频流链接为空直接返回if (StringUtils.isBlank(videoUrl)) {return new JsonResult();}//获取当前登录用户主键String userId = "1";//String userId = UserContext.getCurrentUser().getId();//使用主键获取当前所有转换器ConcurrentHashMap<String, Converter> conMaps = ConverterRegistration.getAllConverters(userId);//通过视频流链接取对应的转换器Converter converter = ConverterRegistration.isExist(videoUrl, conMaps);if (null != converter) {//暂停转换器线程,1分钟无新线程创建,该线程即被销毁converter.exit();}} catch (Exception e) {e.printStackTrace();}return new JsonResult();}}

2.视频流转换接口

public interface IFLVService {/*** 打开一个流地址** @param url rtsp流链接* @param userId 用户主键* @param response 响应请求* @author xufeng*/void open(String url,String userId, Object response);}
FLV流转换
@Service("flvService")
public class FLVService implements IFLVService {/*** 打开一个流地址,写入response* @param url 流地址* @param userId 用户主键* @param object HttpServletResponse* @author xufeng*/@Overridepublic void open(String url, String userId, Object object) {//创建转换器线程并启动Converter c = ConverterRegistration.open(url, userId);//UUID设置一个key值String key = UUID.randomUUID().toString();//创建输出字节流OutputStreamEntity outEntity = new OutputStreamEntity(new ByteArrayOutputStream(), System.currentTimeMillis(),key);//添加流输出System.out.println("==添加流输出=="+key);c.addOutputStreamEntity(key, outEntity);try {HttpServletResponse response = (HttpServletResponse) object;//设置响应头response.setContentType("video/x-flv");response.setHeader("Connection", "keep-alive");response.setStatus(HttpServletResponse.SC_OK);//写出缓冲信息,并清空response.flushBuffer();//循环读取outEntity里的流输出给前台System.out.println(c.getConverterState()+"==(response)循环读取outEntity里的流输出给前台==");readFlvStream(c, outEntity, response);} catch (Exception e) {//客户端长连接过程中被异常关闭,关闭该长连接对应的转换器线程c.exit();e.printStackTrace();//c.removeOutputStreamEntity(outEntity.getKey());}}/*** 递归读取转换好的视频流** @param c 转换器* @param outEntity 输出流* @param response 响应* @author xufeng* @throws Exception*/public void readFlvStream(Converter c, OutputStreamEntity outEntity, HttpServletResponse response)throws Exception {//根据转换器状态来决定是继续等待、读取、结束流输出switch (c.getConverterState()) {case INITIAL:Thread.sleep(300);readFlvStream(c, outEntity, response);break;case OPEN:Thread.sleep(100);//System.out.println("=== OPEN递归读取转换好的视频流=="+c.getUrl());readFlvStream(c, outEntity, response);break;case RUN:if (outEntity.getOutput().size() > 0) {byte[] b = outEntity.getOutput().toByteArray();outEntity.getOutput().reset();response.getOutputStream().write(b);outEntity.setUpdateTime(System.currentTimeMillis());}System.out.println("=== RUN递归读取转换好的视频流=="+c.getUrl());c.setUpdateTime(System.currentTimeMillis());Thread.sleep(100);readFlvStream(c, outEntity, response);break;case CLOSE://log.info("close");break;default:break;}}}

3.转换

public class ConverterRegistration {/*** 转换器集合(根据用户ID分类)*/private static ConcurrentHashMap<String, ConcurrentHashMap<String, Converter>> converters = new ConcurrentHashMap<>();/*** 线程池*/private static ExecutorService executorService = Executors.newCachedThreadPool();/*** 开始一个转换<br/>* 如果已存在这个流的转换就直接返回已存在的转换器* @author xufeng* @param url 视频流链接* @param userId 用户主键* @return converter*/public static Converter open(String url, String userId) {System.out.println("===开始一个转换==="+url);//判断当前用户是否存在转换器线程集合,没有则新建ConcurrentHashMap<String, Converter> concurrentHashMap = converters.get(userId);if (concurrentHashMap == null) {concurrentHashMap = new ConcurrentHashMap<>(16);converters.put(userId, concurrentHashMap);}//判断是否已存在该转换器Converter c = isExist(url, concurrentHashMap);System.out.println("===判断是否已经存在转换器=="+c);try {if (null == c) {String key = UUID.randomUUID().toString();//创建线程c = new ConverterFactories(url, UUID.randomUUID().toString(), converters.get(userId));//记录到集合concurrentHashMap.put(key, c);//c.start();//用线程池启动executorService.execute((Runnable) c);}}catch (Exception e) {e.printStackTrace();}//如果该线程存在,但处于停止状态,则重新设置状态播放if (!c.isRuning()) {//设置运行状态c.setRuning(true);//设置初始化标志c.setState(ConverterState.INITIAL);//线程池启动executorService.execute((Runnable) c);}return c;}/*** 如果流已存在,就共用一个* @author xufeng* @param url 链接* @param concurrentHashMap 转换器集合* @return converter*/public static Converter isExist(String url, ConcurrentHashMap<String, Converter> concurrentHashMap) {//遍历集合,根据url判断是否已存在该流视频for (Converter c : concurrentHashMap.values()) {if (url.equals(c.getUrl())) {return c;}}return null;}/*** 返回集合中的所有转换器* @author xufeng* @param userId 用户主键* @return converters*/public static ConcurrentHashMap<String, Converter> getAllConverters(String userId){return converters.get(userId);}
}

4.使用javacv

public class ConverterFactories extends Thread implements Converter {/*** 运行状态*/public volatile boolean runing = true;/*** 读流器*/private FFmpegFrameGrabber grabber;/*** 转码器*/private FFmpegFrameRecorder recorder;/*** 转FLV格式的头信息<br/>* 如果有第二个客户端播放首先要返回头信息*/private byte[] headers;/*** 保存转换好的流*/private ByteArrayOutputStream stream;/*** 流地址,h264,aac*/private String url;/*** 流输出*/private Map<String, OutputStreamEntity> outEntitys;/*** 当前转换器状态*/private ConverterState state = ConverterState.INITIAL;/*** key用于表示这个转换器*/private String key;/*** 上次更新时间<br/>* 客户端读取是刷新<br/>* 如果没有客户端读取,会在一分钟后销毁这个转换器*/private long updateTime;/*** 转换队列*/private Map<String, Converter> factories;public ConverterFactories(String url, String key, Map<String, Converter> factories) {this.url = url;this.key = key;this.factories = factories;this.updateTime = System.currentTimeMillis();}@Overridepublic void run() {try {//使用ffmpeg抓取流,创建读流器grabber = new FFmpegFrameGrabber(url);//如果为rtsp流,增加配置if ("rtsp".equals(url.substring(0, 4))) {//设置打开协议tcp / udpgrabber.setOption("rtsp_transport", "tcp");//设置未响应超时时间 0.5秒grabber.setOption("stimeout", "500000");//设置缓存大小,提高画质、减少卡顿花屏//grabber.setOption("buffer_size", "1024000");//设置视频比例//grabber.setAspectRatio(1.7777);} else {grabber.setOption("timeout", "500000");}grabber.start();stream = new ByteArrayOutputStream();outEntitys = new ConcurrentHashMap<>();//设置转换状态为打开state = ConverterState.OPEN;//创建转码器recorder = new FFmpegFrameRecorder(stream, grabber.getImageWidth(),grabber.getImageHeight(),grabber.getAudioChannels());//配置转码器recorder.setFrameRate(grabber.getFrameRate());recorder.setSampleRate(grabber.getSampleRate());if (grabber.getAudioChannels() > 0) {recorder.setAudioChannels(grabber.getAudioChannels());recorder.setAudioBitrate(grabber.getAudioBitrate());recorder.setAudioCodec(grabber.getAudioCodec());//设置视频比例//recorder.setAspectRatio(grabber.getAspectRatio());}recorder.setFormat("flv");recorder.setVideoBitrate(grabber.getVideoBitrate());recorder.setVideoCodec(grabber.getVideoCodec());recorder.start(grabber.getFormatContext());//进入写入运行状态state = ConverterState.RUN;if (headers == null) {headers = stream.toByteArray();stream.reset();for (OutputStreamEntity o : outEntitys.values()) {o.getOutput().write(headers);}}int errorNum = 0;//线程运行时while (runing) {//FFmpeg读流压缩AVPacket k = grabber.grabPacket();if (k != null) {try {//转换器转换recorder.recordPacket(k);} catch (Exception e) {}byte[] b = stream.toByteArray();stream.reset();for (OutputStreamEntity o : outEntitys.values()) {if (o.getOutput().size() < (1024 * 1024)) {o.getOutput().write(b);}}errorNum = 0;} else {errorNum++;if (errorNum > 500) {break;}}}} catch (Exception e) {//log.error(e.getMessage(), e);state = ConverterState.ERROR;} finally {closeConverter();//log.info("exit");state = ConverterState.CLOSE;factories.remove(this.key);}}/*** 退出转换*/public void closeConverter() {try {//停止转码器if (null != recorder) {recorder.stop();}//停止、关闭读流器grabber.stop();grabber.close();//关闭转码器if (null != recorder) {recorder.close();}//关闭流if (null != stream) {stream.close();}if (null != outEntitys) {for (OutputStreamEntity o : outEntitys.values()) {o.getOutput().close();}}} catch (Exception e) {e.printStackTrace();//log.error(e.getMessage(), e);}}@Overridepublic String getKey() {return this.key;}@Overridepublic String getUrl() {return this.url;}@Overridepublic ConverterState getConverterState() {return this.state;}@Overridepublic void addOutputStreamEntity(String key, OutputStreamEntity entity) {try {switch (this.state) {case INITIAL:Thread.sleep(100);addOutputStreamEntity(key, entity);break;case OPEN:outEntitys.put(key, entity);break;case RUN:entity.getOutput().write(this.headers);outEntitys.put(key, entity);break;default:break;}} catch (Exception e) {//log.error(e.getMessage(), e);}}@Overridepublic void setUpdateTime(long updateTime) {this.updateTime = updateTime;}@Overridepublic long getUpdateTime() {return this.updateTime;}@Overridepublic void exit() {//设置线程状态为非运行状态,最后会进入finally块关闭读流器、转码器、流this.runing = false;try {this.join();} catch (Exception e) {e.printStackTrace();//log.error(e.getMessage(), e);}}@Overridepublic OutputStreamEntity getOutputStream(String key) {if (outEntitys.containsKey(key)) {return outEntitys.get(key);}return null;}@Overridepublic Map<String, OutputStreamEntity> allOutEntity() {return this.outEntitys;}@Overridepublic void removeOutputStreamEntity(String key) {this.outEntitys.remove(key);}@Overridepublic boolean isRuning() {return runing;}@Overridepublic void setRuning(boolean runing) {this.runing = runing;}@Overridepublic void setState(ConverterState state) {this.state = state;}
}

rtsp流转换器接口

public interface Converter {/*** 设置线程状态* @param state 状态标志*/void setState(ConverterState state);/*** 获取该转换的key*/public String getKey();/*** 获取该转换的url** @return*/public String getUrl();/*** 获取转换的状态** @return*/public ConverterState getConverterState();/*** 添加一个流输出** @param entity*/public void addOutputStreamEntity(String key, OutputStreamEntity entity);/*** 所有流输出** @return*/public Map<String, OutputStreamEntity> allOutEntity();/*** 移除一个流输出** @param key*/public void removeOutputStreamEntity(String key);/*** 设置修改时间** @param updateTime*/public void setUpdateTime(long updateTime);/*** 获取修改时间** @return*/public long getUpdateTime();/*** 退出转换*/public void exit();/*** 启动*/public void start();/*** 获取输出的流** @param key* @return*/public OutputStreamEntity getOutputStream(String key);/*** 判断线程是否在运行* @return boolean*/public boolean isRuning();/*** 设置运行状态* @param runing 运行标志*/public void setRuning(boolean runing);
}

6.输出视频流

public class OutputStreamEntity {public OutputStreamEntity(ByteArrayOutputStream output, long updateTime, String key) {super();this.output = output;this.updateTime = updateTime;this.key = key;}/*** 字节数组输出流*/private ByteArrayOutputStream output;/*** 更新时间*/private long updateTime;/*** key标识*/private String key;public ByteArrayOutputStream getOutput() {return output;}public void setOutput(ByteArrayOutputStream output) {this.output = output;}public long getUpdateTime() {return updateTime;}public void setUpdateTime(long updateTime) {this.updateTime = updateTime;}public String getKey() {return key;}public void setKey(String key) {this.key = key;}}
转换器状态(初始化、打开、关闭、错误、运行)
public enum ConverterState {INITIAL, OPEN, CLOSE, ERROR, RUN
}
public class JsonResult extends HashMap<String, Object> implements Serializable {private static final long serialVersionUID = 1L;public static final int SUCCESS = 200;public JsonResult() {}/*** 返回成功*/public static JsonResult ok() {return ok("操作成功");}/*** 返回成功*/public static JsonResult okFallBack() {return okFallBack("操作成功");}/*** 返回成功*/public JsonResult put(Object obj) {return this.put("data", obj);}/*** 返回成功*/public static JsonResult ok(String message) {return result(200, message);}/*** 降级函数 - 返回成功*/public static JsonResult okFallBack(String message) {return result(205, message);}/*** 返回成功*/public static JsonResult result(int code, String message) {JsonResult jsonResult = new JsonResult();jsonResult.put("timestamp", System.currentTimeMillis());jsonResult.put("status", code);jsonResult.put("message", message);return jsonResult;}/*** 返回失败*/public static JsonResult error() {return error("操作失败");}/*** 返回失败*/public static JsonResult error(String message) {return error(500, message);}/*** 返回失败*/public static JsonResult error(int code, String message) {JsonResult jsonResult = new JsonResult();jsonResult.put("timestamp", System.currentTimeMillis());jsonResult.put("status", code);jsonResult.put("message", message);return jsonResult;}/*** 设置code*/public JsonResult setCode(int code) {super.put("status", code);return this;}/*** 设置message*/public JsonResult setMessage(String message) {super.put("message", message);return this;}/*** 放入object*/@Overridepublic JsonResult put(String key, Object object) {super.put(key, object);return this;}/*** 权限禁止*/public static JsonResult forbidden(String message) {JsonResult jsonResult = new JsonResult();jsonResult.put("timestamp", System.currentTimeMillis());jsonResult.put("status", 401);jsonResult.put("message", message);return jsonResult;}/*@Overridepublic String toString() {return JSONObject.toJSONString(this);}public JSONObject toJSONObject() {return JSONObject.parseObject(toString());}*/}

7.前端展现

<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Insert title here</title>
<style type="text/css">
*{margin: 0px;padding: 0px;overflow: hidden;
}
video{object-fit:fill;width: 100%;height: 100%;
}
</style>
</head>
<body><div class="video-video-div"><video id="video" width="100%" height="100%"></video></div><input type="text" id="url" value="rtsp://127.0.0.1/myvideo"><button id="play">play</button>
</body>
<script type="text/javascript" src="js/jquery-2.1.4.min.js"></script>
<script type="text/javascript" src="js/flv.min.js"></script>
<script type="text/javascript">var videoObject={init:function(id,src){var self=this;this.src=src;this.id=id;this.flvPlayer = flvjs.createPlayer({type: 'flv',url:src,isLive: true,hasAudio: false,hasVideo: true,enableStashBuffer: true},{});this.flvPlayer.attachMediaElement(document.getElementById(id));this.flvPlayer.load();this.flvPlayer.play();this.reLoad=function(){self.flvPlayer.unload();self.flvPlayer.destroy();window.v=videoObject.init(self.id,self.src);}return this;}}
$(function(){$("#play").click(function(){var src=$("#url").val();if($.trim(src)!=""){if(window.v){window.v.flvPlayer.unload();window.v.flvPlayer.destroy();}window.v=videoObject.init("video","/flv/open/"+window.btoa(src));}	});});//获取地址栏参数
function getParameter(name,win){var params;if(null==win||undefined==win){params = window.location.search;}else{params = win.location.search;}params = params.substring(1, params.length);params = params.split("&");for (var i =0; i < params.length; i++){var items = params[i].split("=");var pname = items[0];if(pname == name){return items[1];}}
}
</script>
</html>

这篇关于使用javacv对摄像头视频转码并实现播放的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1019159

相关文章

Python实现精确小数计算的完全指南

《Python实现精确小数计算的完全指南》在金融计算、科学实验和工程领域,浮点数精度问题一直是开发者面临的重大挑战,本文将深入解析Python精确小数计算技术体系,感兴趣的小伙伴可以了解一下... 目录引言:小数精度问题的核心挑战一、浮点数精度问题分析1.1 浮点数精度陷阱1.2 浮点数误差来源二、基础解决

Java实现在Word文档中添加文本水印和图片水印的操作指南

《Java实现在Word文档中添加文本水印和图片水印的操作指南》在当今数字时代,文档的自动化处理与安全防护变得尤为重要,无论是为了保护版权、推广品牌,还是为了在文档中加入特定的标识,为Word文档添加... 目录引言Spire.Doc for Java:高效Word文档处理的利器代码实战:使用Java为Wo

SpringBoot日志级别与日志分组详解

《SpringBoot日志级别与日志分组详解》文章介绍了日志级别(ALL至OFF)及其作用,说明SpringBoot默认日志级别为INFO,可通过application.properties调整全局或... 目录日志级别1、级别内容2、调整日志级别调整默认日志级别调整指定类的日志级别项目开发过程中,利用日志

Java中的抽象类与abstract 关键字使用详解

《Java中的抽象类与abstract关键字使用详解》:本文主要介绍Java中的抽象类与abstract关键字使用详解,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录一、抽象类的概念二、使用 abstract2.1 修饰类 => 抽象类2.2 修饰方法 => 抽象方法,没有

SpringBoot 多环境开发实战(从配置、管理与控制)

《SpringBoot多环境开发实战(从配置、管理与控制)》本文详解SpringBoot多环境配置,涵盖单文件YAML、多文件模式、MavenProfile分组及激活策略,通过优先级控制灵活切换环境... 目录一、多环境开发基础(单文件 YAML 版)(一)配置原理与优势(二)实操示例二、多环境开发多文件版

MyBatis ParameterHandler的具体使用

《MyBatisParameterHandler的具体使用》本文主要介绍了MyBatisParameterHandler的具体使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参... 目录一、概述二、源码1 关键属性2.setParameters3.TypeHandler1.TypeHa

Spring 中的切面与事务结合使用完整示例

《Spring中的切面与事务结合使用完整示例》本文给大家介绍Spring中的切面与事务结合使用完整示例,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考... 目录 一、前置知识:Spring AOP 与 事务的关系 事务本质上就是一个“切面”二、核心组件三、完

使用docker搭建嵌入式Linux开发环境

《使用docker搭建嵌入式Linux开发环境》本文主要介绍了使用docker搭建嵌入式Linux开发环境,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面... 目录1、前言2、安装docker3、编写容器管理脚本4、创建容器1、前言在日常开发全志、rk等不同

Java实现远程执行Shell指令

《Java实现远程执行Shell指令》文章介绍使用JSch在SpringBoot项目中实现远程Shell操作,涵盖环境配置、依赖引入及工具类编写,详解分号和双与号执行多指令的区别... 目录软硬件环境说明编写执行Shell指令的工具类总结jsch(Java Secure Channel)是SSH2的一个纯J

使用Python实现Word文档的自动化对比方案

《使用Python实现Word文档的自动化对比方案》我们经常需要比较两个Word文档的版本差异,无论是合同修订、论文修改还是代码文档更新,人工比对不仅效率低下,还容易遗漏关键改动,下面通过一个实际案例... 目录引言一、使用python-docx库解析文档结构二、使用difflib进行差异比对三、高级对比方