時価総額: $2.8313T 0.35%
ボリューム(24時間): $138.9602B -22.27%
恐怖と貪欲の指数:

28 - 恐れ

  • 時価総額: $2.8313T 0.35%
  • ボリューム(24時間): $138.9602B -22.27%
  • 恐怖と貪欲の指数:
  • 時価総額: $2.8313T 0.35%
暗号
トピック
暗号化
ニュース
暗号造園
動画
トップクリプトスペディア

言語を選択する

言語を選択する

通貨の選択

暗号
トピック
暗号化
ニュース
暗号造園
動画

Solidity ではアレイ ストレージはどのように機能し、そのコストはいくらですか?

Dynamic arrays in Solidity store length in a slot, with elements placed at `keccak256(slot) + index`, ensuring deterministic, collision-free storage.

2025/11/23 11:59

Solidity の配列ストレージを理解する

1. Solidity の配列はコントラクト ストレージに保存され、関数呼び出しとトランザクションにわたって永続的になります。 uint[] public valueなどの動的配列が宣言されている場合、配列に割り当てられたスロットには配列の長さのみが保持されます。実際のデータ要素は、アレイのスロット番号の keccak256 ハッシュに保存され、安全で決定的な位置マッピングが可能になります。

2. 配列内の各要素は、式keccak256(スロット) + インデックスから導出されたストレージ スロットに配置されます。このメカニズムにより、コントラクト内に複数の配列が存在する場合でも、それらのデータが衝突しないことが保証されます。 Ethereum ストレージは 256 ビットにアライメントされているため、各スロットは 1 つの完全な uint256 または同等のサイズの値を保持できます。大きなタイプは複数のスロットにまたがる場合があります。

3. 固定サイズの配列の場合、コンパイラは、宣言された変数の位置から始まる連続したストレージ スロットを予約します。 uint[3] idsのような固定配列は、ちょうど 3 つの隣接するスロットを使用します。サイズはコンパイル時に既知であるため、長さなどのメタデータは保存されず、アクセスがより直接的になり、ガスの点でわずかに安くなります。

4. ネストされた配列により、複雑さが大幅に増加します。 2 次元動的配列は、同じハッシュ法を使用して基本位置を計算し、内部配列インデックスに基づいてオフセットを適用します。アクセス パターンは、複数レベルのハッシュと読み取りまたは書き込み時の SLOAD 操作の増加によりコストが高くなります。

5. 可能であれば、値を構造体にパックするか、より小さな整数型 (たとえば、uint256 の代わりに uint128) を使用すると、複数の変数を 1 つのストレージ スロット内に収めることができます。これにより、使用されるスロットの合計数が減り、これらの変数に関係する展開と状態変更の両方にかかるガス コストが削減されます。

アレイ操作に関連するガスコスト

1. Push()を介して動的配列に追加すると、追加された要素がゼロか非ゼロかによって異なるコストが発生します。保存されている値がゼロ以外の場合、以前に空だったスロットを設定すると、ガスコストが 20,000 の SSTORE 操作が実行されます。スロットに既にデータが含まれていると仮定すると、EIP-1283 ルールに基づいてゼロ以外の値を上書きすると、5,000 ガスのコストがかかります。

2. 配列から要素を読み取ると SLOAD 操作がトリガーされ、アクセスごとに 2,100 ガスのコストがかかります。オンチェーン ロジックでは、読み取りごとにこの基本コストが増加するため、大規模な配列に対する頻繁な反復は避けるべきです。オフチェーンのインデックス作成サービスは、完全なデータセットを取得するのに適しています。

3. 手動シフトが実装されていない限り、動的配列では要素の削除は本質的にサポートされていません。 Pop() を呼び出すと、最後のアイテムが削除され、ストレージ スロットがクリアされると 15,000 ガスが返金され、未使用状態のクリーンアップが促進されます。ただし、項目を中間から削除するには、後続のすべての要素を移動する必要があり、O(n) の計算と大量のガス使用量が発生します。

4. コントラクトの作成中に大規模なアレイを初期化すると、展開コストが大幅に増加します。すべての事前設定値はストレージ初期化ガスを消費します。多くの場合、配列を空に初期化し、ユーザーの操作を通じて徐々に配列を設定し、コストを複数のトランザクションに分散する方が効率的です。

5. 関数内で一時的に使用されるメモリ配列は永続ストレージに書き込まないため、SSTORE コストを完全に回避できます。これらは中間計算には最適ですが、トランザクションの範囲を超えてデータを保持することはできません。選択した結果をストレージに書き込む前にバッチ処理にメモリ アレイを使用すると、全体の効率が最適化されます。

配列を効率的に使用するための最適化手法

1. キーによるランダム アクセスが必要な場合は、配列よりもマッピングを優先します。マッピング (uint => address)のようなマッピングは、反復や長さの管理のオーバーヘッドなしで定数時間のルックアップを提供します。マッピングにより、インデックスの境界やサイズ変更に関する懸念も解消されます。

2. 個々の配列要素に対して削除は慎重に使用します。特に払い戻しが適用されない場合は、要素を手動でゼロに設定する方が、削除に頼るよりもコストがかからない可能性があります。完全にクリーンアップするには、配列が空になるまで繰り返し Pop() を呼び出すことで、クリアされたストレージ スロットからのガスの払い戻しを最大化します。

3. 配列内容のオンチェーン公開を制限します。配列データを使用してイベントを発行すると、外部システムはストレージから直接読み取ることなく状態を再構築できます。イベントは永続ストレージへの書き込みよりコストが低く、インデックス付きパラメーターによるフィルター処理をサポートします。

4. 追加の多いワークロード用にチェックポイントまたはリンク リスト パターンを実装します。コントラクトは、エントリ間のデルタまたは参照のみを保存することにより、冗長なデータ ストレージを削減し、コストのかかる再編成を最小限に抑えます。このアプローチは、履歴状態やユーザー アクティビティ ログを追跡するのに適しています。

5. アセンブリレベルの最適化を慎重に活用します。インライン Yul コードは、高レベルの Solidity 構造よりも高速にストレージの場所を計算できますが、誤ったアドレス指定のリスクが生じます。上級開発者のみが低レベルのストレージ操作を試み、さまざまなコンパイラ バージョン間で徹底的なテストを行う必要があります。

よくある質問

範囲外の配列インデックスにアクセスするとどうなりますか?現在の配列の長さを超えたインデックスにアクセスすると、無効なオペコードがトリガーされ、残りのガスがすべて消費され、トランザクションが元に戻されます。境界チェックは、アクセスする前にコード内で明示的に処理する必要があります。

コントラクト間で配列を引数として渡すことはできますか?はい、配列は内部および外部の関数呼び出しで渡すことができます。 Calldata 配列はメモリへのコピーを避けるため、大規模な入力に対して効率的です。外部インターフェイスには、静的型と動的型の両方をサポートする ABI エンコードが必要です。

動的配列のサイズに制限はありますか?理論上の制限は、利用可能なガスとブロック サイズの制限によって制限されます。実際の制約には、追加と反復のコストが増大することが含まれます。コントラクトでは、オンチェーンで過度に大規模なアレイを処理する場合、トランザクション ガス制限に達するリスクがあります。

空の配列宣言は記憶域を消費しますか?初期化せずに配列を宣言すると、配列の長さがゼロに設定され、1 つのストレージ スロットが占有されます。要素が追加されるまで、追加のスロットは割り当てられません。この最小限のフットプリントにより、初期化されていない配列の遅延実装に対するコスト効率が高くなります。

免責事項:info@kdj.com

提供される情報は取引に関するアドバイスではありません。 kdj.com は、この記事で提供される情報に基づいて行われた投資に対して一切の責任を負いません。暗号通貨は変動性が高いため、十分な調査を行った上で慎重に投資することを強くお勧めします。

このウェブサイトで使用されているコンテンツが著作権を侵害していると思われる場合は、直ちに当社 (info@kdj.com) までご連絡ください。速やかに削除させていただきます。

関連知識

LayerZero コントラクトを使用してクロスチェーン メッセージを実行するにはどうすればよいですか?

LayerZero コントラクトを使用してクロスチェーン メッセージを実行するにはどうすればよいですか?

2026-01-18 13:19:39

LayerZero アーキテクチャを理解する1. LayerZero は、信頼できる仲介者やラップされた資産に依存せずにブロックチェーン間の通信を可能にする、軽量で許可のない相互運用性プロトコルとして動作します。 2. 各チェーンに展開されたウルトラ ライト ノード (ULN) を利用して、ブロック...

安全な署名検証のために EIP-712 を実装するにはどうすればよいですか?

安全な署名検証のために EIP-712 を実装するにはどうすればよいですか?

2026-01-20 22:20:26

EIP-712 の概要と主な目的1. EIP-712 は、イーサリアム アプリケーションにおける型付き構造化データのハッシュと署名の標準を定義します。 2. これにより、ウォレットは、署名リクエスト中に未加工の 16 進文字列の代わりに人間が判読できるドメインおよびメッセージ フィールドを表示できる...

新しい契約を交わしてエアドロップの資格を得るにはどうすればよいですか?

新しい契約を交わしてエアドロップの資格を得るにはどうすればよいですか?

2026-01-24 21:00:23

契約のやり取りの要件を理解する1. ほとんどのエアドロップ キャンペーンでは、Ethereum、Arbitrum、Base などのサポートされているブロックチェーンにデプロイされたスマート コントラクトとの直接対話が義務付けられています。 2. インタラクションには通常、dApp インターフェースに...

スマート コントラクトのセキュリティ アラートを監視するにはどうすればよいですか?

スマート コントラクトのセキュリティ アラートを監視するにはどうすればよいですか?

2026-01-21 07:59:57

オンチェーン監視ツール1. Etherscan や Blockscout などのブロックチェーン エクスプローラーを使用すると、コントラクト バイトコード、トランザクション ログ、内部呼び出しをリアルタイムで検査できます。 2. オンチェーンデータを信頼する前に、契約の検証ステータスを確認する必要が...

自動支払いのための契約を設定して資金を調達するにはどうすればよいですか?

自動支払いのための契約を設定して資金を調達するにはどうすればよいですか?

2026-01-26 08:59:35

スマートコントラクトの展開を理解する1. 開発者は、ガス効率とセキュリティ要件に基づいて、Ethereum、Polygon、Arbitrum などの互換性のあるブロックチェーン プラットフォームを選択する必要があります。 2. Solidity は、特に定期的な送金や条件付き送金の場合、支払い自動化...

OpenZeppelin コントラクトを使用して安全な dApp を構築するにはどうすればよいですか?

OpenZeppelin コントラクトを使用して安全な dApp を構築するにはどうすればよいですか?

2026-01-18 11:19:49

OpenZeppelin コントラクトの基礎を理解する1. OpenZeppelin Contracts は、イーサリアムおよび EVM 互換ブロックチェーン用に構築された、再利用可能なコミュニティ監査済みのスマート コントラクト コンポーネントのライブラリです。 2. ライブラリ内の各コントラクト...

LayerZero コントラクトを使用してクロスチェーン メッセージを実行するにはどうすればよいですか?

LayerZero コントラクトを使用してクロスチェーン メッセージを実行するにはどうすればよいですか?

2026-01-18 13:19:39

LayerZero アーキテクチャを理解する1. LayerZero は、信頼できる仲介者やラップされた資産に依存せずにブロックチェーン間の通信を可能にする、軽量で許可のない相互運用性プロトコルとして動作します。 2. 各チェーンに展開されたウルトラ ライト ノード (ULN) を利用して、ブロック...

安全な署名検証のために EIP-712 を実装するにはどうすればよいですか?

安全な署名検証のために EIP-712 を実装するにはどうすればよいですか?

2026-01-20 22:20:26

EIP-712 の概要と主な目的1. EIP-712 は、イーサリアム アプリケーションにおける型付き構造化データのハッシュと署名の標準を定義します。 2. これにより、ウォレットは、署名リクエスト中に未加工の 16 進文字列の代わりに人間が判読できるドメインおよびメッセージ フィールドを表示できる...

新しい契約を交わしてエアドロップの資格を得るにはどうすればよいですか?

新しい契約を交わしてエアドロップの資格を得るにはどうすればよいですか?

2026-01-24 21:00:23

契約のやり取りの要件を理解する1. ほとんどのエアドロップ キャンペーンでは、Ethereum、Arbitrum、Base などのサポートされているブロックチェーンにデプロイされたスマート コントラクトとの直接対話が義務付けられています。 2. インタラクションには通常、dApp インターフェースに...

スマート コントラクトのセキュリティ アラートを監視するにはどうすればよいですか?

スマート コントラクトのセキュリティ アラートを監視するにはどうすればよいですか?

2026-01-21 07:59:57

オンチェーン監視ツール1. Etherscan や Blockscout などのブロックチェーン エクスプローラーを使用すると、コントラクト バイトコード、トランザクション ログ、内部呼び出しをリアルタイムで検査できます。 2. オンチェーンデータを信頼する前に、契約の検証ステータスを確認する必要が...

自動支払いのための契約を設定して資金を調達するにはどうすればよいですか?

自動支払いのための契約を設定して資金を調達するにはどうすればよいですか?

2026-01-26 08:59:35

スマートコントラクトの展開を理解する1. 開発者は、ガス効率とセキュリティ要件に基づいて、Ethereum、Polygon、Arbitrum などの互換性のあるブロックチェーン プラットフォームを選択する必要があります。 2. Solidity は、特に定期的な送金や条件付き送金の場合、支払い自動化...

OpenZeppelin コントラクトを使用して安全な dApp を構築するにはどうすればよいですか?

OpenZeppelin コントラクトを使用して安全な dApp を構築するにはどうすればよいですか?

2026-01-18 11:19:49

OpenZeppelin コントラクトの基礎を理解する1. OpenZeppelin Contracts は、イーサリアムおよび EVM 互換ブロックチェーン用に構築された、再利用可能なコミュニティ監査済みのスマート コントラクト コンポーネントのライブラリです。 2. ライブラリ内の各コントラクト...

すべての記事を見る

User not found or password invalid

Your input is correct