Как получить полный путь запуска процесса? - PullRequest
90 голосов
/ 31 марта 2011

У меня есть приложение, которое изменяет некоторые настройки другого приложения (это простое приложение C #, которое запускается двойным щелчком (настройка не требуется)).

После изменения настроек мне нужно перезапуститьдругое приложение, чтобы оно отражало измененные настройки.

Итак, мне нужно убить запущенный процесс и запустить процесс заново, но проблема в том, что после убийства я не могу найти процесс.(Причина в том, что система не знает, где находится исполняемый файл ..)

Есть ли способ узнать путь запуска процесса или исполняемого файла, если он запущен?

Я не хочу указывать путь вручную, т. Е. Если он запущен, получите путь, завершите процесс и начните снова, иначе .... Я займусь позже

Ответы [ 12 ]

130 голосов
/ 31 марта 2011
 using System.Diagnostics;
 var process = Process.GetCurrentProcess(); // Or whatever method you are using
 string fullPath = process.MainModule.FileName;
 //fullPath has the path to exe.

В этом API есть одна загвоздка: если вы запускаете этот код в 32-битном приложении, вы не сможете получить доступ к 64-битным путям приложения, поэтому вам придется скомпилировать и запустить ваше приложение.как 64-битное приложение (Свойства проекта → Сборка → Цель платформы → x64).

103 голосов
/ 31 марта 2011

Что вы можете сделать, это использовать WMI, чтобы получить пути. Это позволит вам получить путь независимо от того, 32-битное или 64-битное приложение. Вот пример, демонстрирующий, как вы можете получить его:

// include the namespace
using System.Management;

var wmiQueryString = "SELECT ProcessId, ExecutablePath, CommandLine FROM Win32_Process";
using (var searcher = new ManagementObjectSearcher(wmiQueryString))
using (var results = searcher.Get())
{
    var query = from p in Process.GetProcesses()
                join mo in results.Cast<ManagementObject>()
                on p.Id equals (int)(uint)mo["ProcessId"]
                select new
                {
                    Process = p,
                    Path = (string)mo["ExecutablePath"],
                    CommandLine = (string)mo["CommandLine"],
                };
    foreach (var item in query)
    {
        // Do what you want with the Process, Path, and CommandLine
    }
}

Обратите внимание, что вам придется ссылаться на сборку System.Management.dll и использовать пространство имен System.Management.

Для получения дополнительной информации о том, какую другую информацию вы можете получить из этих процессов, такую ​​как командная строка, используемая для запуска программы (CommandLine), см. Класс Win32_Process и WMI .NET для получения дополнительной информации.

23 голосов
/ 31 марта 2011

Полагаю, у вас уже есть объект процесса запущенного процесса (например, с помощью GetProcessesByName ()). Затем вы можете получить имя исполняемого файла, используя

Process p;
string filename = p.MainModule.FileName;
11 голосов
/ 18 января 2018

Решение для:

  • Оба 32-битных и 64-битных процесса
  • Только System.Diagnostics (без System.Management)

Я использовал решение от Рассела Гантмана и переписал его как метод расширения, который вы можете использовать следующим образом:

var process = Process.GetProcessesByName("explorer").First();
string path = process.GetMainModuleFileName();
// C:\Windows\explorer.exe

С этой реализацией:

internal static class Extensions {
    [DllImport("Kernel32.dll")]
    private static extern bool QueryFullProcessImageName([In] IntPtr hProcess, [In] uint dwFlags, [Out] StringBuilder lpExeName, [In, Out] ref uint lpdwSize);

    public static string GetMainModuleFileName(this Process process, int buffer = 1024) {
        var fileNameBuilder = new StringBuilder(buffer);
        uint bufferLength = (uint)fileNameBuilder.Capacity + 1;
        return QueryFullProcessImageName(process.Handle, 0, fileNameBuilder, ref bufferLength) ?
            fileNameBuilder.ToString() :
            null;
    }
}
8 голосов
/ 03 февраля 2013

Комбинируя ответы Сандживакумара Хиремата и Джеффа Меркадо, вы действительно можете обойти проблему, когда извлекаете значок из 64-битного процесса в 32-битном процессе.

using System;
using System.Management;
using System.Diagnostics;

namespace ConsoleApplication1
{
    class Program
    {
        static void Main(string[] args)
        {
            int processID = 6680;   // Change for the process you would like to use
            Process process = Process.GetProcessById(processID);
            string path = ProcessExecutablePath(process);
        }

        static private string ProcessExecutablePath(Process process)
        {
            try
            {
                return process.MainModule.FileName;
            }
            catch
            {
                string query = "SELECT ExecutablePath, ProcessID FROM Win32_Process";
                ManagementObjectSearcher searcher = new ManagementObjectSearcher(query);

                foreach (ManagementObject item in searcher.Get())
                {
                    object id = item["ProcessID"];
                    object path = item["ExecutablePath"];

                    if (path != null && id.ToString() == process.Id.ToString())
                    {
                        return path.ToString();
                    }
                }
            }

            return "";
        }
    }
}

Это может бытьнемного медленный и не работает на каждом процессе, в котором отсутствует иконка "valid".

6 голосов
/ 09 марта 2016

Вот надежное решение, которое работает как с 32-битными , так и 64-битными приложениями.

Добавьте следующие ссылки:

с использованием System.Diagnostics;

с использованием System.Management;

Добавьте этот метод в свой проект:

public static string GetProcessPath(int processId)
{
    string MethodResult = "";
    try
    {
        string Query = "SELECT ExecutablePath FROM Win32_Process WHERE ProcessId = " + processId;

        using (ManagementObjectSearcher mos = new ManagementObjectSearcher(Query))
        {
            using (ManagementObjectCollection moc = mos.Get())
            {
                string ExecutablePath = (from mo in moc.Cast<ManagementObject>() select mo["ExecutablePath"]).First().ToString();

                MethodResult = ExecutablePath;

            }

        }

    }
    catch //(Exception ex)
    {
        //ex.HandleException();
    }
    return MethodResult;
}

Теперь используйте его так:

int RootProcessId = Process.GetCurrentProcess().Id;

GetProcessPath(RootProcessId);

Обратите внимание, что если вы знаете идентификатор процесса, то этот метод вернет соответствующий ExecutePath.

Extra, для заинтересованных:

Process.GetProcesses() 

... даст вам массив всех запущенных в данный момент процессов, и ...

Process.GetCurrentProcess()

... выдаст вам текущий процесс вместе с их информацией, например, Id и т. д., а такжеограниченный контроль, например, убийство и т. д. *

4 голосов
/ 10 октября 2017

Вы можете использовать pInvoke и собственный вызов, например, следующий.Кажется, это не имеет 32/64-битного ограничения (по крайней мере, в моем тестировании)

Вот код

using System.Runtime.InteropServices;

    [DllImport("Kernel32.dll")]
    static extern uint QueryFullProcessImageName(IntPtr hProcess, uint flags, StringBuilder text, out uint size);

    //Get the path to a process
    //proc = the process desired
    private string GetPathToApp (Process proc)
    {
        string pathToExe = string.Empty;

        if (null != proc)
        {
            uint nChars = 256;
            StringBuilder Buff = new StringBuilder((int)nChars);

            uint success = QueryFullProcessImageName(proc.Handle, 0, Buff, out nChars);

            if (0 != success)
            {
                pathToExe = Buff.ToString();
            }
            else
            {
                int error = Marshal.GetLastWin32Error();
                pathToExe = ("Error = " + error + " when calling GetProcessImageFileName");
            }
        }

        return pathToExe;
    }
0 голосов
/ 16 ноября 2016
using System;
using System.Diagnostics;

class Program
{
    public static void printAllprocesses()
    {
        Process[] processlist = Process.GetProcesses();

        foreach (Process process in processlist)
        {
            try
            {
                String fileName = process.MainModule.FileName;
                String processName = process.ProcessName;

                Console.WriteLine("processName : {0},  fileName : {1}", processName, fileName);
            }catch(Exception e)
            {
                /* You will get access denied exception for system processes, We are skiping the system processes here */
            }

        }
    }

    static void Main()
    {
        printAllprocesses();
    }

}
0 голосов
/ 24 сентября 2013

Попробуйте:

using System.Diagnostics;

ProcessModuleCollection modules = Process.GetCurrentProcess().Modules;
string processpathfilename;
string processmodulename;
if (modules.Count > 0) {
    processpathfilename = modules[0].FileName;
    processmodulename= modules[0].ModuleName;
} else {
    throw new ExecutionEngineException("Something critical occurred with the running process.");
}
0 голосов
/ 06 декабря 2012

Я попал в этот поток, когда искал текущий каталог выполняющегося процесса.В .net 1.1 Microsoft представила:

Directory.GetCurrentDirectory();

Кажется, работает хорошо (но не возвращает имя самого процесса).

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...