C ポインタ 新しいページはコチラ

提供: yonewiki
移動: 案内, 検索
(ポインタ)
3行: 3行:
 
<br />
 
<br />
 
== '''ポインタ''' ==
 
== '''ポインタ''' ==
プログラムを組む上で、各種の型の変数を使いますが、変数自身はコンピュータ上のメモリ空間に保存されます。<br />
+
プログラムを組む上で、各種の型の変数を使いますが、変数自身はコンピュータ上のメモリ空間に保存されます。
 
ポインタはそのメモリ空間のアドレスを保持するための変数です。int型変数のためのint型のアドレス変数みたいなもの<br />
 
ポインタはそのメモリ空間のアドレスを保持するための変数です。int型変数のためのint型のアドレス変数みたいなもの<br />
<br />
+
 
 +
 
 
ここで少しメモリ空間についてのイメージを少しだけもってもらえるために、補足しておきます。<br />
 
ここで少しメモリ空間についてのイメージを少しだけもってもらえるために、補足しておきます。<br />
メモリ空間にはアドレス番号と呼んでいるものが割り振られています。日本の空間の地図上のどこにあなたの家があるかを指し示す住所と同じようなものを<br />
+
メモリ空間にはアドレス番号と呼んでいるものが割り振られています。日本の空間の地図上のどこにあなたの家があるかを指し示す住所と同じようなものを、単なる番号だけで定義しているのです。半導体メモリDRAMについて少し調べると長方形上の敷地3cm*4cm位の中に4~8つほどの長方形の敷地があって、全部で4G個(個って書きましたが、情報処理用語としてはbit(ビット)という単位が使われます。Gはギガで10の9乗で10億)ほどの記憶領域があることがわかります。<br />
単なる番号だけで定義しているのです。半導体メモリDRAMについて少し調べると長方形上の敷地3cm*4cm位の中に4~8つほどの長方形の敷地があって、<br />
+
 
全部で4G個(個って書きましたが、情報処理用語としてはbit(ビット)という単位が使われます。Gはギガで10の9乗で10億)ほどの記憶領域があることがわかります。<br />
+
 
更にその3cm~4cmの敷地を同じ樹脂の中に複数枚重ねたり、その樹脂の塊を電子回路基盤に1個~16個くらい<br />
+
更にその3cm~4cmの敷地を同じ樹脂の中に複数枚重ねたり、その樹脂の塊を電子回路基盤に1個~16個くらいとりつけていたり、その電子回路基板を複数個とりつけたりして、現在のNotePCとかでは8GByte(8bitで1Byte(バイト))という容量のものを取り付けて販売している状況です。ある人はプログラム的には一本に並んだ数十億の箱だと言ったりしますが、物理的には、DRAMメモリのある敷地の中のある一点だったり、HDD上においやられた領域だったりするわけです。<br />
とりつけていたり、その電子回路基板を複数個とりつけたりして、現在のNotePCとかでは8GByte(8bitで1Byte(バイト))という容量のものを取り付けて販売している状況です。<br />
+
 
ある人はプログラム的には一本に並んだ数十億の箱だと言ったりしますが、物理的には、DRAMメモリのある敷地の中のある一点だったり、HDD上においやられた領域だったりするわけです。<br />
+
 
<br />
+
※単位についてですが、K(キロ)は普通10の3乗で1000ですが、厳密にはコンピュータでは2の10乗の1024Byteが1KByteになります。2の10乗KByteの1024KByteで1MByte(メガバイト)ですし、1024MByteで1GByteとなります。但し、普通に1000Byteを1KByteとしたり、1000KByteを1MByte、1000MByteを1GByteとしている。ざっくりとした仕様でコンピュータの性能を表現しているものが一般的です。情報処理用語として1TByte(テラバイト テラは10の12乗)と表現した場合、1兆0995億1162万7776Byteですが、ざっくり仕様が使われている各種メーカの容量では、実際には10の12乗Byteで995億1162万7776Byte少ないです。仕様書の隅っこに1000kByteを1Byteとして計算していますとか明示してあります。<br />
※単位についてですが、K(キロ)は普通10の3乗で1000ですが、厳密にはコンピュータでは2の10乗の1024Byteが1KByteになります。<br />
+
 
2の10乗KByteの1024KByteで1MByte(メガバイト)ですし、1024MByteで1GByteとなります。但し、普通に1000Byteを1KByteとしたり、1000KByteを1MByte、1000MByteを1GByteとしている<br />
+
 
ざっくりとした仕様でコンピュータの性能を表現しているものが一般的です。<br />
+
CPU(Central Processing Unit)とよばれる中央処理装置にも記憶領域があります。キャッシュとよばれています。HDD(Hard Disk Drive)やSSD(Solid State Drive)にも処理記憶領域は作れます。いろいろなところに作れますがアドレスは番号だけで管理されています。CPUの中にある記憶領域を使った場合が最も早く処理され、次にDRAM、そしてSSD、HDD、その他という具合に保存されている場所によっても処理速度が異なるのも特徴です。記憶している領域により処理速度が低下する問題を解決する手法もよく使われる命令を処理速度の高いところにおいたりと、いろいろな試みがなされています。いろいろな試みについてはWindowsやUnixといったOS(Operating System)側が制御します。<br />
情報処理用語として1TByte(テラバイト テラは10の12乗)と表現した場合、1兆0995億1162万7776Byteですが、<br />
+
 
ざっくり仕様が使われている各種メーカの容量では、実際には10の12乗Byteで995億1162万7776Byte少ないです。<br />
+
 
仕様書の隅っこに1000kByteを1Byteとして計算していますとか明示してあります。<br />
+
<br />
+
CPU(Central Processing Unit)とよばれる中央処理装置にも記憶領域があります。キャッシュとよばれています。<br />
+
HDD(Hard Disk Drive)やSSD(Solid State Drive)にも処理記憶領域は作れます。いろいろなところに作れますがアドレスは番号だけで管理されています。<br />
+
CPUの中にある記憶領域を使った場合が最も早く処理され、次にDRAM、そしてSSD、HDD、その他という具合に保存されている場所によっても処理速度が異なるのも特徴です。<br />
+
記憶している領域により処理速度が低下する問題を解決する手法もよく使われる命令を処理速度の高いところにおいたりと、いろいろな試みがなされています。<br />
+
いろいろな試みについてはWindowsやUnixといったOS(Operating System)側が制御します。<br />
+
<br />
+
 
少し、脱線しましたが、記憶領域にはアドレス番号が割り振られているということでした。また更に脱線すると、32bitOSとか64bitOSという種類がありますが、<br />
 
少し、脱線しましたが、記憶領域にはアドレス番号が割り振られているということでした。また更に脱線すると、32bitOSとか64bitOSという種類がありますが、<br />
このアドレス番号の割り振りに使われる桁数に違いがあります。32bitなら2の32乗個のアドレス42億9496万7296個、64bitなら2の64乗個のアドレス1844京6744兆0737億0955万1616個が使えるということになります。<br />
+
このアドレス番号の割り振りに使われる桁数に違いがあります。32bitなら2の32乗個のアドレス42億9496万7296個、64bitなら2の64乗個のアドレス1844京6744兆0737億0955万1616個が使えるということになります。
 
1Gは10億7374万1824Byteでしたから、40億ちょいっつうのは4GByteのことで、32bitOSでは4GByteまでしか使えないということになります。<br />
 
1Gは10億7374万1824Byteでしたから、40億ちょいっつうのは4GByteのことで、32bitOSでは4GByteまでしか使えないということになります。<br />
<br /><br />
+
 
 +
 
 
という簡単な基礎知識を頭の片隅に本題に入ります。<br />
 
という簡単な基礎知識を頭の片隅に本題に入ります。<br />
例えば<br />
+
例えば、Visual Studio2012ではint型は4Byteの変数で使える値の種類は2の4*8bit=32bitで2の32乗=42億9496万7296となります。実際には負の数も表現できますので、先頭1ビットは正負の記号を表し、0なら正、1なら負数で全部32bit全部が1のときが-1です。2進数の補数ってのを勉強しないとダメかも、負数の最大は100000…0000001で、-21億4748万3646~21億4748万3647が表現できる値です。その4Byteを収めてるアドレスは64Bitアドレスなら1844京…の中のどこかから4Byte分のアドレスを使ってるということです。全部のアドレス分の資源16EByte(Eはエクサで10の18乗で100京)がコンピュータに備わっていればの話ですが...
Visual Studio2012では<br />
+
int型は4Byteの変数で使える値の種類は2の4*8bit=32bitで2の32乗=42億9496万7296となります。<br />
+
実際には負の数も表現できますので、先頭1ビットは正負の記号を表し、0なら正、1なら負数で全部32bit全部が1のときが-1です。2進数の補数ってのを勉強しないとダメかも<br />
+
負数の最大は100000…0000001で、-21億4748万3646~21億4748万3647が表現できる値です。<br />
+
その4Byteを収めてるアドレスは64Bitアドレスなら1844京…の中のどこかから4Byte分のアドレスを使ってるということです。<br />
+
全部のアドレス分の資源16EByte(Eはエクサで10の18乗で100京)がコンピュータに備わっていればの話ですが...<br />
+
 
とにかく、そういったアドレスを覚えさせるのが、ポインタ変数です。<br />
 
とにかく、そういったアドレスを覚えさせるのが、ポインタ変数です。<br />
<br />
+
 
 +
 
 
ポインタ変数は<br />
 
ポインタ変数は<br />
 
型名 *変数名;<br />
 
型名 *変数名;<br />
75行: 64行:
 
</syntaxhighlight>
 
</syntaxhighlight>
  
pintNumはint型のアドレスを保持するための変数ですが、ポインタ変数の宣言以外の所で*pintNumと*をポインタ変数の前につけるとアドレスが指し示す中身を、<br />
+
pintNumはint型のアドレスを保持するための変数ですが、ポインタ変数の宣言以外の所で*pintNumと*をポインタ変数の前につけるとアドレスが指し示す中身を、定義されている型にしたがって代入したり、内容を取り出したりすることができます。但し、最初に宣言された段階ではint型のアドレスを保持できるだけで、アドレスは保持していません。つまり、その実体、内容つまり中身、も*pintNumと指定したとしても、中身への代入はできません。したがって宣言した直後に *pintNum=10; としてもエラーになります。<br />
定義されている型にしたがって代入したり、内容を取り出したりすることができます。<br />
+
 
但し、最初に宣言された段階ではint型のアドレスを保持できるだけで、アドレスは保持していません。つまり、その実体、内容つまり中身、も*pintNumと指定したとしても、<br />
+
 
中身への代入はできません。したがって宣言した直後に *pintNum=10; としてもエラーになります。<br />
+
その問題を解決するのがnew演算子です。new演算子はポインタ変数の型と同じint型で、ポインタが指し示す先頭アドレスの生成と先頭アドレスからint型の分だけメモリ領域を確保し、ポインタ変数にアドレスを格納します。この結果アドレスの中身の方には名前がない状態ではありますが、ポインタ変数からでのみ操作できる変数が生成されたことになります。名もなき実体のアドレスをポインタ変数だけがしっているということです。delete演算子では、そうやって生成した名もなき実体のメモリ領域を解放することになります。もうここから先では使いません。どうぞご自由にという感じです。その宣言がされるまではずっとメモリ領域を確保し続けます。最近ではプログラムが終了したときに、このような手法でメモリ領域を確保した部分の解放処理をしてくれるようになりましたが、その昔、これを解放せずにプログラムが終わるという事態が多発していたのも事実です。使い終わったらいち早くかたずけることで資源を有効に利用できるのです。プログラム終了時に全部の片づけをやってくれてるのかは知りませんが…そういうことらしいです。<br />
その問題を解決するのがnew演算子です。new演算子はポインタ変数の型と同じint型で、ポインタが指し示す先頭アドレスの生成と先頭アドレスからint型の分だけメモリ領域を確保し、<br />
+
 
ポインタ変数にアドレスを格納します。この結果アドレスの中身の方には名前がない状態ではありますが、ポインタ変数からでのみ操作できる変数が生成されたことになります。<br />
+
名もなき実体のアドレスをポインタ変数だけがしっているということです。delete演算子では、そうやって生成した名もなき実体のメモリ領域を解放することになります。<br />
+
もうここから先では使いません。どうぞご自由にという感じです。その宣言がされるまではずっとメモリ領域を確保し続けます。最近ではプログラムが終了したときに、このような手法で<br />
+
メモリ領域を確保した部分の解放処理をしてくれるようになりましたが、その昔、これを解放せずにプログラムが終わるという事態が多発していたのも事実です。<br />
+
使い終わったらいち早くかたずけることで資源を有効に利用できるのです。プログラム終了時に全部の片づけをやってくれてるのかは知りませんが…そういうことらしいです。<br />
+
  
<br />
 
 
そして、アドレスそのものを使っての変数同士のやりとりは以下の例ように記述します。<br />
 
そして、アドレスそのものを使っての変数同士のやりとりは以下の例ように記述します。<br />
 
例3:ポインタ変数で実際にアドレスを使う手法<br />
 
例3:ポインタ変数で実際にアドレスを使う手法<br />
111行: 94行:
 
</syntaxhighlight>
 
</syntaxhighlight>
  
上記のように変数名の前に&をつけるとアドレスを返してくれる。このときの&をアドレス演算子と呼んでいます。<br />
+
上記のように変数名の前に&をつけるとアドレスを返してくれる。このときの&をアドレス演算子と呼んでいます。したがってもともとアドレス変数であるpintNumに対しても&アドレス演算子をつけることができ、アドレス番号を格納しているアドレスが返ります。プログラム上ではポインタ変数はint型の変数と同じアドレスを見ることで、名前は違っても同じものを指し示すことができるというのが上記の例です。アドレス番号っていったのに数字じゃないのが返って来てるよ?って言う方は16進数や2進数を勉強する必要があります。16進数での表示がされています。<br />
したがってもともとアドレス変数であるpintNumに対しても&アドレス演算子をつけることができ、アドレス番号を格納しているアドレスが返ります。<br />
+
 
プログラム上ではポインタ変数はint型の変数と同じアドレスを見ることで、名前は違っても同じものを指し示すことができるというのが上記の例です。<br />
+
 
アドレス番号っていったのに数字じゃないのが返って来てるよ?って言う方は16進数や2進数を勉強する必要があります。16進数での表示がされています。<br />
+
pintNumのアドレスはポインタ変数のアドレスで&pintNumですが、更に*をつけると*&pintNumとできこれはpintNumと同じになります。sizeof関数は、引数に指定した型や値の大きさ、つまりアドレスをいくつ分、使っているのか、使うのかをbyte単位で返却してくれる関数です。sizeof(pintNum)とすると32bitアプリケーションのプロジェクトならアドレスの大きさは4Byteなので4が返ってくるし、64bitアプリケーションのプロジェクトならアドレスの大きさは8Byteですから8が返ってきます。64bitPCでもプロジェクト設定がWin32なら4が返ってきます。64bitPCでVisualStudio2012をお使いの場合には、変更できると思っているのですが、※どういった組み合わせなら64bitアプリが作れるか掌握していないです。プロジェクトの設定をx64(64bit対応)に変更するには、Visual Studio2012のメニューのプロジェクト(P)->「プロジェクト名(ココも名前の付け方で違うと思います)」のプロパティ(P)で表示されるダイアログの右上隅にある 構成マネージャー(O)...を押下して表示されるダイアログでプロジェクトのコンテキスト(ビルドまたは配置するプロジェクト構成をチェック)(R)からアクティブソリューションプラットフォーム(P)のリストから<nowiki><新規作成...></nowiki>を選択して表示されるダイアログで、新しいプラットフォーム(P)からx64を選択して、あとは変更しないでOKボタンを押下すると新しいプラットフォーム設定にx64が追加され、アクティブになるので、これでx64環境向けのプロジェクトにできます。独り言になりますが、プラットフォームにARMも選択できるんすね。<br />
<br />
+
 
pintNumのアドレスはポインタ変数のアドレスで&pintNumですが、更に*をつけると*&pintNumとできこれはpintNumと同じになります。<br />
+
 
sizeof関数は、引数に指定した型や値の大きさ、つまりアドレスをいくつ分、使っているのか、使うのかをbyte単位で返却してくれる関数です。<br />
+
アドレス覚えるだけだから、ポインタの宣言で型名なんているの?って思った人はC言語初心者の中でも少し洞察力が高いと思う。残念ながら型名はいります。その指し示しているアドレスからどこまでを変数として使っているかを把握する必要があるからです。代入もできるわけですから、さらには、変数が配列になっている場合には、インクリメント演算子で次の配列番号に移動できたりもします。まぁ少しづつ応用していきましょう。ここではポインタとは何かを理解できればよいと思います。文字列は配列ありきなので、ポインタの扱いがさらに複雑化してきますし、構造体やクラスのポインタも複雑な操作が必要になります。<br />
sizeof(pintNum)とすると32bitアプリケーションのプロジェクトならアドレスの大きさは4Byteなので4が返ってくるし、<br />
+
 
64bitアプリケーションのプロジェクトならアドレスの大きさは8Byteですから8が返ってきます。<br />
+
 
64bitPCでもプロジェクト設定がWin32なら4が返ってきます。<br />
+
<br />
+
64bitPCでVisualStudio2012をお使いの場合には、変更できると思っているのですが、※どういった組み合わせなら64bitアプリが作れるか掌握していないです。<br />
+
プロジェクトの設定をx64(64bit対応)に変更するには<br />
+
Visual Studio2012のメニューのプロジェクト(P)->「プロジェクト名(ココも名前の付け方で違うと思います)」のプロパティ(P)<br />
+
で表示されるダイアログの右上隅にある 構成マネージャー(O)...を押下して表示されるダイアログで<br />
+
プロジェクトのコンテキスト(ビルドまたは配置するプロジェクト構成をチェック)(R)から<br />
+
アクティブソリューションプラットフォーム(P)のリストから<nowiki><新規作成...></nowiki>を選択して表示されるダイアログで<br />
+
新しいプラットフォーム(P)からx64を選択して、あとは変更しないでOKボタンを押下すると新しいプラットフォーム設定にx64が追加され、アクティブになるので<br />
+
これでx64環境向けのプロジェクトにできます。<br />
+
独り言になりますが、プラットフォームにARMも選択できるんすね。<br />
+
<br />
+
アドレス覚えるだけだから、ポインタの宣言で型名なんているの?って思った人はC言語初心者の中でも少し洞察力が高いと思う。残念ながら型名はいります。<br />
+
その指し示しているアドレスからどこまでを変数として使っているかを把握する必要があるからです。代入もできるわけですから<br />
+
さらには、変数が配列になっている場合には、インクリメント演算子で次の配列番号に移動できたりもします。<br />
+
まぁ少しづつ応用していきましょう。<br />
+
<br />
+
ここではポインタとは何かを理解できればよいと思います。<br />
+
文字列は配列ありきなので、ポインタの扱いがさらに複雑化してきますし、構造体やクラスのポインタも複雑な操作が必要になります。<br />
+
<br />
+
 
ポインタ関連記事:<br />
 
ポインタ関連記事:<br />
 
[[C 文字列]]<br />
 
[[C 文字列]]<br />

2013年12月7日 (土) 00:00時点における版



個人用ツール
名前空間

変種
操作
案内
ツールボックス