草庐IT

分类,多继承

唔嗷嗷嗷 2023-03-28 原文

谈谈你对分类Category的理解

使用场景:可以减少单个文件体积;可以按照功能分组,放到不同的分类里,使结构更清晰;降低耦合性,一个类可以有多个开发人员进行开发;模拟多继承。

Category的实现原理:Category编译之后的底层结构是结构体 category_t ,里面存储着分类的对象方法、类方法、属性、协议等信息。在程序运行的时候,runtime会将category中的数据,合并到类信息中(类对象、元类对象)


Category与 Extension的区别是什么:

1、Category 原则上只能增加方法,能添加属性但是需要配合runtime才能解决没有setter和getter的问题,这一点前文中有说明,下文中等下也再提一下。

2、Extension不仅可以添加方法,还可以增加属性,只是属性默认的访问权限是private。

3、Extension 中生命的方法没有在.m 文件中实现,编译器会报警,但是在Category中的方法没有实现,编译器没有任何警告。这是因为Extension是在编译阶段添加到类中的,而Category是在运行时添加到类中的。

如何通过Category实现多继承?

比如有TestA和TestB两个类,如图


现在想实现一个TestC,让TestC同时继承TestA,和TestB。实现如下




测试结果如下


以上是通过分类的方式。还可以通过消息转发的方式:(先注释掉分类中的所有代码,再加消息转发代码)


这里方法签名、和forwardInvocation的方式我就不展示了,之前的文章中有讲过

测试结果:


可以看到。我们在调用c 执行playBasketBall和playFootBall时,其实调用到的是A 和B的方法。这里抛一个问题出来,假设我在消息转发时这样写:把setAge:  age这些也加到消息转发里,那么,外面在给C设置age的时候,调用的是谁的set方法、get方法?

答案是,调用c自己的set方法,因为进入消息转发前,需要先经过消息发送、动态方法解析。 当c本身就有setAge方法时,在消息发送阶段就可以找到自己的setAge,直接调用自己的setter了,根本走不到消息转发这个环节来。

当然上述两个方法应该也可以配合使用,有兴趣可以自己写demo试试。

还有什么方式可以实现呢?

比如组合拳,C 先继承A,然后C再强引用一个B的对象,这样不管是调A的方法还是B的方法,都可以。

再比如协议,把A、B中需要继承的方法,写到协议里,然后C遵循这两个协议,就可以在C中拥有A、B的方法了。

有兴趣可以写demo试试

有关分类,多继承的更多相关文章

  1. ruby-on-rails - Rails 单表继承 : How to override the value written to the type field - 2

    在我的系统中,我已经定义了STI。Dog继承自Animal,在animals表中有一个type列,其值为"Dog"。现在我想让SpecialDog继承自dog,只是为了在某些特殊情况下稍微修改一下行为。数据还是一样。我需要通过SpecialDog运行的所有查询,以返回数据库中类型为Dog的值。我的问题是因为我有一个type列,rails将WHERE"animals"."type"IN('SpecialDog')附加到我的查询中,所以我不能获取原始的Dog条目。所以我想要的是以某种方式覆盖rails在通过SpecialDog访问数据库时使用的值,使其表现得像Dog。有没有办法覆盖用于类型

  2. ruby-on-rails - 为什么 DataMapper 使用混合与继承? - 2

    所以我只是对此感到好奇:DataMapper为其模型使用混合classPostincludeDataMapper::Resource虽然active-record使用继承classPost有谁知道为什么DataMapper选择这样做(或者为什么AR选择不这样做)? 最佳答案 它允许您从另一个不是DM类的类继承。它还允许动态地将DM功能添加到类中。这是我正在处理的模块中的类方法:defdatamapper_classklass=self.dupklass.send(:include,DataMapper::Resource)klass

  3. ruby-on-rails - Ruby on Rails 单表继承(STI)和单元测试问题(使用 PostgreSQL) - 2

    我正在使用带有单个“帐户”表的STI模型来保存用户和技术人员的信息(即用户...8)错误:test_the_truth(用户测试):ActiveRecord::StatementInvalid:PGError:ERROR:关系“技术人员”不存在:从“技术人员”中删除...从本质上讲,标准框架不承认Technicians和Users表(或PostgreSQL称它们为“关系”)不存在,事实上,应该别名为Accounts。有什么想法吗?我对RoR比较陌生,不知道如何解决这个问题而又不完全删除STI。 最佳答案 原来问题是由于存在:./te

  4. ruby - 为什么 Ruby 模块继承不像类继承那样工作? - 2

    假设我有一个名为Flight的模块,其中包含类方法和实例方法。我可以使用include、extend或两者将其方法放入类中:classBatinclude会将Flight添加到Bat.ancestors,但extend不会。我的问题是,为什么模块与类不同?当我对Mammal进行子类化时,我总是同时获得类和实例方法。然而,当我混入一个模块时,我不能同时获得类和实例方法(除非我使用self.included钩子(Hook)或类似ActiveSupport::Concern的东西)。这种差异背后是否存在语言设计问题? 最佳答案 Modul

  5. ruby - 为什么 Object 在 Ruby 中既包含内核又继承它? - 2

    在Ruby(1.8.X)中为什么Object既继承了内核又包含了内核?仅仅继承还不够吗?irb(main):006:0>Object.ancestors=>[Object,Kernel]irb(main):005:0>Object.included_modules=>[Kernel]irb(main):011:0>Object.superclass=>nil请注意,在Ruby1.9中情况类似(但更简洁):irb(main):001:0>Object.ancestors=>[Object,Kernel,BasicObject]irb(main):002:0>Object.included

  6. Ruby 获取继承类 - 2

    我正在为Rails创建我的第一个插件。我对ruby​​还是很陌生,我想知道是否有可能获得继承类?例如,我正在尝试创建一个插件,在您不使用迁移时允许进行单元测试和功能测试。我要做的是初始化一个名为controller的类变量,以初始化为正在测试的Controller类型。如果我有一个基类ControllerTest:classControllerTest所以我目前坚持的是获取继承类的名称。这可能吗?如果没有,有没有人知道我可以如何着手实现它的另一种方式?提前致谢。 最佳答案 非常简单:使用“继承”回调。来自Class类的RDoc:in

  7. ruby-on-rails - 是吗? Rails 3 中的单表继承失败 - 2

    Object#is_a?在Rails3中以最奇怪的方式失败。我将单表继承设置如下(为简洁起见进行了简化):#resource.rbclassResource在我的Controller中,我有这个:defcreate@resource=Resource.findparams[:resource_id]logger.info'@resourceclass:'+@resource.class.namelogger.info'@resourcesuperclass:'+@resource.class.superclass.namelogger.info'@resourceis_a?(Video

  8. ruby-on-rails - Rspec Controller 在 Rails 中测试继承自 AbstractController::Base 的 Controller - 2

    我正在为我未构建的应用程序编写Controller测试,因此这绝对是一个学习过程。这是我第一次遇到直接继承自AbstractController::Base的Controller。显然,它的行为与其他Controller不同。其格式大致为:classSchwadGenericController我尝试了正常测试,这是我目前要让任何事情发生的地方。require'rails_helper'describeSchwadGenericControllerdo#before(:each)do#SchwadGenericController.skip_authorize_resource#end

  9. ruby - 我如何从 Rational(或任何没有构造函数的类)继承? - 2

    例如,我可以很容易地继承自String,如下所示:classMyString'thingsandstuff'但是我如何继承没有构造函数的Rational呢?例如:defMyRatNoMethodError:undefinedmethod`new'forMyRat:ClassMyRat(10).inc#=>NoMethodError:undefinedmethod`MyRat'formain:ObjectMyRat.send(:initialize,10).inc#=>TypeError:alreadyinitializedclass#???#Noneofitworks!我找不到初始化新

  10. ruby - Ruby 如何处理嵌套类的继承? - 2

    在下面的测试用例中:classPackageclassComponentdefinitializep[:initialize,self]endendendclassPackage_A结果:[:initialize,#]#[:initialize,#]#如何获取特定的Package_A.component和Package_B.component? 最佳答案 ClassComponent是在Package中声明的,所以看起来是正确的。::指示在Package_A范围内查找名称Component。由于那里没有Component,它会查找父

随机推荐