0

我正在创建一个第三方库,我需要找到一种以幂等方式创建唯一标识符的方法。我会尽力解释。

假设你点UIViewControllerMyViewController。该库公开了两个不同的 API 来注册/注销组件。

当我执行register视图控制器时,我需要为控制器创建一个唯一标识符,该标识符在被释放之前应该是有效的。如果我创建此控制器的另一个实例,则标识符应该不同。

换句话说,假设我有

MyViewController* a = // alloc-init

现在在里面a,我像这样注册自己

// I'm within viewDidLoad for example
[[LibraryManager sharedManager] registerEntry:self];

此时,我的需求如下:根据我传入的实例创建一个唯一标识符(在我的库中)

如果a想注销,它可以简单地做

// I'm within dealloc for example
[[LibraryManager sharedManager] deregisterEntry:self];

基于 ( self) 中传递的实例,我需要(以某种方式)检索先前创建的标识符,以便清除使用该register方法创建的内容。

如果我创建另一个控制器,比如说b,标识符应该与使用创建的那个不同a

我不会依赖委托模式来要求控制器(在这种情况下)通过[[NSUUID UUID] UUIDString];或类似的方式提供唯一标识符。

有没有办法根据传入的实例创建唯一标识符?我正在考虑存储内存地址或类似的字符串表示形式,但恕我直言,这是一种黑客行为。

4

1 回答 1

1

尚不完全清楚您想要做什么,但您也许可以使用关联的引用。例如,当您想要一个对象的唯一标识符时,您将尝试使用以下方法获取它:

NSUUID* uuid = objc_getAssociatedObject(theObject, myKey);

如果对象尚未分配唯一 ID,则返回nil. 在这种情况下,您可以生成唯一 ID 并将其设置为对象:

if (!uuid)
{
    uuid = [NSUUID UUID];
    objc_setAssociatedObject(theObject, myKey, uuid, OBJC_ASSOCIATION_RETAIN);
}

关键myKey是您的图书馆独有的东西。一种常见的策略(例如,用于 KVO 的上下文)是创建一个静态变量并使用其地址:

static const int foo;
static const void* const myKey = &foo;

您可以将这些组合起来以使密钥拥有自己的地址:

static void* const myKey = (void*)&myKey;

无论如何,关联对象将在与其关联的对象被释放时被释放(或者如果在其上设置了不同的关联对象,尽管在您的情况下不会这样做)。

不幸的是,这种方法不是线程安全的,因为两个线程都可以尝试查找唯一 ID,发现尚未分配任何 ID,然后生成一个新 ID,然后分配它。最后一个分配将获胜,但与此同时,另一个可能返回的 ID 最终被替换。实际上,该对象将具有两个不同的 ID。

因此,如果两个线程有​​可能同时尝试查询 ID,则需要在方法中提供同步。例如,创建一个串行 GCD 队列和dispatch_sync()一个包含上述代码的块到该队列。

于 2015-02-11T12:43:13.890 回答