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_915921435 小时前
iOS 虚拟位置设置实战,多工具协同打造精准调试与场景模拟环境
android·ios·小程序·https·uni-app·iphone·webview
QuantumLeap丶5 小时前
《Flutter全栈开发实战指南:从零到高级》- 11 -状态管理Provider
android·flutter·ios
2501_916008895 小时前
App 上架需要什么?从开发者账号到开心上架(Appuploader)免 Mac 上传的完整流程指南
macos·ios·小程序·uni-app·objective-c·cocoa·iphone
QuantumLeap丶1 天前
《Flutter全栈开发实战指南:从零到高级》- 09 -常用UI组件库实战
flutter·ios·dart
2501_915918411 天前
App 上架苹果商店全流程详解 从开发者账号申请到开心上架(Appuploader)跨平台免 Mac 上传实战指南
macos·ios·小程序·uni-app·objective-c·cocoa·iphone
2501_916007471 天前
从零开始学习iOS App开发:Xcode、Swift和发布到App Store完整教程
android·学习·ios·小程序·uni-app·iphone·xcode
Pluto5381 天前
第一个app产品的迭代
ios·github
2501_915921431 天前
iOS 26 CPU 使用率监控策略 多工具协同构建性能探索体系
android·ios·小程序·https·uni-app·iphone·webview
狂团商城小师妹1 天前
JAVA国际版同城打车源码同城服务线下结账系统源码适配PAD支持Android+IOS+H5
android·java·ios·小程序·交友
游戏开发爱好者81 天前
iOS 应用逆向对抗手段,多工具组合实战(iOS 逆向防护/IPA 混淆/无源码加固/Ipa Guard CLI 实操)
android·ios·小程序·https·uni-app·iphone·webview