近来需要做一个在线浏览视频的功能,刚开始考虑的思路是:将视频上传之后,在页面上根据不同的视频格式嵌入不同的播放代码,可是这样的话页面上的代码量会非常的多,而且有的视频格式还得使用不同的播放器来播放,这样觉得很麻烦,实用性不强。后来查询得知:可以将各种视频格式转为flv的格式,然后在页面上只需要有一个flash播放器即可播放flv文件了,这样的话就不用考虑使用不同的播放器来播放的情况,而且页面代码简洁。以下为转换代码:
(1)将视频转换过程做成一个帮助类:
import java.io.BufferedReader;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.util.List;
public class ConvertSingleVideo {
private static String mencoder_home = "D:\\javaserve\\mencoder\\mencoder.exe";//mencoder.exe所放的路径
private static String ffmpeg_home = "D:\\javaserve\\ffmpeg\\ffmpeg.exe";//ffmpeg.exe所放的路径
public static String inputFile_home = "F:\\java\\work\\jingpinkecheng\\WebRoot\\upload\\input\\";//需转换的文件的位置
public static String outputFile_home = "D:\\javaserve\\tomcat\\apache-tomcat-7.0.32\\webapps\\jingpinkecheng\\upload\\output\\";//转换后的flv文件所放的文件夹位置
private String tempFile_home;//存放rm,rmvb等无法使用ffmpeg直接转换为flv文件先转成的avi文件
public ConvertSingleVideo(String tempFilePath){
this.tempFile_home = tempFilePath;
}
/**
* 功能函数
* @param inputFile 待处理视频,需带路径
* @param outputFile 处理后视频,需带路径
* @return
*/
public boolean convert(String inputFile, String outputFile)
{
if (!checkfile(inputFile)) {
System.out.println(inputFile + " is not file");
return false;
}
if (process(inputFile,outputFile)) {
System.out.println("ok");
return true;
}
return false;
}
//检查文件是否存在
private boolean checkfile(String path) {
File file = new File(path);
if (!file.isFile()) {
return false;
}
return true;
}
/**
* 转换过程 :先检查文件类型,在决定调用 processFlv还是processAVI
* @param inputFile
* @param outputFile
* @return
*/
private boolean process(String inputFile,String outputFile) {
int type = checkContentType( inputFile);
boolean status = false;
if (type == 0) {
status = processFLV(inputFile,outputFile);// 直接将文件转为flv文件
} else if (type == 1) {
String avifilepath = processAVI(type,inputFile);
if (avifilepath == null)
return false;// avi文件没有得到
status = processFLV(avifilepath,outputFile);// 将avi转为flv
}
return status;
}
/**
* 检查视频类型
* @param inputFile
* @return ffmpeg 能解析返回0,不能解析返回1
*/
private int checkContentType(String inputFile) {
String type = inputFile.substring(inputFile.lastIndexOf(".") + 1,inputFile.length()).toLowerCase();
// ffmpeg能解析的格式:(asx,asf,mpg,wmv,3gp,mp4,mov,avi,flv等)
if (type.equals("avi")) {
return 0;
} else if (type.equals("mpg")) {
return 0;
} else if (type.equals("wmv")) {
return 0;
} else if (type.equals("3gp")) {
return 0;
} else if (type.equals("mov")) {
return 0;
} else if (type.equals("mp4")) {
return 0;
} else if (type.equals("asf")) {
return 0;
} else if (type.equals("asx")) {
return 0;
} else if (type.equals("flv")) {
return 0;
}
// 对ffmpeg无法解析的文件格式(wmv9,rm,rmvb等),
// 可以先用别的工具(mencoder)转换为avi(ffmpeg能解析的)格式.
else if (type.equals("wmv9")) {
return 1;
} else if (type.equals("rm")) {
return 1;
} else if (type.equals("rmvb")) {
return 1;
}
return 9;
}
/**
* ffmepg: 能解析的格式:(asx,asf,mpg,wmv,3gp,mp4,mov,avi,flv等)
* @param inputFile
* @param outputFile
* @return
*/
private boolean processFLV(String inputFile,String outputFile) {
if (!checkfile(inputFile)) {
System.out.println(inputFile + " is not file");
return false;
}
File file = new File(outputFile);
if(file.exists()){
System.out.println("flv文件已经存在!无需转换");
return true;
} else {
System.out.println("正在转换成flv文件……");
List<String> commend = new java.util.ArrayList<String>();
//低精度
commend.add(ffmpeg_home);
commend.add("-i");
commend.add(inputFile);
commend.add("-ab");
commend.add("128");
commend.add("-acodec");
commend.add("libmp3lame");
commend.add("-ac");
commend.add("1");
commend.add("-ar");
commend.add("22050");
commend.add("-r");
commend.add("29.97");
// 清晰度 -qscale 4 为最好但文件大, -qscale 6就可以了
commend.add("-qscale");
commend.add("4");
commend.add("-y");
commend.add(outputFile);
StringBuffer test=new StringBuffer();
for(int i=0;i<commend.size();i++)
test.append(commend.get(i)+" ");
System.out.println(test);
try {
ProcessBuilder builder = new ProcessBuilder();
builder.command(commend);
builder.start();
return true;
} catch (Exception e) {
e.printStackTrace();
return false;
}
}
}
/**
* Mencoder:
* 对ffmpeg无法解析的文件格式(wmv9,rm,rmvb等),可以先用别的工具(mencoder)转换为avi(ffmpeg能解析的)格式.
* @param type
* @param inputFile
* @return
*/
private String processAVI(int type,String inputFile) {
File file =new File(tempFile_home);
if(file.exists()){
System.out.println("avi文件已经存在!无需转换");
return tempFile_home;
}
List<String> commend = new java.util.ArrayList<String>();
commend.add(mencoder_home);
commend.add(inputFile);
commend.add("-oac");
commend.add("mp3lame");
commend.add("-lameopts");
commend.add("preset=64");
commend.add("-ovc");
commend.add("xvid");
commend.add("-xvidencopts");
commend.add("bitrate=600");
commend.add("-of");
commend.add("avi");
commend.add("-o");
commend.add(tempFile_home);
StringBuffer test=new StringBuffer();
for(int i=0;i<commend.size();i++)
test.append(commend.get(i)+" ");
System.out.println(test);
try
{
ProcessBuilder builder = new ProcessBuilder();
builder.command(commend);
Process p=builder.start();
/**
* 清空Mencoder进程 的输出流和错误流
* 因为有些本机平台仅针对标准输入和输出流提供有限的缓冲区大小,
* 如果读写子进程的输出流或输入流迅速出现失败,则可能导致子进程阻塞,甚至产生死锁。
*/
final InputStream is1 = p.getInputStream();
final InputStream is2 = p.getErrorStream();
new Thread() {
public void run() {
BufferedReader br = new BufferedReader(new InputStreamReader(is1));
try {
String lineB = null;
while ((lineB = br.readLine()) != null ){
if(lineB != null)System.out.println(lineB);
}
} catch (IOException e) {
e.printStackTrace();
}
}
}.start();
new Thread() {
public void run() {
BufferedReader br2 = new BufferedReader(new InputStreamReader(is2));
try {
String lineC = null;
while ( (lineC = br2.readLine()) != null){
if(lineC != null)System.out.println(lineC);
}
} catch (IOException e) {
e.printStackTrace();
}
}
}.start();
//等Mencoder进程转换结束,再调用ffmpeg进程
p.waitFor();
System.out.println("who cares");
return tempFile_home;
}catch (Exception e){
System.err.println(e);
return null;
}
}
}
(2)action中调用该类的方法:
public String showResourceJiaoxueshipin() throws IOException{
resourcejiaoxueshipin = itemService.findById(resourcejiaoxueshipinid);
String fileName = resourcejiaoxueshipin.getFirst_img();
ConvertSingleVideo conver = new ConvertSingleVideo("F:\\java\\work\\jingpinkecheng\\WebRoot\\upload\\temp\\" + fileName.substring(0,fileName.lastIndexOf("."))+".avi");
conver.convert(ConvertSingleVideo.inputFile_home + fileName, ConvertSingleVideo.outputFile_home + fileName.substring(0,fileName.lastIndexOf("."))+".flv");
HttpSession session = request.getSession();
fileName = new String(fileName.getBytes("UTF-8"),"GBK");//存到session后在jsp页面取出的值是GBK("乱码"),所以这里先变成乱码,传输过去之后即可消除
session.setAttribute("jiaoxueshipinName",fileName.substring(0,fileName.lastIndexOf("."))+".flv" );
System.out.println("我是测试:"+session.getAttribute("jiaoxueshipinName"));
return "success";
}
(3)jsp代码:
只需要在页面嵌入一个flash播放器,我这里使用vcastr22.swf,并且将flv文件的路径填写正确即可。嵌入代码如下:
<object classid="clsid:D27CDB6E-AE6D-11cf-96B8-444553540000" codebase="http://download.macromedia.com/pub/shockwave/cabs/flash/swflash.cab#version=6,0,29,0" height="120" width="190">
<param name="movie" value="../upload/vcastr22.swf?vcastr_file=../upload/output/${sessionScope.jiaoxueshipinName }">
<param name="quality" value="high">
<param name="allowFullScreen" value="true" />
<!-- src里就是播放器的路径以及需要显示的flv文件的路径,路径一定要正确! -->
<embed
src="../upload/vcastr22.swf?vcastr_file=../upload/output/${sessionScope.jiaoxueshipinName }"
quality="high"
pluginspage="http://www.macromedia.com/go/getflashplayer"
type="application/x-shockwave-flash" width="500" height="350">
</embed>
</object>
经过以上步骤,即可成功的完成视频的转换即在线浏览。浏览效果如下:
参考资料:
文章浏览阅读1.4k次。在Python编程中,列表(List)是一种非常常用的数据结构,用于存储一系列有序的元素。有时候我们需要从列表中提取特定的元素,可以通过索引(Indexing)和切片(Slicing)来实现。通过掌握这些技巧,可以灵活地应对不同的需求,提取列表中需要的数据。需要注意的是,切片操作是左闭右开区间,即包含起始索引对应的元素,但不包含结束索引对应的元素。需要注意的是,当步长为负数时,提取元素的方向也会相应地反向。提取了列表中索引为1和2的元素(不包括索引为3的元素),即。,我们提取了列表中的第一个元素。_python list获取元素
文章浏览阅读780次。TLD(Tracking-Learning-Detection)是英国萨里大学的一个捷克籍博士生在其攻读博士学位期间提出的一种新的单目标长时间(long term tracking)跟踪算法。该算法与传统跟踪算法的显著区别在于将传统的跟踪算法和传统的检测算法相结合来解决被跟踪目标在被跟踪过程中发生的形变、部分遮挡等问题。同时,通过一种改进的在线学习机制不断更新跟踪模块的“显著特征点”和检测模块的目_tld demo 目标跟踪
文章浏览阅读355次。Go反射的实现与 interface和 unsafe.Pointer密切相关。(本文的部署环境是 Go 1.12.9)。interface回顾首先我们简单回顾一下 interface 的结构,总体上是:细分下来分为有函数的 iface 和无函数的 eface(就是 interface{});无函数的 eface有函数的 iface静态类型(static interface type)和动态混合类型...
文章浏览阅读3.7k次,点赞2次,收藏6次。前言:我在做javaweb实验的时候,遇到了一件很无语的事,我感觉我的代码没有问题,但是运行起来总是404,我总以为是我的哪个跳转的页面写错了,然后总在检查。其实后来想了一想,我是遇到问题就慌了,就乱检查,这样很浪费时间。出现了404总归就是找不到呗,发现页面跳转没有什么问题的时候,就应该检查一下xml文件里面是不是写错了。自己是真的蠢,一个很简单的问题,浪费了一天的时间。每次提交表单的时候就显示这样的页面。(表单提交到AddStudentServlet)后来我总是找页面跳转是不是出问题了,但是问题出_项目下 jsp 404
文章浏览阅读5.5w次,点赞27次,收藏203次。简介 在局部特征点检测快速发展的时候,人们对于特征的认识也越来越深入,近几年来许多学者提出了许许多多的特征检测算法及其改进算法,在众多的特征提取算法中,不乏涌现出佼佼者。 从最早期的Moravec,到Harris,再到SIFT、SUSAN、GLOH、SURF算法,可以说特征提取算法层出不穷。各种改进算法PCA-SIFT、ICA-SIFT、P-ASURF、R-ASURF、..._fast特征检测
文章浏览阅读163次。因为之前装过opencv2.4.9和cuda,这次的opencv3.3.1装起来问题很多,先下载opencv3.3.1,cmake那一步的时候,用如下指令cmake -D CMAKE_BUILD_TYPE=bulid -D CMAKE_INSTALL_PREFIX=/usr/local -D CUDA_GENERATION=Kepler ..make -j15多线程指令可以大大缩短时间最后sudo..._opencv2.4.9与opencv3.3.1
文章浏览阅读423次。Easycode是idea的一个插件,可以直接对数据的表生成entity,controller,service,dao,mapper,无需任何编码,简单而强大。1、安装(EasyCode)我这里的话是已经那装好了。建议大家在安装一个插件,叫做Lombok。Lombok能通过注解的方式,在编译时自动为属性生成构造器、getter/setter、equals、hashcode、toString方法。出..._idea自动生成mysqlsql插件
文章浏览阅读736次。ajax的使用教程_ajax 完整
文章浏览阅读5.9k次。Redis 简介Redis 是完全开源免费的,遵守BSD协议,是一个高性能的key-value数据库。Redis 与其他 key - value 缓存产品有以下三个特点:· Redis支持数据的持久化,可以将内存中的数据保存在磁盘中,重启的时候可以再次加载进行使用。· Redis不仅仅支持简单的key-value类型的数据,同时还提供list,set,zset,hash等数据结构的存储。· Red..._jfinal redis
文章浏览阅读1.3w次,点赞7次,收藏55次。新项目挖新坑,公司又需要做新的小程序,没有其他人支持,这次还得前后台一起做...不懂前端的我又来挖坑了...先做小程序“我的”页面,需要注意的有两个点:考虑到前几个版本getUserInfo不再弹出授权窗口,所以增加了不可见的button来唤起授权窗口;"我的"页面功能列表虽然现在不多,但是指不定什么时候就要增加或者修改,所以将该列表做成可配置..._小程序我的页面设计
文章浏览阅读6.1k次,点赞7次,收藏31次。Odrive无刷驱动器 开发过程学习记录(2)——电机系统的参数配置和校准开发说明:*ODrive 硬件版本: v3.6-56VODrive 硬件内部固件版本: fw-v0.5.1odrivetool版本: 0.5.1.post0 (pip install odrive==0.5.1.post0)*配置环境: windows10上次在windows环境下,将odrive的基础配置进行了详细的讲解,本次文章将进行电机参数的配置,并使电机进入闭环控制状态。首先按下述方式连接好odrive:连_耗散电阻
文章浏览阅读3.4k次。Java爬虫,信息抓取的实现 标签: