使用Singleton在线程之间传递时数据丢失

时间:2011-04-19 08:52:39

标签: iphone objective-c multithreading singleton alloc

修改

谢谢@BlackFrog。我想现在我已经接近了,但价值观仍然没有得到实现......

值设置如[progressController updateProgressSummary:...]中的日志所示,但是当我在progressUpdate initWithProgressUpdate:....中记录它们时,它们是nil,如下所示。

我对于使用了为progressUpdate设置的属性或为progressUpdate的3个组件中的每个组件设置的属性,我感到有些困惑。我已根据建议将3个单独的属性从assign更改为retain,并且还尝试对整个progressUpdate属性执行相同的操作(此处未显示)。

progressController.h

...

    @property (nonatomic, assign)   ProgressUpdate  *progressUpdate;

progressController.m

// Ask delegate to update and display Progress text
-(void) updateProgressSummary:(NSString *)summary detail:(NSString *)detail percentComplete:(NSNumber *)complete {
// These report the proper values
    DLog(@"Reporting Summary - %s", [summary UTF8String]);
    DLog(@"Reporting Detail -  %s", [detail UTF8String]);
    DLog(@"Reporting Complete -  %i", [complete intValue]); 

    if (summary != nil)
        self.progressUpdate.summaryText = summary;
    self.progressUpdate.detailText = detail;
    self.progressUpdate.percentComplete = complete;

    ProgressUpdate *progressUpdateForIssue = [[ProgressUpdate alloc] initWithProgressUpdate:progressUpdate];    
    [self.delegate performSelectorOnMainThread:@selector(displayProgress:) withObject:progressUpdateForIssue waitUntilDone:NO]; 

    [progressUpdateForIssue release];

}

但是几毫秒之后......,在对象里面......他们是零。

progressUpdate.h

.....

@property (nonatomic, retain) NSString  *summaryText;
@property (nonatomic, retain) NSString  *detailText;
@property (nonatomic, retain) NSNumber  *percentComplete;

progressUpdate.m

-(id) initWithProgressUpdate:(ProgressUpdate *)update {
    if ((self = [super init])) {
        summaryText = [update.summaryText copy];
        detailText = [update.detailText copy];
        percentComplete = [[NSNumber alloc] initWithFloat:[update.percentComplete floatValue]];
    }
    // These report nil values
    DLog(@"Reporting in progUpdate summaryText - %s", [summaryText UTF8String]);
    DLog(@"Reporting in progUpdate detailText -  %s", [detailText UTF8String]);
    DLog(@"Reporting in progUpdate percentComplete -  %i", [percentComplete intValue]);
return self;
}

更新结束

我需要一些帮助,将自定义类中的数据从一个线程传递到另一个线程。它在通行证之前但在抵达后消失。我已经尝试过我所知道的一切,但无济于事。

我的后台线程调用ProgressController并将其传递给当前进度的详细信息。这反过来会在ProgressController的委托(视图控制器)上执行SelectorOnMainThread来显示细节。

当我通过单个NSString时,一切正常,但我需要传递两个字符串和一个数字,而performSelectorOnMainThread只能传递一个对象,我已将它们封装在一个自定义对象中 - ProgressUpdate。

数据正确传入ProgressController,但在视图控制器中出现时为空。我知道这一点,因为我把NSLog放在了不同的地方。

我想知道它是否与之相关:

  • 多线程和自定义对象

  • ProgressController是一个单例,这就是为什么我每次调用时都会分配一个新的ProgressUpdate,但这没有帮助。

欢迎任何想法。为清楚起见,代码如下。

ProgressUpdate.h

#import <Foundation/Foundation.h>


@interface ProgressUpdate : NSObject {
    NSString    *summaryText;    
    NSString    *detailText;
    NSNumber    *percentComplete;
}
@property (nonatomic, assign) NSString  *summaryText;
@property (nonatomic, assign) NSString  *detailText;
@property (nonatomic, assign) NSNumber  *percentComplete;

-(id) initWith:(ProgressUpdate *)update;

@end

ProgressUpdate.m

#import "ProgressUpdate.h"

@implementation ProgressUpdate

@synthesize summaryText, detailText, percentComplete;

-(id) initWith:(ProgressUpdate *)update {
    self = [super init];

    self.summaryText = update.summaryText;
    self.detailText = update.detailText;
    self.percentComplete = update.percentComplete;
    return self;
}

@end

ProgressController.m

static ProgressController *sharedInstance;

+ (ProgressController *)sharedInstance {
    @synchronized(self) {
        if (!sharedInstance)
            [[ProgressController alloc] init];              
    }
    return sharedInstance;
}

+(id)alloc {
    @synchronized(self) {
        NSAssert(sharedInstance == nil, NSLocalizedString(@"Attempted to allocate a second instance of a singleton ProgressController.", @"Attempted to allocate a second instance of a singleton ProgressController."));
        sharedInstance = [super alloc];
    }
    return sharedInstance;
}
-(id) init {
    if (self = [super init]) {
        [self open];
    }
    return self;
}

.........

// Ask delegate to update and display Progress text
-(void) updateProgressSummary:(NSString *)summary detail:(NSString *)detail percentComplete:(NSNumber *)complete {

    if (summary != nil)
        self.progressUpdate.summaryText = summary;
    self.progressUpdate.detailText = detail;
    self.progressUpdate.percentComplete = complete;
    ProgressUpdate *progressUpdateForIssue = [[ProgressUpdate alloc] initWith:progressUpdate];  
    [self.delegate performSelectorOnMainThread:@selector(displayProgress:) withObject:progressUpdateForIssue waitUntilDone:NO]; 
    [progressUpdateForIssue release];

}

RootViewController.m

// Delegate method to display specific text in Progress label
- (void) displayProgress:(ProgressUpdate *)update {
    [progressSummaryLabel setText:update.summaryText];
    [progressDetailLabel setText:update.detailText];
    [progressBar setProgress:[update.percentComplete intValue]];
    [progressView setNeedsDisplay];
}

2 个答案:

答案 0 :(得分:2)

在init方法中,您只是分配ivars而不是将它们保留在新对象中。 重做您的init方法如下:

-(id) initWithProgressUpdate:(ProgressUpdate *)update {
    if ((self = [super init])) {
        summaryText = [update.summaryText copy];
        detailText = [update.detailText copy];
        percentComplete = [[NSNumber alloc] initWithFloat:[update.percentComplete floatValue];
    }
    return self;
}

几点:

  • 您不应在init方法中使用访问者
  • 重命名您的init方法非常清楚
  • 在@property中,将assign更改为retain

答案 1 :(得分:0)

尝试删除声明'[progressUpdateForIssue release];'在方法

' - (void)updateProgressSummary:(NSString *)摘要详细信息:(NSString *)detail percentComplete:(NSNumber *)complete'。

还要在类ProgressUpdate中将属性属性从“assign”更改为“retain”。 您可以在dealloc方法中释放这些属性。

祝你好运。