类似的Go函数在追加字符串和数组时表现不如预期。

huangapple go评论125阅读模式
英文:

Similar Go functions appending strings and arrays do not behave as expected

问题

我有两个Go函数:

  1. func permutation(prefix, str []int) {
  2. n := len(str)
  3. if n == 0 {
  4. fmt.Println(prefix)
  5. } else {
  6. for i := 0; i < n; i++ {
  7. permutation(
  8. append(prefix, str[i]),
  9. append(str[0:i], str[i+1:]...),
  10. )
  11. }
  12. }
  13. }
  14. func perms(prefix, str string) {
  15. n := len(str)
  16. if n == 0 {
  17. fmt.Println(prefix)
  18. } else {
  19. for i := 0; i < n; i++ {
  20. perms(
  21. prefix+string(str[i]),
  22. string(str[0:i])+string(str[i+1:]),
  23. )
  24. }
  25. }
  26. }

第一个函数接受一个整数数组,第二个函数接受一个字符串。它们都计算数组或字符串的所有排列。

你可以这样运行它们:

  1. permutation([]int{}, []int{1, 2, 3})
  2. perms("", "123")

它们的输出不一样:

  1. $ go run main.go
  2. [1 2 3]
  3. [1 3 3]
  4. [3 3 3]
  5. [3 3 3]
  6. [3 3 3]
  7. [3 3 3]
  8. 123
  9. 132
  10. 213
  11. 231
  12. 312
  13. 321

我猜测我可能忽略了一些关于追加数组的细微差别。我似乎无法弄清楚问题所在。你有什么想法吗?

英文:

I have two Go functions:

<!-- language: lang-go -->

  1. func permutation(prefix, str []int) {
  2. n := len(str)
  3. if n == 0 {
  4. fmt.Println(prefix)
  5. } else {
  6. for i := 0; i &lt; n; i++ {
  7. permutation(
  8. append(prefix, str[i]),
  9. append(str[0:i], str[i+1:]...),
  10. )
  11. }
  12. }
  13. }
  14. func perms(prefix, str string) {
  15. n := len(str)
  16. if n == 0 {
  17. fmt.Println(prefix)
  18. } else {
  19. for i := 0; i &lt; n; i++ {
  20. perms(
  21. prefix+string(str[i]),
  22. string(str[0:i])+string(str[i+1:]),
  23. )
  24. }
  25. }
  26. }

The first takes an array of ints, the second takes a string. They both then calculate all permutations of the array, or the string.

I can run them like so:

  1. permutation([]int{}, []int{1, 2, 3})
  2. perms(&quot;&quot;, &quot;123&quot;)

Their output is not the same:

  1. $ go run main.go
  2. [1 2 3]
  3. [1 3 3]
  4. [3 3 3]
  5. [3 3 3]
  6. [3 3 3]
  7. [3 3 3]
  8. 123
  9. 132
  10. 213
  11. 231
  12. 312
  13. 321

I guess there is some nuance to appending arrays that I am missing. I can't seem to figure it out. Any idea what's going on?

答案1

得分: 2

str1+str2返回一个新的字符串(在内存中与原字符串无关),但append的行为不同。例如,append(str[0:i], str[i+1:]...)会破坏str的原始内容,用str[i+1:]覆盖str[i:]。这是因为str[0:i]有足够的容量来附加str[i+1:]而不需要分配新的缓冲区。

解决方法是在每次迭代中创建一个全新的数组,至少对于str来说是这样,因为append(prefix, str[i])不会受到这个问题的影响。例如:

  1. for i := 0; i < n; i++ {
  2. var s []int
  3. s = append(s, str[0:i]...)
  4. s = append(s, str[i+1:]...)
  5. permutation(append(prefix, str[i]), s)
  6. }

有关切片和append机制的更多信息,请参考以下链接:

英文:

While str1+str2 does return new (unrelated in terms of memory) string, append doesn't behave this way. For example append(str[0:i], str[i+1:]...) will destroy original content of str, overwriting str[i:] with str[i+1:]. This is because str[0:i] will have capacity to append str[i+1:] without allocating new buffer.

The solution would be to create a completely new array in every iteration. At least for str, as append(prefix, str[i]) is immune to this problem. For example:

  1. for i := 0; i &lt; n; i++ {
  2. var s []int
  3. s = append(s, str[0:i]...)
  4. s = append(s, str[i+1:]...)
  5. permutation(append(prefix, str[i]), s)
  6. }

https://play.golang.org/p/lXwu39AA0V

More about slices and mechanism of append:

http://blog.golang.org/go-slices-usage-and-internals

https://blog.golang.org/slices

huangapple
  • 本文由 发表于 2015年11月22日 04:30:41
  • 转载请务必保留本文链接:https://go.coder-hub.com/33848258.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定