
昔 Delphi で作成したものを C で書き直しました。Delphi ではまったく問題なかったのですが、C の場合、方程式の係数によって、掃き出し処理をした後 0 にマイナスが付加されることがあります。書式指定しない printf でも同じです。
なぜでしょうか?
※解は正しいです。
#include <stdio.h>
#include <math.h>
#define M 5
/* これは掃き出し処理、後単位行列になった 0 成分にマイナスがついてしまう
double A[M][M+1] = {
{ 5.0, -3.0, -3.0, 3.0, 2.0, -8.0 },
{ 9.0, -7.0, 1.0, -6.0, -5.0, -3.0 },
{ -1.0, -1.0, -8.0, 7.0, 4.0, -4.0 },
{ -7.0, 2.0, -6.0, 0.0, -1.0, -1.0 },
{ 8.0, 5.0, -3.0, 9.0, -7.0, 5.0 }
};
*/
double A[M][M+1] = { // こちらはOK
{ 2.0, 0.0, 4.0, 2.0, 0.0, 0.0 },
{ 2.0, 2.0, 0.0, 4.0, 8.0,12.0 },
{ 2.0, 4.0, 4.0, 0.0, 2.0, 0.0 },
{ 4.0, 2.0, 0.0, 0.0, 4.0, 0.0 },
{ 4.0, 0.0, 2.0, 2.0, 4.0, 8.0 }
};
//Gauss Jordan
int GaussJordan(int N)
{
double mMax, R_pivot, pivot, temp;
int pRow = -1;
for (int pv = 0; pv < N; pv++) //行ループ(一番外側のループ)
{
mMax = 0.000000001;
for (int k = pv; k < N; k++) {//行ループ ピボット行以下の最大値探索
if (abs(A[k][pv]) > mMax){
mMax = abs(A[k][pv]);
pRow = k;
}
}
if (mMax <= 0.000000001){ //0対策
printf("解が存在しないか、または不定です!");
return (0);
}
if (pv != pRow) //行の入れ替え
{
for (int k = 0; k < N + 1; k++) { //列ループ
temp = A[pv][k];
A[pv][k] = A[pRow][k];
A[pRow][k] = temp;
}
}
//ピボット行 A[pv][j] の処理
pivot = A[pv][pv];
R_pivot = 1.0 / pivot;
for (int j = 0; j < N + 1; j++) //列ループ
A[pv][j] = A[pv][j] * R_pivot; //A[pv][pv] = 1 となる
//ピボット行以外の処理 ⇒ ピボット列を 0 にする。
for (int k = 0; k < N; k++){ //行ループ
temp = A[k][pv]; //各行のピボット列成分
for (int j = pv; j < N + 1; j++) //ピボット列以降を処理
if (k != pv)
A[k][j] = A[k][j] - temp * A[pv][j];
}
}
return (1);
}
int main(void) {
printf("連立一次方程式\n");
for (int i = 0; i < M; i++) {
for (int j = 0; j < M+1; j++)
printf("%8.3g",A[i][j]);
printf("\n");
}
GaussJordan(M);
printf("\n解\n");
for (int i = 0; i < M; i++) {
for (int j = 0; j < M+1; j++)
printf("%8.3g",A[i][j]);
printf("\n");
}
}

No.2ベストアンサー
- 回答日時:
IEEE754の浮動小数点数にはゼロの正負がありますから、値として負のゼロならマイナス記号が付くのは当然のことです。
あるいは-0と表示されていても実際には-1e-10とかの小さな負の値の可能性もあります。書式指定していないときは本当に-0だと思いますけど。
-0の表示を回避したければ-0を0に置き換える処理を入れる必要があるでしょう。
お探しのQ&Aが見つからない時は、教えて!gooで質問しましょう!
関連するカテゴリからQ&Aを探す
おすすめ情報
デイリーランキングこのカテゴリの人気デイリーQ&Aランキング
-
C言語での引数の省略方法
-
#define _CRT_SECURE_NO_WARNIN...
-
実数の整数部,小数部の取得
-
C言語 エラーの原因がわからな...
-
複数桁10進数の*桁目だけを抽出...
-
「指定されたキャストは有効で...
-
int16_t の _t は何?
-
「{ } で囲むだけ」は正しい?
-
非再帰のマージソートについて
-
毎回違う乱数を生成するにはど...
-
opencvで特定の座標を指定しそ...
-
ブロックで };ついているのと...
-
数字列を3桁ごとにカンマで区切...
-
エラー 添字が付けられた値が、...
-
課題でつまってます・・・
-
Cでボリュームコントロールを制...
-
system関数がうまくいかない
-
if と配列の組み合わせ
-
std::set<int> で、ある値が何...
-
シグマ公式・・・C言語
マンスリーランキングこのカテゴリの人気マンスリーQ&Aランキング
-
複数桁10進数の*桁目だけを抽出...
-
「指定されたキャストは有効で...
-
C言語での引数の省略方法
-
#define _CRT_SECURE_NO_WARNIN...
-
C言語 エラーの原因がわからな...
-
ラップ関数とはどんなものですか?
-
【C++】関数ポインタの使い方
-
if と配列の組み合わせ
-
(int *)の意味
-
ColorをRGBで指定する方法
-
未解決の外部シンボル _printf...
-
構造体の勉強中です 合計点の高...
-
商と剰余を同時に求める(C言語)
-
C言語で分からないところがあり...
-
式は定数値が必要です」という...
-
C言語での奇数の和
-
std::set<int> で、ある値が何...
-
課題でつまってます・・・
-
int16_t の _t は何?
-
比較回数と交換回数表示について
おすすめ情報
> %.20f みたいにぎりぎりまで表示するように出力を指定すれば, どっちかは区別がつく.
やはり、マイナスがつきますね。
ま、これを表示させる必要はほとんどないのでとくに問題はないのですけど、何か引っかかる(笑)。
ちなみにC#ではこんな症状は生じませんでした。