我正在尝试构建一个C#
应用程序,该应用程序使用MPC-HC's
API
来控制视频播放器。它来回发送Windows消息,它正在工作。
除了MPC-HC
有时会冻结,当它发生时,我的应用程序也会冻结。
我有办法吗?
1.防止我的申请冻结
2.检测MPC-HC
窗口是否响应
3.如果冻结,则终止进程并重新打开媒体播放器窗口
我已经能够编写2和3的代码,以及在发送命令之前检测响应性,但如果MPC-HC
在发送API命令时冻结,我的应用程序会冻结,我无能为力。
有没有解决方法?也许在中间有第三个进程,我的应用程序只能以不能冻结主应用程序的方式与该中间进程通信(如果可能的话)?
编辑:原来我之前使用'await'关键字测试了多线程,这实际上并没有启动任何新线程,这就是为什么这个过程仍然冻结了!如果我正确地创建了一个新线程,那就没问题......这段代码可以工作并取代我在下面写的整个程序
Dim T As New Task(Function() Comm.SendMsg(CommWnd.SYSMSG.WM_COPYDATA, cmd, param))
T.Start()
If T.Wait(3000) = False Then
MpcProcess.Kill()
End If
当你在学习新功能方面落后时会发生什么
答案 0 :(得分:0)
我首先尝试创建单独的线程,但这不起作用。整个申请仍然冻结。
但是,我确实找到了一个解决方案:创建一个监视视频播放器的新进程。如果它冻结,它会冻结我的主应用程序,但是单独的进程仍然在运行并且可以杀死它。以下是此代码,以防其他人遇到类似问题。
文件:Program.cs
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace ProcessWatcher {
public class Program {
private static WatcherThread watcher;
/// <summary>
/// Application entry point
/// </summary>
/// <param name="args">Command-line parameters</param>
static void Main(string[] args) {
// args = new string[] { "mpc-hc", "" };
if (args.Length < 2) {
System.Windows.Forms.MessageBox.Show(
@"Call with these parameters
ProcessWatcher [WatchProcess] [HostProcess] [Interval]
[WatchProcess]: Name of the process to monitor and kill if frozen.
[HostProcess]: When specified host process stops, end program.
[Interval]: (optional) Interval in milliseconds between checks. Default is 2000.
Example:
ProcessWatcher mpc-hc NaturalGroundingPlayer 2000");
return;
}
string WatchProcess = args[0];
string HostProcess = args[1];
int Interval = args.Length > 2 ? Int32.Parse(args[2]) : 2000;
watcher = new WatcherThread(WatchProcess, HostProcess, Interval);
System.Windows.Forms.Application.Run();
}
}
}
文件:ProcessWatcher.cs
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Text;
using System.Threading;
namespace ProcessWatcher {
/// <summary>
/// Checks regularly whether specified process is responsive. If not, kill it.
/// </summary>
public class WatcherThread {
private string processWatch;
private string processHost;
private Timer timer;
public WatcherThread(string processWatch, string processHost, int interval) {
this.processWatch = processWatch;
this.processHost = processHost;
timer = new Timer(KillIfFrozen, null, interval, interval);
}
/// <summary>
/// Kills the watched process if it is not responsive.
/// </summary>
/// <param name="state">null</param>
public void KillIfFrozen(Object state) {
Process[] ProcessList = Process.GetProcessesByName(processWatch);
foreach (Process item in ProcessList) {
if (item.Responding == false) {
// Wait 1500ms for the process to respond
for (int i = 0; i < 6; i++) {
Thread.Sleep(250);
if (item.Responding)
break;
}
// If still not responsive, kill process.
if (item.Responding == false) {
try {
// This throws an "Access denied" exception but still stops the process.
item.Kill();
} catch {}
}
}
}
// If host process is closed, end program.
if (!string.IsNullOrEmpty(processHost)) {
ProcessList = Process.GetProcessesByName(processHost);
if (ProcessList.Length == 0) {
System.Windows.Forms.Application.Exit();
}
}
}
}
}
然后,在主应用程序中使用Process.Start来运行此应用程序。当主应用程序停止时,它会检测到它并停止。
现在,为了使主应用程序进程在Visual Studio中结束执行时停止,在主应用程序项目属性下,您必须进入Debug并禁用“启用Visual Studio托管进程”