iOS ------代理 分类 拓展

news2025/1/10 23:53:00

代理协议

一,概念:

代理,又称委托代理(delegate),是iOS中常用的一种设计模式。顾名思义,它是把某个对象要做的事委托给别的对象去做。那么别的对象就是这个对象的代理,代替它来打理要做的事,这个对象就是别的对象的委托。

协议,是多个类共享的一个方法列表,在协议中所列出的方法没有响应的实现,由其他类来实现

委托,指给一个对象提供机会对另一个对象中的变化做出反应或者响应另一个对象的行为,其基本思想就是协同解决问题。

在iOS程序设计中,委托通过一种@protocol的方式来实现,所以又称为协议。UITableView, UITextField等都用到这种设计模式。

二,委托模式的好处

  1. 避免子类化带来的过多的子类以及子类与父类的耦合
    委托模式通过将特定功能的实现委托给另一个对象(委托对象),避免了通过创建大量子类来实现不同功能的需要。相比于继承,委托模式更加灵活,可以在运行时动态地改变委托对象,而无需修改或创建新的子类。这样可以减少类的层级结构,提高代码的可维护性和可扩展性,并降低类之间的耦合度。
  2. 通过委托传递消息机制实现分层解耦
    委托模式通过委托对象作为中介,将消息的传递和处理分离。委托对象负责接收并处理某些特定的消息或事件,而委托方只需调用委托对象的方法,将消息传递给委托对象进行处理。这种分层解耦的设计使得委托方和委托对象之间的关注点分离,提高了代码的可读性和可维护性。同时,委托模式也支持多个对象作为委托对象,实现更灵活的消息传递和处理。

三,各自的工作

委托的工作

  1. 定义协议和方法
  2. 声明委托变量
  3. 设置代理
  4. 通过委托变量调用委托方法

代理的工作

  1. 遵循协议
  2. 实现委托方法

具体实现代码参考博客:

iOS代理协议

分类

1.适用范围

  • 当你已经封装好了一个类(也可能是系统类,第三方库),不想再改动这个类,随着程序功能的增加需要在类中添加一个方法,这时我们不必修改主类,只需要给你原来的类增加一个分类。
  • 将一个大型的类拆封成不同的分类,在不同分类中实现类别声明的方法,这样可以将一个类的实现写到多个.m文件中,方便管理和开发。

2.语法格式

.h

#import "ECOPerson.h"

NS_ASSUME_NONNULL_BEGIN

@interface ECOPerson (Work)
- (void)performDaysWork;
- (void)takeVacationFromWork;
@end

NS_ASSUME_NONNULL_END

.m

#import "ECOPerson+Work.h"

@implementation ECOPerson (Work)
- (void)performDaysWork {
    
}
- (void)takeVacationFromWork {
    
}
@end

文件名:主类名+分类名

调用方法时,只需要向主类引用发送消息即可

3.注意事项

  • 分类中的方法的优先级比原来的类中的方法高,如果在分类中重写了原来类的方法,那么分类中的方法会覆盖原来类中的方法。
  • 分类中只能声明方法,不能添加属性变量
  • 通常,分类.h文件,但也可以定义在.m文件,此时分类的方法就变成了私有方法。

4,为什么分类中只能声明方法,不能添加属性变量

先看一段代码

@interface Person (Friendship)
@property (nonatomic, strong) NSString *friends;

- (void)addFriend:(Person*)person;
- (void)removeFriend:(Person*)person;
- (BOOL)isFriendWith:(Person*)person;

@end

编译会报错
在这里插入图片描述
意思是说此分类无法合成与friends属性相关的实例变量.

我们先看一看分类的本质

分类(Category)是OC中的特有语法,它是表示一个指向分类的结构体的指针。原则上它只能增加方法,不能增加成员(实例)变量。具体原因看源码组成:

在这里插入图片描述

在分类的结构体指针中,没有属性列表,只有方法列表。所以原则上它只能添加方法,不能添加属性(成员变量),实例上可以通过其他方式添加属性

问题:
为什么在分类中声明属性时,运行不会出错?既然分类不让添加属性,为什么写@property仍然可以编译通过?

我们知道用@property声明属性时,编译器会自动帮我们生成_成员变量和setter和getter方法,但在分类的结构体指针中根本没有成员列表,所以在分类中用@property声明属性,即无法生成_成员变量和setter和getter方法只有getter和setter和声明。编译和运行都可以通过。如果调用了_成员变量和setter/getter方法,报错就会发生。

5.虽然不能再类别中定义成员属性,但是也可以让它支持添加属性和成员变量

OC是动态的,我们可以手动添加。
一种常见的方法就是通过runtime.h 中objc_getAssociatedObject / objc_setAssociatedObject来访问和生成关联对象。通过这种方法来模拟生成对象。

在这里插入图片描述

在这里插入图片描述

在其他地方使用
在这里插入图片描述

注意,以上代码只是手动实现了setter/getter方法,但调用_成员变量依然会报错

6.关联对象

1.主要函数:
void objc_setAssociatedObject(id object, const void *key, id value, objc_AssociationPolicy policy);
//相当于 setValue:forKey 进行关联value对象
//objc_AssociationPolicy  属性 是设定该value在object内的属性,即 assgin, (retain,nonatomic)...等
id objc_getAssociatedObject(id object, const void *key); 
//用来读取对象
void objc_removeAssociatedObjects(id object);
//函数来移除一个关联对象,或者使用objc_setAssociatedObject函数将key指定的关联对象设置为nil。
2.相关参数
  • 参数一:id object : 给哪个对象添加属性,这里要给自己添加属性,用self。
  • 参数二:void * == id key : key值,根据key获取关联对象的属性的值,在objc_getAssociatedObject中通过次key获得属性的值并返回。
  • 参数三:id value : 关联的值,也就是set方法传入的值给属性去保存。
  • 参数四:objc_AssociationPolicy policy : 策略,属性以什么形式保存。
OBJC_ASSOCIATION_ASSIGN 等价于 @property(assign)。
OBJC_ASSOCIATION_RETAIN_NONATOMIC等价于 @property(strong, nonatomic)。
OBJC_ASSOCIATION_COPY_NONATOMIC等价于@property(copy, nonatomic)。
OBJC_ASSOCIATION_RETAIN等价于@property(strong,atomic)。
OBJC_ASSOCIATION_COPY等价于@property(copy, atomic)

我们在代码中使用了 OBJC_ASSOCIATION_RETAIN_NONATOMIC 就相当于使用了 nonatomic 和 strong 修饰符。

4.关联对象的原理
实现关联对象技术的核心对象
  • AssociationsManager
  • AssociationsHashMap
  • ObjectAssociationMap
  • ObjcAssociation

源码实现

objc_setAssociatedObject函数的源码实现
void
objc_setAssociatedObject(id object, const void *key, id value, objc_AssociationPolicy policy)
{
    _object_set_associative_reference(object, key, value, policy);
}

再看_object_set_associative_reference方法

void _object_set_associative_reference(id object, const void *key, id value, uintptr_t policy)
{
    // 这段代码在以前当 object 和 key 为 nil 时是有效的。某些代码可能依赖于此以避免崩溃。显式检查和处理。
    // rdar://problem/44094390
    if (!object && !value) return;
    // 如果 object 和 value 均为 nil,直接返回,避免崩溃

    if (object->getIsa()->forbidsAssociatedObjects())
        _objc_fatal("objc_setAssociatedObject called on instance (%p) of class %s which does not allow associated objects", object, object_getClassName(object));
    // 如果对象所属的类禁止使用关联对象,则触发致命错误

    DisguisedPtr<objc_object> disguised{(objc_object *)object};
    ObjcAssociation association{policy, value};
    // 创建 DisguisedPtr 包装的对象指针和 ObjcAssociation 对象,用于关联对象的存储

    // 在锁之外对新值进行 retain 操作
    association.acquireValue();
    
    bool isFirstAssociation = false;
    {
        AssociationsManager manager;//
        AssociationsHashMap &associations(manager.get());//
        // 获取关联对象的管理器和关联对象的哈希映射

        if (value) {
            auto refs_result = associations.try_emplace(disguised, ObjectAssociationMap{});//
            if (refs_result.second) {
                /* it's the first association we make */
                isFirstAssociation = true;
            }
            // 如果 value 不为空,则将对象和一个新的关联对象映射插入哈希映射中,并标记为第一个关联对象

            /* establish or replace the association */
            auto &refs = refs_result.first->second;
            auto result = refs.try_emplace(key, std::move(association));
            if (!result.second) {
                association.swap(result.first->second);
            }
            // 建立或替换关联对象,将关联对象存储在哈希映射的指定键下
        } else {
            auto refs_it = associations.find(disguised);
            if (refs_it != associations.end()) {
                auto &refs = refs_it->second;
                auto it = refs.find(key);
                if (it != refs.end()) {
                    association.swap(it->second);
                    refs.erase(it);
                    if (refs.size() == 0) {
                        associations.erase(refs_it);
                    }
                }
            }
            // 如果 value 为空,则从哈希映射中删除指定键对应的关联对象
        }
    }

    // 在锁之外调用 setHasAssociatedObjects 方法,如果对象实现了 _noteAssociatedObjects 方法,则会触发调用,并且可能触发 +initialize 方法,从而执行其他操作,包括设置更多的关联对象
    if (isFirstAssociation)
        object->setHasAssociatedObjects();

    // 在锁之外释放旧值
    association.releaseHeldValue();
}

运行思路可以总结如下:

  1. 首先,检查传入的对象和值是否为 nil,如果都是 nil,则直接返回,避免崩溃。
  2. 检查对象所属的类是否禁止使用关联对象,如果是,则触发致命错误。
  3. 创建一个 DisguisedPtr 对象,用于包装对象指针,以及一个 ObjcAssociation 对象,用于存储关联对象的策略和值。
  4. 在锁之外对新值进行 retain 操作,确保新值在函数执行期间有效。
  5. 获取关联对象的管理器和关联对象的哈希映射。
  6. 如果新值不为空,将对象和一个新的关联对象映射插入哈希映射中,并标记为第一个关联对象。
  7. 建立或替换关联对象,将关联对象存储在哈希映射的指定键下。
  8. 如果新值为空,从哈希映射中删除指定键对应的关联对象。
  9. 在锁之外调用对象的 setHasAssociatedObjects 方法,如果对象实现了 _noteAssociatedObjects 方法,则会触发调用,并且可能触发 +initialize 方法,从而执行其他操作,包括设置更多的关联对象。
  10. 在锁之外释放旧值,确保旧值在函数执行期间正确释放。
_object_get_associative_reference函数的源码实现
id _object_get_associative_reference(id object, const void *key) {
    ObjcAssociation association{}; // 创建一个空的 ObjcAssociation 对象,用于存储关联对象的策略和值

    {
        AssociationsManager manager; // 创建 AssociationsManager 对象,用于管理关联对象的哈希表
        AssociationsHashMap &associations(manager.get()); // 获取关联对象的哈希表
        AssociationsHashMap::iterator i = associations.find((objc_object *)object); // 在哈希表中查找指定对象的关联对象集合
        if (i != associations.end()) {
            ObjectAssociationMap &refs = i->second; // 获取指定对象的关联对象集合
            ObjectAssociationMap::iterator j = refs.find(key); // 在关联对象集合中查找指定键的关联对象
            if (j != refs.end()) {
                association = j->second; // 将查找到的关联对象赋值给 association
                association.retainReturnedValue(); // 对返回的关联对象进行保留操作,以便在返回后保持有效
            }
        }
    }

    return association.autoreleaseReturnedValue(); // 对返回的关联对象进行自动释放操作,并返回该关联对象
}

这段代码的运行思路如下:

  1. 首先,创建一个空的 ObjcAssociation 对象 association,用于存储关联对象的策略和值。

  2. 创建 AssociationsManager 对象 manager,用于管理关联对象的哈希表。

  3. 通过 manager.get() 获取关联对象的哈希表 associations

  4. associations 哈希表中查找指定对象 (objc_object *)object 的关联对象集合,使用 associations.find() 方法进行查找,返回一个迭代器 i

  5. 如果找到了指定对象的关联对象集合,进入内部的 if 代码块。

  6. 在关联对象集合 refs 中查找指定键 key 的关联对象,使用 refs.find() 方法进行查找,返回一个迭代器 j

  7. 如果找到了指定键的关联对象,进入内部的第二个 if 代码块。

  8. 将找到的关联对象赋值给 association

  9. 对返回的关联对象进行保留操作,使用 association.retainReturnedValue() 方法。

  10. 返回自动释放后的关联对象,使用 association.autoreleaseReturnedValue() 方法。

总体而言,这段代码的目的是在给定的对象中查找指定键的关联对象,并返回关联对象的值。它通过迭代访问嵌套的哈希表结构来实现这一目标,并使用 ObjcAssociation 类来封装和管理关联对象的策略和值

_object_remove_associations函数的源码实现
void _object_remove_associations(id object, bool deallocating) {
    ObjectAssociationMap refs{}; // 创建一个空的 ObjectAssociationMap 对象,用于存储关联对象的集合

    {
        AssociationsManager manager; // 创建 AssociationsManager 对象,用于管理关联对象的哈希表
        AssociationsHashMap &associations(manager.get()); // 获取关联对象的哈希表
        AssociationsHashMap::iterator i = associations.find((objc_object *)object); // 在哈希表中查找指定对象的关联对象集合
        if (i != associations.end()) {
            refs.swap(i->second); // 交换指定对象的关联对象集合和空的 refs,将关联对象集合存储在 refs 中

            // 如果不是正在释放对象,则保留 SYSTEM_OBJECT 类型的关联对象
            bool didReInsert = false;
            if (!deallocating) {
                for (auto &ref : refs) {
                    if (ref.second.policy() & OBJC_ASSOCIATION_SYSTEM_OBJECT) {
                        i->second.insert(ref); // 将 SYSTEM_OBJECT 类型的关联对象重新插入到关联对象集合中
                        didReInsert = true;
                    }
                }
            }
            if (!didReInsert)
                associations.erase(i); // 如果没有重新插入 SYSTEM_OBJECT 类型的关联对象,则从哈希表中删除关联对象集合
        }
    }

    SmallVector<ObjcAssociation *, 4> laterRefs; // 创建一个 SmallVector 对象,用于存储稍后释放的关联对象

    // 释放所有关联对象(在锁的外部进行)
    for (auto &i : refs) {
        if (i.second.policy() & OBJC_ASSOCIATION_SYSTEM_OBJECT) {
            // 如果不是正在释放对象,则将 RELEASE_LATER 类型的关联对象存储在 laterRefs 中
            if (deallocating)
                laterRefs.append(&i.second);
        } else {
            i.second.releaseHeldValue(); // 释放普通类型的关联对象的值
        }
    }
    for (auto *later : laterRefs) {
        later->releaseHeldValue(); // 释放稍后释放的关联对象的值
    }
}

这段代码的运行思路如下:

  1. 首先创建一个空的 ObjectAssociationMap 对象 refs,用于存储关联对象的集合。

  2. 创建 AssociationsManager 对象 manager,用于管理关联对象的哈希表。

  3. 通过 manager.get() 获取关联对象的哈希表 associations

  4. associations 哈希表中查找指定对象 (objc_object *)object 的关联对象集合,使用 associations.find() 方法进行查找,返回一个迭代器 i

  5. 如果找到了指定对象的关联对象集合,进入内部的 if 代码块。

  6. 将指定对象的关联对象集合和空的 refs 进行交换,将关联对象集合存储在 refs 中。

  7. 如果不是正在释放对象(deallocating 为假),则遍历 refs 中的关联对象。

  8. 对于具有 OBJC_ASSOCIATION_SYSTEM_OBJECT 策略的关联对象,重新插入到关联对象集合中,以保留这些关联对象。

  9. 如果没有重新插入 OBJC_ASSOCIATION_SYSTEM_OBJECT 类型的关联对象,则从哈希表中删除关联对象集合。

  10. 创建一个 SmallVector 对象 laterRefs,用于存储稍后需要释放的关联对象。

  11. 遍历 refs 中的关联对象。

  12. 对于具有 OBJC_ASSOCIATION_SYSTEM_OBJECT 策略的关联对象,如果正在释放对象(deallocating 为真),将其存储在 laterRefs 中,稍后释放。

  13. 对于普通类型的关联对象,调用 releaseHeldValue() 方法释放其值。

  14. 遍历 laterRefs 中的稍后释放的关联对象。

  15. 对于 laterRefs 中的关联对象,调用 releaseHeldValue() 方法释放其值。

这段代码的功能是移除指定对象的关联对象,并根据标志 deallocating 决定是否保留特定类型的关联对象。它通过使用哈希表和向量来存储和管理关联对象,并根据不同的条件执行相应的操作,最终释放关联对象的值。

下面具体看一看实现关联对象技术的四个核心对象

ASSociatonsManager
class AssociationsManager {
    using Storage = ExplicitInitDenseMap<DisguisedPtr<objc_object>, ObjectAssociationMap>;
    // 使用 DisguisedPtr<objc_object> 作为键,ObjectAssociationMap 作为值的哈希映射类型别名

    static Storage _mapStorage;
    // 静态成员变量,用于存储对象的关联对象的哈希映射

public:
    AssociationsManager() { AssociationsManagerLock.lock(); }
    // 构造函数,在创建 AssociationsManager 实例时加锁

    ~AssociationsManager() { AssociationsManagerLock.unlock(); }
    // 析构函数,在销毁 AssociationsManager 实例时解锁

    AssociationsHashMap& get() {
        return _mapStorage.get();
    }
    // 返回对象的关联对象的哈希映射

    static void init() {
        _mapStorage.init();
    }
    // 静态方法,用于初始化关联对象的哈希映射
};

AssociationsManager 类具体实现了以下功能:

  1. 它使用静态成员变量 _mapStorage 存储对象的关联对象的哈希映射。
  2. 通过构造函数和析构函数的加锁和解锁操作,确保在创建和销毁 AssociationsManager 实例时对关联对象的访问是线程安全的。
  3. AssociationsManager 提供了 get() 方法,用于获取对象的关联对象的哈希映射。这样,其他代码可以通过调用 get() 方法来获取特定对象的关联对象,并进行操作。
  4. AssociationsManager 还提供了 init() 方法,用于初始化关联对象的哈希映射。这个方法在适当的时候被调用,以确保在使用关联对象之前,相关的数据结构已经被正确初始化。
ObjcAssociation
class ObjcAssociation {
    uintptr_t _policy;
    id _value;

public:
    ObjcAssociation(uintptr_t policy, id value) : _policy(policy), _value(value) {}
    // 构造函数,使用给定的策略和值初始化 ObjcAssociation 对象

    ObjcAssociation() : _policy(0), _value(nil) {}
    // 默认构造函数,将策略和值初始化为默认值

    ObjcAssociation(const ObjcAssociation& other) = default;
    ObjcAssociation& operator=(const ObjcAssociation& other) = default;
    // 默认拷贝构造函数和赋值运算符重载

    ObjcAssociation(ObjcAssociation&& other) : ObjcAssociation() {
        swap(other);
    }
    // 移动构造函数,通过交换操作,将源对象的值转移到当前对象

    inline void swap(ObjcAssociation& other) {
        std::swap(_policy, other._policy);
        std::swap(_value, other._value);
    }
    // 交换当前对象和另一个对象的值

    inline uintptr_t policy() const { return _policy; }
    // 返回对象的关联策略

    inline id value() const { return _value; }
    // 返回对象的关联值

    inline void acquireValue() {
        if (_value) {
            switch (_policy & 0xFF) {
                case OBJC_ASSOCIATION_SETTER_RETAIN:
                    _value = objc_retain(_value);
                    break;
                case OBJC_ASSOCIATION_SETTER_COPY:
                    _value = ((id(*)(id, SEL))objc_msgSend)(_value, @selector(copy));
                    break;
            }
        }
    }
    // 获取关联值并根据策略进行内存管理

    inline void releaseHeldValue() {
        if (_value && (_policy & OBJC_ASSOCIATION_SETTER_RETAIN)) {
            objc_release(_value);
        }
    }
    // 释放持有的关联值,如果策略需要释放内存

    inline void retainReturnedValue() {
        if (_value && (_policy & OBJC_ASSOCIATION_GETTER_RETAIN)) {
            objc_retain(_value);
        }
    }
    // 对返回的关联值进行保留操作,如果策略需要保留内存

    inline id autoreleaseReturnedValue() {
        if (slowpath(_value && (_policy & OBJC_ASSOCIATION_GETTER_AUTORELEASE))) {
            return objc_autorelease(_value);
        }
        return _value;
    }
    // 对返回的关联值进行自动释放操作,如果策略需要自动释放内存
};

ObjcAssociation 类的实例对象可以用于存储一个对象的关联对象的策略和值。它的成员变量 _policy 存储关联对象的策略信息_value 存储关联对象的值
该类提供了以下主要功能和方法:

  1. 构造函数:通过构造函数可以创建一个 ObjcAssociation 对象,并指定关联对象的策略和值。
  2. 值获取和设置:通过 policy() 方法可以获取关联对象的策略,通过 value() 方法可以获取关联对象的值。
  3. 内存管理:acquireValue() 方法用于根据策略进行内存管理,它会根据策略对关联对象的值进行保留操作或复制操作。releaseHeldValue() 方法用于释放持有的关联值,如果策略需要释放内存。retainReturnedValue() 方法用于对返回的关联值进行保留操作,如果策略需要保留内存。autoreleaseReturnedValue() 方法用于对返回的关联值进行自动释放操作,如果策略需要自动释放内存。
AssociationsHashMap和 ObjectAssociationMap
typedef DenseMap<const void *, ObjcAssociation> ObjectAssociationMap;
typedef DenseMap<DisguisedPtr<objc_object>, ObjectAssociationMap> AssociationsHashMap;

这段代码定义了两个类型别名:ObjectAssociationMapAssociationsHashMap,它们都是基于 LLVM 的 DenseMap 实现的哈希表。

  1. ObjectAssociationMap 是一个哈希表,用于存储指向 ObjcAssociation 类对象的指针和对应的关联对象。它的键类型是 const void *,即指向常量空指针的指针类型,通常用作关联对象的标识符。值类型是 ObjcAssociation,即 ObjcAssociation 类的对象,用于存储关联对象的策略和值。
  2. AssociationsHashMap 是一个嵌套的哈希表,用于存储 Objective-C 对象和其关联对象的映射关系。它的键类型是 DisguisedPtr<objc_object>,是一个包装了 objc_object 类型的指针的类模板,用于隐藏指针的真实类型。值类型是 ObjectAssociationMap,即前面提到的 ObjectAssociationMap 类型,用于存储指向 ObjcAssociation 类对象的指针和对应的关联对象。
    这些哈希表的设计是为了高效地存储和检索 Objective-C 对象的关联对象。

ObjectAssociationMap 通过将关联对象的标识符作为键,与实际的 ObjcAssociation 对象建立映射关系。而 AssociationsHashMap 则通过将 Objective-C 对象的指针作为键,与 ObjectAssociationMap 建立映射关系,从而实现了快速查找某个对象的关联对象的能力

总结一下这几个核心类的关系

在这里插入图片描述

总结
  • 关联对象并不存储在被关联对象本身的内存中,而是有一个全局统一的AssociationManager中
  • AssociationsHashMap中存储着多个实例对象的指针DisguisedPtr<objc_object>和对应的ObjectAssociationMap
  • 一个实例对象就对应一个ObjectAssociationMap,ObjectAssociationMap中存储多个此实例对象的关联对象的key和objcAssciation
  • objcAssciation中存储着关联对象的value和policy策略
使用场景
  • 添加额外的属性:当你需要为现有的类添加属性,但又无法修改该类的源代码时,可以使用关联对象。例如,你可以为系统类添加自定义属性,以满足特定的业务需求。
  • 为分类添加属性:分类无法直接添加实例变量,但通过关联对象,你可以为分类添加属性。这允许你为现有类的不同分类添加自定义属性,以扩展其功能。
  • 在类别中添加方法实现:通过关联对象,你可以在运行时动态地将方法实现关联到一个对象上。这样,你可以在不修改原始类的情况下,为特定的对象或类别添加自定义行为。
  • 实现代理模式:当一个对象需要充当另一个对象的代理时,可以使用关联对象将代理对象与被代理对象关联起来。这样,代理对象就可以接收并处理被代理对象的消息。
  • 实现观察者模式:通过关联对象,你可以动态地添加观察者对象,并将其与被观察的对象关联起来。这样,观察者对象可以接收到被观察对象的状态变化通知。
  • 扩展第三方库或框架:当你需要在第三方库或框架中添加自定义属性或行为时,可以使用关联对象。这样,你可以在不修改第三方代码的情况下,为其添加额外的功能。

拓展

1,适用范围

拓展是分类的一种特殊形式

2,语法格式

@interface 主类类名()

@end

拓展通常定义在主类的.m文件中,拓展声明的方法直接在主类的.m文件中实现

3,注意事项

  • 拓展可以声明实例变量,可以声明属性
  • 因为拓展通常定义在.m文件中,所以拓展声明的文件和属性通常是私有的

4,使用

方式一:以单独的文件定义

XYZPopViewController_ExViewController.h文件
#import"XYZPopViewController.h"

@interface XYZPopViewController()
@property(nonatomic,strong)NSString*stringOfEx;
- (void)testEx;
@end

在主类的.m文件中定义

XYZPopViewController.m文件
#import"XYZPopViewController.h"

@interface XYZPopViewController()
@property(nonatomic,strong)NSString*stringOfEx;
- (void)testEx;
@end

在主类的.m文件中实现拓展定义的方法

@implementation XYZPopViewController
- (void)testEx {
     self.stringOfEx=@"给扩展里面定义的属性字符串赋值";
     NSLog(@"定义的属性String是:%@",self.stringOfEx);
 }
 @end

拓展和分类的区别

  1. 分类是不可以声明实例变量,通常是公开的,文件名是:主类名+分类名.h
  2. 扩展是可以声明实例变量,是私有的,扩展只能在自身类中使用,而不是子类或者其他地方,
    文件名为:主类名_扩展标识.h,在主类的.m文件中#import该头文件
  3. 类拓展是在编译阶段添加到类中,而分类是在运行时添加到类中

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1619029.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【简单讲解下如何学习C++】

&#x1f3a5;博主&#xff1a;程序员不想YY啊 &#x1f4ab;CSDN优质创作者&#xff0c;CSDN实力新星&#xff0c;CSDN博客专家 &#x1f917;点赞&#x1f388;收藏⭐再看&#x1f4ab;养成习惯 ✨希望本文对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出…

P1106 删数问题

本题为洛谷&#xff1a; #include<iostream> #include<string> using namespace std; int main(){string n;int k;cin>>n>>k;while(k--){for(int i0;i<n.length();i){if(n[i]>n[i1]){n.erase(i,1); break;} }for(int i0;i<n.length()-1&&…

uniapp APP检测更新

需求&#xff1a; 1.首次进入APP给出弹窗提示是否存在最新版本APP&#xff0c;可选择更新或者取消 2.选择取消后&#xff0c;在使用期间不再弹出该弹窗 3.在设置中增加按钮&#xff0c;点击进行版本检测&#xff0c;再弹窗 效果图&#xff1a; 使用到的插件&#xff1a;APP升…

如何理解自然语言处理中的位置编码(Positional Encoding)

在自然语言处理和特别是在使用Transformer模型中,位置编码(Positional Encoding)是一个关键的概念。它们的作用是为模型提供序列中各个元素的位置信息。由于Transformer架构本身并不像循环神经网络(RNN)那样具有处理序列的固有能力,位置编码因此显得尤为重要。 为什么需…

大语言模型微调过程中的 RLHF 和 RLAIF 有什么区别?

目前想要深入挖掘大型语言模型&#xff08;LLM&#xff09;的全部潜力需要模型与我们人类的目标和偏好保持一致。从而出现了两种方法&#xff1a;来自人类反馈的人力强化学习&#xff08;RLHF&#xff09;和来自人工智能反馈的人工智能驱动的强化学习&#xff08;RLAIF&#xf…

linux磁盘管理和挂载和镜像源管理

一.分清文件类型 所有的设备都在/dev中 我们知道d开头是目录 c开头是外部设备比如打印机&#xff0c;鼠标等外接设备 b开头才是我们要找的块文件&#xff0c;即硬盘 二. 3个磁盘管理 1.mount负责挂载外部磁盘2.fdisk查看磁盘分区情况3.lsblk查看可用设备信息以及它们的挂载点…

Android Studio Iguana | 2023.2.1配置优化

一. 前言 本篇文章记录最新版本的Android Studio的配置优化&#xff0c;写这篇文章的是由于电脑中的AS工具更新版本覆盖安装后&#xff0c;AS会经常卡死&#xff0c;Debug的时候也经常莫名其妙的断掉&#xff0c;非常影响工作效率&#xff0c;所以重新把配置环境整理一下&#…

模型部署的艺术:让深度学习模型跃入生产现实

模型部署的艺术&#xff1a;让深度学习模型跃入生产现实 1 引言 1.1 部署的意义&#xff1a;为何部署是项目成功的关键 在深度学习项目的生命周期中&#xff0c;模型的部署是其成败的关键之一。通常&#xff0c;一个模型从概念构思、数据收集、训练到优化&#xff0c;最终目的…

Python网络爬虫-详解XPath匹配网页数据

前言 XPath&#xff0c;全称XML Path Language&#xff0c;即XML路径语言&#xff0c;它是一门在XML文档中查找信息的语言。XPath使用路径表达式来选取XML文档中的节点或节点集。这些节点是通过沿着路径&#xff08;path&#xff09;或者步&#xff08;steps&#xff09;来选取…

grafana报错This panel requires Angular (deprecated)

1.原因 报错解释&#xff1a; Grafana在更新到7.0版本后&#xff0c;弃用了AngularJS&#xff08;一种用于构建大型Web应用的JavaScript框架&#xff09;。在早期的Grafana版本中&#xff0c;某些面板可能依赖于AngularJS&#xff0c;但这种依赖已经逐步被新的React或Vue面板所…

[论文笔记] EcomGPT:COT扩充数据的电商大模型

社区供稿 | EcomGPT:基于任务链数据的电商大模型(附魔搭推理实践) - 知乎 https://arxiv.org/pdf/2312.15696.pdf EcomInstruct指令数据集构建 数据集组成 COT方式构造垂域训练数据:把原本的垂域任务分解成了原子任务,构造了基于解决原子任务的数据。这样能用类似…

更易使用,OceanBase开发者工具 ODC 4.2.4 版本升级

亲爱的朋友们&#xff0c;大家好&#xff01;我们的ODC&#xff08;OceanBase Developer Center &#xff09;再次迎来了重要的升级V 4.2.4&#xff0c;这次我们诚意满满&#xff0c;从五个方面为大家精心打造了一个更加易用、贴心&#xff0c;且功能更强的新版本&#xff0c;相…

网动统一通信平台(Active UC) downloadDocument.action 任意文件读取漏洞复现

0x01 产品简介 网动统一通信平台(Active UC) 是一个涵盖了多种通信功能的综合平台&#xff0c;通常包括文字、语音、视频通讯等功能&#xff0c;并且可能提供了一系列的通讯工具和服务。这样的平台通常旨在提升用户的沟通效率和便利性&#xff0c;为用户提供一个统一的通信环境…

探索未来的区块链DApp应用,畅享数字世界的无限可能

随着区块链技术的飞速发展&#xff0c;分布式应用&#xff08;DApp&#xff09;正成为数字经济中的一股强劲力量。DApp以其去中心化、透明公正的特点&#xff0c;为用户带来了全新的数字体验&#xff0c;开创了数字经济的新潮流。作为一家专业的区块链DApp应用开发公司&#xf…

服务器(AIX、Linux、UNIX)性能监视器工具【nmon】使用介绍

目录 ■nmon简介 1.安装 2.使用简介 3.使用&#xff08;具体使用的例子【CPU】【内存】&#xff09; 4.采集数据 5.查看log&#xff08;根据结果&#xff0c;生成报表&#xff09; 6.分析结果 ■nmon简介 nmon&#xff08;"Nigels performance Monitor"&…

贪吃蛇(C语言版)

在我们学习完C语言 和单链表知识点后 我们开始写个贪吃蛇的代码 目标&#xff1a;使用C语言在Windows环境的控制台模拟实现经典小游戏贪吃蛇 贪吃蛇代码实现的基本功能&#xff1a; 地图的绘制 蛇、食物的创建 蛇的状态&#xff08;正常 撞墙 撞到自己 正常退出&#xf…

IDEA pom.xml依赖警告

IDEA中&#xff0c;有时 pom.xml 中会出现如下提示&#xff1a; IDEA 2022.1 升级了检测易受攻击的 Maven 和 Gradle 依赖项&#xff0c;并建议修正&#xff0c;通过插件 Package Checker 捆绑到 IDE 中。 这并不是引用错误&#xff0c;不用担心。如果实在强迫症不想看到这个提…

pycharm远程连接server

1.工具–部署–配置 2.部署完成后&#xff0c;将现有的项目的解释器设置为ssh 解释器。实现在远端开发 解释器可以使用/usr/bin/python3

C#-使用Harmony库实现DLL文件反射调用

一. Harmony工作原理 利用C#运行时Runtime的反射机制,动态加载dll中的方法,字段,属性,实现对DLL方法的重写和代码注入。 二. Harmony下载及安装 1.下载Harmony_lib库lib.harmony.2.3.3.nupkg 霸王•吕布 / CSharpHarmonyLib GitCodehttps://gitcode.net/qq_35829452/csharph…

万界星空科技电机行业MES+商业电机行业开源MES+项目合作

要得出mes系统解决方案在机电行业的应用范围&#xff0c;我们先来看一下传统机电行业的管理难题&#xff1a; 1、 产品标准化程度较低&#xff0c;制造工艺复杂&#xff0c;生产周期较长&#xff0c;产品质量不稳定&#xff1b; 2、 自动化程度低&#xff0c;大多数工序以手工…