squeak(smalltalk) 如何将 'inject' 字符串转换成字符串

squeak(smalltallk) how to 'inject' string into string

我正在写一个名为 "MyObject" 的 class。 class 方法之一是:

addTo: aCodeString assertType: aTypeCollection

当使用 aCodeString 调用该方法时,我想(在运行时)向 "MyObject" class 添加一个新方法,其中 aCodeString 是它的源代码并且将类型检查代码注入源代码。 例如,如果我这样调用 addTo: assertType:

a := MyObject new.
a addTo:  'foo: a boo:b baz: c
    ^(a*b+c)'
assertType: #(SmallInteger SmallInteger SmallInteger).

希望以后可以写:

answer := (a foo: 2 boo: 5 baz: 10). 

并在 answer 中获得 20 个。 如果我写:

a foo: 'someString' boo: 5 baz: 10.

我收到了正确的消息,因为 'someString' 不是 SmallInteger。

我知道如何编写类型检查代码,并且我知道要在运行时将方法添加到 class,我可以使用行为 class 中的 'compile' 方法。

问题是我想在源代码中添加类型检查代码。 我不是很熟悉 squeak class 的所有内容,所以我不确定我是否宁愿将 aCodeString 编辑为 addTo: assertType: 中的字符串,然后使用 compile:(而且我不知道 怎么做 ),或者有一种方法可以将代码注入 Behavior class 或其他 squeak class 中的现有方法.

基本上,我要问的是如何将字符串注入现有字符串或将代码注入现有方法。

有很多方法可以实现这种类型检查...

您建议的方法是修改源代码(一个字符串),以便插入额外的前提条件类型检查。

此方法的关键点是您必须在正确的位置插入类型检查。这意味着以某种方式解析原始来源(或至少 selector 和参数)以便找到它的确切范围(和参数名称)。

参见 Parser 中的方法 initPattern:return: 及其发送者。您会发现相当低级(不是最漂亮的)代码,这些代码为块(通过 return: 关键字传递)提供 sap 3 个对象的数组:方法 select 或方法参数和方法优先级(一个代码告诉方法是否连接到一元、二进制或关键字消息)。从那里,您将获得足够的 material 来实现源代码操作(使用 copyReplace:from:to:with: 将一个字符串插入另一个字符串)。

不要犹豫,编写一小段代码并在调试器中执行(select 要调试的代码,然后使用调试菜单或 ALT+Shift+D)。还广泛使用检查员来更深入地了解事情的运作方式!

另一种解决方案是解析源代码的整个抽象语法树 (AST),并操纵该 AST 以插入类型检查。通常,解析器构建 AST,因此请观察它是如何工作的。从修改后的 AST 中,您可以生成新的 CompiledMethod(字节码指令)并将其安装在 methodDictionary 中 - 查看 compile: 的源代码并按照发送的消息进行操作,直到您发现 generateMethodFromNode:trailer:。这有点复杂,并且有一个不好的副作用,即源代码现在与生成的代码不同步,一旦您想调试该方法,这可能会成为一个问题(幸运的是,Squeak 可以使用反编译代码代替源代码代码!)。

最后,您还可以为您的某些 classes 安排备用编译器和解析器(参见 compilerClass and/or parserClass)。替代 TypeHintParser 将接受源代码中带有类型提示的修改语法(曾几何时,它是在尖括号 foo: x <Integer> bar: y <Number> 内的 args 后面使用类型提示实现的)。并且替代 TypeHintCompiler 将根据这些类型提示自动安排编译前提条件。因为你将在 Squeak 中非常先进,你还将在源代码索引和字节码之间创建特殊映射,以便拥有健全的调试器甚至特殊的反编译器 class 可以识别前提条件类型检查并将它们转换回类型提示以防万一。

我的建议是从您提出的第一种方法开始。

编辑

我忘了说,还有另一种方法,但它目前在 Pharo 而不是 Squeak 中可用:Pharo 编译器(名为 OpalCompiler)确实将字节码指令具体化为对象(class名称以 IR 开头)在生成阶段。所以在这个阶段也可以通过适当的黑客攻击直接操纵字节码指令......我很确定我们可以找到使用示例。可能是最先进的技术了。