Проблема заключается в следующем:
При подключении клиента к внепроцессорному серверу автоматизации со следующим кодом:
While (1<>3) do
begin
Aserv.Connect;
Aserv.Disconnect;
end;
Происходит утечка памяти сервера автоматизации.
Каким образом можно производить очистку памяти на стороне сервера автоматизации?
Какой код нужно добавить в перегруженный деструктор destructor Destroy; override?
Единственным выходом пока является переписать сервер на C++ Builder. На C++ Builder 5 данной проблемы нет.
Re: Delphi 5-7. Утечка памяти сервера автоматизации.
Здравствуйте, Tesla, Вы писали:
T>При подключении клиента к внепроцессорному серверу автоматизации со следующим кодом: T>While (1<>3) do T>begin T>Aserv.Connect; T>Aserv.Disconnect; T>end; T>Происходит утечка памяти сервера автоматизации.
А как ты это видишь? Память какого процесса растет?
Ну и, докучи, покажи методы Connect и Disconnect
T>Каким образом можно производить очистку памяти на стороне сервера автоматизации?
Есть правила управления памятью (здесь), их придерживаться.
T>Какой код нужно добавить в перегруженный деструктор destructor Destroy; override?
БМП.
GS
Re[2]: Delphi 5-7. Утечка памяти сервера автоматизации.
Суть проблемы в том, что при подключении и отключении сервера автоматизации происходит утечка памяти.
Т.е. при подключении и отключении клиента происходит увеличении размера внепроцессорного сервера автоматизации вплоть до переполнения виртуальной памяти.
Данные примеры не из моего проекта, а из книги Delphi 6 руководство разработчика Стив Тейксер. Подобные примеры можно найти в книге Delphi 6 и Com Елманова, А Тенцер.
В каждом примере утечка памяти сервера автоматизации. Могу их выслать по E-mail.
Если вы писали сервер автоматизации на Delphi 5-7, то не могли не заметить этого подводного камня.
Вот пример приложения.
Внепроцессорный сервер автоматизации:
TEventSink = class(TObject, IUnknown, IDispatch)
private
FController: TMainForm;
{ IUnknown }
function QueryInterface(const IID: TGUID; out Obj): HResult; stdcall;
function _AddRef: Integer; stdcall;
function _Release: Integer; stdcall;
{ IDispatch }
function GetTypeInfoCount(out Count: Integer): HResult; stdcall;
function GetTypeInfo(Index, LocaleID: Integer; out TypeInfo): HResult; stdcall;
function GetIDsOfNames(const IID: TGUID; Names: Pointer;
NameCount, LocaleID: Integer; DispIDs: Pointer): HResult; stdcall;
function Invoke(DispID: Integer; const IID: TGUID; LocaleID: Integer;
Flags: Word; var Params; VarResult, ExcepInfo, ArgErr: Pointer): HResult; stdcall;
public
constructor Create(Controller: TMainForm);
end;
var
MainForm: TMainForm;
implementation
uses ActiveX;
{$R *.DFM}
{ TMainForm }
procedure TMainForm.FormCreate(Sender: TObject);
begin
FServer := CoServerWithEvents.Create;
FEventSink := TEventSink.Create(Self);
InterfaceConnect(FServer, IServerWithEventsEvents, FEventSink, FCookie);
end;
procedure TMainForm.FormDestroy(Sender: TObject);
begin
InterfaceDisconnect(FEventSink, IServerWithEventsEvents, FCookie);
FEventSink.Free;
end;
procedure TMainForm.SendButtonClick(Sender: TObject);
begin
FServer.AddText(Edit.Text);
end;
procedure TMainForm.ClearButtonClick(Sender: TObject);
begin
FServer.Clear;
end;
procedure TMainForm.CloseButtonClick(Sender: TObject);
begin
Close;
end;
procedure TMainForm.OnServerMemoChanged(const NewText: string);
begin
Memo.Text := NewText;
end;
procedure TMainForm.OnClear;
begin
Memo.Clear;
end;
{ TEventSink }
constructor TEventSink.Create(Controller: TMainForm);
begin
FController := Controller;
inherited Create;
end;
{ TEventSink.IUnknown }
function TEventSink._AddRef: Integer;
begin
// No need to implement, since lifetime is tied to client
Result := 1;
end;
function TEventSink._Release: Integer;
begin
// No need to implement, since lifetime is tied to client
Result := 1;
end;
function TEventSink.QueryInterface(const IID: TGUID; out Obj): HResult;
begin
// First look for my own implementation of an interface
// (I implement IUnknown and IDispatch).
if GetInterface(IID, Obj) then
Result := S_OK
// Next, if they are looking for outgoing interface, recurse to return
// our IDispatch pointer.
else if IsEqualIID(IID, IServerWithEventsEvents) then
Result := QueryInterface(IDispatch, Obj)
// For everything else, return an error.
else
Result := E_NOINTERFACE;
end;
{ TEventSink.IDispatch }
function TEventSink.GetIDsOfNames(const IID: TGUID; Names: Pointer;
NameCount, LocaleID: Integer; DispIDs: Pointer): HResult;
begin
Result := E_NOTIMPL;
end;
function TEventSink.GetTypeInfo(Index, LocaleID: Integer;
out TypeInfo): HResult;
begin
Pointer(TypeInfo) := nil;
Result := E_NOTIMPL;
end;
function TEventSink.GetTypeInfoCount(out Count: Integer): HResult;
begin
Count := 0;
Result := S_OK;
end;
function TEventSink.Invoke(DispID: Integer; const IID: TGUID;
LocaleID: Integer; Flags: Word; var Params; VarResult, ExcepInfo,
ArgErr: Pointer): HResult;
var
V: OleVariant;
begin
Result := S_OK;
case DispID of
1:
begin
// First parameter is new string
V := OleVariant(TDispParams(Params).rgvarg^[0]);
FController.OnServerMemoChanged(V);
end;
2: FController.OnClear;
end;
end;
end.
Re[3]: Delphi 5-7. Утечка памяти сервера автоматизации.
Если в FormCreate FServer создается, то в FormDestroy я бы попробовал его освобождать. Или в FormCreate проверять, что он уже создан, и не создавать по новой.
GS
Re[4]: Delphi 5-7. Утечка памяти сервера автоматизации.
Все-таки выбор пал на то, что нужно переписать COM сервер на C++ Builder.
Как не странно в C++ Builder сервер работает шустрее на 25 процентов.
Решения с Delphi найти не удалось. Видимо эта ошибка с 4-7 Delphi для того чтобы выбирали технологию Cobra.