【初心者向け】C++におけるlvalue、rvalue、xvalue、glvalue、prvalueをわかりやすく解説

2024-07-27

C++における値のカテゴリ:lvalue、rvalue、xvalue、glvalue、prvalue

lvalue(左辺値)

lvalueは、"left-value"の略で、プログラムが値を取得したり書き換えたりできる式を表します。具体的には、以下の要素がlvalueに該当します。

  • 変数名
  • 配列要素
  • 構造体・共用体のメンバ変数
  • 関数呼び出し(lvalue参照を返す場合)
  • アドレス演算子(&)で取得したポインタ

lvalueの例:

int x = 10; // 変数名
int arr[] = {1, 2, 3}; // 配列要素
struct Point {
  int x, y;
} p; // 構造体メンバ変数
int func() &; // lvalue参照を返す関数
int* ptr = &x; // アドレス演算子で取得したポインタ

rvalueは、"right-value"の略で、lvalueとは異なり、プログラムが値を取得することしかできない式を表します。具体的には、以下の要素がrvalueに該当します。

  • リテラル(整数、文字列、浮動小数点など)
  • 演算式
  • 対象を持たない一時オブジェクト
10; // 整数リテラル
"Hello, world!" // 文字列リテラル
3.14159; // 浮動小数点リテラル
func(); // lvalue参照を返さない関数
x + y; // 演算式

xvalue(移動可能なrvalue)

xvalueは、"movable rvalue"の略で、移動演算子を用いて所有権を移動できるrvalueを表します。具体的には、以下の要素がxvalueに該当します。

  • rvalue参照を返す関数呼び出し
  • rvalue参照型へのキャスト
std::move(x); // rvalue参照を返す関数呼び出し
static_cast<std::move_iterator<int*>>(ptr); // rvalue参照型へのキャスト

glvalueは、**"generalized left-value"**の略で、lvalueとxvalueを包括する概念です。つまり、glvalue式は、プログラムが値を取得したり書き換えたり、所有権を移動したりすることができます。

glvalue = lvalue ∪ xvalue

prvalueは、**"pure rvalue"**の略で、xvalueではないrvalueを表します。具体的には、以下の要素がprvalueに該当します。

  • リテラル

lvalue、rvalue、xvalue、glvalue、prvalueは、C++における式の値を表現するための重要な概念です。これらの概念を理解することで、C++プログラムの動作をより深く理解することができます。

  • C++11では、rvalue参照が導入され、xvalueの概念が追加されました。
  • rvalue参照とxvalueは、移動セマンティクスと呼ばれる機能と密接に関連しています。移動セマンティクスは、リソースのコピーではなく所有権の移動を行うことで、パフォーマンスを向上させることができます。



int x = 10; // 変数 (lvalue)

// xの値を取得
int y = x;

// xの値を書き換える
x = 20;
int y = 10; // リテラル (rvalue)

// yの値を取得 (コピー)
int z = y;

// yの値を書き換えることはできない
y = 20; // エラー
std::vector<int> vec = {1, 2, 3}; // コンテナ (lvalue)

// vecの要素をxvalueとして取得
auto it = vec.begin();
int value = *it;

// vecの要素を書き換えることはできない
*it = 10; // エラー

// vecの所有権を移動
std::vector<int> new_vec = std::move(vec);

glvalueとprvalueの例

int x = 10; // 変数 (glvalue)
int y = 10; // リテラル (prvalue)

// xとyを比較
if (x == y) {
  // ...
}



図表による説明

値のカテゴリの関係を図表で表すことで、理解しやすくなります。以下は、lvalue、rvalue、xvalueの関係を示す図表の一例です。

式
  /\
 /  \
lvalue  rvalue
   /\
  /  \
 glvalue  prvalue
        /\
       /  \
      xvalue  ...

類似概念との比較

lvalue、rvalue、xvalueなどの概念は、他のプログラミング言語における類似概念と比較することで理解しやすくなります。例えば、以下のように比較することができます。

C++JavaC#
lvalue左辺値左辺値
rvalue右辺値右辺値
xvalue移動可能なrvalue移動可能なrvalue
glvalue一般化左辺値一般化左辺値
prvalue純粋な右辺値純粋な右辺値

実践的な例

これらのリソースは、lvalue、rvalue、xvalueなどの概念をより深く理解し、実際のコードでどのように使用するのかを学ぶのに役立ちます。


c++ c++11 expression



スマートポインタとは何ですか?いつ使うべきですか? (C++、ポインタ、C++11)

スマートポインタは、C++におけるポインタの安全性を向上させるためのテンプレートクラスです。通常のポインタとは異なり、メモリリークやダングリングポインタの問題を自動的に解決します。メモリリークの防止: スマートポインタは、オブジェクトが不要になったときに自動的にメモリを解放します。これにより、メモリリークを防止することができます。...


C++/Cにおける構造体のsizeofとメンバーの和の関係について

日本語解説C++やC言語において、構造体のsizeofは、その構造体内の各メンバーのsizeofの合計と必ずしも一致しません。これは、構造体のメモリレイアウトやパディングによる影響です。メモリアライメント: 多くのプロセッサは、特定のデータ型を特定のアドレス境界に配置することを要求します。例えば、4バイトの整数型は通常4バイト境界に配置されます。...


C++における基底クラスコンストラクタの呼び出し規則の代替方法

C++において、派生クラスのコンストラクタは、その基底クラスのコンストラクタを必ず呼び出さなければなりません。これは、基底クラスの初期化が派生クラスの初期化に先立つ必要があるためです。明示的な呼び出し:class Derived : public Base { public: Derived() : Base(initial_value) { // 派生クラスの初期化 } }; この場合、Base(initial_value)の部分が、基底クラスのコンストラクタを明示的に呼び出しています。...


C++におけるexplicitキーワードの代替方法

explicitキーワードは、C++においてコンストラクタのオーバーロードを制限するために使用されます。コンストラクタは、クラスのオブジェクトを初期化するための特別なメンバ関数です。コンストラクタをオーバーロードすると、異なる引数リストを持つ複数のコンストラクタを定義することができます。...


C++におけるPOD型以外のデータ型 (日本語)

POD (Plain Old Data) 型 は、C++において、C言語の構造体と互換性のある基本的なデータ型のことです。POD型は、メモリレイアウトが単純であり、C言語のデータ型と直接対応しています。これにより、C++とC言語の間でのデータのやり取りが容易になります。...



c++ c++11 expression

C++におけるキャストの比較: Regular Cast, static_cast, dynamic_cast

C++では、異なるデータ型間で値を変換する操作をキャストと呼びます。キャストには、regular cast、static_cast、dynamic_castの3種類があります。最も単純なキャスト方法です。コンパイル時に型チェックが行われますが、実行時に型安全性が保証されません。


C/C++ ビット操作入門: 単一ビットの設定、クリア、トグルの代替方法

C++とCでは、ビットレベルでの操作を行うことができます。これは、低レベルなシステムプログラミングや、効率的なデータ処理において重要です。ビット演算子& : AND| : OR~ : NOT<< : 左シフト>> : 右シフトビット位置は、通常0から始まり、右から左にインデックスされます。


C++におけるクラスと構造体の使い分け:具体的なコード例

C++では、クラスと構造体はどちらもデータと関数をカプセル化するための手段ですが、その使用目的とデフォルトのアクセス修飾子に違いがあります。デフォルトのアクセス修飾子: private主な用途:オブジェクト指向プログラミング (OOP) における抽象的なデータ型を定義する。データの隠蔽とカプセル化を実現する。継承やポリモーフィズムなどのOOPの概念を活用する。


C++におけるポインタ変数と参照変数の違い

ポインタ変数と参照変数は、どちらも他の変数のメモリアドレスを保持するという意味で似ています。しかし、その使用方法や特性にはいくつかの重要な違いがあります。宣言方法: データ型 *変数名;値: 変数のアドレスを保持する。操作:アドレスの変更が可能。*演算子を使って間接参照が可能。->演算子を使って構造体やクラスのメンバにアクセス可能。


C++のswitch文で変数宣言ができない理由:具体的なコード例と解説

C++では、switch文の内部で変数を宣言することができません。この制限は、C++の構文規則によるものです。switch文は、特定の値と比較して、それに対応する処理を実行する制御構造です。変数を宣言した場合、その変数のスコープがswitch文の内部に限定され、switch文の外部からアクセスできなくなります。これは、switch文の構造と目的と相容れないためです。