将字符串与指向字符串的指针作为函数参数传递时,时间复杂度有什么区别?

Meg*_*ori 1 string pointers pass-by-value go

假设我有一个字符串 s。

 s := "helloworld"
Run Code Online (Sandbox Code Playgroud)

现在,我的问题是,如果 s 有 'n' 个字节,那么如果我将 s 传递给函数与将 &s 传递给函数然后访问字符串的第 i 个字节,则相对于 'n' 的时间复杂度是多少。

如果将 &s 传递给函数并访问字符串的第 i 个字节需要 O(1) 时间,那么当我将 s 传递给函数然后访问字符串的第 i 个字节时是否需要 O(n) 时间(因为整个字符串将被复制)?

我试过这个,发现复制一个字符串确实改变了指向它的指针。希望能更清楚地了解这一点。

func main() {
    str := "helloworld"
    fmt.Println("string pointer 1:", &str)
    printStringPointer(str)
}

func printStringPointer(s string) {
    fmt.Println("string pointer 2:", &s)
}
Run Code Online (Sandbox Code Playgroud)

输出:

string pointer 1: 0xc000010200
string pointer 2: 0xc000010210
Run Code Online (Sandbox Code Playgroud)

Mar*_*arc 5

go 中的字符串类似于切片,它们只是一个细长的描述符,包含一个指向底层数据的指针和一个长度。

您可以在reflect.StringHeader类型中看到这一点:

type StringHeader struct {
    Data uintptr
    Len  int
}
Run Code Online (Sandbox Code Playgroud)

将字符串传递给函数时,会复制标头,但不会复制底层数据。我们可以更改您的示例以打印 的值Data,表明它指向内存中的相同地址:playground link

func main() {
    str := "helloworld"
    fmt.Println("string pointer 1: ", &str)
    fmt.Println("string Data 1:    ", (*reflect.StringHeader)(unsafe.Pointer(&str)).Data)
    printStringPointer(str)
}

func printStringPointer(s string) {
    fmt.Println("string pointer 2: ", &s)
    fmt.Println("string Data 2:    ", (*reflect.StringHeader)(unsafe.Pointer(&s)).Data)
}
Run Code Online (Sandbox Code Playgroud)

输出:

string pointer 1:  0xc000010200
string Data 1:     4970654
string pointer 2:  0xc000010210
string Data 2:     4970654
Run Code Online (Sandbox Code Playgroud)

传递字符串是常数时间(复制标题),不复制底层数据