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. 注意命名冲突
相关推荐
2501_915918414 小时前
Fiddler中文版全面评测:功能亮点、使用场景与中文网资源整合指南
android·ios·小程序·https·uni-app·iphone·webview
不知名It水手5 小时前
uniapp运行项目到ios基座
ios·uni-app·cocoa
Digitally6 小时前
[5种方法] 如何将iPhone短信保存到电脑
ios·iphone
杂雾无尘9 小时前
Swift 5.9 新特性揭秘:非复制类型的安全与高效
ios·swift·apple
Thomas_YXQ13 小时前
Unity3D iOS闪退问题解决方案
ios
Daniel_Coder15 小时前
iOS Widget 开发-7:TimelineProvider 机制全解析:构建未来时间线
ios·swift·widget
Daniel_Coder16 小时前
iOS Widget 开发-3:Widget 的种类与尺寸(主屏、锁屏、灵动岛)
ios·swift·widget
Engandend1 天前
Flutter与iOS混合开发交互
flutter·ios·程序员
山水域1 天前
GoogleAdsOnDeviceConversion 库的作用与用法
ios
Lucifer晓1 天前
记录一次Flutter项目上传App Store Connect出现“Validation failed”错误的问题
flutter·ios