VC++2010にて下記コードのビルドは通るのですが、
vVecotrInt, vVectorIntPtrの要素数0のとき、iptr代入時に
Debug assertaion Failed! vector iterator not dereferencableとなります。
そもそもbegin()はイテレータなので、ポインタに代入しようとしていることが間違いかと思うのですが。
質問1.int* iptr = ~ ではなく、std:vector<int>::iteratorとすれば、
要素数が0でもエラーがおきません。この違いは何でしょうか?
質問2.そもそもイテレータをポインタに代入して何か得することがあるんでしょうか?
ただイテレータとポインタは同じようなものだと思って、コーディングしてるだけなんでしょうか・・・
コード:
// vVectorIntの要素を間接参照して(参照先はint)、そのアドレスをポインタに格納
std::vector<int> vVectorInt;
int* iptr = &*vVectorInt.begin(); // ここでvector iterator not dereferencable
// vVectorIntPtrの要素を間接参照して(参照先はint*)、ポインタに格納
std::vector<int*> vVectorIntPtr;
int* iptr = *vVectorIntPtr.begin(); // ここでvector iterator not dereferencable
// vVecotrIntPtr2の要素数0のときでも、イテレータを使えば問題ない
std::vector<int*> vVectorIntPtr2;
std::vector<int*>::iterator itr = vVectorIntPtr2.begin();
No.4ベストアンサー
- 回答日時:
まずbegin()関数は反復子(イテレータ)を返すことを理解してください。
イテレータはポインタとは異なりますが、単項*演算子を使ってその要素を取り出すことができます。
これを元にコードを解釈すると、
std::vector<int> vVectorInt;
int* iptr = &*vVectorInt.begin();
のvVectorInt.begin() はintを要素とするベクタのイテレータ。
従って*vVectorInt.begin();はその要素であるintデータそのものです。
iptrはintへのポインタであるので、その要素であるintデータにアドレス演算子&を使用することによってアドレスを取得しています。
ベクタの要素数が0の場合、begin()は最終要素へのイテレータend()に一致し、無効なデータを指し示します。
従って、*vVectorInt.begin();は無効データとなり、not dereferencableというエラーが発生します。
std::vector<int*> vVectorIntPtr;
int* iptr = *vVectorIntPtr.begin(); // ここでvector iterator not dereferencable
も同様。
std::vector<int*> vVectorIntPtr2;
std::vector<int*>::iterator itr = vVectorIntPtr2.begin();
これはイテレータをとってきているのであって、イテレータを使った要素の取り出しは行っていません。
要素数0の場合、*itrを使えばnot dereferencableが派生します。
ベクタに限らずSTLコンテナではend()に一致するイテレータで要素をアクセスしてはいけません。
>質問1.int* iptr = ~ ではなく、std:vector<int>::iteratorとすれば、
> 要素数が0でもエラーがおきません。この違いは何でしょうか?
根本的な勘違いをしていると思います。
STLコンテナは要素数0でもイテレータを使用できますが、そのイテレータを使って要素を使用するにはイテレータが有効なデータを指している必要があります。そのためにはend()と一致していないことを確認する必要があります。
> 質問2.そもそもイテレータをポインタに代入して何か得することがあるんでしょうか?
> ただイテレータとポインタは同じようなものだと思って、コーディングしてるだけなんでしょうか・
提示されたコードはイテレータをポインタに代入しているわけでは有りません。
イテレータを使って要素のアドレスを取得しているだけです。
#2の補足
>イテレータだったら要素数を気にせずアクセスできるのに、
>アドレスを取得するには要素数を気にしなきゃいけないって、
アドレスを取得するのに要素数を気にしなきゃいけないのではなく、イテレータが有効なデータを指しているかを気にしなきゃいけないのです。
そもそもSTLではポインタを使用する必要はなくてイテレータを使用すれば事足ります。
ポインタを取得しているのはSTLを使用していないcで書かれた古いコードに配列を渡すために使用していると思われます。
>利点がいまいちわからないです・・・
イテレータを使用すれば、コンテナの種類(ベクタ、リスト等)によら同じアルゴリズムで処理を行うことができます。
No.5
- 回答日時:
> 利点がいまいちわからないです・・・
利点は、まさに"Debug assertaion Failed! vector iterator not dereferencable"が発生することではないですかね。
生ポインタは初期化せずに使用することができてしまい、不正なアドレスなら例外が発生してすぐにバグと気付くけど、不正でないアドレスになっていた場合、そのポインタを使用したところは正常に動作したように見えて、プログラムのほかの場所で使用している変数を壊してしまう。
それに対してイテレータは初期化せずに使用したらランタイムでチェックすることが可能です。
No.3
- 回答日時:
すいません。
2つめはよく見たら、int*のvectorなんですね。じゃあ、コンパイルは通ります。これは、たんにベクトルの要素(それ自体がポインタ)を取り出してるだけです。
質問2の、「イテレータをポインタに代入する」ことはできません。なんで利点とかいう議論自体が成り立ちません。
1つめ、2つめのコードは、どちらも、「イテレータをポインタに代入している」コードではありません。
No.2
- 回答日時:
まず、提示のコード2つ目は、コンパイル通らないですよね。
1つ目は、「イテレータをポインタに代入している」わけではありません。
(イテレータをポインタに代入することはできません。多くのコンパイラでコンパイルエラーになるはずです)
そうではなくて「vectorの一番目の要素」の生アドレスを、生ポインタに代入しているだけです。vectorが空荷場合には、「vectorの一番目の要素」が存在しないのですね実行時エラーになります。
この回答への補足
> 「vectorの一番目の要素」の生アドレスを、生ポインタに代入
そうですね、イテレータを直接代入してるわけではないですよね・・・
しかし、イテレータだったら要素数を気にせずアクセスできるのに、
アドレスを取得するには要素数を気にしなきゃいけないって、
利点がいまいちわからないです・・・
2つ目のはコンパイル通りました。(VC++2010)
お探しのQ&Aが見つからない時は、教えて!gooで質問しましょう!
関連するカテゴリからQ&Aを探す
おすすめ情報
- ・漫画をレンタルでお得に読める!
- ・14歳の自分に衝撃の事実を告げてください
- ・架空の映画のネタバレレビュー
- ・「お昼の放送」の思い出
- ・昨日見た夢を教えて下さい
- ・【お題】絵本のタイトル
- ・【大喜利】世界最古のコンビニについて知ってる事を教えてください【投稿~10/10(木)】
- ・メモのコツを教えてください!
- ・CDの保有枚数を教えてください
- ・ホテルを選ぶとき、これだけは譲れない条件TOP3は?
- ・家・車以外で、人生で一番奮発した買い物
- ・人生最悪の忘れ物
- ・【コナン30周年】嘘でしょ!?と思った○○周年を教えて【ハルヒ20周年】
- ・ハマっている「お菓子」を教えて!
- ・最近、いつ泣きましたか?
- ・夏が終わったと感じる瞬間って、どんな時?
- ・10秒目をつむったら…
- ・人生のプチ美学を教えてください!!
- ・あなたの習慣について教えてください!!
- ・牛、豚、鶏、どれか一つ食べられなくなるとしたら?
- ・都道府県穴埋めゲーム
デイリーランキングこのカテゴリの人気デイリーQ&Aランキング
-
セグメントエラー
-
ハンドルはポインタか
-
DelphiでCreateProcessがうまく...
-
オーバーロードされたメンバ関...
-
ポインタ引数をさらにポインタ...
-
init関数の意味
-
C言語での`void**`の意味
-
C言語のポインタに直接アドレス...
-
C++配列の型に関するC++規格に...
-
VC++でmciSendString
-
B-reps(境界表現)をC、C++で...
-
C++とWIN32APIとゲームプログラ...
-
nullの概念
-
基本アルゴリズムの『返す』の...
-
#include <stdio.h> int main(v...
-
構造体メンバがポインタである...
-
仮引数の変数をローカル変数に...
-
関数の引数をvoid*でキャストする
-
ファイルポインタ
-
C言語の習得度別お勧め参考書
マンスリーランキングこのカテゴリの人気マンスリーQ&Aランキング
-
C言語のポインタに直接アドレス...
-
セグメントエラー
-
init関数の意味
-
fopne で失敗する原因
-
Run-Time Check Failure #3とい...
-
戻り値で構造体を返すことは可...
-
ExcelVBAでのkernel32(64bit)
-
LPSTR型の初期化について
-
参照型で受け取った引数をポイ...
-
C言語でのconstを返す関数
-
ハンドルはポインタか
-
ハンドル、アドレス、ポインタ...
-
PASCALとFARの意味
-
CWnd::EnableWindow()の扱い方
-
ポインタについて
-
デバイスハンドルとは?
-
C言語の文字列?処理 strcpyやl...
-
ポインタのミスでOS壊れるの...
-
C++ vectorのbeginについて
-
基本アルゴリズムの『返す』の...
おすすめ情報