我可以用我的方法扩展内置字符串 class 吗

Can I extend built-in String class with my methods

我发现在内置 String class 中没有内置的 trim (strip) 方法可以从字符串中删除前导和尾随空格。我想用我的功能扩展它。可能吗?使用示例 here,我尝试了以下代码:

String extend [
    trimleading: str [ |ch ret flag|
        ret := str.                 "make a copy of sent string"
        flag := true.
        [flag] whileTrue: [         "while first char is space"
            ch := ret first: 1.     "again test first char"
            ch = ' '                "check if space remaining" 
            ifTrue: [ ret := (ret copyFrom: 2 to: ret size)]    "copy from 2nd char"
            ifFalse: [flag := false] 
            ].
        ^ret                        "return modified string"
        ]
    trim: str [ |ret|
        ret := str. 
        ret := (self trimleading: ret).           "trim leading spaces"
        ret := (self trimleading: (ret reverse)). "reverse string and repeat trim leading"
        ^(ret reverse)                            "return re-reversed string"
        ]
].

oristr := '        this is a test  '
('>>',oristr,'<<') displayNl.
('>>',(oristr trim),'<<') displayNl.

以上代码不起作用并给出以下错误:

$ gst string_extend_trim.st 

>>        this is a test  <<
Object: '        this is a test  ' error: did not understand #trim
MessageNotUnderstood(Exception)>>signal (ExcHandling.st:254)
String(Object)>>doesNotUnderstand: #trim (SysExcept.st:1448)
UndefinedObject>>executeStatements (string_extend_trim.st:23)

问题出在哪里,如何解决?谢谢。

编辑:以下代码有效,但不会更改原始字符串:

String extend [
    trimleading [ |ch ret flag|
        ret := self.                    "make a copy of sent string"
        flag := true.
        [flag] whileTrue: [             "while first char is space"
            ch := ret first: 1.         "again test first char"
            ch = ' '                    "check if space remaining" 
            ifTrue: [ ret := (ret copyFrom: 2 to: ret size)]    "copy from 2nd char"
            ifFalse: [flag := false] 
            ].
        ^ret                                "return modified string"
        ]
    trim [ |ret|
        ret := self.    
        ret := (self trimleading).  "trim leading spaces"
        ret := ((ret reverse) trimleading ). "reverse string and repeat trim leading"
        ^(ret reverse)                      "return re-reverse string"
        ]
].

oristr := '        this is a test  '
('>>',oristr,'<<') displayNl.
('>>',(oristr trim),'<<') displayNl.
('>>',oristr,'<<') displayNl.
oristr := (oristr trim).
('>>',oristr,'<<') displayNl.

oristr trim如何改变oristr?我不想写 oristr := oristr trim.

您已经解决的第一个问题:最初您定义了一个带有一个参数的方法 trim:,但发送 trim 时没有参数。

第二个问题是原地修改String。您可以使用 self at: index put: aCharacter 和其他一些方法更改字符以复制和覆盖范围,但您将无法更改字符串的大小(长度)。在我所知道的 Smalltalks 中,对象在创建后不能更改其大小。因此,我建议您坚持在 trim.

中创建一个字符较少的新字符串

系统中到处都有一个对象交换另一个对象的方法。它被称为become:。但我认为你不应该在这里使用它。根据 Smalltalk 的实现,您最终可能会产生不需要的副作用,例如在方法中替换 String 文字(因此下一个方法调用实际上 运行 用不同的、修剪过的字符串代替文字)。

您的代码与您链接的示例之间的区别在于,在示例中,它们扩展了自定义 class,而您扩展了核心 class。不同之处在于您应该加载代码的方式和 运行 它。应该用Packages in GNU-Smalltalk to build it. There is an excellent answer on SO by @lurker ,喜欢的话请点个赞,我不想在这里重复信息

使您的代码适应 String extend

String extend [
    trimleading: str [ |ch ret flag|
        ret := str.                     "make a copy of sent string"
        flag := true.
        [flag] whileTrue: [         "while first char is space"
            ch := ret first: 1.         "again test first char"
            ch = ' '
            ifTrue: [ ret := (ret copyFrom: 2 to: ret size) ]    "copy from 2nd char"
            ifFalse:  [flag := false ] 
        ].
        ^ ret                                "value is modified string"
    ]
    trim [ | ret |
        ret := self trimleading: self.           "trim leading spaces"
        ret := self trimleading: (ret copy reverse). "reverse string and repeat trim leading"
        ^ (ret reverse)                      "return re-reverse string"
    ]
].

oristr := '        this is a test  '.
('>>',oristr,'<<') displayNl.
('>>',(oristr trim),'<<') displayNl.
('>>',oristr,'<<') displayNl.
oristr := (oristr trim).
('>>',oristr,'<<') displayNl.

您正在将消息 #trim 发送到 origstr 变量,因此您必须在没有任何参数的情况下进行定义。但是,这不适用于 #trimleading:,所以我把你以前的代码放在那里了。

注意:您应该认真阅读 self 关键字及其作用并理解它——您使用它的方式不正确。您分配了 ret := self 值但没有使用它,您只是用下一个分配覆盖它。