У меня есть консольное приложение, которое я запускаю из приложения с графическим интерфейсом. Консольное приложение принимает параметры имен файлов для анализа и обработки. В настоящее время я могу захватить его вывод и отобразить его в приложении с графическим интерфейсом, но я хотел бы иметь возможность отправлять ему команды, чтобы контролировать или даже останавливать его выполнение.
Как я могу отправить команду или строку или что-нибудь в консольное приложение, предпочтительно используя каналы, которые я открыл, чтобы прочитать его вывод?
const
CReadBuffer = 2400;
var
saSecurity: TSecurityAttributes;
hRead: THandle;
hWrite: THandle;
suiStartup: TStartupInfo;
piProcess: TProcessInformation;
pBuffer: array[0..CReadBuffer] of AnsiChar;
dRead: DWord;
dRunning: DWord;
dWritten: DWord;
Command: String;
BytesLeft: Integer;
BytesAvail: Integer;
begin
saSecurity.nLength := SizeOf(TSecurityAttributes);
saSecurity.bInheritHandle := True;
saSecurity.lpSecurityDescriptor := nil;
if CreatePipe(hRead, hWrite, @saSecurity, 0) then
begin
FillChar(suiStartup, SizeOf(TStartupInfo), #0);
suiStartup.cb := SizeOf(TStartupInfo);
suiStartup.hStdInput := hRead;
suiStartup.hStdOutput := hWrite;
suiStartup.hStdError := hWrite;
suiStartup.dwFlags := STARTF_USESTDHANDLES or STARTF_USESHOWWINDOW;
suiStartup.wShowWindow := SW_HIDE;
Command := 'messageparser.exe c:\messagefile.msg';
UniqueString(Command);
if CreateProcess(nil, PChar(Command), @saSecurity,
@saSecurity, True, NORMAL_PRIORITY_CLASS, nil, nil, suiStartup, piProcess) then
begin
repeat
dRunning := WaitForSingleObject(piProcess.hProcess, 100);
Application.ProcessMessages;
repeat
dRead := 0;
if not PeekNamedPipe(hread, @pbuffer, CReadBuffer, @dRead, @BytesAvail, @BytesLeft) then
RaiseLastOSError;
if dRead <> 0 then
begin
ReadFile(hRead, pBuffer[0], CReadBuffer, dRead, nil);
pBuffer[dRead] := #0;
OemToCharA(pBuffer, pBuffer);
// do something with the data
// if a condition is present then do the following:
// WriteFile(hWrite, some_command, size_of_buffer, DWritten, nil);
end;
until (dRead < CReadBuffer);
until (dRunning <> WAIT_TIMEOUT);
CloseHandle(piProcess.hProcess);
CloseHandle(piProcess.hThread);
end;
CloseHandle(hRead);
CloseHandle(hWrite);
end;
Затем на стороне консоли есть поток, ожидающий ввода. Вот метод execute:
while not Terminated do
begin
ReadLn(Command);
// process command
Sleep(10);
end;
Это ново для меня, поэтому, если есть советы о том, как это сделать правильно, я приветствую их :). Однако всякий раз, когда я посылаю Команду, она появляется как то, что я читаю в pBuffer из ReadPipe, а не какова команда.
Надеюсь, это поможет.
-
Нашел решение на основе подсказки Нат.
Двунаправленная связь между графическим интерфейсом и консолью