• 设为首页
  • 点击收藏
  • 手机版
    手机扫一扫访问
    迪恩网络手机版
  • 关注官方公众号
    微信扫一扫关注
    公众号

iOS block 和同步性

[复制链接]
菜鸟教程小白 发表于 2022-12-12 17:10:32 | 显示全部楼层 |阅读模式 打印 上一主题 下一主题

我对我的应用程序中的 block 和同步性有疑问。这是场景:

我有一个 NetworkManager,它使用 AFHTTPRequestOperationManager (AFNetworking 2.0) 来调用服务器。我有一个 ContentStore 类单例,它为应用程序提供内容。应用程序中的任何类都可以向内容存储请求内容,并传递一个 block 来接收该内容。如果 ContentStore 在内存或存档中有内容,它会将其传递给请求内容的类中的 block 。如果没有,它会向 NetworkManager 发出请求,并向 NetworkManager 传递一个 block ,当内容从服务器到达时,这个 block 会从原始类调用该 block 。到目前为止,这一切都很好并且运行良好。

问题是,有时一个类可以调用 ContentStore 来获取内容,而该内容已经发出请求,但尚未收到回复。在这种情况下,两个请求向服务器发送相同的内容,这实际上不会导致问题,但效率低。

我想出了一个系统,它执行以下操作:如果正在向服务器请求内容,则 ContentStore 将传递给它的所有 block 保存在一个数组中,当从服务器接收到回复时 ContentStore循环遍历 block 数组并将内容传递给每个 block 。

这很好用,但我想知道是否存在可能的故障。当一个类为内容调用 ContentStore 并且请求正在进行时,是否可以将 block 添加到数组中,而下面代码中的 downloadCompletionBlock 正在循环通过同一个数组?

感谢任何花时间理解和回答我的问题的人。如果您需要更多信息,请告诉我。

@interface ContentStore()
@property (nonatomic, strong) NSDictionary *downloadDictionary;
@property (nonatomic, strong) NSMutableArray *downloadBlocksForRequest;
@property (nonatomic, strong) NetworkManger *networkManager;
@end



@implementation ContentStore

- (void) downloadWithCompletionBlockvoid (^)(NSDictionary *))completionBlock
{

    if (_downloadDictionary) {
        completionBlock(_downloadDictionary);
        return;
    } else {
        _downloadDictionary = [NSKeyedUnarchiver unarchiveObjectWithFile:[self downloadArchivePath]];
        if (_downloadDictionary) {
            completionBlock(_downloadDictionary);
            return;
        } else if (_downloadBlocksForRequest) {
            [_downloadBlocksForRequest addObject:completionBlock];
            return;
        } else {
            _downloadBlocksForRequest = [[NSMutableArray alloc]init];
            [_downloadBlocksForRequest addObject:completionBlock];
        }
    }

    void (^downloadCompletionBlock)(NSDictionary *) = ^(NSDictionary *downloadDictionary)
    {
        // do other work necessary

        [NSKeyedArchiver archiveRootObject:downloadDictionary toFile:[self downloadArchivePath]];
        [self setDownloadDictionary:downloadDictionary];

        for (int i=0; i<_facesBlocksForRequest.count; i++) {
            void (^complBlock)(NSDictionary *) = [_facesBlocksForRequest objectAtIndex:i];
            complBlock(downloadDictionary);

        }
        _downloadBlocksForRequest = nil;

    };


    [_networkManager downloadWithCompletionBlock:downloadCompletionBlock];

}



Best Answer-推荐答案


NSMutableArray 与 iOS 中的大多数(如果不是全部)可变容器一样,是 not thread-safe .所以你的担心是有道理的,这段代码可能会引入一个非常难以捕捉的错误。您可能希望通过这样做来同步对它的访问

@synchronized(_downloadBlocksForRequest) {
     // enumerate _downloadBlocksForRequest, or add an object to it, etc.
}

关于iOS block 和同步性,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20616959/

回复

使用道具 举报

懒得打字嘛,点击右侧快捷回复 【右侧内容,后台自定义】
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

关注0

粉丝2

帖子830918

发布主题
阅读排行 更多
广告位

扫描微信二维码

查看手机版网站

随时了解更新最新资讯

139-2527-9053

在线客服(服务时间 9:00~18:00)

在线QQ客服
地址:深圳市南山区西丽大学城创智工业园
电邮:jeky_zhao#qq.com
移动电话:139-2527-9053

Powered by 互联科技 X3.4© 2001-2213 极客世界.|Sitemap