在 Smalltalk 中,当发送者和参数属于不同类型时,定义可交换二进制方法的最佳方法是什么?

use*_*746 7 oop design-patterns smalltalk

假设您有一个 Foo 类,并且您希望能够将一个 Foo 乘以一个 Number 以获得另一个 Foo,使用“@@”作为乘号。

由于乘法是可交换的,所以能够写出很好:

| f a b |
f := Foo new.
a := 3 @@ f.
b := f @@ 3.
self assert: a = b
Run Code Online (Sandbox Code Playgroud)

这不仅需要将二进制方法“@@”添加到 Foo 中,还需要添加到 Number 类中。所以你最终在两个不同的地方使用了本质上相同的方法(以及循环依赖),这看起来很不雅观。

所以我想知道,在 Smalltalk 中,有没有其他方法可以创建交换二进制方法,其中发送者和参数是不同的类型——一种不需要你在两个不同的类中定义相同消息的方法?

如果没有,是否可以使用 Smalltalk 本身来创建这种能力(即添加自动管理可交换二进制方法的类/方法,而不改变实际的 Smalltalk 语言或 VM)?

Uko*_*Uko 5

在您的情况下,如果您没有将@@消息的参数发送到Foo.

例如:

f @@ 'hello'
Run Code Online (Sandbox Code Playgroud)

要省略这一点,您可以使用双重调度。所以你定义了一个乘以一个数字的方法:

Foo>>#multiplyWithANumber: aNumber
    "do multiplication with a number"
Run Code Online (Sandbox Code Playgroud)

然后在对象层次结构中,定义 @@

Object>>#@@ aFoo
    "signal some error saying that this operation is not supported"
    self shouldNotImplement

Number>>#@@ aFoo
    ^ aFoo multiplyWithANumber: self

Foo>>#@@ anObject
    "pass decision to the parameter"
    "also, what should happen if anObject is a Foo"
    ^ anObject @@ self
    
Run Code Online (Sandbox Code Playgroud)

这可能过于复杂,在一个简单的情况下,如果您不太关心类型,并且想要避免重复,您可以:

Foo>>#multiplyWithANumber: aNumber
    "do multiplication with a number"

Foo>>#@@ aNumber
    ^ self multiplyWithANumber: aNumber

Number>>#@@ aFoo
    ^ aFoo multiplyWithANumber: self
Run Code Online (Sandbox Code Playgroud)

当然,您可以multiplyWithANumber:全部跳过,只使用一个@@with 实现(可能在 Foo 端,因为它是此实现的主要原因),另一个@@只调用@@with 实现。我喜欢有一个详细的方法,所以很清楚发生了什么,你不必写额外的评论。

  • `#multiplyWithANumber:` 中的不定冠词“A”可能会产生误导。例如,考虑“foomultiplyWithANumber:3”。我建议改为“#multiplyWithNumber:”(或“#multiplyByNumber:”)。正如您所知,选择器是消息,而不是过程名称。 (2认同)