您的位置:首页 > 编程语言 > C#

【c#】关于终止以及删除主进程创建的线程以及进程的方法

2015-08-27 13:38 603 查看
今天在开发一款工具软件的时候,出现了一个问题:c#在Start新的Process之后,主动关闭窗口,在进程中发现窗口创建的另一个应用进程的优先级变为了主进程,导致了重启工具的时候,无法通过Process新建新的子进程,顺便再填上一个由主进程创建Thread线程之后,导致的关闭窗口,线程升级为进程的问题。
首先,先截取Frame窗口的关闭事件,将关闭方法写在里面:

private int WM_SYSCOMMAND = 0x112;
        private long SC_MAXIMIZE = 0xF030;
        private long SC_MINIMIZE = 0xF020;
        private long SC_CLOSE = 0xF060;
        protected override void WndProc(ref Message m)
{//拦截事件使用
if (m.Msg == WM_SYSCOMMAND)
{
if (m.WParam.ToInt64() == SC_MAXIMIZE)
{

}
if (m.WParam.ToInt64() == SC_MINIMIZE)
{

}
if (m.WParam.ToInt64() == SC_CLOSE)
{
Trace.WriteLine("点击了关闭按钮");
ProcessExtend.KillProcessTree(Process.GetCurrentProcess());//可关闭进程
Process.GetCurrentProcess().Kill();//可关闭线程
}
}

base.WndProc(ref m);
}
关闭线程很简单,只需要调用Process.GetCurrentProcess().Kill();就可以;关闭相关的子进程有点复杂,需要从进程树里挨个停止以及杀掉,在网上查找资料,发现可以直接写一个工具类,静态调用:

using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Linq;
using System.Runtime.InteropServices;
using System.Text;
using System.Threading.Tasks;

namespace DsidealAssistant.Tools
{
    static class ProcessExtend
    {
        private struct ProcessBasicInformation
        {
            public int ExitStatus;
            public int PebBaseAddress;
            public int AffinityMask;
            public int BasePriority;
            public uint UniqueProcessId;
            public uint InheritedFromUniqueProcessId;
        }

        [DllImport("ntdll.dll")]
        static extern int NtQueryInformationProcess(
           IntPtr hProcess,
           int processInformationClass,
           ref ProcessBasicInformation processBasicInformation,
           uint processInformationLength,
           out uint returnLength
        );

        public static void KillProcessTree(this Process parent)
        {
            var processes = Process.GetProcesses();
            foreach (var p in processes)
            {
                var pbi = new ProcessBasicInformation();
                try
                {
                    uint bytesWritten;
                    if (NtQueryInformationProcess(p.Handle, 0, ref pbi, (uint)Marshal.SizeOf(pbi), out bytesWritten) == 0)
                        if (pbi.InheritedFromUniqueProcessId == parent.Id)
                            using (var newParent = Process.GetProcessById((int)pbi.UniqueProcessId))
                                newParent.KillProcessTree();
                }
                catch { }
            }
            parent.Kill();
        }
    }
}

如果在64位程序上无效的话,可是试试这个:

static void KillProcessAndChildren(int pid)
{
ManagementObjectSearcher searcher = new ManagementObjectSearcher("Select * From Win32_Process Where ParentProcessID=" + pid);
ManagementObjectCollection moc = searcher.Get();
foreach (ManagementObject mo in moc)
{
KillProcessAndChildren(Convert.ToInt32(mo["ProcessID"]));
}
try
{
Process proc = Process.GetProcessById(pid);
Console.WriteLine(pid);
proc.Kill();
}
catch (ArgumentException)
{
/* process already exited */
}
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息