ホームページ >バックエンド開発 >Golang >文字列の Go スライスを文字列ポインタのスライスに変換するときに、元のメモリ アドレスを保持する方法は?

文字列の Go スライスを文字列ポインタのスライスに変換するときに、元のメモリ アドレスを保持する方法は?

Patricia Arquette
Patricia Arquetteオリジナル
2024-12-26 11:09:14488ブラウズ

How to Preserve Original Memory Addresses When Converting a Go Slice of Strings to a Slice of String Pointers?

スライス変換での識別子アドレスの保持

Go では、文字列のスライス (values1) を文字列へのポインターのスライス (values2) に変換します。 )は独特の課題を提示します。以下に示すコードは、この問題を示しています。

values1 := []string{"a", "b", "c"}
var values2 []*string
for _, v := range values1 {
    fmt.Printf("%p | %T\n", v, v)
    values2 = append(values2, &v)
}

予想どおり、出力では、values2 の各要素が、値が異なるにもかかわらず、同じメモリ アドレスを指していることがわかります。これは、範囲ループ変数 v が元の文字列値へのポインタではなく、そのコピーであるために発生します。その結果、ループ内で新しい値が割り当てられた場合でも、v は文字列値のままになります。

ソリューションの実装

この問題を解決するには、アドレスを追加する必要があります。元のスライス要素を value2 に変換します。考えられる解決策は 2 つあります。

1.元のスライスのインデックス作成

for i, _ := range values1 {
    values2 = append(values2, &values1[i])
}

インデックス作成を通じてスライス要素にアクセスすることで、必要なポインター アドレスを取得し、それらの個別の値を保存します。

2.一時変数の使用

for _, v := range values1 {
    v2 := v
    values2 = append(values2, &v2)
}

このアプローチでは、各文字列値を保持する一時変数 (v2) を宣言します。 v2 を &v2 に割り当てることで、values2 に一意のポインタが含まれるようにする独立したポインタ値を作成します。

識別子アドレスを保持することの影響

識別子の保持には注意することが重要です。アドレスには特定の影響があります:

  • のバッキング配列value1 は、value1 と value2 の両方が到達可能な限りメモリに保持されます。
  • values1 要素への変更は、values2 に反映されます。
  • values1 がガベージ コレクションされないようにするには、インデックス作成または一時的なものを使用します。上記の解決策で説明したように、変数。

以上が文字列の Go スライスを文字列ポインタのスライスに変換するときに、元のメモリ アドレスを保持する方法は?の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。