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

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

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

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

プログラミング言語

プログラミング言語はパソコン上で実行することができるソースコードを記述する為に扱う言語の総称です。

Q&A

解決済

3回答

4400閲覧

C言語でニュートン法を用いて解を求めたい (2つ)

ALTERNATIVE17

総合スコア2

C

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

プログラミング言語

プログラミング言語はパソコン上で実行することができるソースコードを記述する為に扱う言語の総称です。

0グッド

0クリップ

投稿2020/09/02 02:48

前提・実現したいこと

C言語を用いてニュートン法による非線形方程式の解を求めたいです。

非線形方程式 e^x=3x をニュートン法を用いて求め、その解を出力したいです。
解は約0.6と約1.5となるはずなのですが、初期値を変更しても約0.6の解しか出力されません。

私が考えている理想的な出力結果は
初期値に1を入力:解=約0.6
初期値に2以上を入力:解=約1.5
です。

発生している問題・エラーメッセージ

エラーメッセージではないのですが、初期値を変更しても以下の解しか出力されません。

初期値を入力してください x0= 1 方程式 e^x=3x の解は 0.61906 です
初期値を入力してください x0= 2 方程式 e^x=3x の解は 0.61906 です

該当のソースコード

C

1#include<stdio.h> 2#include<math.h> 3#define eps 0.0001 4 5double f(double x){ 6 return exp(x)-3*x; //求める方程式// 7} 8 9double df(double x){ 10 return exp(x)-3; //求める方程式の1回微分// 11} 12 13int main(void){ 14 double xn,x0; 15 16 printf("初期値を入力してください x0= "); 17 scanf("%lf" ,&x0); 18 19 while(fabs(xn-x0) > eps){ 20 x0 = xn; 21 xn = -1 * f(x0)/df(x0) + x0; //xn+1の式// 22 } 23 24 printf("方程式 e^x=3x の解は %.5f です", xn); 25 26 return(0); 27}

試したこと

初期値を1~10まで変更 → どの初期値でも約0.6の解しか出力されませんでした

補足情報(FW/ツールのバージョンなど)

コンパイル・実行はCygwin64 Terminalを使用しております。
また、質問文や考え方等に至らない点があるかと思いますがよろしくお願いいたします。

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

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

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

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

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

guest

回答3

0

ベストアンサー

処理を追っていくとわかると思いますが、ループに入った時点でxnが不定値です。
コンパイラが気を利かせて0にしてくれているとすると、
ループに入った時点で
x0 = xn;となるのでx00にされます。

というわけで、初期値として設定すべきはxnのほうで、
x0は最初のwhileの条件を満たすようxn+1とかに初期化する。

など。

投稿2020/09/02 03:06

ozwk

総合スコア13521

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

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

ALTERNATIVE17

2020/09/02 03:21

ご回答ありがとうございます。 確かに x0 が 0 になってしまいますね。。 初期値を代入する変数を xn に変更したところ、無事に理想とした出力がされました! 本当にありがとうございました!
guest

0

xnが未初期化の状態でwhileに入っているようです。それがx0に代入されているせいではないでしょうか?

投稿2020/09/02 02:58

swordone

総合スコア20651

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

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

ALTERNATIVE17

2020/09/02 03:03

ご回答ありがとうございます。 ご教授頂いたようにxnを定義する際に「double xn=0」としてみましたが、出力は変わりませんでした。。 (もし初期化の方法やその他何か誤り等がありましたら遠慮なくご教授頂けますと幸いです)
swordone

2020/09/02 03:06

そうすると今度はx0が0で固定されてしまいます。 最初の入力をxnに入れるべきでは?
ALTERNATIVE17

2020/09/02 03:19

ご返信ありがとうございます。 ご指摘の通り初期値を代入する変数を xn に変更したところ、無事に理想としていた出力がされました! 確かに私が先ほどやった方法ですと x0 に xn(=0) が代入されてしまうので、0で固定されてしまいますね。。 本当にありがとうございました!
guest

0

  • 新しい値(元のコードではxn)を求めるのには古い値(元のコードではx0)が必要
  • 繰り返しを終了するかどうかを決めるのにも、新しい値と古い値が必要
  • 古い値は、上記2つの処理が終わってから、新しい値で上書きすることが必要

C

1int main(void) 2{ 3 double x_new, x_old; 4 5 printf("初期値を入力してください x0= "); 6 scanf("%lf", &x_old); 7 8 while (1) { 9 x_new = -1 * f(x_old) / df(x_old) + x_old; //xn+1の式// 10 if (fabs(x_new - x_old) <= eps) { 11 break; 12 } 13 x_old = x_new; 14 } 15 16 printf("方程式 e^x=3x の解は %.5f です", x_new); 17 18 return 0; 19}

result

1$ newton 2初期値を入力してください x0= 1 3方程式 e^x=3x の解は 0.61906 です 4$ newton 5初期値を入力してください x0= 2 6方程式 e^x=3x の解は 1.51213 です

投稿2020/09/02 03:23

Daregada

総合スコア11990

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

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

ALTERNATIVE17

2020/09/02 03:26

ご回答ありがとうございます。 確かに、ご指摘の3つのポイントが重要ですね。 丁寧にプログラムコードまで回答して頂きありがとうございました!
Daregada

2020/09/02 03:40

素直にループさせると回答のように「途中で脱出する」処理になるので、whileで書くにはx0, x1の初期値をうまく仕込まないと行けません。元のコードのまま、xnにキー入力した値を設定するようにしたようですが、x0にも適切な初期値を設定しないと「偶然うまく動いている」だけの状態です。
ALTERNATIVE17

2020/09/02 04:39

そうなのですね。。 改良してみます。。
ALTERNATIVE17

2020/09/02 04:46

初期化を x0=xn+100 としました。この場合初期値に 100 を入力しない限りは正常に動作したのですが、このような記述でよろしいのでしょうか?
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問