javacv-ffmpeg(四)拉取时错误处理

说明

该错误处理不区分点直播

一、连接时报错

超时、地址错误、参数缺失

//该类错误在拉取流是try
	try {
		FFmpegFrameGrabber ff = new FFmpegFrameGrabber(videofile);
		// 微秒 大概为设置时间的两倍 
		ff.setOption(TimeoutOption.RW_TIMEOUT.getKey(), "10000000");
		// rtsp 默认udp 丢包 改为tcp
		ff.setOption("rtsp_transport", "tcp");
		ff.start();
		
		}catch (Exception e) { 
		//TODO 日志
		}finally {
			if (grabber != null) {
				grabber.stop();
				grabber.release();
			}
		}

二、空包处理

int nullCount = 0;

			if (null != f) {
			//处理逻辑
			}} else {
				log.info("frame:" + null);
				nullCount++;
				if (nullCount >= 5) {
					log.info("frame is null count:5  关闭任务");
					break;
				}else {continue;}
			}

三、无数据传输

这个问题是最恼火的。你被限制了或者对方出了问题,导致你连接成功后什么都没有,空包都没有,一直占用资源不是释放还没有任何产出
解决方案:重写FFmpegFrameGrabber
复制FFmpegFrameGrabber类后 在特定位置加 回调(亲测:在调用FFmpegFrameGrabber的时候加回调不好使)
效果:规定时间(可设置)内无数据传输杀掉该线程

	//1、在createDefault方法后添加  我都是直接杀线程 不用返回值
		/**
	 * 读流超时时间 : 60秒
	 */
	private static final int FRAME_READ_TIMEOUT = 60000;
	// 增加callbacjk 监听
	private final AtomicLong lastFrameTime = new AtomicLong(System.currentTimeMillis());
	private final org.bytedeco.ffmpeg.avformat.AVIOInterruptCB.Callback_Pointer cp = new org.bytedeco.ffmpeg.avformat.AVIOInterruptCB.Callback_Pointer() {
		@Override
		public int call(Pointer pointer) {
			// 0:continue, 1:exit
			if (lastFrameTime.get() + FRAME_READ_TIMEOUT < System.currentTimeMillis()) {
				try {
					log.info("frame read timeout 60s.---{}", System.currentTimeMillis());
					Thread t = Thread.currentThread();
					log.info(t.getId() + ":" + t.getName());
					t.interrupt();
				} catch (java.lang.Exception e) {
					log.error("Thread interrupt", e);
				}
				return 1;
			}
			return 0;
		}
	};

//2、在startUnsafe()方法中
//if ((ret = avformat_open_input(oc, filename, f, options)) < 0)  代码前 添加
		// 回调
		lastFrameTime.set(System.currentTimeMillis());
		oc = avformat_alloc_context();
		org.bytedeco.ffmpeg.avformat.AVIOInterruptCB cb = new org.bytedeco.ffmpeg.avformat.AVIOInterruptCB();
		cb.callback(cp);
		oc.interrupt_callback(cb);

//3、在grabFrame方法中if (oc == null || oc.isNull()) 判断结束后添加
		// 回调
		lastFrameTime.set(System.currentTimeMillis());
  • 3
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
在Java中,使用FFmpegJavaCV库可以很容易地实现收流器的功能。收流器可以接收来自RTSP或RTMP服务器的视频流,并保存为视频文件。以下是一个基于JavaCV-FFmpeg的简单实现。 首先需要添加JavaCVFFmpeg的依赖。可以在Maven中使用以下依赖: ``` <dependency> <groupId>org.bytedeco</groupId> <artifactId>javacv-platform</artifactId> <version>1.5.4</version> </dependency> ``` 接下来是收流器的实现代码: ```java import org.bytedeco.javacv.FFmpegFrameGrabber; import org.bytedeco.javacv.Java2DFrameConverter; import org.bytedeco.javacv.JavaFXFrameConverter; import java.awt.image.BufferedImage; import java.io.File; public class StreamReceiver { private FFmpegFrameGrabber grabber; private Java2DFrameConverter frameConverter; private JavaFXFrameConverter fxConverter; private boolean isRunning = false; public void start(String streamUrl, String outputFile) { try { grabber = new FFmpegFrameGrabber(streamUrl); grabber.start(); frameConverter = new Java2DFrameConverter(); fxConverter = new JavaFXFrameConverter(); isRunning = true; while (isRunning) { BufferedImage image = frameConverter.convert(grabber.grab()); if (image != null) { fxConverter.convert(image); } } grabber.stop(); grabber.release(); } catch (Exception e) { e.printStackTrace(); } } public void stop() { isRunning = false; } public static void main(String[] args) { StreamReceiver receiver = new StreamReceiver(); receiver.start("rtmp://live.hkstv.hk.lxdns.com/live/hks", "output.mp4"); } } ``` 在这个实现中,`start`方法会创建一个`FFmpegFrameGrabber`对象,并使用给定的`streamUrl`开始抓取。然后使用`Java2DFrameConverter`将抓取的帧转换为`BufferedImage`对象,最后使用`JavaFXFrameConverter`将`BufferedImage`转换为JavaFX中的`Image`对象。在循环中,我们检查`BufferedImage`是否为空,以确保我们只保存有内容的帧。最后,当`stop`方法被调用,循环中止并关闭抓取器。 可以通过调用`StreamReceiver`的`start`方法启动收流器,并使用RTMP或RTSP URL作为参数。收流器将不断抓取视频流,并将其保存到指定的输出文件中。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值