《Effective Ruby:改善Ruby程序的48条建议》一第7条:了解super的不同行为

第7条:了解super的不同行为

假设你已经写好了一个类,这个类继承自一个基类。而被继承的类定义了一个不那么适合新类的方法。因此你决定改进那个方法,但是你不想完全替代既有方法,因为它承担了90%的必要且繁重工作。你也不想改变基类,因为那将破坏其他代码的功能。先忽略掉使用组合而非继承的这个更好的选项,让我们费点力气来重载这个方法。你最后做了这些改动:

你让自己陷入了困境。最大的问题是:你如何调用超类中的m1方法?如果你试图调用衍生类中的方法,你将陷入无限循环中。那没什么帮助。这就到了super出场的时候了:

现在我们有些眉目了。新版的m1方法使用super来调用父类的m1方法。本例中,super有点替代Base#m1的意思。你在能够直接使用Base#m1的任何地方都能使用super。这包括通过super传递你希望传递给Base类的m1的任何参数。(当然仅限于目标方法能够接收的那些参数。)小心这个陷阱。
即使super被用作并扮演得像个方法,它其实是Ruby的关键字。这之间的不同很重要,因为super是基于Ruby中设定为可选的元素——括号,来改变行为的。我说得很严肃哦,请仔细想上一秒。当使用super时,如果省略了括号是会改变其行为的。这比听上去还要严重。要知道括号改变了什么,我们需要回顾一下super的三种写法:
super的第一种用法最为平常——就是我们之前所见的那样。如果你至少给super一个参数,那么它的行为就好像常规的Ruby方法,并且括号是可选的。通过这种形式,super将参数全部传递给目标方法。
如果没有给super任何参数和括号,它的行为可能会和你的预期不同。这种用法之下,super在调用目标方法时将宿主方法(enclosing method)的所有参数全部传递过去。如果存在方法块它也会一并传递。
你应该可以看出,这种形式下的super会带来一些副作用。首先,这样使用super仅仅在目标方法和宿主方法接收相同数量的参数时才可用。比如,这对我们之前看到的例子中的方法Base#m1和Derived#m1就不适用。试图在Derived#m1方法中使用无参的super会引发一个ArgumentError异常。如果被重载的方法不接收参数而宿主方法接收,那么着同样会为你带来一个相同的异常。
可以看出的第二件事与传递给重载方法的参数的值有关。如果你在使用无参形式调用super之前,在宿主方法中改变了其参数中的一个值,那么super会将改变后的当前值传递给目标方法,而非原始的值。这看起来非常合理,但有时你也需要注意。
当你希望在使用super时不将任何参数传递给重载方法,你需要使用空括号,即super()。在Ruby中这种形式看起来实在很怪。即使我们喜欢使用括号,在无参调用时也不会加上它们。这种super的用法看起来有些不自然,但这是无参(也没有块)调用重载方法的唯一途径。
以下是对上述规则的代码表述:

下一个需要关心的是super如何寻找要重载的方法。你也许会简单地想,只要调用其超类的同一方法就可以了,但这也简单地过分了哦。事实上,super是从整个继承体系中寻找的,见第6条,不过有些许微小的差异。使用super时,它从继承体系的上一层去寻找和当前方法同名的方法。所以,它将从当前类的上一层开始寻找。
如果你还记得我们对继承体系的研究,你将明白上一层可能会是一个单例类。这很棒啊,因为这意味着super能够对其引用的模块中的方法进行重载。

这个例子也指出了使用super的一个限制。如果你希望调用一个定义在超类中的方法,而与此同时如果包含的模块中也定义了同名方法,super就无法帮你达到目的。显然,super会在找到第一个匹配的同名方法后停下来,而那是包含的模块中的方法,而不是超类中的。但如果你真的意外遇到了这种情况,那么可能在设计上存在严重的问题。可以考虑使用组合而非继承了。
最后一个需要警示的是super是如何与method_missing相互作用的。如果你思考过这个问题,你会知道,只要在继承体系中存在另一个同名方法,使用super就不会引发method_missing。另一方面,如果你恰巧是一个瞌睡的开发者或者正在使用一些元编程的黑魔法,你最终可能会在使用super时引发method_missing。如果是这样,你读到这节会非常开心。
就像一个普通的方法查找一样,如果super无法找到它要寻找的方法,将调用method_missing来报告这个错误。如你料想的那样,Ruby寻找method_missing时会从原接收者所在的类开始查找。这一切都很好,我们还可以从异常结果中得到有用的错误信息。

Ruby会在内部追踪method_missing方法是否由一个错误的super调用引发,若是如此,会在异常中给出一些额外的信息。但这仅发生在Ruby没能在继承体系中找到已定义的method_missing方法而使用默认实现时。一旦继承体系中存在任何定义了method_missing方法的类,你就会丢失这个有用的细节。这个信息无法找回,即使你在自己的method_missing实现中调用super也不行。
再多说点,method_missing在被调用时描述的细节是说,方法是存在的,只是它在错误的地方。如果上面的SuperHappy类定义了method_missing方法,它将在调用laugh时被调用,因为那个方法试图使用super调用另一个不存在的laugh方法。那么method_missing的第一个参数是什么呢?对的,就是缺少的方法的名字,本例中就是:laugh。但是SuperHappy本身是有laugh方法的。有点迷茫了是不是?
假如你想细数避免自行定义method_missing方法的原因,请确保你也算上了这个(包括第30条里提到的)。本例中,来自method_missing方法的默认实现中的NoMethodError异常显然比我们自己定义的method_missing方法更好。
要点回顾
当你想重载继承体系中的一个方法时,关键字super可以帮你调用它。
不加括号地无参调用super等价于将宿主方法的所有参数传递给要调用的方法。
如果希望使用super并且不向重载方法传递任何参数,必须使用空括号,即super()。
当super调用失败时,自定义的method_missing方法将丢弃一些有用的信息。在第30条中有method_missing的替代解决方案。

时间: 2025-01-10 03:13:30

《Effective Ruby:改善Ruby程序的48条建议》一第7条:了解super的不同行为的相关文章

编写高质量代码改善C#程序的157个建议[IEnumerable<T>和IQueryable<T>、LINQ避免迭代、LINQ替代迭代]

原文:编写高质量代码改善C#程序的157个建议[IEnumerable<T>和IQueryable<T>.LINQ避免迭代.LINQ替代迭代] 前言 本文已更新至http://www.cnblogs.com/aehyok/p/3624579.html .本文主要学习记录以下内容: 建议29.区别LINQ查询中的IEnumerable<T>和IQueryable<T> 建议30.使用LINQ取代集合中的比较器和迭代器 建议31.在LINQ查询中避免不必要的迭代

编写高质量代码改善C#程序的157个建议[为泛型指定初始值、使用委托声明、使用Lambda替代方法和匿名方法]

原文:编写高质量代码改善C#程序的157个建议[为泛型指定初始值.使用委托声明.使用Lambda替代方法和匿名方法] 前言 泛型并不是C#语言一开始就带有的特性,而是在FCL2.0之后实现的新功能.基于泛型,我们得以将类型参数化,以便更大范围地进行代码复用.同时,它减少了泛型类及泛型方法中的转型,确保了类型安全.委托本身是一种引用类型,它保存的也是托管堆中对象的引用,只不过这个引用比较特殊,它是对方法的引用.事件本身也是委托,它是委托组,C#中提供了关键字event来对事件进行特别区分.一旦我们

编写高质量代码改善C#程序的157个建议[用抛异常替代返回错误、不要在不恰当的场合下引发异常、重新引发异常时使用inner Exception]

原文:编写高质量代码改善C#程序的157个建议[用抛异常替代返回错误.不要在不恰当的场合下引发异常.重新引发异常时使用inner Exception] 前言 自从.NET出现后,关于CLR异常机制的讨论就几乎从未停止过.迄今为止,CLR异常机制让人关注最多的一点就是"效率"问题.其实,这里存在认识上的误区,因为正常控制流程下的代码运行并不会出现问题,只有引发异常时才会带来效率问题.基于这一点,很多开发者已经达成共识:不应将异常机制用于正常控制流中.达成的另一个共识是:CLR异常机制带来

编写高质量代码改善C#程序的157个建议[4-9]

原文:编写高质量代码改善C#程序的157个建议[4-9] 前言 本文首先亦同步到http://www.cnblogs.com/aehyok/p/3624579.html.本文主要来学习记录一下内容: 建议4.TryParse比Parse好 建议5.使用int?来确保值类型也可以为null 建议6.区别readonly和const的使用方法 建议7.将0值设为枚举的默认值 建议8.避免给枚举类型的元素提供显式的值 建议9.习惯重载运算符 建议4.TryParse比Parse好 如果注意观察,除st

编写高质量代码改善C#程序的157个建议[C#闭包的陷阱、委托、事件、事件模型]

原文:编写高质量代码改善C#程序的157个建议[C#闭包的陷阱.委托.事件.事件模型] 前言 本文已更新至http://www.cnblogs.com/aehyok/p/3624579.html .本文主要学习记录以下内容: 建议38.小心闭包中的陷阱 建议39.了解委托的实质 建议40.使用event关键字对委托施加保护 建议41.实现标准的事件模型 建议38.小心闭包中的陷阱 首先我们先来看一段代码: class Program { static void Main(string[] arg

编写高质量代码改善C#程序的157个建议[为类型输出格式化字符串、实现浅拷贝和深拷贝、用dynamic来优化反射]

原文:编写高质量代码改善C#程序的157个建议[为类型输出格式化字符串.实现浅拷贝和深拷贝.用dynamic来优化反射] 前言 本文已更新至http://www.cnblogs.com/aehyok/p/3624579.html .本文主要学习记录以下内容: 建议13.为类型输出格式化字符串 建议14.正确实现浅拷贝和深拷贝 建议15.使用dynamic来简化反射实现 建议13.为类型输出格式化字符串   有两种方法可以为类型提供格式化的字符串输出. 一种是意识到类型会产生格式化字符串输出,于是

编写高质量代码改善C#程序的157个建议[匿名类型、Lambda、延迟求值和主动求值]

原文:编写高质量代码改善C#程序的157个建议[匿名类型.Lambda.延迟求值和主动求值] 前言 从.NET3.0开始,C#开始一直支持一个新特性:匿名类型.匿名类型由var.赋值运算符和一个非空初始值(或以new开头的初始化项)组成.匿名类型有如下基本特性: 1.既支持简单类型也支持复杂类型.简单类型必须是一个非空初始值,复杂类型则是一个以new开头的初始化项. 2.匿名类型的属性是只读的,没有属性设置器,它一旦倍初始化就不可更改. 3.如果两个匿名类型的属性值相同,那么就任务这两个匿名类型

编写高质量代码改善C#程序的157个建议[动态数组、循环遍历、对象集合初始化]

原文:编写高质量代码改善C#程序的157个建议[动态数组.循环遍历.对象集合初始化] 前言   软件开发过程中,不可避免会用到集合,C#中的集合表现为数组和若干集合类.不管是数组还是集合类,它们都有各自的优缺点.如何使用好集合是我们在开发过程中必须掌握的技巧.不要小看这些技巧,一旦在开发中使用了错误的集合或针对集合的方法,应用程序将会背离你的预想而运行. 本文已更新至http://www.cnblogs.com/aehyok/p/3624579.html .本文主要学习记录以下内容: 建议16.

编写高质量代码改善C#程序的157个建议[10-12]

原文:编写高质量代码改善C#程序的157个建议[10-12] 前言 本文已更新至http://www.cnblogs.com/aehyok/p/3624579.html .本文主要学习记录以下内容: 建议10.创建对象时需要考虑是否实现比较器 建议11.区别对待==和Equals 建议12.重写Equals时也要重写GetHashCode 建议10.创建对象时需要考虑是否实现比较器 有对象的地方就会存在比较,就像小时候每次拿着考卷回家,妈妈都会问你隔壁的那谁谁谁考了多少分呀.下面我们也来举个简单

编写高质量代码改善C#程序的157个建议[泛型集合、选择集合、集合的安全]

原文:编写高质量代码改善C#程序的157个建议[泛型集合.选择集合.集合的安全] 前言   软件开发过程中,不可避免会用到集合,C#中的集合表现为数组和若干集合类.不管是数组还是集合类,它们都有各自的优缺点.如何使用好集合是我们在开发过程中必须掌握的技巧.不要小看这些技巧,一旦在开发中使用了错误的集合或针对集合的方法,应用程序将会背离你的预想而运行. 本文已更新至http://www.cnblogs.com/aehyok/p/3624579.html .本文主要学习记录以下内容: 建议20.使用