◎1------------------------------
#include<stdio.h>
int main(void)
{
char *pt="ABC";
printf("pt=%s\n",pt);
char dt[10]="ABCDE";
char *pp;
pp=dt;
printf("pp=%s\n",pp);
return 0;
}
--------------------------------------
◎2----------------------------------
#include<stdio.h>
int main(void)
{
char *pt="ABC";
printf("*pt=%s\n",*pt);
char dt[10]="ABCDE";
char *pp;
pp=dt;
printf("*pp=%s\n",*pp);
return 0;
}
-----------------------------------
◎3---------------------------------------
#include<stdio.h>
int main(void)
{
int ary[5]={111,222,333,444,-1};
int* pt=ary;
while(1){
printf("%d ",*pt);
++pt;
if(*pt==-1){
break;
}
}
puts("");
return 0;
}
-----------------------------------------------
以上3つのプログラムで、◎1はprintfで「*」が付いてなく、正常に実行出来ました。
◎2はprintfで「*」が付いてなく、エラーは出ませんが、文字列が表示されませんでした。
◎3は文字列ではなく値ですが、printfで「*」が付いていて正常に実行できます。
これは、値の場合は「*pt」とすることで、ptのアドレスに値を代入しているという事で、「printf("%d ",*pt);」で実行できたということですかね?
文字列の場合は、先頭のアドレスを渡すだけなので、「printf("pt=%s\n",pt);」のようにしてアドレスを参照しないとダメであるということですか?
◎2で「printf("*pt=%s\n",*pt);」としてしまうと、何が起きてしまうのかわかりません。
以上、教えていただけると嬉しいです。
No.4ベストアンサー
- 回答日時:
◎1について
printfの%sは「ポインタ」を受け取り、そのポインタの先にある文字列を表示します。
>char *pt="ABC";
「ptはcharのポインタであり、そのポインタの先にある文字列は"ABC"だ」として、宣言と初期化を行っています。
>printf("pt=%s\n",pt);
「pt」は「ポインタ」であり「そのポインタの先にある文字列は"ABC"」です。なので「ABC」が表示されます。
>char dt[10]="ABCDE";
「ptはcharの配列であり、その配列の各要素はA,B,C,D,E,\0だ」として、宣言と初期化を行っています。
>char *pp;
「ppはcharのポインタである」と宣言を行っています。
>pp=dt;
「ppが指すポインタは、dtを指せ」と代入しています。結果、ppのポインタはdtを指し、ppのポインタの先にある文字列は"ABCDE"になります。
>printf("pp=%s\n",pp);
「pp」は「ポインタ」であり「そのポインタの先にある文字列は"ABCDE"」です。なので「ABCDE」が表示されます。
◎2について
printfの%sは「ポインタ」を受け取り、そのポインタの先にある文字列を表示します。これは◎1も◎2も変わりません。
>char *pt="ABC";
>char dt[10]="ABCDE";
>char *pp;
>pp=dt;
は◎1と同一なので説明を省きます。
>printf("*pt=%s\n",*pt);
>printf("*pp=%s\n",*pp);
どちらも、%sに対して「指してるポインタの中から取り出した文字」つまり「A」を渡しています。「文字A」はポインタではありません。
しかし、printfは「文字A」(アスキーコードで0x41、65である可能性が高い)を「ポインタだと信じて動く」ので「メモリの65番地にある内容を文字列だと思って表示しよう」とします。
もし「メモリの65番地にある内容が0」なら「1文字も表示せずに文字列の終りが来る」ので、何も表示しません。
もし「メモリの65番地から始まって、ずっ~と、数メガバイト、文字列の終りを示す0が無かった」としたら「メモリの65番地から数メガバイトに渡って、延々と文字列として表示し続け、プログラムを止められなくなる」と言う事態に陥った筈です。
今回は「偶然、何も表示しなかっただけ」で済みましたが、下手したらパソコンを強制リセットするしか無い状態になってたかも。
◎3について
printfの%dは「数値」を受け取り、その数値を表示します。
>printf("%d ",*pt);
ポインタとか*とはは何の関係もありません。「数値」を受け取るのだから「数値」を渡さなければなりません。
ptは「intを指してるポインタ」ですから、printfの引数に「*pt」と書くと「ポインタが指してる所にあるintの数値」が取り出されます。
そして、その「intの数値」がprintfに渡され、表示されます。
ここで注意しないとならないのは
[[[ printfは、渡された物がすべて正しいと思って、疑いもしない ]]]
と言う事です。
引数が足りなかったり、引数に指定した物の内容が変でも、何も疑わず、言われた通りに表示しようとします。
--------
一連の質問を見るに、質問者さんは「ポインタや配列で、*が付いている時、宣言、初期化、代入、参照がきちんと区別が出来ていない」ようです。
Cでは「見た目に、まったく同じ書き方」をしていても「それがどこに書いてあるかとか、前後の文脈の繋がりとかで、まったく全然別の意味」になってしまう文があります。
また、printfの%sと%dのように「見た目にそっくりだけど、まったく全然別のモノを要求してる」という事もあります。
この辺りの「見極め」が出来るまで「似たような書き方だから、同じように出来る筈」と思ってはいけません。
と言うか「似たような書き方だから、同じように出来る筈」と思ってしまうのが「質問者さんの悪い所」です。
Cでは「1文字でも違えば、何かが絶対に違う筈」という「疑う能力」が要求されます。
>printfの%sは「ポインタ」を受け取り、そのポインタの先にある文字
>列を表示します。
>printf("*pp=%s\n",*pp);
>どちらも、%sに対して「指してるポインタの中から取り出した文字」
>つまり「A」を渡しています。「文字A」はポインタではありません。
>しかし、printfは「文字A」(アスキーコードで0x41、65である可能性>が高い)を「ポインタだと信じて動く」ので「メモリの65番地にある
>内容を文字列だと思って表示しよう」とします。
上記の内容等理解できました。
細かくありがとうございます。
後、ご指摘いただいた辺りがまだ、ずばり欠点だと思います。
そこをしっかり把握して努力していきます!
No.5
- 回答日時:
>(1)の*ptでアドレスに格納している内容、"ABC"を参照できると思ったんのですが、出来ません。
。私の先の回答には誤解を招きかねない箇所がありました。
pt = "ABC";
という文で、"ABC" の先頭文字 'A' のアドレスが pt に入ります。
その後、"ABC" 全体を取得するには、*pt ではなく pt を使います。
*pt は、pt が指している文字列 "ABC" の先頭文字 'A' 「だけ」を取得します。
>*pt は、pt が指している文字列 "ABC" の先頭文字 'A' 「だけ」を取>得します。
上記の内容理解できました。
確かに「printf("*pt=%c\n",*pt);」として、'A'は取得できていました。ありがとうございます。
No.3
- 回答日時:
>値の場合は「*pt」とすることで、ptのアドレスに値を代入しているという
理解が正確ではありません。
ptがポインタ変数の場合、ptの値は、何かの変数のアドレスです。
そのとき、*ptは、その、何かの変数のアドレスに格納している内容を参照します。
これは、ptがchar *型であろうがint *型であろうが他の何型であろうが同じです。
ご回答ありがとうございます。
char *pt="ABC";
printf("*pt=%s\n",*pt); ////(1)
(1)の*ptでアドレスに格納している内容、"ABC"を参照できると思ったんのですが、出来ません。。
やはり文字列の場合、ptにしてアドレスを参照しないとダメなのですかね?
No.2
- 回答日時:
> ◎2はprintfで「*」が付いてなく、エラーは出ませんが、文字列が表示されませんでした。
#include<stdio.h>
int main(void)
{
char dt[10]="ABCDE";
char *pp; // ポインタ型の変数を使いますよ!
pp=dt;
printf("*pp=%c\n",*pp); // 値を渡している(dt[0])
printf("*pp=%s\n",pp); // アドレスを渡している
return 0;
}
宣言時の*と式の中の*は意味が違うっていうことですね。
No.1
- 回答日時:
printf()の書式と、対応する引数のデータ型を調べれば、答えは明らかだと思いますが、
リファレンスとか参照されているのでしょうか。
"%s"は文字列に対応する書式制御文字で、char*を引数として受け取り処理します。
"%d"は整数に対応する書式制御文字で、intを引数として受け取りしょりします。
それぞれ、対応するデータ型で引数を受け渡す事が決められていますので、
間違った引数で実行した場合、printf()はどのような動作になるのかは不明です。
お探しのQ&Aが見つからない時は、教えて!gooで質問しましょう!
関連するカテゴリからQ&Aを探す
おすすめ情報
- ・漫画をレンタルでお得に読める!
- ・街中で見かけて「グッときた人」の思い出
- ・「一気に最後まで読んだ」本、教えて下さい!
- ・幼稚園時代「何組」でしたか?
- ・激凹みから立ち直る方法
- ・1つだけ過去を変えられるとしたら?
- ・【あるあるbot連動企画】あるあるbotに投稿したけど採用されなかったあるある募集
- ・【あるあるbot連動企画】フォロワー20万人のアカウントであなたのあるあるを披露してみませんか?
- ・映画のエンドロール観る派?観ない派?
- ・海外旅行から帰ってきたら、まず何を食べる?
- ・誕生日にもらった意外なもの
- ・天使と悪魔選手権
- ・ちょっと先の未来クイズ第2問
- ・【大喜利】【投稿~9/7】 ロボットの住む世界で流行ってる罰ゲームとは?
- ・推しミネラルウォーターはありますか?
- ・都道府県穴埋めゲーム
- ・この人頭いいなと思ったエピソード
- ・準・究極の選択
- ・ゆるやかでぃべーと タイムマシンを破壊すべきか。
- ・歩いた自慢大会
- ・許せない心理テスト
- ・字面がカッコいい英単語
- ・これ何て呼びますか Part2
- ・人生で一番思い出に残ってる靴
- ・ゆるやかでぃべーと すべての高校生はアルバイトをするべきだ。
- ・初めて自分の家と他人の家が違う、と意識した時
- ・単二電池
- ・チョコミントアイス
デイリーランキングこのカテゴリの人気デイリーQ&Aランキング
-
init関数の意味
-
戻り値で構造体を返すことは可...
-
C言語のポインタに直接アドレス...
-
C言語でのconstを返す関数
-
LPSTR型の初期化について
-
ハンドルはポインタか
-
Run-Time Check Failure #3とい...
-
セグメントエラー
-
PASCALとFARの意味
-
NULLポインタが0でない処理系と...
-
関数の引数をvoid*でキャストする
-
ハンドル、アドレス、ポインタ...
-
fopne で失敗する原因
-
構造体の削除について
-
NULLとブランクの違い
-
文字列
-
プーさんのマウスポインタを教...
-
DelphiでCreateProcessがうまく...
-
参照型で受け取った引数をポイ...
-
アプリを32bitから64bit移行
マンスリーランキングこのカテゴリの人気マンスリーQ&Aランキング
-
C言語のポインタに直接アドレス...
-
init関数の意味
-
セグメントエラー
-
fopne で失敗する原因
-
戻り値で構造体を返すことは可...
-
Run-Time Check Failure #3とい...
-
ExcelVBAでのkernel32(64bit)
-
LPSTR型の初期化について
-
C言語でのconstを返す関数
-
ハンドルはポインタか
-
【C言語】戻り値が構造体の関数
-
nullポインタを逆参照とは?
-
基本アルゴリズムの『返す』の...
-
参照型で受け取った引数をポイ...
-
#define NULL ((void *)0) の弊害
-
アプリを32bitから64bit移行
-
ハンドル、アドレス、ポインタ...
-
デバイスハンドルとは?
-
C++で関数ポインタから関数名を...
-
Cで作成したDLL関数をVBから呼...
おすすめ情報