我使用带有进度条指示的.net ffmpeg wrapper http://www.mediasoftpro.com实现后台视频处理,以计算处理的视频数量,并将信息发送到网页以更新进度条指示器。如果只有一个进程一次工作,它的工作正常,但是在两个并发进程的情况下(从两台不同的计算机一次开始两个视频发布),进度条突然混合进度状态。 这是我的代码,我使用静态对象将单个实例的信息正确发送到进度条。
static string FileName = "grey_03";
protected void Page_Load(object sender, EventArgs e)
{
if (!Page.IsPostBack)
{
if (Request.Params["file"] != null)
{
FileName = Request.Params["file"].ToString();
}
}
}
public static double ProgressValue = 0;
public static MediaHandler _mhandler = new MediaHandler();
[WebMethod]
public static string EncodeVideo()
{
// MediaHandler _mhandler = new MediaHandler();
string RootPath = HttpContext.Current.Server.MapPath(HttpContext.Current.Request.ApplicationPath);
_mhandler.FFMPEGPath = HttpContext.Current.Server.MapPath("~\\ffmpeg_july_2012\\bin\\ffmpeg.exe");
_mhandler.InputPath = RootPath + "\\contents\\original";
_mhandler.OutputPath = RootPath + "\\contents\\mp4";
_mhandler.BackgroundProcessing = true;
_mhandler.FileName = "Grey.avi";
_mhandler.OutputFileName =FileName;
string presetpath = RootPath + "\\ffmpeg_july_2012\\presets\\libx264-ipod640.ffpreset";
_mhandler.Parameters = " -b:a 192k -b:v 500k -fpre \"" + presetpath + "\"";
_mhandler.OutputExtension = ".mp4";
_mhandler.VCodec = "libx264";
_mhandler.ACodec = "libvo_aacenc";
_mhandler.Channel = 2;
_mhandler.ProcessMedia();
return _mhandler.vinfo.ErrorCode.ToString();
}
[WebMethod]
public static string GetProgressStatus()
{
return Math.Round(_mhandler.vinfo.ProcessingCompleted, 2).ToString();
// if vinfo.processingcomplete==100, then you can get complete information from vinfo object and store it in database and perform other processing.
}
这里是jquery函数,负责每秒后更新进度条指示等。
$(function () {
$("#vprocess").on({
click: function (e) {
ProcessEncoding();
var IntervalID = setInterval(function () {
GetProgressValue(IntervalID);
}, 1000);
return false;
}
}, '#btn_process');
});
function GetProgressValue(intervalid) {
$.ajax({
type: "POST",
url: "concurrent_03.aspx/GetProgressStatus",
data: "{}",
contentType: "application/json; charset=utf-8",
dataType: "json",
success: function (msg) {
// Do something interesting here.
$("#pstats").text(msg.d);
$("#pbar_int_01").attr('style', 'width: ' + msg.d + '%;');
if (msg.d == "100") {
$('#pbar01').removeClass("progress-danger");
$('#pbar01').addClass("progress-success");
if (intervalid != 0) {
clearInterval(intervalid);
}
FetchInfo();
}
}
});
}
问题是由静态mediahandler对象引起的
public static MediaHandler _mhandler = new MediaHandler();
我需要一种方法来保持两个并发进程信息彼此分离,以便更新具有完全属于该进程的值的进度条。
答案 0 :(得分:2)
我认为你在这里有误解。
因为您使用的是static
变量,所以您会看到并发问题。这是因为在ASP.NET中,static
变量由所有请求共享,因为它们是每个AppDomain一次,直到应用程序池回收。
您应该公开一个WCF端点,您可以使用它来询问您的服务是否已完成。此服务应该在队列中工作,并为队列中的每个项目调用ffmpeg
程序。您还没有提到有多少用户会调用您的页面,但最好以这种方式设计它,这样您就可以更简单地控制服务器允许一次处理的项目数。您还将遇到更少的问题,因为您在服务上下文而不是ASP.NET下运行进程。