我们有一个程序集,我们称之为Foo.exe
。此可执行文件将由其他应用程序启动,例如Bar1.exe
或Bar2.exe
。
如果Foo.exe
运行,则每隔10秒检查一次Bar1
或Bar2
进程是否正在运行。如果没有,它会清理一些东西并关闭。
这适用于普通用户场景。但是,如果我们发展,我们有一个大问题:
应用程序无法查看应用程序Bar1
或Bar2
是否正在调试,在这两种情况下{1}都可以在任务管理器中使用。
这意味着,如果Bar.svhost.exe
将被忽略,Bar.svhost.exe
在调试过程中结束,这是不可能的。
但是如果Foo.exe
将被看到,Bar.svhost.exe
永远不会结束,我们必须手动杀死,但之后却没有正确清理。
知道如何解决问题吗?
(从Foo.exe
或Foo.exe
结束Bar1
是不可能的,因为可以在计算机上运行多个Bar2
或Bar1
Bar2
Foo
1}}只需要运行一次。因此Foo.exe
必须自行检查。并且“杀死”过程将让它清理干净)
添加: 这里是问题的伪代码示例
//Bar1.exe and Bar2.exe
void Main()
{
if (!FooIsRunning())
StartFoo();
DoSomething();
}
//Foo.exe
void Main()
{
Initialize();
while (BarIsRunning());
Cleanup();
}
private bool BarIsRunning()
{
var processes = Process.GetProcesses();
if (processes.Any(p => p.ProcessName.Contains("Bar.exe"))
return true;
var vshostProcess = processes.FirstOrDefault(p => p.ProcessName.Contains("Bar.vshost.exe");
return vshostProcess != null && ProcessIsDebugging(process);
}
private bool ProcessIsDebugging(Process process)
{
// How to...
return true;
}
答案 0 :(得分:4)
您检查过Debugger.IsAttached吗?
答案 1 :(得分:0)
我想我解决了这个问题。我发现如果调试Bar1或Bar2,MainWindowTitle会有所不同。 在调试之前,Bar1.vshost.exe的MainWindowTitle将是真正的应用程序标题,如果不是MainWindowTitle是空的。
foo.exe的
private static bool IsLastClientClosed()
{
var processes = Process.GetProcesses();
var debuggingProcesses = processes.Where(p => p.ProcessName.Contains("Bar1.vshost") || p.ProcessName.Contains("Bar2.vshost")).ToList();
if (debuggingProcesses.Any())
return !DebuggingRuns(debuggingProcesses);
return processes.Any(p => p.ProcessName.StartsWith("Bar1") || p.ProcessName.StartsWith("Bar2"));
}
private static bool DebuggingRuns(IEnumerable<Process> processes)
{
foreach (var process in processes)
{
try
{
if (!string.IsNullOrEmpty(process.MainWindowTitle))
return true;
}
catch
{
}
}
return false;
}
答案 2 :(得分:0)
如果要检查其他应用程序是否附加了调试程序,请使用CheckRemoteDebuggerPresent
我为这些流程编写了一个扩展方法:
public static class ProcessExtensions
{
[DllImport("kernel32.dll", SetLastError = true, ExactSpelling = true)]
static extern bool CheckRemoteDebuggerPresent(IntPtr hProcess, ref bool isDebuggerPresent);
public static bool IsDebuggerAttached(this Process process)
{
try
{
var isDebuggerAttached = false;
CheckRemoteDebuggerPresent(process.Handle, ref isDebuggerAttached);
return isDebuggerAttached;
}
catch (Exception)
{
return false;
}
}
}