In plain C#, it looks like you have to loop through them all: // IntPtr myHandle = ... Process myProcess = Process.GetProcesses().Single( p => p.Id != 0 && p.Handle == myHandle); The above example intentionally fails if the handle isn't found. Otherwise, you could of course use SingleOrDefault. Apparently, it doesn't like you requesting the handle of process ID 0, hence the extra condition. Using the WINAPI, you can use GetProcessId. I couldn't find it on pinvoke.net, but this should do: [DllImport("kernel32.dll")] static extern int GetProcessId(IntPtr handle); (signature uses a DWORD, but process IDs are represented by ints in the .NET BCL) It seems a bit odd that you'd have a handle, but not a process ID however. Process handles are acquired by calling OpenProcess, which takes a process ID. ---------- using System; using System.Threading; using System.Security.Permissions; using System.Security.Principal; using System.Diagnostics; class ProcessDemo { public static void Main() { Process notePad = Process.Start("notepad"); Console.WriteLine("Started notepad process Id = " + notePad.Id); Console.WriteLine("All instances of notepad:"); // Get Process objects for all running instances on notepad. Process[] localByName = Process.GetProcessesByName("notepad"); int i = localByName.Length; while (i > 0) { // You can use the process Id to pass to other applications or to // reference that particular instance of the application. Console.WriteLine(localByName[i - 1].Id.ToString()); i -= 1; } Process chosen; i = localByName.Length; while (i > 0) { Console.WriteLine("Enter a process Id to kill the process"); string id = Console.ReadLine(); if (id == "") break; try { chosen = Process.GetProcessById(Int32.Parse(id)); } catch (Exception e) { Console.WriteLine("Incorrect entry."); continue; } if (chosen.ProcessName == "notepad") { chosen.Kill(); chosen.WaitForExit(); } i -= 1; } } }