この章ではポインタについての具体例と配列のアドレスとポインタについて説明します。
以下の例は変数のアドレスを関数に渡し、ポインタ変数で受け取っている例です。 関数本体では以下のようにアドレスをポインタを受け取ります。 tashizan関数の中で*aと5を足して*aに代入し、*bと5を足して*bに代入しています。
*aには変数xのアドレスが入っているので、*aとxは同じ意味になります。
*bには変数yのアドレスが入っているので、*bとyは同じ意味になります。
つまり、*aは2と5を足して7になりましたが、*aが7になることによって、main関数の中のxが7に変わっています。
*bは4と5を足して9になりましたが、*bが9になることによって、main関数の中のyが9に変わっています。 次の例では配列のアドレスとポインタについて説明します。 もしくは以下のように書くことができます。 printf関数の変換指定文字を「%p」にして配列を出力すると配列の先頭のアドレスを取得することが出来ます。
つまり、配列のアドレスは配列の先頭の要素のアドレスになります。
つまり、配列名を書くだけで配列xの一番前のアドレスが取得出来ます。
と同じ結果はと書いても得ることが出来ます。
つまり、xと&x[0] (配列の先頭の要素のアドレス)はまったく同じアドレスです。
配列名でアドレスを取得するのとは違ってのアドレスを取得するには「&」を付けなくてはいけません。 それはのようにと記述してください。
xが先頭のアドレスを示すので、+1を付け加えることで先頭から一つアドレスを進めると言う意味になります。
つまり、と同じ意味になります。
は前から2番目のアドレスです。
*(x+0)と書くと配列の先頭要素を取り出すことが出来ます。
これを書き換えるととなります。
配列の2番目の要素を取り出すにはと書きます。
+1は先頭から一つアドレスを進めると言う意味になります。 結果は以下の通りです。次の例ではポインタの加算を使った値の書き換えの方法について説明します。 hairetsu(x);の箇所で配列xのアドレスをhairetsu関数のポインタ変数に渡しています。 hairetsu関数内で書き換えた結果、main関数の配列xの要素である「1,2,3」が「34,46,98」に書き換えられます。
先ほどの例をfor文を使って書き換えると以下のようになります。 次はポインタに関する演算について説明します。 この例ではポインタ *gに配列xのアドレスを入れています。
1 gにはxの先頭要素のアドレスが入っているので、「*」を付けることで1番前の要素を得ることが出来ます。
結果は4です。 g++;とありますが、これはポインタの指している位置を1つ先に進めると言う意味です。
ポインタを進めた後にと出力すると、前から2番目の要素の値である6を得ることが出来ます。
さらにポインタの指している位置を進めると8が得られます。
3 この箇所はポインタを戻しています。
ポインタをのように減算子を使って書くと、その時点のポインタから1つ後ろに戻る動作をします。
つまり8の時点から1つ下がるので6が得られます。
では次の例を見てみましょう。 これまでは配列のアドレスを関数本体に渡す時、ポインタで受け取っていましたが今度は他の方法で配列を関数に渡します。
hairetsu(x);のように配列のアドレスを渡すまでは同じですが、関数本体のほうはのように配列で受け取っています。
そして要素の数(3)だけ繰り返しループして、の結果をg配列に代入しています。
つまりg[i]のiに0から2までが代入され、その都度の計算結果が代入されることになります。 配列xと配列gはまったく同じなので(xもgも同じアドレスを示しています)、この例の結果は「10と10と10」になります。
では次の例に進みます。 仮引数にポインタを指定した場合、これまでは のように代入していましたが、実はこのような難しい表現を使わなくてものようにポインタにを付けて値を代入することもできます。
g[i]のiに0から2までが代入され、その都度の計算結果を代入します。 結果は「6と6と6」です。
次は文字列を扱う配列とポインタの関係について説明します。 配列の章で学習しましたが、文字列を配列で扱う時には型をcharにして、のように書きます。
と同じことをポインタを使って書き換えるとのように書くことが出来ます。
出力する時には もしくは と書きます。 ポインタを使ったのような書き方では再度のように代入(上書き)することができます。
では次の例に進みます。 関数の引数にポインタを使用している場合に、引数の値を変更したくない時があります。
そのような時にconstを変数の前に指定します。
constをつけることにより、引数の中身を書き換えないことを明確にできるメリットがあります。
関数の引数に const を指定すると、その関数の中では値を書き換えることができなくなります。
ですので、のような記述はできませんので、この例はエラーになります。
次の例は2次元配列とポインタの関係を学習します。 複数の文字列を配列に入れるには配列のポインタを使用します。 例ではの箇所です。
出力する方法は普通の配列と同じように添え字を使って出力します。
タグ: アドレス
この章ではポインタについて説明します。
「ポインタ」を理解するためにはコンピュータがどのようにデータを記憶しているかの理解が必要です。
変数の値をコンピュータが記憶する場合、そのデータはメモリ上に置かれます。
メモリとは, 1バイトの幅の区画がビルの階のように連続してつながっているものと思ってください。
例えば、ビルは下から順番に1,2…と階数が付けられますが、メモリも同じで「1バイトの幅の区画」に順番に「番号」が付けられています。
この番号を「アドレス」と言います。
このアドレスは例えば0026FDE8のような16進数で表します。
変数を記憶するには1バイトで済むとは限らず、型によっては何バイトも必要になることがあります。
変数の型によって、使うバイト数が決まると言うことです。 では最初に戻ってアドレスについて詳しく説明します。
例えば2147483647と言う大きな整数を使用して4バイト分を使うとします。
この整数のアドレス(メモリ上で)はどこにあると思いますか?
実はこの整数のアドレスはその整数に割り当てられたアドレスの先頭のアドレスにあります。 ではアドレスが分かったところで次の例でポインタについて説明します。 ここではsizeof演算子で変数の大きさを測っています。
結果をバイト単位で表します。
変数iはsizeof演算子で調べた結果、4バイトあることがわかります。 変数iのアドレスを知るには変数の前に「&」を付けます。
これを付けることによって、変数iの値がメモリのどこに記憶されているか知ることが出来ます。
「&」をアドレス演算子と言います。
このアドレスは使っているコンピュータによって違います。
例えば「0044FA6C」などと表示されます。
そして、そのアドレスを出力するためのprintf関数の変換指定文字は「%p」を使います。
もしくは「std::cout」を使って以下のように書いてもいいです。 ここで「ポインタ」について説明します。
ポインタとは変数のアドレスを記憶する特殊な変数のことです。
これをポインタ変数と言います。
つまり、アドレスを記憶するだけの変数を作ることが出来るのです。
ポインタ変数も普通の変数と同じく宣言しなくてはいけません。 例ではと宣言しています。
そしてポインタの宣言をした後で、のように変数iのアドレスをポインタ変数gに代入します。
以下のように宣言と初期化を一緒に書くこともできます。 これで変数iのアドレスがポインタgに代入されましたので、の結果もと同じく0044FA6Cになります。
以下のように書いても同じ意味です。 ここでさらにポインタについての補足をします。先ほどiのアドレスをポインタgに入れましたが、これにより、iとgは同じアドレスを保有することになりました。
抽象的に言いますと、i君とgさんが同居することを意味していまして、gに起こった出来事はiに起こったことと同じ意味を持つようになります。
ですので一身同体のゆえ、次の5番ではポインタgからiの値を取り出しています。 gには変数iのアドレスが入っていますが、「*g」のようにという形で出力すると変数iの値を得ることが出来ます。
つまり、iとgには同じアドレスが入っているので、「*g」と書くことでiに入っている値が取り出すことが出来ます。
この例の結果は2147483647となり、iの値と同じになります。「*g」に2147483600を代入した後にiを出力するとどうなると思いますか?
結果は「2147483600」になります。 ポインタ変数「*g」は変数iの代わりに使うことが出来ることを意味します。
次の章で具体的なポインタの使い方について説明します。
ポインタを使ってみよう
この章ではポインタについての具体例と配列のアドレスについて説明します。
以下の例は変数のアドレスを関数に渡し、ポインタ変数で受け取っている例です。 tashizan関数の中では*aと5を足して*aに代入し、*bと5を足して*bに代入しています。
*aには変数xのアドレスが入っているので、*aとxは同じ意味になります。
*bには変数yのアドレスが入っているので、*bとyは同じ意味になります。
次の例では配列のアドレスとポインタについて説明します。初めに配列のアドレスを取得する方法について説明します。 つまり、配列名を書くだけで配列xの一番前のアドレスが取得できます。
配列自体のアドレスは配列の先頭アドレスを指すということです。
xが先頭のアドレスを示すので、「+1」を付け加えることで先頭から一つアドレスを進めると言う意味になります。
これはと同じ意味になります。
は前から2番目のアドレスを意味します。
では前から3番目のアドレスを得るためにはどうしたらいいでしょうか?
結果は以下のようにもしくはと書いてください。 次に配列の要素をポインタで取り出す方法について説明します。 =>*(x+0)と書くと配列の先頭要素を取り出すことが出来ます。 「数値」に添え字を書きますとその添え字に相当する要素を取得できます。
「*(x+0)」は「*x」と同じですので、以下のようにも書くことができます。 つまり、「*配列名」と書いても配列の0番目の要素を取得できます。
配列の前から2番目の要素を取得するにはと書きます。
配列の前から3番目の要素を取得するにはと書きます。
結果は以下の通りです。
次の例ではポインタの加算を使った値の書き換え方法について説明します。 hairetsu(x);の箇所で配列xのアドレスをhairetsu関数のポインタ変数であるに渡しています。
そして、*(g+0)=34;の箇所で配列の一番前の要素に34を代入して、
*(g+1)=46;の箇所で配列の前から二番目の要素に46を代入して、
*(g+2)=98;の箇所で配列の前から三番目の要素に98を代入しています。 hairetsu関数内で書き換えた結果、main関数の配列xの要素である「1,2,3」が「34,46,98」に書き換えられます。
先ほどの例をfor文を使って書き換えると以下のようになります。
次はポインタに関する演算について説明します。この例ではポインタ*gに配列xのアドレスを入れています。 gにはxの先頭要素のアドレスが入っているので、「*」を付けることで1番前の要素の値を得ることができます。
結果は4です。 g++;(加算子)とありますが、これはポインタを1つ先に進めると言う意味です。
つまり、ポインタに加算子を使っているわけです。
最初のg++;でポインタを進めた後にと出力すると、前から2番目の要素の値である6を得ることが出来ます。
さらに2つ目のg++;でポインタを進めると8が得られます。
では次の例に進みます。 これまでは配列のアドレスを関数本体の仮引数に渡す時、ポインタで受け取っていましたが、今度は他の方法で配列を関数に渡します。
hairetsu(x);のように配列のアドレスを渡すまでは同じですが、関数本体のほうはのように配列で受け取っています。
そして要素の数(3)だけ繰り返しループして、の結果をg配列に代入しています。
つまりg[i]のiに0から2までが代入され、その都度の計算結果が代入されることになります。 配列xと配列gはまったく同じなので(xもgも同じアドレスを示しています)、この例の結果はになります。
では次の例に進みます。仮引数にポインタを指定した場合、これまでは のように代入していましたが、実はこのような難しい表現を使わなくても、のようにポインタにを付けて値を代入することもできます。
g[i]のiに0から2までが代入され、その都度の計算結果を代入します。 結果は「6と6と6」です。
次は文字列を扱う配列とポインタの関係について説明します。 配列の章で学習しましたが、文字列を配列で扱う時には型をcharにして、のように書きます。
この配列をポインタを使って書き換えると以下のように書くことが出来ます。 このようにchar型のポインタに文字列を入れることができます。
では次の例に進みます。 関数の引数にポインタを使用している場合に、引数の値を変更したくない時があります。
そのような時にconstを仮引数の前に指定します。
関数の引数に const を指定すると、その関数の中では値を書き換えることができなくなります。
ですので、のような記述はできませんので、この例はエラーになります。
次の例は2次元配列とポインタの関係について説明します。 複数の文字列を配列に入れるには配列のポインタを使用します。
基本構文は以下の通りです。 配列のポインタを出力する方法は普通の配列と同じように添え字を使って出力します。
この章ではポインタについて説明します。
「ポインタ」を理解するためにはコンピュータがどのようにデータを記憶しているのかの理解が必要です。
変数の値をコンピュータが記憶する場合、そのデータはメモリ上に置かれます。
メモリとは, 1バイトの幅の区画がビルの階のように連続してつながっているものと思ってください。
例えば、ビルは下から順番に「1,2…」と階数が付けられますが、メモリも同じで順番に「1バイトの幅の区画」に番号が付けられています。
この番号をアドレスと言います。
このアドレスは例えば0026FDE8のように16進数で表します。
変数を記憶するには1バイトで済むとは限らず、型によっては何バイトも必要になることがあります。
変数の型によって、使う大きさ(バイト数)が決まると言うことです。 では最初に戻って「アドレス」について詳しく説明します。
例えば2147483647と言う大きな整数を使用して4バイト分を使うとします。
この整数のアドレス(メモリ上において)はどこにあると思いますか?
実はこの整数のアドレスは「その整数に割り当てられたアドレスの先頭のアドレス」にあります。
では次の例でポインタについて説明します。
結果は以下の通りです。 ここではsizeof演算子で変数の大きさを測っています。
結果はバイト単位で表します。
変数iはsizeof演算子で調べた結果、4バイトあることがわかります。 変数iのアドレスを知るには変数の前に&を付けます。これを付けることによって、変数iの値がメモリのどこに記憶されているか知ることができます。
&をアドレス演算子と言います。
このアドレスは使っているコンピュータによって違います。
例えばなどと出力されます。
そして、そのアドレスを出力するためのprintf関数の変換指定文字はを使います。 ここで「ポインタ」について説明します。
ポインタとは変数のアドレスを記憶する特殊な変数のことです(ポインタ変数と言います)。
つまり、アドレス(住所)を記憶するだけの変数を作ることができるのです。
ポインタ変数も普通の変数と同じく宣言しなくてはいけません。 そしてポインタの宣言をした後で、のように変数iのアドレスをポインタ変数gに代入します(ポインタとは変数のアドレスを記憶する変数のことでしたね)。 ここでさらにポインタについての補足をします。
先ほどiのアドレスをポインタgに入れましたが、これにより、iとgは同じアドレスを保有することになりました。
抽象的に言いますと、i君とgさんが同居することを意味していまして、gに起こった出来事はiに起こったことと同じ意味を持つようになります。
ですので一身同体のゆえ、次の5番ではポインタgからiの値を取り出しています。 *gに2147483600を代入して、printfでiを出力するとどうなると思いますか? 次の章で具体的なポインタの使い方について説明します。