NSCondition - >目标C.

时间:2011-06-06 11:15:59

标签: objective-c

我是Objective-C的新手。我正在研究线程。

我必须同步执行线程。我正在使用NSInvocationOperaion来生成一个线程。

我有两个主题。我需要等待第一个线程发出事件或超时信号。 可以通过NSConditionLock发出事件信号。如何发出超时信号。我不能在这里使用waitUntilDate方法,因为超时不是固定值。 有没有办法做到这一点?

EDITED

main.m
------
#import "PseudoSerialQueue.h"
#import "PseudoTask.h"

int main()
{
    PseudoSerialQueue* q = [[[PseudoSerialQueue alloc] init] autorelease];
    [q addTask:self selector:@selector(test0)];
    [q addTask:self selector:@selector(test1)];
    [q addTask:self selector:@selector(test2)];
    [q quit];
    return 0;
}

PseudoTask.h
-----------------

#import <Foundation/Foundation.h>


@interface PseudoTask : NSObject {

    id target_;
    SEL selector_;
    id queue_;

}

@property(nonatomic,readonly)id target;

-(id)initWithTarget:(id)target selector:(SEL)selector queue:(id)queue;
-(void)exec;

@end

PseudoTask.m
-----------------

#import "PseudoTask.h"


@implementation PseudoTask

@synthesize target = target_;

-(id)initWithTarget:(id)target selector:(SEL)selector queue:(id)queue
{
    self = [super init];
    if (self) {
        target_ = [target retain];
        selector_ = selector;
        queue_ = [queue retain];
    }
    return self;
}

-(void)exec
{
    [target_ performSelector:selector_];
}

-(void)dealloc
{
    [super dealloc];
    [target_ release];
    [queue_ release];
}

@end


PseudoSerialQueue.h
----------------------------

#import <Foundation/Foundation.h>
#import "PseudoTask.h"

@interface PseudoSerialQueue : NSObject {

    NSCondition* condition_;
    NSMutableArray* array_;
    NSThread* thread_;

}

-(void)addTask:(id)target selector:(SEL)selector;

@end

PseudoSerialQueue.m
----------------------------

#import "PseudoSerialQueue.h"

@implementation PseudoSerialQueue

-(id)init
{
    self = [super init];
    if (self) {
        array_ = [[NSMutableArray alloc]init];
        condition_ = [[NSCondition alloc]init];
        thread_ = [[NSThread alloc] initWithTarget:self selector:@selector(execQueue) object:nil];
        [thread_ start];
    }
    return self;
}

-(void)addTask:(id)target selector:(SEL)selector
{
    [condition_ lock];
    PseudoTask* task = [[PseudoTask alloc] initWithTarget:target selector:selector queue:self];
    [array_ addObject:task];
    [condition_ signal];
    [condition_ unlock];
}

-(void)quit
{
    [self addTask:nil selector:nil];
}

-(void)execQueue
{
    for(;;)
    {
        NSAutoreleasePool* pool = [[NSAutoreleasePool alloc]init];

        [condition_ lock];

        if (array_.count == 0) {
            [condition_ wait];
        }

        PseudoTask* task = [array_ objectAtIndex:0];
        [array_ removeObjectAtIndex:0];

        [condition_ unlock];

        if (!task.target) {
            [pool drain];
            break;
        }

        [task exec];
        [task release];

        [pool drain];
    }
}

-(void)dealloc
{
    [array_ release];
    [condition_ release];
    [super dealloc];
}

@end

我无法从主传递自己。希望我错误地称之为。 错误:“自我”未宣布即将到来。

我无法理解 - (无效)EXEC {   [target_ performSelector:selector_]; } 在PseudoTask.m中

target_不是方法,而是ivar。 我没有收到任何错误或警告。但我无法理解该代码。

我正在写你从你的程序中理解的内容。如果我理解程序是错误的,请纠正我。

当初始化PseudoSerialQueue并且它等待来自addTask方法的信号时,会产生Thread execQueue。 在quit方法中调用addTask方法,传递的参数为nil.I无法理解为什么传递nil参数。

如果你解释它会很有帮助。谢谢。

2 个答案:

答案 0 :(得分:2)

你的意思是NSCondition?您可以使用waitUntilDate:作为相对时间。

[condition lock];
// wait 5 seconds. 
[condition waitUntilDate:[NSDate dateWithTimeIntervalSinceNow:5]];
[condition unlock];

<强>编辑:

我的PseudoSerialQueue类需要从一个派生自NSObject的类中调用,如下所示。

@interface Test : NSObject
@end

@implementation Test
- (void)test0
{
}

- (void)test1
{
}

- (id)init
{
    self = [super init];
    return self;
}

- (void)exec
{
    PseudoSerialQueue *q = [[PseudoSerialQueue alloc] init];
    [q addTask:self selector:@selector(test0)];
    [q addTask:self selector:@selector(test1)];
    [q addTask:self selector:@selector(test0)];
    [q quit];
}
@end

您可以从主要功能调用它。

Test *test = [[Test alloc] init];
[test exec];
  

我无法理解为什么传递一个nil参数。

我只是选择它作为在PseudoSerialQueue中退出循环的消息。

答案 1 :(得分:0)

让两个情况下第一个线程发出第二个信号;然后在第二个线程中你可以告诉你在哪种情况下你是基于第一个控制器或模型中的一些只读标志(比如isDataAvailable)。