java 视频切片_关于视频播放、视频切片、跨域访问视频

关于视频播放、视频切片、跨域访问视频

前言

最近在着手部署上线做的一个视频网站,当我们部署到云服务器上后并开始测试视频观看并发量,发现了一个很严重的问题:带宽不足。9 或 10 个人同时观看视频的时候,就会出现有些用户加载不了视频的问题。

我们的云服务器是 1块钱的腾讯云,本身带宽就很低了,所以肯定没有足够大的带宽来支撑这么大的流量传输。这对于一个视频网站来说,带宽这个问题是非常致命的。在不能改变带宽的条件下,于是我们去找了一些方法来提高性能。

分析不足

我们的视频网站使用的内置 Flash 播放器来播放视频,用户观看视频的时候是直接完全加载整个视频的(不管你看了多久),从一开始播放就开始加载,并且并不会因为用户暂停而暂停加载, 它是一直持续加载直到加载完全的。对于绝大多数用户来说,他们不一定会把视频看完,如果是加载一个小视频,那还没有什么大问题,但如果是加载一个大视频的话,这就会浪费的大量的流量,并且加载过程会持续占用带宽,使得用户量多的时候,视频加载就会出现问题。

修改第一步

了解到这个问题之后,我们去看了别人的视频网站是如何撑起高用户量的,在视频播放的时候,我们发现它们并不是一开始就完全加载视频的,而是一段段的加载,去搜索之后发现这是一种切片的技术,用于控制流量传输。具体的切片的原理可参看 http://www.cnblogs.com/flash3d/archive/2013/11/02/3403109.html。

了解了切片技术之后,我们于是就开始在我们项目中应用切片的技术,我们使用的是 ffmpeg 来对视频进行切片。方法就是在程序中调用 ffmpeg 程序,然后调用切片命令对我们的视频进行切片,生成 m3u8 文件和 ts 文件,然后使用 flash 播放器播放,能够看到的确能够一段段的加载视频。

修改到这一步,这似乎解决了我们的问题,但是新的问题又出现了,我们发现当我们对大视频进行切片的时候,服务器的内存会占用很大,至于为什么会占用那个大,我们猜想可能是因为对视频切片时,ffmpeg 把整个视频加载到内存,所以导致内存占用高。当同时对多个视频进行切片的时候,服务器就炸掉了。于是我们又寻求新的方法去解决。

修改第二步

因为没想到好的方法去解决本地切片内存占用问题,于是我们使用了新的途径去存储播放视频,就是使用云端存储来存储视频,我们选用的是七牛云服务器来存储。它也提供了不同语音的 SDK 供开发者参考。

使用七牛云,我在我的 Java Web 项目里面导进必须的 4 个包,以及编写了上传视频并进行切片预处理的工具类。刚开始使用的时候也遇到许多问题。

问题1:上传不同格式的视频,有些播放不了

一开始,我们对任何格式的视频都调用同一个切片命令,以为会生成同一种格式的视频文件。但是当我们上传的 MP4 格式的视频,切片上传后,直接在浏览器输入外链(文件的访问链接),此时能够正常播放;但是上传 avi 或者 flv 格式的视频,上传切片后,直接输入外链会变成下载文件。

当时一直想不通这个问题的原因,因为明明都是调用了同一个命令切片,按理来说应该格式是一样的,但是却出现不同的行为。后来通过七牛云的问答平台寻求解决方案,才发现如果在上传的时候,没有使用 saveas 参数对结果另存为 xxx.m3u8 格式,他还是任然会以原有的格式去保存。所以基于浏览器对不同视频格式的支持,对 mp4、avi、flv等格式的视频则出现不同的效果

问题2: 使用外链播放出现跨域拒绝的问题

在我们解决完视频上传的问题之后,在播放器通过外链来播放视频的时候,发现出现跨域被拒绝的问题 (ERROR:HLSError(code/url/msg)=1tp:Cannot load M3U8: crossdomain access denied:Error #2048),google 了问题,发现原来Flash 播放器在加载跨域视频时,会先去加载云端的 corssdomain.xml 文件,然后判断是否被允许加载。

解决方法:需要在七牛云端上传 crossdomian.xml 文件

Java 七牛云上传视频的工具类

基于七牛云的 API,写了一个上传视频并切片的接口,可供参考;如有错 ,可一起讨论

该接口的配置信息采用配置文件 video.properties 来加载,具体的配置文件配置内容为:

access_key=your access key

secert_key=your secert key

bucketname=你的存储空间

pipeline=你的多媒体处理队列名

fops=avthumb/m3u8/noDomain/1/vb/500k/t/120(这是切片命令)若是要做其他处理,请参照七牛云 SDK

domain=你的七牛云映射域名

public class QiNiuUtil {

private static String DEFAULT_PROPERTIES = "video.properties";

private static Properties properties = new Properties();

static {

String path = QiNiuUtil.class.getResource("/").toString();

path = path.substring(6, path.length() - 8) + DEFAULT_PROPERTIES;

System.out.println(path);

try {

FileInputStream fileInputStream = new FileInputStream(path);

properties.load(fileInputStream);

System.out.println(properties.toString());

} catch (IOException e) {

System.out.println("配置文件不存在,加载配置文件失败");

}

}

public static String domian = properties.getProperty("domain");

private static String ACCESS_KEY = properties.getProperty("access_key");

private static String SECRET_KEY = properties.getProperty("secert_key");

// 要上传的空间

private static String bucketname = properties.getProperty("bucketname");

// 设置切片操作参数

private static String fops =properties.getProperty("fops");

// 设置转码的队列

private static String pipeline = properties.getProperty("pipeline");

//密钥配置

private static Auth auth = Auth.create(ACCESS_KEY, SECRET_KEY);

//创建上传对象

private static UploadManager uploadManager = new UploadManager();

//上传策略中设置persistentOps字段和persistentPipeline字段

public static String getUpToken(String pfops){

return auth.uploadToken(bucketname,null,3600,new StringMap()

.putNotEmpty("persistentOps", pfops)

.putNotEmpty("persistentPipeline", pipeline), true);

}

public static boolean upload(byte[] data, String key) throws IOException{

Response res = null;

try {

// 调用put方法上传

// 指定文件以 m3u8 格式另存

String urlbase64 = UrlSafeBase64.encodeToString(bucketname + ":" + key + ".m3u8");

res = uploadManager.put(data, key, getUpToken(fops + "|saveas/"+ urlbase64));

//打印返回的信息

System.out.println(res.bodyString());

} catch (QiniuException e) {

Response r = e.response;

// 请求失败时打印的异常的信息

System.out.println(r.toString());

try {

//响应的文本信息

System.out.println(r.bodyString());

} catch (QiniuException e1) {

//ignore

}

}

return res.isOK();

}

}

你可能感兴趣的:(java,视频切片)