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

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

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

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

Q&A

解決済

3回答

1656閲覧

C言語で小数点以下6桁まで出力したい。gettimeofdayを使用し、構造体でsec, usec保持

nohope2nolife

総合スコア13

C

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

0グッド

0クリップ

投稿2019/06/25 17:44

編集2019/06/25 20:28

C言語でgettimeofdayを使っているのですが、出力が期待した値でないときがあるので、解決法をご教示いただけたらと思います。
C言語で出力結果を小数点以下6桁で出したい、というものです。

以下は骨組みを簡略化したコードとなります。問題が起きているほぼそのままの実装です。
また、このコードでは正常に動きますが、私の手元にある少々複雑なコードでは、%06luとしているのに出力が9.18446744073709270220となって、小数点以下6桁で切れていないときがあります。
何が原因と考えられますか。

C

1#include <stdio.h> 2#include <sys/time.h> 3#include <unistd.h> 4 5struct timeval tv; 6 7struct struct_sec { 8 time_t time_sec; 9}; 10 11struct struct_usec { 12 suseconds_t time_usec; 13}; 14 15static struct struct_sec struct_sec; 16static struct struct_usec struct_usec; 17 18int main(int argc, char **argv) { 19 struct_sec.time_sec = 0; 20 struct_usec.time_usec = 0; 21 22 gettimeofday(&tv, NULL); 23 time_t time_before_sec, time_after_sec; 24 suseconds_t time_before_usec, time_after_usec; 25 time_before_sec = tv.tv_sec; 26 time_before_usec = tv.tv_usec; 27 28 /* ここになんかの処理 この処理の時間を測る*/ 29 sleep(1); // 今回はsleep 30 31 gettimeofday(&tv, NULL); 32 33 time_after_sec = tv.tv_sec; 34 time_after_usec = tv.tv_usec; 35 36 time_t time_1 = time_after_sec - time_before_sec; 37 suseconds_t time_2 = time_after_usec - time_before_usec; 38 39 struct_sec.time_sec += time_1; // 今回は1つの和だが、実際は何回か足して合計時間を測る 40 struct_usec.time_usec += time_2; 41 42 printf("%ld.%06lu\n", struct_sec.time_sec, struct_usec.time_usec); // この出力が小数点以下6桁を指定しているのにそうじゃないときがある。 43 44 return 0; 45}

どうかよろしくお願いいたします。

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

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

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

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

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

guest

回答3

0

ベストアンサー

あなたは2つの点で誤解をしています。
問題点1
書式設定を%06luとすれば、かならず、6桁で表示されると考えている。
これは誤りです。表示するデータが0~999999の場合は6桁で表示されます。
1->000001
20->000020
123456789->123456789
のようになります。
従って、
struct_usec.time_usecに18446744073709270220のような大きな数値が格納されていれば
結果的にその数字が表示されます。

問題点2
時間のusecの扱い方に問題がある。
suseconds_t time_2 = time_after_usec - time_before_usec;
これを行うと、time_after_usec < time_before_usecの場合、time_2に非常に大きな意味のない値が格納される。

time_after_usec = 1
time_before_usec = 4 なら
time_2は18446744073709551613となる。

このケースを想定して

C

1time_t time_1; 2suseconds_t time_2; 3if (time_after_usec < time_before_usec){ 4 time_2 = 100000 + time_after_usec - time_before_usec; 5 time1 = time_after_sec - time_before_sec -1; 6}else{ 7 time_1 = time_after_sec - time_before_sec; 8 time_2 = time_after_usec - time_before_usec; 9}

のようにしないといけない。

時刻の加算については
struct_usec.time_usec += time_2;
これを行うと、struct_usec.time_usecが1000000以上になる場合があるが、その場合は
struct_usec.time_usecを1000000で割った商をstruct_sec.time_secに加算しないといけない。
そしてstruct_usec.time_usecを1000000で割った余りをstruct_usec.time_usecに設定する必要がある。
要はstruct_usec.time_usecが1秒以上なら、その秒数をstruct_sec.time_secに加算し、
struct_usec.time_usecは常に0~999999の範囲に収まるようにしないといけないということです。

時間の管理をusec単位で行うなら、catsforepawさんの方法(秒をusecに換算してusecで管理する方法)が最も簡単でかつ正確かと思います。

投稿2019/06/26 01:56

編集2019/06/26 02:05
tatsu99

総合スコア5438

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

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

nohope2nolife

2019/06/26 03:00

ありがとうございます!! 助かりました!
guest

0

ありがちなミスですね。POSIXの時間管理のダサさがミスを誘発しています。

まず、timeval構造体のtv_usecには、秒未満をマイクロ秒で表す値が格納されるので必ず6桁で収まります。したがってその部分の書式は%06luで合っています。

では何故ご質問のコードでは6桁を超えるかというと、時間間隔の計算が間違っているからです。

ご質問のコードでは秒と秒未満のそれぞれを別個に計算していますが、time_after_usectime_before_usecより小さい場合を考慮していません。例えば、afterが12.3秒でbeforeが10.5秒だったとして、整数部分と小数部分を単純に別個に計算してしまうとどうなるでしょう。3から5を引いたらマイナスになってしまいますが、それと同様なことが起きています。
マイナスの値を%06luと正数値として表示しようとすると、大きな値が表示されることになります。

小学校で習った筆算を思い出してください。同じ桁位置で引かれる値よりも引く値が多きい場合は、上の桁から一つ借りてくる(借りてきた上の桁は一つ減らす)ということをしますよね。もし秒部分とマイクロ秒部分を別個で計算したいのなら、同じように繰り下がりを処理しなければいけません。

よりスマートに処理するのなら、時間を秒部分とマイクロ秒部分に分けて扱うのではなく、秒部分をマイクロ秒に換算して秒未満と合算して一つの値として扱いましょう。そうすれば時間間隔は一回の引き算で求まります。その際、long型だと処理系によっては桁あふれを起こすので、long long型を利用してください。

投稿2019/06/26 01:43

編集2019/06/26 01:51
catsforepaw

総合スコア5938

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

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

nohope2nolife

2019/06/26 03:01

ありがとうございます!! わかりやすかったので私でも理解でき,とても勉強になりました.
guest

0

%06lu → %06.6lu
フォーマット指定子一覧
「追記」

text

1usr ~/Project/test % ./a.out 21.000408 3usr ~/Project/test % ./a.out 41.000410 5usr ~/Project/test % ./a.out 61.000066 7usr ~/Project/test % ./a.out 81.000114 9usr ~/Project/test % ./a.out 101.000096 11

投稿2019/06/25 21:42

編集2019/06/25 21:50
cateye

総合スコア6851

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

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

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問