gpt4 book ai didi

ios - 返回具有 Core Foundation 属性的 Objective-C 对象

转载 作者:塔克拉玛干 更新时间:2023-11-02 09:46:28 25 4
gpt4 key购买 nike

我有一个方法返回名为“Credential”的自定义对象的 NSArray,该对象具有两个属性:NSString 和 CFDataRef。

正如您所注意到的,该对象具有两种类型的属性,NS Objective-C 属性和 Core-Foundation 属性。

对象在每个交互循环中初始化,因为它像这样填充 NSArray:

cred = [[Credential alloc] init];
cred.cn = [NSString stringWithString:(__bridge NSString *)(summary)];
cred.serialNumber = CFDataCreateCopy(kCFAllocatorDefault, serialNumber);

当我运行分析器时,我收到消息:

Object leaked: allocated object is not referenced later in this 
execution path and has a retain count of +1

我假设发生此警告是因为我正在初始化一个 CF 对象并从该方法返回而不释放它,但是负责释放该对象的 new 应该是调用方法的代码。

我应该在哪里调用 Credential 类的 CFDataRef 属性的 CFRelease?

编辑:

我用的是ARC所以把发布NSString(cred.cn)的责任交给他。不过,关于 CFDataRef (cred.serialNumber),我不会发布它,因为稍后我会从另一个类和部分代码中需要它。然后,我不确定如何管理它。当对象“Credential”被处置时,ARC 会释放它吗?如果没有,我可以覆盖Credential的dealloc方法来做serialNumber的CFRelease吗?

下面是初始化并返回 Credential 对象的 NSArray 的完整方法:

- (NSArray *) retrieveIdentities
{
CFArrayRef identities = NULL;
NSMutableArray *returnIdentities = nil;
OSStatus sanityCheck = NULL;

const void *keys[] = {kSecClass, kSecMatchLimit, kSecReturnRef, kSecReturnData, kSecReturnAttributes};
const void *values[] = {kSecClassIdentity, kSecMatchLimitAll, kCFBooleanTrue, kCFBooleanTrue, kCFBooleanTrue};
CFDictionaryRef query = CFDictionaryCreate(NULL, keys, values, sizeof(values)/sizeof(const void *), NULL, NULL);

sanityCheck = SecItemCopyMatching(query, (CFTypeRef *)&identities);

if (query)
CFRelease(query);
if (sanityCheck == errSecItemNotFound)
return nil;
if (sanityCheck != noErr)
@throw [[KeychainException alloc] initWithName:@"KeychainException" reason:@"ERROR_LISTING_IDENTITIES" userInfo:[NSDictionary dictionaryWithObjectsAndKeys:[NSNumber numberWithLong: sanityCheck], @"osstatus", nil]];

CFDictionaryRef result = NULL;
CFStringRef summary = NULL;
SecCertificateRef certificate = NULL;
CFDataRef serialNumber = NULL;
Credential *cred = nil;
CFIndex resultCount = CFArrayGetCount(identities);

returnIdentities = [[NSMutableArray alloc] init];

for (CFIndex i = 0; i<resultCount; i++)
{
result = CFArrayGetValueAtIndex(identities,i);
SecIdentityRef identity = (SecIdentityRef) CFDictionaryGetValue(result, kSecValueRef);

if ((sanityCheck = SecIdentityCopyCertificate(identity, &certificate)) != noErr)
@throw [[KeychainException alloc] initWithName:@"KeychainException" reason:@"ERROR_EXTRACTING_CERTIFICATE" userInfo:[NSDictionary dictionaryWithObjectsAndKeys:[NSNumber numberWithLong: sanityCheck], @"osstatus", nil]];

CFTypeRef keyClass = CFDictionaryGetValue(result, kSecAttrKeyClass);
if ([[(__bridge id)keyClass description] isEqual:(__bridge id)(kSecAttrKeyClassPrivate)])
{
summary = SecCertificateCopySubjectSummary(certificate);
serialNumber = CFDataCreateCopy(NULL, CFDictionaryGetValue(result, kSecAttrSerialNumber));
cred = [[Credential alloc] init];
cred.cn = [NSString stringWithString:(__bridge NSString *)(summary)];
cred.serialNumber = CFDataCreateCopy(kCFAllocatorDefault, serialNumber);
[returnIdentities addObject:cred];

if (summary)
CFRelease(summary);

if (serialNumber)
CFRelease(serialNumber);
}
}

if (certificate)
CFRelease(certificate);


return returnIdentities;
}

最佳答案

如果字符串是使用 CF 函数创建的,这意味着您拥有它,您应该将所有权转移给 ARC。目前,您只是桥接对 ARC 的引用,不会取得所有权。

对于数据,您需要重写dealloc 并在数据上调用CFRelease


分析器并不完美。有些事情它觉得很困难,所以它会谨慎行事,并告诉您可能存在问题。

可以肯定的是,尤其是在泄漏和内存管理方面,您应该使用 Instruments 来检查发生了什么。

关于ios - 返回具有 Core Foundation 属性的 Objective-C 对象,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17319590/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com