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

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

新規登録して質問してみよう
ただいま回答率
85.48%
C

C言語は、1972年にAT&Tベル研究所の、デニス・リッチーが主体となって作成したプログラミング言語です。 B言語の後継言語として開発されたことからC言語と命名。そのため、表記法などはB言語やALGOLに近いとされています。 Cの拡張版であるC++言語とともに、現在世界中でもっとも普及されているプログラミング言語です。

for

for文は、様々なプログラミング言語で使われている制御構造です。for文に定義している条件から外れるまで、for文内の命令文を繰り返し実行します。

C++

C++はC言語をもとにしてつくられた最もよく使われるマルチパラダイムプログラミング言語の1つです。オブジェクト指向、ジェネリック、命令型など広く対応しており、多目的に使用されています。

Q&A

解決済

2回答

2450閲覧

ガウスの後退代入の二重ループに関する質問です。(※ガウスはあんまり関係ありません。二重ループがメインです。)

carnage0216

総合スコア194

C

C言語は、1972年にAT&Tベル研究所の、デニス・リッチーが主体となって作成したプログラミング言語です。 B言語の後継言語として開発されたことからC言語と命名。そのため、表記法などはB言語やALGOLに近いとされています。 Cの拡張版であるC++言語とともに、現在世界中でもっとも普及されているプログラミング言語です。

for

for文は、様々なプログラミング言語で使われている制御構造です。for文に定義している条件から外れるまで、for文内の命令文を繰り返し実行します。

C++

C++はC言語をもとにしてつくられた最もよく使われるマルチパラダイムプログラミング言語の1つです。オブジェクト指向、ジェネリック、命令型など広く対応しており、多目的に使用されています。

0グッド

0クリップ

投稿2018/07/30 10:27

編集2018/08/18 03:49
for (i = N - 1; i >= 0; i--) { d = a[i][N]; for (j = i + 1; j < N; j++) d -= a[i][j] * a[j][N]; a[i][N] = d / a[i][i]; }

を皆さんの力を借りて以下のように分解しました。そこで疑問がわいたのですが、

//for (i = N - 1; i >= 0; i--) { i = N - 1; //i == 3 d = a[i][N]; //for (j = i + 1; j < N; j++) { j = i + 1; //j == 4 //} a[i][N] = d / a[i][i]; i--; //i == 2 d = a[i][N]; //for (j = i + 1; j < N; j++) { j = i + 1; //j == 3 d -= a[i][j] * a[j][N]; //} a[i][N] = d / a[i][i]; i--; //i == 1 d = a[i][N]; //for (j = i + 1; j < N; j++) { j = i + 1; //j == 2 d -= a[i][j] * a[j][N]; j++; //j == 3 d -= a[i][j] * a[j][N]; //} a[i][N] = d / a[i][i]; i--; //i == 0 d = a[i][N]; //for (j = i + 1; j < N; j++) { j = i + 1; //j == 1 d -= a[i][j] * a[j][N]; j++; //j == 2 d -= a[i][j] * a[j][N]; j++; //j == 3 d -= a[i][j] * a[j][N]; //} a[i][N] = d / a[i][i]; //}

以下の部分のjは j++より1ずつ増えていくはずが減っていっているのは二重ループが働いているためでしょうか?
もちろんiを考慮してj=i+1となっているためj==4からj==3となるのはわかるのですが、では(j = i + 1; j < N; j++)のj++の意味は何なのかと混乱しています。
iはi = N - 1よりi == 3となりfor (i = N - 1; i >= 0; i--) のi--によりi == 2となるのに、やはり二重ループが原因なのでしょうか?
どうかわかりやすく解説していただけないでしょうか?
どうか絶対に理解したいのでよろしくお願いいたします。

//for (i = N - 1; i >= 0; i--) { i = N - 1; //i == 3 d = a[i][N]; //for (j = i + 1; j < N; j++) { j = i + 1; //j == 4 //} a[i][N] = d / a[i][i]; i--; //i == 2 d = a[i][N]; //for (j = i + 1; j < N; j++) { j = i + 1; //j == 3 d -= a[i][j] * a[j][N]; //}

8/17<編集>
質問ではないのですが、以下のような三重ループのプログラムを練習したいと思うのですが似たようなプログラムを載せて頂けないでしょうか?出来れば詳しい解説なども表示して頂けるとありがたいです。

for (i = 0; i < N - 1; i++) { for (j = i + 1; j <= N - 1; j++) { for (k = i + 1; k <= N - 1; k++) { A[j][k] = A[j][k] - A[i][k] * A[j][i] / A[i][i];  } b[j] = b[j] - b[i] * A[j][i] / A[i][i];  }  }

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

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

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

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

rubato6809

2018/08/17 14:02

インデントって何のことか知ってますか?
carnage0216

2018/08/18 02:52

はい、字下げのことでございます。きっと、またインデントがうまく活用できてなかったんですね。わざとやっているわけではないのですが、なぜかうまくいかなくて....。
rubato6809

2018/08/18 03:23

「きっと、また」って、貴方がこのページを見たらインデントがついて表示されるんですか?昨日追加した三重ループのコードにインデントを付けてください。空白文字(スペースキー)で問題なくインデントできますよ。普段、手元のソースファイルにインデントをつけないのですか?
carnage0216

2018/08/18 03:50

付けないです。インデントの付け方が読んでみてもわからず、付けていないことが多いです。
guest

回答2

0

ベストアンサー

以下の部分のjは j++より1ずつ増えていくはずが減っていっているのは二重ループが働いているためでしょうか?

何が疑問なのか?と思ったけど、二重ループがどう動くかわからん、ということならお手伝いできますよ。

まずはインデントを正しくしよう笑。インデントがいい加減なのは、プログラムの構造がわかっていない、ということを白状してるようなもので、そのまま二重ループがわからないことに通じている・・・かもしれない。

C

1 for (i = N - 1; i >= 0; i--) { 2 d = a[i][N]; 3 for (j = i + 1; j < N; j++) { 4 d -= a[i][j] * a[j][N]; 5 } 6 a[i][N] = d / a[i][i]; 7 }

i = N - 1; //i == 3 で始まるなら#define N 4ですね。配列 a[?][?] の大きさがわからないが、それは置いておいて

  • i, j 変数の値、だけでなく
  • 配列 a[?][?] の、どこをアクセスするのか
  • どこでループするのか

を可視化するために、プログラムをこんな風に書き変えてみました。最初からCだし、こちらが少しだけ役にたつはず。まあ、ちょっとしたオモチャですが笑。

C

1#include <stdio.h> 2#define N 4 3int main(void) 4{ 5 printf("for (i = N - 1; i >= 0; i--) {\n"); 6 for (int i = N - 1; i >= 0; i--) { 7 printf(" i = %d;\n", i); 8 printf(" d = a[%d][%d];\n", i, N); 9 10 printf(" for (j = i + 1; j < N; j++) {\n"); 11 for (int j = i + 1; j < N; j++) { 12 printf(" j = %d\n", j); 13 printf(" d -= a[%d][%d] * a[%d][%d];\n", 14 i, j, j, N); 15 } 16 17 printf(" }\n\n"); 18 printf(" a[%d][%d] = d / a[%d][%d];\n", 19 i, N, i, i); 20 } 21 printf("}\n\n"); 22 return 0; 23}

質問者が期待した動きに合っているかどうか、実行結果をよく見てほしい。他に表示させたいものがあれば・表示の仕方を変えたければ、プログラムを変更して構わないですよ。
実行結果です。

$ ./a.out for (i = N - 1; i >= 0; i--) { i = 3; d = a[3][4]; for (j = i + 1; j < N; j++) { } a[3][4] = d / a[3][3]; i = 2; d = a[2][4]; for (j = i + 1; j < N; j++) { j = 3 d -= a[2][3] * a[3][4]; } a[2][4] = d / a[2][2]; i = 1; d = a[1][4]; for (j = i + 1; j < N; j++) { j = 2 d -= a[1][2] * a[2][4]; j = 3 d -= a[1][3] * a[3][4]; } a[1][4] = d / a[1][1]; i = 0; d = a[0][4]; for (j = i + 1; j < N; j++) { j = 1 d -= a[0][1] * a[1][4]; j = 2 d -= a[0][2] * a[2][4]; j = 3 d -= a[0][3] * a[3][4]; } a[0][4] = d / a[0][0]; }

投稿2018/07/30 13:26

rubato6809

総合スコア1380

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

carnage0216

2018/07/30 15:58

どうもお久しぶりです。 どうもありがとうございます。 rubato6809さんから回答がもらえるとは嬉しいです。 完全に見捨てられたかと思っていました。(今もそうかもしれませんが) 是非、じっくり読ませて頂きます。
guest

0

まず、ソースコードのインデントをなんとかしましょう。理解できるものも理解できない。あと{}の省力をやめる。あと変数のスコープを小さく。

cpp

1for (auto i = N - 1; i >= 0; i--) { 2 d = a[i][N]; 3 for (auto j = i + 1; j < N; j++) { 4 d -= a[i][j] * a[j][N]; 5 } 6 a[i][N] = d / a[i][i]; 7}

jは内側のループのみで利用されています。(いや、外側のループの外側がどうなっているかに依存しますけど、ここ以外でjを使うことが有意義な結果を生むと思えない)

まあもうちょっとわかりやすい例を持ってきますか。

cpp

1#include <iostream> 2 3int main() 4{ 5 constexpr int N = 5; 6 for (auto i = N - 1; i >= 0; i--) { 7 std::cout << "i:" << i << std::endl; 8 for (auto j = i + 1; j < N; j++) { 9 std::cout << "j:" << j; 10 } 11 std::cout << std::endl; 12 } 13}

こういうコードがあったとき、実行結果は

https://wandbox.org/permlink/ed5EdE2hdOeQW9Q6

i:4 i:3 j:4 i:2 j:3j:4 i:1 j:2j:3j:4 i:0 j:1j:2j:3j:4

となりますよね?

そういうことです。


なんか私のコードを弄りたい需要があるようなのであえて明示しておくと、コードはCC0でライセンスされます。

投稿2018/07/30 10:40

編集2018/07/30 12:46
yumetodo

総合スコア5850

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

carnage0216

2018/07/30 11:53

yumetodoさん。どうもありがとうございます。書いていただいたコードを参考に勉強させていただきます。
carnage0216

2018/07/30 12:30 編集

質問とは関係ないのですがyumetodoさんから頂いたコードを腕試しとしてⅭ言語で書いてみてもいいでしょうか? 決してⅭ++が読みにくいとか嫌なわけではありません。何か不快に感じることがあれば遠慮なくいってください。 いいえ変える必要はありません。すいませんでした。
yumetodo

2018/07/30 12:29

include変えてprintf使うようにしてconstexpr消せばいいだけですが?わざわざ書き換える必要が???
pepperleaf

2018/07/30 12:30

ほとんど、C言語じゃないですか? 昔の C言語には、 // コメントが無かったですが、、それくらい。
carnage0216

2018/07/30 12:31

すいませんでした。C++をほとんど読んだことがなかったもので、二度としません。
yumetodo

2018/07/30 12:33

>昔の C言語には、 // コメントが無かったですが 四半世紀近く前だ・・・ >二度としません。 いや、別にご自由に。
carnage0216

2018/07/30 12:36

どうもありがとうございます。
yumetodo

2018/07/30 14:18

まあというか、デバッガでステップ実行しましょう案件
asm

2018/07/30 17:11

C++読めないならタグ付けなきゃいいのにね
carnage0216

2018/07/30 17:50 編集

え!?ついてましたか!? 消しますね。消し方がわからないので調べます。 お騒がせしました。
pepperleaf

2018/07/31 11:30

yumetodo さん、 > 四半世紀近く前だ・・・ 2010年頃まで、使っていた コンパイラは、サポートしてましたが、非標準でした。 (もっとも 20世紀作だったかも知れない) 他にも C++コードがありました。これはうっかり。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問