我是 objc/Cocoa 的新手,以前从未使用过 C。
我在获取先前定义的 C 结构数据时遇到问题......
这是我的代码:
AppController.h
#import <Cocoa/Cocoa.h>
@interface AppController : NSObject {
AuthorizationRef authRef;
AuthorizationRights authRights;
AuthorizationFlags authFlags;
}
- (IBAction)toggleAuthentification:(id)sender;
@end
AppController.m
#import "AppController.h"
@implementation AppController
- (id)init {
if (![super init])
return nil;
AuthorizationCreate(NULL, kAuthorizationEmptyEnvironment, kAuthorizationFlagDefaults, &authRef);
AuthorizationItem rightItems[1] = {{"com.myname.myapp.adminRights", 0, NULL, 0}};
authRights.count = 1;
authRights.items = rightItems;
authFlags = kAuthorizationFlagDefaults | kAuthorizationFlagExtendRights | kAuthorizationFlagInteractionAllowed | kAuthorizationFlagPreAuthorize;
return self;
}
- (IBAction)toggleAuthentification:(id)sender {
NSLog(@"%d", AuthorizationCopyRights(authRef, &authRights, kAuthorizationEmptyEnvironment, authFlags ^ kAuthorizationFlagInteractionAllowed, NULL));
}
@end
当我点击我的应用程序中调用 toggleAuthentification:
的按钮时,我收到错误代码 -60008 (errAuthorizationInternal)。
在调试器中,我可以看到 authRights.count
= 1,这是正确的,但是 authRights.items
与 init
中定义的数据没有任何对应关系>.
我尝试了很多不同的方法,但我没有找到解决方案。
拜托,任何人都可以解释为什么它不能像我希望的那样工作以及如何解决我的问题。
比尔
最佳答案
您的代码中的问题是内存管理。
当您创建 rightItems[]
时,它是 AuthorizationItem
元素的数组,它是在堆栈中创建的,因为 rightItems[]
是自动的(本地)变量。这意味着数组使用的内存在方法结束时被释放。由于您将 rightItems[]
分配给 authRights.items
,因此 authRights.items
指向 -init< 之后的虚假内存地址
已完成执行。对 authRights.items
的任何进一步引用都具有未定义的行为,并且可能会在您的程序中产生错误。
你需要做的是在堆中创建数组,这样它就不会在 -init
完成执行时被释放。您可以通过 malloc()
来做到这一点。例如:
const size_t numberOfRightItems = 1;
AuthorizationItem *rightItems = malloc(sizeof(AuthorizationItem)
* numberOfRightItems);
rightItems[0] = (AuthorizationItem){"com.myname.myapp.adminRights", 0, NULL, 0};
authRights.items = rightItems;
理论上,您负责在不再需要 authRights.items 时释放它,因为您已经在堆中为它分配了内存。因此,在您的 -dealloc
方法中:
- (void)dealloc {
free(authRights.items);
super[dealloc];
}
我说理论上是因为您将授权权限存储在您的应用程序 Controller 中。由于应用程序 Controller 存在于整个应用程序生命周期中,从技术上讲它不会接收 -release
或 -dealloc
。但是,如果您重新构建应用程序并且该代码最终位于一个类中,该类的对象在应用程序退出之前不一定是事件的,那么这是一种很好的做法并且可能有用。
进一步注意:您应该在-init
方法中将[super init]
的返回值分配给self
。这也是一种很好的做法,因为 [super init]
可能会返回与当前 self
不同的对象。
关于objective-c - Obj-c 中访问先前定义的 C 结构数据的问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5643041/