Главная страница
Top.Mail.Ru    Яндекс.Метрика
Текущий архив: 2006.03.26;
Скачать: CL | DM;

Вниз

Многопоточность и COM   Найти похожие ветки 

 
Evloev Oleg   (2005-05-04 19:24) [0]

Hi All,

Вопрос такой.

Есть поток, в котором вызывается метод COM-сервера, который, в свою очередь, выполняется довольно медленно. Для контроля его работы на экране висит окно, отображающее текущее состояние дел. На этом окне есть кнопочка "Отменить", прерывающая работу сначала COM-сервера, а потом потока.
Вопрос. Как правильно остановить COM-сервер, чтобы он потом не вешал сам поток, в котором его запускали.
Код примерно такой.

В потоке, работающем с COM-объектом.

var Server:TComServer;

procedure DoingSome;
begin
 try
   Server.Doing; // вызов медленной процедуры
 except
  .. все плохо
 end;
end;

procedure KillServer;
begin
 FreeAndNil(Server);
end;


Факт в том что процедура KillServer вызывается во время работы Server.Doing, из другого потока. Все вроде проходит нормально, но потом этот поток, который работал с COM-сервером, вешает программу при вызове метода Terminate. Хотя в этот метод ничего не прописано. Если вместо Server.Doing написать например Sleep(4000), то Terminate проходит без проблем.

Ну вот, надеюсь внятно спросил. Пытался использовать критические секции для синхронизации процедур. Они действительно синхронизировались, то есть вторая вызывалась только после отработки первой. Однако конечный результат (Terminate) не изменился.


 
Набережных С. ©   (2005-05-04 22:00) [1]

Отключать объект из другого потока - не лучшая идея, мягко говоря. Под W2k and latter - CoCancelCall. Либо сервер должен поддерживать такую возможность. Например, метод принимает интерфейс обратного вызова и вызывает его метод периодически, по мере обработки запроса. Возвращаемый результат можно использовать для отмены.

PS: Надеюсь, а счет маршалинга ты в курсе.


 
Набережных С. ©   (2005-05-04 22:00) [2]

Отключать объект из другого потока - не лучшая идея, мягко говоря. Под W2k and latter - CoCancelCall. Либо сервер должен поддерживать такую возможность. Например, метод принимает интерфейс обратного вызова и вызывает его метод периодически, по мере обработки запроса. Возвращаемый результат можно использовать для отмены.

PS: Надеюсь, а счет маршалинга ты в курсе.


 
Evloev Oleg   (2005-05-05 11:09) [3]

Наверное более точно код приведу. Так как зависание происходит даже не доходя до метода FreeAndNil(Server).

Существует Thread, в котором находится объект Interpreter. В нем две функции:

// Сама медленная функция
function TInterpreter.Interpret(Script:widestring):boolean;
begin
 EnterCriticalSection(FLock);
 try
   try
     ComServer.Execute(Script);
   except
     on E:Exception do begin
       RaiseExceptionFmt("Ошибка обработки скрипта: %s",[E.Message]);
     end;
   end;
 finally
   LeaveCriticalSection(FLock);
 end;
end;

// Её останов
procedure TInterpreter.DeActivate;
begin
 EnterCriticalSection(FLock);
 try
   FreeAndNil(ComServer);
 finally
   LeaveCriticalSection(FLock);
 end;
end;


Первая функция вызывается в данном потоке, а вторая по событию нажатия клавиши, то есть из другого потока, как я понимаю. Чтобы их разграничить, я выставил критические секции.
Если вместо ComServer.Execute(Script) поставить любую другую локальную медленную функцию, то EnterCriticalSection(FLock) в методе DeActivate спокойно дождется ее выполнения и потом убьет сервер. Однако в текущем коде EnterCriticalSection(FLock) в DeActivate никогда не дождется своей очереди, даже если сервер закончил обрабатывать функцию. Все просто виснет насмерть.


 
Slym ©   (2005-05-06 05:08) [4]

Пока ComServer не будет поддерживать отмену выполнения или калбаки ничего у тебя не получится. FreeAndNil(ComServer) не выход, а ошибка



Страницы: 1 вся ветка

Текущий архив: 2006.03.26;
Скачать: CL | DM;

Наверх




Память: 0.48 MB
Время: 0.05 c
9-1126253501
RzCoDer
2005-09-09 12:11
2006.03.26
ладшафт в OpenGL


15-1141323235
Desdechado
2006-03-02 21:13
2006.03.26
Что-то с форумом...


1-1140478735
kilop
2006-02-21 02:38
2006.03.26
Интерпретатор Pascal на Delphi


1-1140796941
pasha_golub
2006-02-24 19:02
2006.03.26
Граф, но не дерево???


2-1141019874
Рафик
2006-02-27 08:57
2006.03.26
DBGrid