php小编草莓将为您介绍关于`sort.Slice`函数的一些重要信息。在Go语言中,`sort.Slice`函数用于对切片进行排序,但其排序结果的顺序是不确定的。也就是说,对于相同的输入切片,每次排序的结果可能会有所不同。这是因为`sort.Slice`函数使用了一种快速且高效的排序算法,但排序的具体顺序是基于输入数据的特定条件而定的。因此,在使用`sort.Slice`函数时,我们应该注意到排序结果的不确定性,以避免在依赖特定排序顺序的场景中出现问题。
我正在尝试使用 go 标准库中的 sort.slice
对字符串切片进行排序。我希望它们按字母顺序排序,除了我希望空字符串出现在所有其他字符串之后(因此我不能只使用 sort.strings
)。
对于 less 函数,我认为这会起作用:
func(i, j int) bool { return s[j] == "" || s[i] < s[j] }
但是,根据输入顺序,我似乎得到了随机答案。这是 mwe:
package main import ( "fmt" "math/rand" "sort" "time" ) func main() { s := []string{"", "foo", "bar", "baz"} rand.seed(time.now().unix()) rand.shuffle(len(s), func(i, j int) { s[i], s[j] = s[j], s[i] }) fmt.printf("%q\n", s) sort.slice(s, func(i, j int) bool { return s[j] == "" || s[i] < s[j] }) fmt.printf("%q\n", s) }
这是运行几次的输出:
$ go run ./z ["" "foo" "baz" "bar"] ["bar" "baz" "foo" ""] $ go run ./z ["baz" "" "foo" "bar"] ["bar" "" "baz" "foo"] $ go run ./z ["bar" "foo" "" "baz"] ["" "bar" "baz" "foo"] $ go run ./z ["bar" "foo" "baz" ""] ["" "bar" "baz" "foo"]
这是因为您的 less()
函数没有说出您想要的内容。
您说您希望将空字符串排序在所有非空字符串之后。你的逻辑:
return s[j] == "" || s[i] < s[j]
这确实告诉我们第二个是否是 ""
,那么第一个就更少。这或多或少是正确的(除非两者都是空的,“is-less”并不是真的:它们是相等的)。但是,如果第一个是 ""
而第二个不是怎么办?那么你的函数应该返回 false
但它返回 s[i] < s[j]
。如果第二个不为空,则为 true
,告诉 ""
小于另一个,与你想要的正好相反。
正确的“is-less”关系是这样的:
sort.slice(s, func(i, j int) bool { if s[j] == "" && s[i] != "" { return true } if s[i] == "" && s[j] != "" { return false } return s[i] < s[j] })
如果只有第二个是 ""
,则您希望第一个更少。如果只有第一个是空的,您希望它“不少于”。否则使用正常顺序(按字节)。
在 go playground 上尝试一下。
请注意,如果第一个和第二个值都为空,则此函数将返回 false
,因为 false
,因为 ""
不小于 ""
(它们相等)。这是要返回的正确值,尽管在此处返回 true
不小于
true
仍会导致正确的顺序(交换空元素将导致相同的结果),但这可能会导致更少的交换。
请注意,在自定义逻辑中,如果只有一个字符串为空,则会偏离正常顺序。这是逻辑异或(异或)关系a xor b
是 true
如果只有 a
或只有 b
是 true
。在 go 中,没有逻辑 xor
运算符,但 a xor b
相当于 a != b
:
true
(否则为 false
如果“检测到”一个空字符串,如果第二个空字符串为空,则结果为
sort.Slice(s, func(i, j int) bool { // Move empty elements to the end: if (s[i] == "") != (s[j] == "") { // If only one is empty return s[j] == "" } return s[i] < s[j] })
以上是`sort.Slice` 顺序是不确定的的详细内容。更多信息请关注PHP中文网其他相关文章!