objective-c - Obj-c 中访问先前定义的 C 结构数据的问题

标签 objective-c c struct memory-management

我是 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.itemsinit 中定义的数据没有任何对应关系>.

我尝试了很多不同的方法,但我没有找到解决方案。

拜托,任何人都可以解释为什么它不能像我希望的那样工作以及如何解决我的问题。

比尔

最佳答案

您的代码中的问题是内存管理。

当您创建 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/

相关文章:

iphone - 是否可以在 Objective-C 中重定向系统 API?

c - g_object_unref : assertion 'G_IS_OBJECT (object)' failed

c++ - 在多线程上下文中卸载 Mono 域

string - 如何连接两个字符串并将它们存储到相同的结构键中

ios - 如何找出哪个类(class)插入了我?

objective-c - 如何在 Objective-C 中使用 FFmpeg 在 macOS 中录制屏幕?

objective-c - 通过OperationQueue加载UIImageView的问题

c++ - abs(unsigned long) 有什么意义吗?

c - 结构不存储值和 scanf 失败

c++ - 结构作为C++中的参数