Какао-оболочка для интерактивной команды Unix
Итак, я знаю, что вы можете сделать NSTask для запуска инструментов командной строки с Objective-C:
NSTask *task;
task = [[NSTask alloc] init];
[task setLaunchPath: @"/usr/bin/gdb"];
[task launch];
Мне просто интересно, есть ли способ общения с интерактивными инструментами командной строки, такими как gdb
, Это будет включать в себя ввод команд на основе взаимодействия с пользователем (например, run
, kill
или же quit
с gdb
) и затем реагирует на основании информации, которую он выводит.
2 ответа
Вы можете использовать NSTask's setStandardInput:
, setStandardOutput:
а также setStandardError:
селекторы в сочетании с экземплярами NSPipe для связи с запущенной программой.
Например, чтобы прочитать вывод задачи:
task = [[NSTask alloc] init];
[task setStandardOutput: [NSPipe pipe]];
[task setStandardError: [task standardOutput]]; // Get standard error output too
[task setLaunchPath: @"/usr/bin/gdb"];
[task launch];
Затем вы можете получить NSFileHandle
экземпляр, который вы можете использовать, чтобы прочитать вывод задачи с помощью:
NSFileHandle *readFromMe = [[task standardOutput] fileHandleForReading];
Чтобы настроить канал для отправки команд в GDB, вы должны добавить
[task setStandardInput: [NSPipe pipe]];
прежде чем запустить задачу. Тогда вы получите NSFileHandle
с
NSFileHandle *writeToMe = [[task standardInput] fileHandleForWriting];
Использование setStandardInput:
а также setStandardOutput:
методы класса NSTaks.
NSTask *task;
task = [[NSTask alloc] init];
[task setLaunchPath: @"/usr/bin/gdb"];
NSPipe *outputpipe=[[NSPipe alloc]init];
NSPipe *errorpipe=[[NSPipe alloc]init];
NSFileHandle *output,*error;
[task setArguments: arguments];
[task setStandardOutput:outputpipe];
[task setStandardError:errorpipe];
NSLog(@"%@",arguments);
output=[outputpipe fileHandleForReading];
error=[errorpipe fileHandleForReading];
[task launch];
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(receivedData:) name: NSFileHandleReadCompletionNotification object:output];
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(receivedError:) name: NSFileHandleReadCompletionNotification object:error];
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(TaskCompletion:) name: NSTaskDidTerminateNotification object:task];
//[input writeData:[NSMutableData initWithString:@"test"]];
[output readInBackgroundAndNotify];
[error readInBackgroundAndNotify];
[task waitUntilExit];
[outputpipe release];
[errorpipe release];
[task release];
-(void) receivedData:(NSNotification*) rec_not {
NSFileHandle *out=[[task standardOutput] fileHandleForReading];
NSData *dataOutput=[[rec_not userInfo] objectForKey:NSFileHandleNotificationDataItem];
if( !dataOutput)
NSLog(@">>>>>>>>>>>>>>Empty Data");
NSString *strfromdata=[[NSString alloc] initWithData:dataOutput encoding:NSUTF8StringEncoding];
[out readInBackgroundAndNotify];
[strfromdata release];
}
/* Called when there is some data in the error pipe */
-(void) receivedError:(NSNotification*) rec_not {
NSFileHandle *err=[[task standardError] fileHandleForReading];
NSData *dataOutput=[[rec_not userInfo] objectForKey:NSFileHandleNotificationDataItem];
if( !dataOutput)
NSLog(@">>>>>>>>>>>>>>Empty Data");
else {
NSString *strfromdata=[[NSString alloc] initWithData:dataOutput encoding:NSUTF8StringEncoding];
[strfromdata release];
}
[err readInBackgroundAndNotify];
}
/* Called when the task is complete */
-(void) TaskCompletion :(NSNotification*) rec_not {
NSLog(@"task ended");
}