Форум: "Corba";
Текущий архив: 2006.03.26;
Скачать: [xml.tar.bz2];
ВнизМногопоточность и 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 вся ветка
Форум: "Corba";
Текущий архив: 2006.03.26;
Скачать: [xml.tar.bz2];
Память: 0.46 MB
Время: 0.059 c