Я пытался сделать подобное.Я пришел к выводу, что предоставленная Microsoft библиотека COM является неполной.Я не использую его, потому что в документе упоминается, что «Примечание: эта тема является предварительной документацией и может быть изменена в будущих выпусках».
Итак, я решил взглянуть на то, что IISExpressTray.exeделается.Похоже, делают похожие вещи.
Я разобрал IISExpressTray.dll и обнаружил, что нет никакого волшебства в перечислении всех процессов IISexpress и остановке процесса IISexpress.
Он не вызывает эту библиотеку COM.Он ничего не ищет из реестра.
Итак, решение, которое я выбрал, очень простое.Чтобы запустить процесс IIS Express, я просто использую Process.Start () и передаю все необходимые параметры.
Чтобы остановить процесс IIS Express, я скопировал код из IISExpressTray.dll с помощью отражателя.Я видел, что он просто отправляет сообщение WM_QUIT целевому процессу IISExpress.
Вот класс, который я написал для запуска и остановки процесса IIS Express.Надеюсь, что это может помочь кому-то еще.
class IISExpress
{
internal class NativeMethods
{
// Methods
[DllImport("user32.dll", SetLastError = true)]
internal static extern IntPtr GetTopWindow(IntPtr hWnd);
[DllImport("user32.dll", SetLastError = true)]
internal static extern IntPtr GetWindow(IntPtr hWnd, uint uCmd);
[DllImport("user32.dll", SetLastError = true)]
internal static extern uint GetWindowThreadProcessId(IntPtr hwnd, out uint lpdwProcessId);
[DllImport("user32.dll", SetLastError = true)]
internal static extern bool PostMessage(HandleRef hWnd, uint Msg, IntPtr wParam, IntPtr lParam);
}
public static void SendStopMessageToProcess(int PID)
{
try
{
for (IntPtr ptr = NativeMethods.GetTopWindow(IntPtr.Zero); ptr != IntPtr.Zero; ptr = NativeMethods.GetWindow(ptr, 2))
{
uint num;
NativeMethods.GetWindowThreadProcessId(ptr, out num);
if (PID == num)
{
HandleRef hWnd = new HandleRef(null, ptr);
NativeMethods.PostMessage(hWnd, 0x12, IntPtr.Zero, IntPtr.Zero);
return;
}
}
}
catch (ArgumentException)
{
}
}
const string IIS_EXPRESS = @"C:\Program Files\IIS Express\iisexpress.exe";
const string CONFIG = "config";
const string SITE = "site";
const string APP_POOL = "apppool";
Process process;
IISExpress(string config, string site, string apppool)
{
Config = config;
Site = site;
AppPool = apppool;
StringBuilder arguments = new StringBuilder();
if (!string.IsNullOrEmpty(Config))
arguments.AppendFormat("/{0}:{1} ", CONFIG, Config);
if (!string.IsNullOrEmpty(Site))
arguments.AppendFormat("/{0}:{1} ", SITE, Site);
if (!string.IsNullOrEmpty(AppPool))
arguments.AppendFormat("/{0}:{1} ", APP_POOL, AppPool);
process = Process.Start(new ProcessStartInfo()
{
FileName = IIS_EXPRESS,
Arguments = arguments.ToString(),
RedirectStandardOutput = true,
UseShellExecute = false
});
}
public string Config { get; protected set; }
public string Site { get; protected set; }
public string AppPool { get; protected set; }
public static IISExpress Start(string config, string site, string apppool)
{
return new IISExpress(config, site, apppool);
}
public void Stop()
{
SendStopMessageToProcess(process.Id);
process.Close();
}
}
Мне не нужно перечислять все существующие процессы IIS Express.Если вам это нужно, то, что я увидел в отражателе, что IISExpressTray.dll делает, это вызывает Process.GetProcessByName("iisexpress", ".")
Чтобы использовать предоставленный мною класс, вот пример программы, которую я использовал для тестирования.
class Program
{
static void Main(string[] args)
{
Console.Out.WriteLine("Launching IIS Express...");
IISExpress iis1 = IISExpress.Start(
@"C:\Users\Administrator\Documents\IISExpress\config\applicationhost.config",
@"WebSite1(1)",
@"Clr4IntegratedAppPool");
IISExpress iis2 = IISExpress.Start(
@"C:\Users\Administrator\Documents\IISExpress\config\applicationhost2.config",
@"WebSite1(1)",
@"Clr4IntegratedAppPool");
Console.Out.WriteLine("Press ENTER to kill");
Console.In.ReadLine();
iis1.Stop();
iis2.Stop();
}
}
Возможно, это не ответ на ваш вопрос, но я думаю, что люди, интересующиеся вашим вопросом, могут найти мою работу полезной.Не стесняйтесь улучшать коды.Есть несколько мест, которые вы можете улучшить.
- Вместо жесткого кодирования местоположения iisexpress.exe, вы можете исправить мой код для чтения из реестра.
- Я не сделалвключить все аргументы, поддерживаемые iisexpress.exe
- Я не делал обработки ошибок.Итак, если процесс IISExpress не удалось запустить по каким-либо причинам (например, используется порт), я не знаю.Я думаю, что самый простой способ исправить это - отслеживать поток StandardError и выдавать исключение, если я получаю что-либо из потока StandardError