搭建rtmp直播流服务之3:java开发ffmpeg实现rtsp转rtmp并实现ffmpeg命令的接口化管理架构设计及代码实现
上一篇文章简单介绍了java如何调用ffmpeg的命令:http://blog.csdn.net/eguid_1/article/details/51777716
上上一篇介绍了nginx-rtmp服务器的搭建:http://blog.csdn.net/eguid_1/article/details/51749830
这一篇将进一步深挖java对ffmepg命令的控制并最终实现服务接口化
本篇文章源码:http://download.csdn.net/detail/eguid_1/9563637
通知:由于很多同学反映本章代码的命令封装设计的不是很好,所以对本章代码重新进行了实现,新版本推翻了本章原有代码内部实现,接口设计更加利于注入自己的实现,并增加可执行原生ffmpeg命令功能
新版本请到这里查看:java封装FFmpeg命令,支持原生ffmpeg全部命令,实现FFmpeg多进程处理与多线程输出控制(开启、关闭、查询),rtsp/rtmp推流、拉流
(一)、简单介绍
该服务接口可实现rtsp协议转换为rtmp协议且可以实现rtmp直播流发布到nginx流媒体服务器,其中最为重要的是如何实现通过参数生成ffmpeg命令并执行,且可以通过接口进行控制ffmpeg命令的停止
(二)、实现ffmpeg接口化服务架构设计
push端接口化管理
一、接口化调用
1、采用多线程方式,每次调用push端口开启一个主进程及两个输出线程
2、可以对每个push端(线程)进行开启和关闭的控制
3、统一接口参数,对ffmpeg的命令做到参数可控制
二、架构设计
1、服务接口
PushManager提供push(开启一个push处理器),closePush(关闭push处理器),viewAppName接口(查看当前已经开启的应用)
1.1、应用名和push处理器的关系
一个处理器对应一个应用名
1.2、push处理器
一个处理器对应一个push主进程和两个输出线程
2、主进程控制
2.1、主进程开启
服务接口调用push处理器开启push主进程,主进程会自动开启两个输出线程用于消息输出,
开启后会将主进程Process和两个输出线程OutHandler通过map返回给服务接口。
2.2、主进程关闭
主进程可通过Process的destroy方法进行安全关闭
3、输出线程控制
3.1、输出线程开启
输出线程从主进程获取到输出流进行输出
3.2、输出线程关闭
输出线程重写了destory方法,用于安全的关闭输出线程
4、持久层控制
持久层分为两个:
1、appName(应用名)-pushId(push处理器的ID)对应关系
用于维护应用名和push处理器ID的对应关系,pushId为随机生成id
2、pushId-主进程-输出线程对应关系
主要用于存放主进程(Process)和两个输出线程,建立两者对应关系,方便服务接口管理
(三)、代码实现
1、PushManager实现
/**
* 实现push管理器的push,delete,view服务
*
* @author eguid
* @see PushMangerImpl
* @since jdk1.7
*/
public class PushManagerImpl implements PushManager
{
/**
* 引用push处理器
*/
private PushHandler pusher = new PushHandlerImpl();
/**
* 管理应用名和push处理器之间的关系
*/
private PushId_AppRelshipDao pard=new PushId_AppRelshipDaoImpl();
/**
* 管理处理器的主进程Process及两个输出线程的关系
*/
private HandlerDao hd = new HandlerDaoImpl();
public void setPusher(PushHandler pusher)
{
this.pusher = pusher;
}
public void setPard(PushId_AppRelshipDao pard)
{
this.pard = pard;
}
public void setHd(HandlerDao hd)
{
this.hd = hd;
}
@Override
public String push(Map<String, Object> map)
{
if(map==null||map.isEmpty()||!map.containsKey("appName"))
{
return null;
}
String appName=null;
ConcurrentMap<String, Object> resultMap = null;
try
{
appName=(String)map.get("appName");
if(appName!=null&&"".equals(appName.trim()))
{
return null;
}
resultMap = pusher.push(map);
// 生成一个标识该命令行线程集的key
String pushId = UUID.randomUUID().toString();
hd.set(pushId, resultMap);
pard.set(appName, pushId);
}
catch (IOException e)
{
// 暂时先写这样,后期加日志
System.err.println("发生一个异常" + e.getMessage());
}
return appName;
}
@Override
public void closePush(String appName)
{
String pushId=null;
if(pard.isHave(appName))
{
pushId= pard.getPushId(appName);
}
if (pushId!=null&&hd.isHave(pushId))
{
ConcurrentMap<String, Object> map = hd.get(pushId);
//关闭两个线程
((OutHandler)map.get("error")).destroy();
((OutHandler)map.get("info")).destroy();
//暂时先这样写,后期加日志
System.out.println("停止命令-----end commond");
//关闭命令主进程
((Process)map.get("process")).destroy();
//删除处理器与线程对应关系表
hd.delete(pushId);
//删除应用名对应关系表
pard.delete(appName);
}
}
@Override
public List<String> viewAppName()
{
return pard.getAll();
}
2、pushHandler实现(push处理器)
/**
* 提供解析参数生成ffmpeg命令并处理push操作
* @see PushHandlerImpl
* @since jdk1.7
*/
public class PushHandlerImpl implements PushHandler
{
/*
* "ffmpeg -i "+ "rtsp://admin:admin@192.168.2.236:37779/cam/realmonitor?channel=1&subtype=0 "+" -f flv -r 25 -s 640x360 -an" + " rtmp://192.168.30.21/live/test"
* 推送流格式: name:应用名;input:接收地址;output:推送地址;fmt:视频格式;fps:视频帧率;rs:视频分辨率;disableAudio:是否开启音频
*/
@Override
public ConcurrentMap<String, Object> push(Map<String, Object> paramMap)
throws IOException
{
// 从map里面取数据,组装成命令
String comm = getComm4Map(paramMap);
ConcurrentMap<String, Object> resultMap = null;
// 执行命令行
final Process proc = Runtime.getRuntime().exec(comm);
System.out.println("执行命令----start commond");
OutHandler errorGobbler = new OutHandler(proc.getErrorStream(), "Error");
OutHandler outputGobbler = new OutHandler(proc.getInputStream(), "Info");
errorGobbler.start();
outputGobbler.start();
// 返回参数
resultMap = new ConcurrentHashMap<String, Object>();
resultMap.put("info", outputGobbler);
resultMap.put("error", errorGobbler);
resultMap.put("process", proc);
return resultMap;
}
/**
* 通过解析参数生成可执行的命令行字符串;
* name:应用名;input:接收地址;output:推送地址;fmt:视频格式;fps:视频帧率;rs:视频分辨率;disableAudio:是否开启音频
*
* @param paramMap
* @return 命令行字符串
*/
protected String getComm4Map(Map<String, Object> paramMap)
{
// -i:输入流地址或者文件绝对地址
StringBuilder comm = new StringBuilder("ffmpeg -i ");
// 是否有必输项:输入地址,输出地址,应用名
if (paramMap.containsKey("input") && paramMap.containsKey("output")
&& paramMap.containsKey("appName"))
{
comm.append(paramMap.get("input")).append(" ");
// -f :转换格式,默认flv
comm.append(" -f ").append(paramMap.containsKey("fmt") ? paramMap.get("fmt") : "flv").append(" ");
// -r :帧率,默认25
comm.append("-r ").append(paramMap.containsKey("fps") ? paramMap.get("fps") : "30").append(" ");
// -s 分辨率 默认是原分辨率
comm.append("-s ").append(paramMap.containsKey("rs") ? paramMap.get("rs") : "").append(" ");
// -an 禁用音频
comm.append("-an ").append(paramMap.containsKey("disableAudio") && ((Boolean)paramMap.get("disableAudio")) ? "-an" : "").append(" ");
// 输出地址
comm.append(paramMap.get("output"));
//发布的应用名
comm.append(paramMap.get("appName"));
//一个视频源,可以有多个输出,第二个输出为拷贝源视频输出,不改变视频的各项参数并且命名为应用名+HD
comm.append(" ").append(" -vcodec copy -f flv -an ").append(paramMap.get("output")).append(paramMap.get("appName")).append("HD");
System.out.println(comm.toString());
return comm.toString();
}
else
{
throw new RuntimeException("输入流地址不能为空!");
}
}
}
3、OutHandler(输出线程)
**
* 用于输出命令行主进程的消息线程(必须开启,否则命令行主进程无法正常执行) 重要:该类重写了destroy方法,用于安全的关闭该线程
*
* @author eguid
* @see OutHandler
* @since jdk1.7
*/
public class OutHandler extends Thread
{
// 控制状态
volatile boolean status = true;
BufferedReader br = null;
String type = null;
public OutHandler(InputStream is, String type)
{
br = new BufferedReader(new InputStreamReader(is));
this.type = type;
}
/**
* 重写线程销毁方法,安全的关闭线程
*/
@Override
public void destroy()
{
status = false;
}
/**
* 执行输出线程
*/
@Override
public void run()
{
String msg = null;
try
{
while (status)
{
if ((msg = br.readLine()) != null)
{
System.out.println(type + "消息:" + msg);
}
}
}
catch (IOException e)
{
e.printStackTrace();
}
}
}
4、两个dao层接口,方便后期实现该接口并实现持久化
4.1、主进程(Process)和两个输出线程Dao
/**
* 命令行执行处理器缓存,方便管理处理器的开启和关闭
* @author eguid
* @see HandlerDao
* @since jdk1.7
*/
public interface HandlerDao
{
/**
* 获取某个处理器
* @param pushId
* @return
*/
public ConcurrentMap get(String pushId);
/**
* 存放一个处理器
* @param handlerMap
*/
public void set(String key, ConcurrentMap<String, Object> resultMap);
/**
* 获取全部处理器的id
* @return
*/
public ConcurrentMap getAll();
/**
* 删除某个处理器
* @param pushId
*/
public void delete(String pushId);
/**
* 是否存在key
*/
public boolean isHave(String pushId);
}
4.2、应用名-pushId对应关系Dao
/**
* 用于维护管理应用名与pushId的关系对应
* @author eguid
* @see PushId_AppRelshipDao
* @since jdk1.7
*/
public interface PushId_AppRelshipDao
{
/**
* 获取应用名对应的pushId
* @param appName
* @return pushId
*/
public String getPushId(String appName);
/**
* 插入一个应用名和pushId对应
* @param appName
* @param pushId
*/
public void set(String appName,String pushId);
/**
* 通过应用名删除对应关系
* @param appName
*/
public void delete(String appName);
/**
* 获取全部应用
*/
public List<String> getAll();
/**
* 是否存在应用名
* @param appName
* @return true:存在;false:不存在
*/
public boolean isHave(String appName);
}
下一篇将介绍一些支持rtmp直播的播放器
搭建rtmp直播流服务之3:java开发ffmpeg实现rtsp转rtmp并实现ffmpeg命令的接口化管理架构设计及代码实现的更多相关文章
- 三、直播整体流程 五、搭建Nginx+Rtmp直播流服务
HTML5实现视频直播功能思路详解_html5教程技巧_脚本之家 https://m.jb51.net/html5/587215.html 三.直播整体流程 直播整体流程大致可分为: 视频采集端:可以 ...
- 搭建rtmp直播流服务之2:使用java实现ffmpeg命令接口化调用(用java执行ffmpeg命令)
欢迎大家积极开心的加入讨论群 群号:371249677 (点击这里进群) 一.环境搭建 1.安装ffmpeg 下载对应系统的ffmpeg安装包,个人采用windows平台进行开发,所以安装了windo ...
- 搭建rtmp直播流服务之1:使用nginx搭建rtmp直播流服务器(nginx-rtmp模块的安装以及rtmp直播流配置)
欢迎大家积极开心的加入讨论群 群号:371249677 (点击这里进群) 一.方案简要 首先通过对开发方案的仔细研究(实时监控.流媒体.直播流方案的数据源-->协议转换-->服务器--&g ...
- 搭建rtmp直播流服务之4:videojs和ckPlayer开源播放器二次开发(播放rtmp、hls直播流及普通视频)
前面几章讲解了使用 nginx-rtmp搭建直播流媒体服务器; ffmpeg推流到nginx-rtmp服务器; java通过命令行调用ffmpeg实现推流服务; 从数据源获取,到使用ffmpeg推流, ...
- 抛开flash,自己开发实现C++ RTMP直播流播放器
抛开flash,自己开发实现C++ RTMP直播流播放器 众所周知,RTMP是以flash为客户端播放器的直播协议,主要应用在B/S形式的场景中.本人研究并用C++开发实现了RTMP直播流协议的播放器 ...
- Windows10环境下 Nginx+ffmpeg自搭服务器制作RTMP直播流
Windows10环境下 Nginx+ffmpeg自搭服务器制作RTMP直播流学习笔记 所需条件: nginx-rtmp-module(带rtmp模块) ,链接:https://link.jiansh ...
- windows下流媒体nginx-rmtp-module服务器搭建及java程序调用fmpeg将rtsp转rtmp直播流【转】
https://github.com/illuspas/nginx-rtmp-win32 http://bashell.sinaapp.com/archives/build-nginx-rtmp-mo ...
- 极速搭建RTMP直播流服务器+webapp (vue) 简单实现直播效果
在尝试使用webRTC实现webapp直播失败后,转移思路开始另外寻找可行的解决方案.在网页上尝试使用webRTC实现视频的直播与看直播,在谷歌浏览器以及safari浏览器上测试是可行的.但是基于基座 ...
- nginx开发(四)调用ffmpeg,搭建rtmp直播流。
1: 修改conf文件,配置rtmp直播 打开usr/local/nginx/conf/nginx.conf,添加红色内容: rtmp {#rtmp点播配置 server { li ...
随机推荐
- Python-3------新年考试周的Python学习
2016一开始就是考试周,准备专业课的考试复习.每天上午复习,晚上复习到8点半,之后到10点这点时间来看Python.庆幸没有在忙碌的时候荒废 Python的学习. 期待寒假,以前寒假在家总是没事做, ...
- java 操作FTP
package comm.ftp; import java.io.ByteArrayInputStream; import java.io.File; import java.io.FileInput ...
- ubuntu下搭建nginx+mysql+php-fpm站点
概述 Nginx ("engine x") 是一个高性能的 HTTP 和 反向代理 服务器,也是一个 IMAP/POP3/SMTP 代理服务器. nginx的优势在于能以低内存高 ...
- MongoDB副本集的搭建
副本集是mongodb提供的一种高可用解决方案.相对于原来的主从复制,副本集能自动感知primary节点的下线,并提升其中一个Secondary作为Primary. 整个过程对业务透明,同时也大大降低 ...
- vector实现最大流EK算法
序: 在之前的文章中实现了不利用STL实现EK算法,效率也较高.这次我们企图简化代码,减少变量的使用与手写模拟的代码. 注意:vector等STL的container在不开O2优化的时候实现同一个效果 ...
- TP5学习基础二:目录结构、URL路由、数据操作
一.安装1.使用git或者composer(composer update)进行实时更新,区别在于git不会清空核心框架目录而composer会清空.2.使用官网打包好的TP压缩包(解压即可用)-&g ...
- 记录——excel导出lua工具(python实现)
项目需要一个从excel导出lua配置表的工具,之前的工具是主程写的,效率极差,i7 CPU 一次全部导出要花掉1个多小时.匪夷所思的是,这么渣的效率,居然用了整整一年.当 然,中途有人反映效率差,主 ...
- 【exp/imp】将US7ASCII字符集的dmp文件导入到ZHS16GBK字符集的数据库中
[exp/imp]将US7ASCII字符集的dmp文件导入到ZHS16GBK字符集的数据库中 1.1 BLOG文档结构图 1.2 前言部分 1.2.1 导读和注意事项 各位技术爱好者,看完本文后 ...
- 【更新WordPress 4.6漏洞利用PoC】PHPMailer曝远程代码执行高危漏洞(CVE-2016-10033)
[2017.5.4更新] 昨天曝出了两个比较热门的漏洞,一个是CVE-2016-10033,另一个则为CVE-2017-8295.从描述来看,前者是WordPress Core 4.6一个未经授权的R ...
- Java线程池总结
前一篇文章Java中实现多线程关键词整理中介绍了Java中创建多线程的各种办法,里面提到了线程池,这里对Java中的线程池做一个总结. 1. 关于ThreadPoolExecutor 为了更好地控制多 ...