質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

88.92%

c言語のポインタを用いた出力から入力を求める

受付中

回答 4

投稿 編集

  • 評価
  • クリップ 2
  • VIEW 1,032

offn

score 14

出力結果からプログラムの?部分を埋めたいのですがどのように考えればよいのかわかりません。

#include<stdio.h>

int main(void){
  int ai[2]={?,?};

  short *ai2 = (short *)(&ai[sizeof(ai)/sizeof(short)-2])-4;
  short *ai3 = (short *)((char *)((short *)(ai+2)-3)-1);
  int n;

  for(n=0;n<7;n += 2)
    printf("%d, %d\n",*(short *)((char *)ai2+n),n);

  for(n=0;n<3;n++)
    printf("%d, %d\n",*(ai3+n),n);
}

実行して出力を
100, 0
?, 2
?, 4
511, 6
0, 0
256, 1 
-100, 2
となるように?の部分を埋めたいのですがやり方がわかりません。
aiの要素をどのようにして考えて求めれば良いのでしょうか?
%pでそれぞれのアドレスを表示させて考えるのかなと思ったのですがそれだと結局aiの要素を求めていないとわからず手が付きません。

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

質問への追記・修正、ベストアンサー選択の依頼

  • Chironian

    2017/05/24 15:05

    ai[2]の内容がなんだろうが、そのプログラムでその出力はでません。?の部分に適当に(123とか456とか)値をいれて実際にもう一度やってみて下さい。

    キャンセル

  • offn

    2017/05/24 17:23

    aiのアドレスを求める部分は自分の確認用で足し加えていたので出力と異なるものになってしまいました;;

    キャンセル

回答 4

+1

無理です。できません。
希望している出力結果の1行目が、
printf("%p, %d\n",&ai[j],ai[j]);の出力で、
100, 0 
となっています。
100はai[0]の番地、0はai[0]の値になりますが、
ai[0]が格納される番地はユーザーが決められるものではありません。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2017/05/24 17:25

    考えるためにいろいろプログラムを変えていたらオリジナルのものではなくなっていましたので修正しました。
    よろしければ回答よろしくお願いします。

    キャンセル

+1

こんにちは。

なるほど、なかなかハードな問題ですね。一旦具体的な値を想定するのが良いです。

short *ai2 = (short *)(&ai[sizeof(ai)/sizeof(short)-2])-4;

について、やってみます。
まず、比較的メジャーなケースで考えましょう。int型が4バイト、short型を2バイトの処理系とします。
すると、sizeof(ai)はint型2の配列のサイズのなで8、sizeof(short)は2ですね。
すると、sizeof(ai)/sizeof(short)-28/2-2=2です。
すると、ai[sizeof(ai)/sizeof(short)-2]ai[2]であることが分かります。

ここはいきなりハードですね。(初心者に出す問題としては「無し」のレベルかも)
aiは要素数が2の配列なので、ai[0]とai[1]は存在しますがai[2]は存在しません。
では、そのアドレスを計算できないかと言うと、アドレスは計算できます。
配列は必ずアドレスが連続しますから、ai[2]はメモリは割り当てられてなくてもアドレスは計算でき、ai[1]のアドレスにその要素のサイズを加えた値となります。
今、ai先頭のアドレスをxと置くとint型は4バイトですから、&ai[0]はx, &ai[1]はx+4, &ai[2]はx+8となります。

&ai[2]はx+8です。そして、(short*)へキャストして4引いてます。
x+8から、short型4つ分のバイト数を引くのでx+8-2*4=xとなります。
(なんか泣きたくなりますね。でも、勉強のためですから我慢しましょう。)
つまり、ai2に入っている値とはaiと同じ値です。

確かめてみましょう。

#include<stdio.h>

int main(void){
  int ai[2]={123,456};

  short *ai2 = (short *)(&ai[sizeof(ai)/sizeof(short)-2])-4;
  short *ai3 = (short *)((char *)((short *)(ai+2)-3)-1);
printf("ai=%p, ai2=%p\n", ai, ai2);

  int n;

  for(n=0;n<7;n += 2)
    printf("%d, %d\n",*(short *)((char *)ai2+n),n);

  for(n=0;n<3;n++)
    printf("%d, %d\n",*(ai3+n),n);
}


Wandboxでやってみました

ai=0x7ffdfceae780, ai2=0x7ffdfceae780
123, 0
0, 2
456, 4
0, 6
0, 0
-14336, 1
1, 2


確かに同じ値になってますね。

これと同様の考え方をすれば解けるのではないかと思います。
後は頑張って下さい。(いや、なかなかハードでした。)

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2017/05/24 21:06

    回答ありがとうございます。
    ポイント演算も大変分かりやすくaiとai2の先頭アドレスが同じことはわかったのですが、回答をいただいてから考えていたのですがaiの要素を求めるやり方とどのように関連付ければわかりませんでした。
    よろしければ詳細を教えてくれませんでしょうか?

    キャンセル

  • 2017/05/24 22:23

    int型やshort 型の変数がメモリの中にどのように仕舞われているのか、習っていると思います。
    それを図にしておき、各printf()文の最初の%dが、そのメモリの「どこ」を表示しているのか見当をつけましょう。回答に記載したような考え方で出来るはずです。残りは「手間」の問題だけです。

    キャンセル

0

for 内の最初に ai の内容を出力しています。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2017/05/24 17:26

    考えるためにいろいろプログラムを変えていたらオリジナルのものではなくなっていましたので修正しました。
    よろしければ回答よろしくお願いします。

    キャンセル

0

本来は、ポインタを理解してプログラム実行せず求める課題だと思います。

答えだけ求めるなら
ポインタ演算を理解せずとも結果を代入してaiを求め表示すればいいんです。
//int n;の後

n=0;*(short *)((char *)ai2+n)=100;
n=6;*(short *)((char *)ai2+n)=511;
n=0;*(ai3+n)=0;
n=1;*(ai3+n)=256;
n=2;*(ai3+n)=-100;


を追加

//最後に表示

printf("%d,%d\n",ai[0],ai[1]);


//最初のai配列の初期値はなくてもいい

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 88.92%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る