繁体   English   中英

如何在Objective-C中创建弱引用?

[英]How to create weak reference in Objective-C?

我有这样的情况:

NSMutableArray * A = [[NSMutableArray alloc]initwithObjects:@"one",nil];
NSMutableArray * B = [[NSMutableArray alloc]initwithObjects:@"two",nil];

[A addObject:B];
[B addObject:A];

现在这里是一个保留周期,我怎样才能打破这个保留周期? (使用弱引用)....在上面的例子中。 谢谢

我首先会质疑你需要这样做的原因 - 但如果你确定它是最好的数据架构,我会使用NSValue 要在Foundation集合类中存储对象的引用(不保留它),可以使用+[NSValue valueWithNonretainedObject:]

NSMutableArray *a = [[NSMutableArray alloc] initwithObjects:@"one", nil];
NSMutableArray *b = [[NSMutableArray alloc] initwithObjects:@"two", nil];

[a addObject:b];
[b addObject:[NSValue valueWithNonretainedObject:a]];

然后,为了稍后从NSValue实例获取您的对象,您将执行以下操作:

NSMutableArray *c = (NSMutableArray *) [[b objectAtIndex:index] nonretainedObjectValue];

附录

我已经将使用+[NSValue valueWithPointer:]更改为+[NSValue valueWithNonretainedObject:] ,因为在ARC下,编译器会抱怨(由于从对象类型转换为const void * )。 虽然功能相同,但是类型转换和方法名称更有意义(并且编译器实际上允许您编译而不诉诸hackery)。

我过去解决这个问题的方法是使用NSProxy的子类来打破循环。 我将有一个对象存储对其中一个数组的弱引用,并将除内存管理之外的所有消息传递给它。

     ┌──── NSArray A <────┐
     │                    │
     │                    │
     v        weak        │
ACWeakProxy ┈ ┈ ┈ ┈ ┈ > NSArray B

@interface ACWeakProxy : NSProxy {
    id _object;
}

@property(assign) id object;

- (id)initWithObject:(id)object;

@end

@implementation ACWeakProxy

@synthesize object = _object;

- (id)initWithObject:(id)object {
    // no init method in superclass
    _object = object;
    return self;
}

- (BOOL)isKindOfClass:(Class)aClass {
    return [super isKindOfClass:aClass] || [_object isKindOfClass:aClass];
}

- (void)forwardInvocation:(NSInvocation *)invocation {
    [invocation setTarget:_object];
    [invocation invoke];
}

- (NSMethodSignature *)methodSignatureForSelector:(SEL)sel {
    return [_object methodSignatureForSelector:sel];
}

@end

然后你的代码变成了

NSMutableArray * A = [[NSMutableArray alloc] initwithObjects:@"one", nil];
NSMutableArray * B = [[NSMutableArray alloc] initwithObjects:@"two", nil];

[A addObject:B];
ACWeakProxy * proxy = [[ACWeakProxy alloc] initWithObject:A];
[B addObject:proxy];
[proxy release];

// will print "two"
NSLog(@"%@", [[[B objectAtIndex:1] objectAtIndex:1] objectAtIndex:0]);

但是,在您仍在使用它时,请确保您的弱参考不会消失。

你可以:

  • 通过删除对象手动打破循环
  • 使用CFArrayCreateMutable()并将retain和release回调设置为NULL

     CFArrayCallBacks acb = { 0, NULL, NULL, CFCopyDescription, CFEqual }; NSMutableArray *noRetain = (NSMutableArray *)CFArrayCreateMutable(NULL, 0, &acb); 

更好的是,退一步思考如何以不同的方式实现自己想要的东西。 通过适当的设计,甚至可能不会出现此问题。

在iPhone中,没有弱引用的概念。 在你的情况下,我认为它会导致其中任何一个过度释放的问题。

通常,对于2个班级,我会做这样的事情:

让一个类通过保留而另一个只使用assign来拥有另一个类。 一些示例代码可能会澄清它

A级

@interface A {

}

@property (nonatomic, retain) B *b;
@end

B级

@interface B {

}
@property (nonatomic, assign) A *a;
@end

在没有垃圾收集的iPhone环境中,弱引用或多或少被定义为不保留对象的引用。 考虑到这一点(实际上即使没有考虑到这一点),您无法控制NSMutableArray是否创建弱引用。 在GC环境中,你仍然有循环,但这并不重要,因为一旦两个对象都无法访问,它们都会消失。

你唯一能做的就是手动打破循环。 您可以通过从另一个阵列中删除其中一个阵列来实现。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM