Форум программистов, компьютерный форум, киберфорум
C# для начинающих
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 4.80/5: Рейтинг темы: голосов - 5, средняя оценка - 4.80
0 / 0 / 0
Регистрация: 30.04.2020
Сообщений: 7

Поиск запущенного одноименного процесса с одинаковым хозяином в терминальной сессии

03.07.2020, 21:01. Показов 1013. Ответов 1

Студворк — интернет-сервис помощи студентам
Добрый день!

Есть задача при запуске приложения проверить на наличие уже запущенной копии приложения у текущего пользователя, если да, то активировать уже запущенный процесс. Само приложение находится на сетевом диске, запуск осуществляется по UNC путям и происходит все на терминальном сервере где несколько пользователей работают с данным приложением и важно, чтобы у каждого пользователя была только один процесс этого приложения.
На текущий момент задача решена и работает, если бы не одно "НО"!
Часть рабочего кода, которая была до сегодня:
Кликните здесь для просмотра всего текста
C#
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
void Form_Load(object sender, EventArgs e)
{
   string procname = Process.GetCurrentProcess().ProcessName;
   string owner = GetProcessOwner(Process.GetCurrentProcess().Id);
   // Если уже запущена одна копия программы под текущим пользователем, сначала закроем текущую и активируем потом уже открытую.
   if (Process.GetProcessesByName(procname).Where(x => GetProcessOwner(x.Id).Contains(owner)).Count()>1)
   {
      //Файл-команда в темповой директории пользователя для уже открытого приложения
      File.AppendAllText(tempo, "");
      //Закрываем новую версию запуска приложения
      Close();
    }
   else
   {
       // Тут уже сам запуск приложения и всех необходимых настроек
       //...
   }
}
   //Получения автора (owner) процесса по его ID
static string GetProcessOwner(int processId)
{
   string query = "Select * From Win32_Process Where ProcessID = " + processId;
   ManagementObjectSearcher searcher = new ManagementObjectSearcher(query);
   ManagementObjectCollection processList = searcher.Get();
 
   foreach (ManagementObject obj in processList)
   {
      string[] argList = new string[] { string.Empty, string.Empty };
      int returnVal = Convert.ToInt32(obj.InvokeMethod("GetOwner", argList));
      if (returnVal == 0)
      {
          return argList[1] + "\\" + argList[0];
       }
   }
   return "System";
}


"НО" заключается в следующем: поскольку пользователей много, то по мере запуска приложения каждым пользователем время запуска следующего процесса приложения в новой терминальной сессии увеличивается. Если первые пользователи в начале рабочего дня запускают приложение и ждут 2-3 секунды максимум, то в обед пользователь может ожидать запуска 30-40 сек. Причина понятна чем больше любых процессов у всех пользователей на терминалке, тем больше время на обработку массива потребуется текущему приложению.

Сегодня переписал немного код, думаю он должен дать небольшую оптимизацию, но проверить под нагрузкой смогу только в понедельник.
Переписанный код:
Кликните здесь для просмотра всего текста
C#
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
void Form_Load(object sender, EventArgs e)
{
    string procname = Process.GetCurrentProcess().ProcessName;
    string userName = WindowsIdentity.GetCurrent().Name;
   // Если уже запущена одна копия программы под текущим пользователем, сначала закроем текущую и активируем потом уже открытую.
    if (GetCountProcess(procname, userName) > 1)
   {
      //Файл-команда в темповой директории пользователя для уже открытого приложения
      File.AppendAllText(tempo, "");
      //Закрываем новую версию запуска приложения
      Close();
    }
   else
   {
       // Тут уже сам запуск приложения и всех необходимых настроек
       //...
   }
}
//Кол-во запущенных процессов
static byte GetCountProcess(string procName, string user)
{
   byte counter = 0;
   string query = $"Select * From Win32_Process Where Name like \'{procName}%\'";
   ManagementObjectSearcher searcher = new ManagementObjectSearcher(query);
   ManagementObjectCollection processList = searcher.Get();
 
   foreach (ManagementObject obj in processList)
   {
    string[] argList = new string[] { string.Empty, string.Empty };
    int returnVal = Convert.ToInt32(obj.InvokeMethod("GetOwner", argList));
    if (returnVal == 0)
    {
        counter += Convert.ToByte((argList[1] + "\\" + argList[0]).Contains(user) ? 1 : 0);                 
    }
    }
    return counter;
}


Обращаюсь к завсегдатаям форума и знатокам C# и не только, возможно Вы знаете более правильные/оптимальные решения данного вопроса по определению/подсчету запущенных приложений под одним хозяином, или у кого то возникнут идеи как еще можно оптимизировать.

P.S. Прошу сильно тапками не забрасывать, изучать C# начал относительно недавно
0
IT_Exp
Эксперт
34794 / 4073 / 2104
Регистрация: 17.06.2006
Сообщений: 32,602
Блог
03.07.2020, 21:01
Ответы с готовыми решениями:

Поиск адреса в памяти запущенного процесса
Здравствуйте, всем доброго здравия и времени суток! У меня есть вот такая тема, пишу программу которая ищет в памяти процесса указанный...

Поиск запущенного процесса Visual Studio C++
Добрый день форум!)хех Мне нужно что-бы при запуске консольного приложения писалось например что процесс "TslGame.exe" -...

Поиск запущенного процесса и его завершение
Доброго всем времени суток. Сам в VBS не силен, но необходимо: Скрипт проверяет наличие запущенного процесса, если процесс запущен...

1
0 / 0 / 0
Регистрация: 30.04.2020
Сообщений: 7
06.07.2020, 09:19  [ТС]
На выходных еще одна идея в голову пришла по оптимизации времени запуска, переписал часть кода с анализом ветки реестра в HKCU:
Кликните здесь для просмотра всего текста
C#
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
void Form_Load(object sender, EventArgs e)
{
    string procname = Process.GetCurrentProcess().ProcessName;
    string userName = WindowsIdentity.GetCurrent().Name;
    string regProc = "Software\\MyProgram\\Startup";
    byte countProc = 0;
   // Работаем с реестром и смотрим есть ли уже наша запись в реестре.
   using (RegistryKey key = Registry.CurrentUser.CreateSubKey(regProc))
   {
        string regParam = "IdRunProcess";
    string idProc = "";
    idProc = key?.GetValue(regParam)?.ToString();
    if (!string.IsNullOrEmpty(idProc))
    {
        countProc = GetCountProcess(procname, userName, idProc);
    }
    if (countProc == 0)
    {
    //Если не нашли записи о запущенном процессе или она не соответствует действительности, сделаем новую запись
        key.SetValue(regParam, Process.GetCurrentProcess().Id);
    }
   }
   // Если уже запущена одна копия программы под текущим пользователем, сначала закроем текущую и активируем потом уже открытую.
    if (countProc > 0)
   {
      //Файл-команда в темповой директории пользователя для уже открытого приложения
      File.AppendAllText(tempo, "");
      //Закрываем новую версию запуска приложения
      Close();
    }
   else
   {
       // Тут уже сам запуск приложения и всех необходимых настроек
       //...
   }
}
static byte GetCountProcess(string procName, string user, string process)
{
    byte counter = 0;
    int processId = Convert.ToInt32(process);
    string query = $"Select * From Win32_Process Where Name like \'{procName}%\' and ProcessID = {processId}";
    ManagementObjectSearcher searcher = new ManagementObjectSearcher(query);
    ManagementObjectCollection processList = searcher.Get();
 
    foreach (ManagementObject obj in processList)
    {
        string[] argList = new string[] { string.Empty, string.Empty };
        int returnVal = Convert.ToInt32(obj.InvokeMethod("GetOwner", argList));
        if (returnVal == 0)
        {
            counter += Convert.ToByte((argList[1] + "\\" + argList[0]).Contains(user) ? 1 : 0);                 
        }
    }
    return counter ;
}

При штатном закрытии приложения, ветка реестра чиститься в соответствующем событие, но даже если произойдет просто убийство процесса не штатным образом, то при следующем запуске приложения запись в реестре изменится на актуальную с новым номером процессом.
Минусов данного способа вроде не увидел, если конечно пользователь сам не зайдет в нужную ветку и не грохнет ее. Но даже если это и произойдет, то максимум что произойдет, это в текущей терминальной сессии будет позволен второй запуск приложения.
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
BasicMan
Эксперт
29316 / 5623 / 2384
Регистрация: 17.02.2009
Сообщений: 30,364
Блог
06.07.2020, 09:19
Помогаю со студенческими работами здесь

Поиск строки в памяти запущенного процесса и её изменение
Подскажи пожалуйста, что я делаю не так. Вот мой код. #include "windows.h" #include <iostream> using namespace std; int...

Поиск строки в памяти запущенного процесса и её изменение
Не могу решить задачку нужно считать строку из памяти процесса запущенной программы. RVA строки $0036D0C8 нужен ведь RVA не VA ...

Отключение терминальной сессии
Всем привет. Собственно вопрос, как отключить терминальную сессию. Использую сервер для авто тестирования, и просто необходимо, что бы был...

Печать в терминальной сессии
Здравствуйте. Имеется win server 2008 32 бит, принтер hp 1102 подключенный к компьютеру (другому). В настройках принтера дан общий...

Выход из терминальной сессии
Добрый день. У меня стоит Windows Server 2008. К нему через терминал подключаются пользователи, причем в настройках RDP прописывается...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
2
Ответ Создать тему
Новые блоги и статьи
Отправка уведомления на почту при изменении наименования справочника
Maks 24.03.2026
Программная отправка письма электронной почты на примере изменения наименования типового справочника "Склады" в конфигурации БП3. Перед реализацией необходимо выполнить настройку системной учетной. . .
модель ЗдравоСохранения 5. Меньше увольнений- больше дохода!
anaschu 24.03.2026
Теперь система здравосохранения уменьшает количество увольнений. 9TO2GP2bpX4 a42b81fb172ffc12ca589c7898261ccb/ https:/ / rutube. ru/ video/ a42b81fb172ffc12ca589c7898261ccb/ Слева синяя линия -. . .
Midnight Chicago Blues
kumehtar 24.03.2026
Такой Midnight Chicago Blues, знаешь?. . Когда вечерние улицы становятся ночными, а ты не можешь уснуть. Ты идёшь в любимый старый бар, и бармен наливает тебе виски. Ты смотришь на пролетающие. . .
Контроль уникальности заводского номера - вариант №2
Maks 24.03.2026
В отличие от предыдущего варианта добавлено прерывание циклов, также добавлены новые переменные для сохранения контекста ошибки перед прерыванием цикла: Процедура ПередЗаписью(Отказ, РежимЗаписи,. . .
SDL3 для Desktop (MinGW): Вывод текста со шрифтом TTF с помощью библиотеки SDL3_ttf на Си и C++
8Observer8 24.03.2026
Содержание блога Финальные проекты на Си и на C++: finish-text-sdl3-c. zip finish-text-sdl3-cpp. zip
Жизнь в неопределённости
kumehtar 23.03.2026
Жизнь — это постоянное существование в неопределённости. Например, даже если у тебя есть список дел, невозможно дойти до точки, где всё окончательно завершено и больше ничего не осталось. В принципе,. . .
Модель здравоСохранения: работники работают быстрее после её введения.
anaschu 23.03.2026
geJalZw1fLo Корпорация до введения программа здравоохранения имела много невыполненных работниками заданий, после введения программы количество заданий выросло. Но на выплатах по больничным это. . .
Контроль уникальности заводского номера - вариант №1
Maks 23.03.2026
Алгоритм контроля уникальности заводского (или серийного) номера на примере нетипового документа выдачи шин для спецтехники с табличной частью, разработанного в конфигурации КА2. Данные берутся из. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru