iOS - 关联对象

详细总结 Objective-C 的关联对象功能:

1. 基本使用

objectivec 复制代码
// 1. 设置关联对象
objc_setAssociatedObject(id object, const void *key, id value, objc_AssociationPolicy policy);

// 2. 获取关联对象
id objc_getAssociatedObject(id object, const void *key);

// 3. 移除关联对象
objc_removeAssociatedObjects(id object);

2. 关联策略

objectivec 复制代码
typedef OBJC_ENUM(uintptr_t, objc_AssociationPolicy) {
    OBJC_ASSOCIATION_ASSIGN = 0,           // 弱引用
    OBJC_ASSOCIATION_RETAIN_NONATOMIC = 1, // 强引用,非原子
    OBJC_ASSOCIATION_COPY_NONATOMIC = 3,   // 复制,非原子
    OBJC_ASSOCIATION_RETAIN = 01401,       // 强引用,原子
    OBJC_ASSOCIATION_COPY = 01403          // 复制,原子
};

3. 实现示例

3.1 分类中添加属性

objectivec 复制代码
// UIView+Extension.h
@interface UIView (Extension)
@property (nonatomic, strong) NSString *identifier;
@end

// UIView+Extension.m
@implementation UIView (Extension)

static void *IdentifierKey = &IdentifierKey;

- (void)setIdentifier:(NSString *)identifier {
    objc_setAssociatedObject(self, 
                            IdentifierKey, 
                            identifier, 
                            OBJC_ASSOCIATION_COPY_NONATOMIC);
}

- (NSString *)identifier {
    return objc_getAssociatedObject(self, IdentifierKey);
}

@end

3.2 使用宏简化

objectivec 复制代码
// 定义关联属性的宏
#define OBJC_ASSOCIATION_PROPERTY_STRONG(_getter_, _setter_, _association_) \
- (void)_setter_:(id)object { \
    objc_setAssociatedObject(self, @selector(_getter_), object, _association_); \
} \
- (id)_getter_ { \
    return objc_getAssociatedObject(self, @selector(_getter_)); \
}

// 使用宏
@implementation UIView (Extension)
OBJC_ASSOCIATION_PROPERTY_STRONG(identifier, setIdentifier, OBJC_ASSOCIATION_COPY_NONATOMIC)
@end

4. 关键点

4.1 Key 的选择

objectivec 复制代码
// 1. 静态变量地址
static char kAssociatedObjectKey;
objc_setAssociatedObject(obj, &kAssociatedObjectKey, value, policy);

// 2. 选择器
objc_setAssociatedObject(obj, @selector(propertyName), value, policy);

// 3. 字符串常量
static NSString *const kAssociatedKey = @"kAssociatedKey";
objc_setAssociatedObject(obj, (__bridge void *)kAssociatedKey, value, policy);

4.2 内存管理

objectivec 复制代码
// 强引用示例
objc_setAssociatedObject(self, 
                        @selector(strongProperty), 
                        object, 
                        OBJC_ASSOCIATION_RETAIN_NONATOMIC);

// 弱引用示例
objc_setAssociatedObject(self, 
                        @selector(weakProperty), 
                        object, 
                        OBJC_ASSOCIATION_ASSIGN);

5. 线程安全

objectivec 复制代码
// 原子操作
objc_setAssociatedObject(self, 
                        key, 
                        value, 
                        OBJC_ASSOCIATION_RETAIN); // 使用带原子性的关联策略

// 手动加锁
@synchronized(self) {
    objc_setAssociatedObject(self, 
                            key, 
                            value, 
                            OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}

6. 最佳实践

6.1 属性封装

objectivec 复制代码
@interface NSObject (Associated)

@property (nonatomic, strong) id associatedObject;
@property (nonatomic, weak) id weakAssociatedObject;
@property (nonatomic, copy) NSString *copyAssociatedObject;

@end

6.2 清理处理

objectivec 复制代码
- (void)dealloc {
    // 移除特定关联对象
    objc_setAssociatedObject(self, @selector(associatedObject), nil, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
    
    // 或移除所有关联对象
    objc_removeAssociatedObjects(self);
}

7. 注意事项

objectivec 复制代码
// 1. 避免循环引用
@implementation MyClass
- (void)setupBlock {
    __weak typeof(self) weakSelf = self;
    self.associatedBlock = ^{
        __strong typeof(weakSelf) strongSelf = weakSelf;
        [strongSelf doSomething];
    };
}

// 2. 合理选择关联策略
- (void)setDelegate:(id<MyDelegate>)delegate {
    objc_setAssociatedObject(self, 
                            @selector(delegate), 
                            delegate, 
                            OBJC_ASSOCIATION_ASSIGN); // 代理使用 ASSIGN
}

关联对象的优点:

  1. 无需修改原类即可添加属性
  2. 支持各种属性特性(strong/weak/copy)
  3. 线程安全选项
  4. 运行时动态性

注意事项:

  1. 合理使用关联策略
  2. 注意内存管理
  3. 避免过度使用
  4. 考虑性能影响
  5. 注意命名冲突
相关推荐
木木黄木木3 小时前
Theos环境搭建与XM文件开发指南,以及iOS弹窗源码分享
ios·c#
木木黄木木3 小时前
iOS插件,Theos环境搭建与XM文件开发指南(完善版本)
ios
帅次3 小时前
Flutter:StatelessWidget vs StatefulWidget 深度解析
android·flutter·ios·小程序·swift·webview·android-studio
货拉拉技术5 小时前
货拉拉基于“声明式”的埋点方案实践
ios·程序员
帅次5 小时前
Flutter Widget 体系结构解析
android·flutter·ios·小程序·xcode·web app·dalvik
Mr.NickJJ20 小时前
Swift系列02-Swift 数据类型系统与内存模型
开发语言·ios·swift
二流小码农20 小时前
鸿蒙开发:wrapBuilder传递参数
android·ios·harmonyos
火柴就是我1 天前
flutter rust bridge 编译成so 文件 或者 .a文件 依赖到主项目
flutter·ios·rust
YungFan1 天前
iOS开发之网络代码进化史
ios·swift
SunshineBrother1 天前
shell脚本,怎么查找项目中的重复图片
ios