shell浮點數運算問題 - Linux

Hedda avatar
By Hedda
at 2018-12-04T23:26

Table of Contents

忍不住回一下...

※ 引述《zchien (小建)》之銘言:
: 我用unix 系統

UNIX的商標權由國際開放標準組織所擁有
只有符合單一UNIX規範的UNIX系統才能使用UNIX這個名稱
否則只能稱為類UNIX(UNIX-like)。

既然你來到這個版, 我猜你是用 Linux, 就直接說用 Linux 吧

: 寫sh時,遇到了一個bug ,就是浮點數無法運算

我猜你的意思是 寫 shell script.

: 我使用ntpdate去校正時間, 並且記錄校正了多少時間
: 例如:需要校正的時間是 0.000304 s
: 我寫
: ntpdate -p 10.37.82.23 > time.txt

不知道你是用哪一個系統?
Linux 用 ntpdate -p 10.37.82.23 應該會報錯吧?!
usage: ntpdate [-46bBdqsuv] [-a key#] [-e delay] [-k file] [-p samples] \
[-o version#] [-t timeo] server ...
或是你是指 ntpdate -q 10.37.82.23 ?

: NTP= grep delay time.txt | awk ' ' {print $6}

上面的語法應該也會報錯吧?

NTP=$(grep delay time.txt | awk '{print $6}')

: # $NTP will show 類似 0.000304
: if [ $NTP -le 0.100001 ];then
: ntpdate -u 10.37.82.23
: fi

先不管 能不能用小數點的問題.
為什麼 小於等於 0.100001 秒才更新?
如果 offset 大於 0.100001 秒就不更新?

: # 如果NTP server 連線異常, $NTP will show 0.000000
: if [ $NTP -eq 0.000000 ];then
: echo "NTP server not connect"
: fi
: 上面第一個if 會 不成立 , 可能類似浮點數運算, 他以為兩個都是0
: 第二個if 也會顯示錯誤, [0.000000 -eq 0.000000 ] 他竟然顯示不相等
: 請問這是什麼原因,我可以怎麼修正

應該不會是不成立或是不相等吧?
-eq 應該會直接罵人, 類似:
./test.sh: line n: [: 0.000000: integer expression expected

首先, 既然你是用 > time.txt, 會一直被覆蓋, 所以假設 time.txt 並沒有要保留.
那其實可以直接一路 | 到底
NTP=$(ntpdate -q 10.37.82.23 | grep delay | awk '{print $6}')
然後都已經用 awk 了, 可以不用特別再叫 grep 出來跑龍套
NTP=$(ntpdate -q 10.37.82.23 | awk '/delay/{print $6}')

另外, 不知道你有沒有發現, 這樣你得到的 NTP 後面其實會有一個 ,
$ ntpdate -q clock.stdtime.gov.tw
server 211.22.103.158, stratum 2, offset 0.000697, delay 0.03252
4 Dec 22:41:21 ntpdate[1947]: adjust time server 211.22.103.158 offset 0.000697 sec
這應該會影響你後面的判斷式

把它去掉比較好一些
NTP=$(ntpdate -q 10.37.82.23 | awk '/delay/{print $6}'|sed -e 's/,$//')

或是可以在 awk print 時先乘 1000000, 然後往後都可以用 us(微秒) 來處理
這樣 awk 會自己把 , 去掉, 也間接避開了小數點運算的限制

NTP=$(ntpdate -q 10.37.82.23 | awk '/delay/{print $6*1000000}')

if [ "$NTP" -le "100001" ];then
ntpdate -u 10.37.82.23
fi

if [ "$NTP" -eq "0" ];then
echo "NTP server not connect"
fi

先不管不能判斷小數點, 邏輯判斷上 0.000000 也是 -le 0.100001 呀...
把兩個 if 合在一起吧.

NTP=$(ntpdate -q 10.37.82.23 | awk '/delay/{print $6*1000000}')

if [ "$NTP" -eq "0" ];then
echo "NTP server not connect"
elif [ "$NTP" -le "100001" ];then
ntpdate -u 10.37.82.23
fi

雖然機率不比中樂透高, 有考慮過如果剛剛好 offset 是 0.000000 秒嗎?
man ntpdate:
DIAGNOSTICS
ntpdate’s exit status is zero if it finds a server and updates the
clock, and nonzero otherwise.
ntpdate 有錯誤時的 return code 不是 0, 建議用這個來判斷比較好.

#!/bin/sh

ntp_response=$(ntpdate -q 10.37.82.23 2>&1)
rc=$?
if [ "${rc}" -ne "0" ];then
echo "ntpdate error, return code: ${rc}"
else
ntp_offset=$(echo ${ntp_response} | awk '/delay/{print $6*1000000}')
if [ "${ntp_offset}" -le "100001" ];then
echo "offset: ${ntp_offset}"
ntpdate -u 10.37.82.23
fi
fi

大概是醬~

最後有兩個問題還是不解.
1. 為什麼 offset -le 0.100001 才要 -u, 超過 0.100001 不處理嗎?
2. 有考慮直接改用 ntpd 嗎?

--
Tags: Linux

All Comments

Ethan avatar
By Ethan
at 2018-12-06T07:30
看以前發的文很可能是solaris系列,所以參數之類不同不奇怪
Todd Johnson avatar
By Todd Johnson
at 2018-12-06T21:58
感謝回覆,solaris和你上面指令應該相同,很多指令都是硬
記的,所以上面可能寫的不完全,明天上班,我在試看看,感
Edward Lewis avatar
By Edward Lewis
at 2018-12-09T05:43
Server上面有跑程式,如果時間瞬間差太大,程式會當機,
所以寫crontab 讓他每小時校時
Emma avatar
By Emma
at 2018-12-13T17:41
那就更適合ntpd,他本身就考慮到時間不能突然跳動的問題了
Hazel avatar
By Hazel
at 2018-12-13T23:27
只是你系統上不一定有現成的
Aaliyah avatar
By Aaliyah
at 2018-12-17T08:42
喔喔,高手過招果然很多學習的主體地方
Valerie avatar
By Valerie
at 2018-12-18T19:04
版上高手好多
Suhail Hany avatar
By Suhail Hany
at 2018-12-21T01:36
這是 高手高手高高手了吧

shell浮點數運算問題

Margaret avatar
By Margaret
at 2018-12-03T22:44
我用unix 系統 寫sh時,遇到了一個bug ,就是浮點數無法運算 我使用ntpdate去校正時間, 並且記錄校正了多少時間 例如:需要校正的時間是 0.000304 s 我寫 ntpdate -p 10.37.82.23 andgt; time.txt NTP= grep delay time. ...

請益家目錄設定檔備份與維護方式

Jacky avatar
By Jacky
at 2018-12-03T00:33
不好意思打擾各位大大,想請教一下關於家目錄備份的原則, 同時也煩請大大們看一下我目前的策略還有哪些改進空間了 目前的工作環境:實驗室Server 跟 NFS、自己的主機 (Ubuntu) 因為前陣子才開始接觸 Linux,怕一個不小心動亂 Server 的東西, 所以基本上都是在自己主機上忙碌, ...

fil 一個執行檔分析工具

Lucy avatar
By Lucy
at 2018-12-02T19:34
有時候需要分析執行檔 Linux 上有 file 指令可以使用 可是 Windows 上一直找不到好的替代品 只好自己來了 https://github.com/joeky888/fil - ...

常常系統變成唯讀狀態,必須要執行fsck才正常

Damian avatar
By Damian
at 2018-11-28T23:11
最近常常出現用到一半系統不能更新、某些程式沒音效, 重開機後都會進入一個命令的模式,然後使用 fsck 修復, 這個月已經出現了三到四次了,會是新的 kernel 造成的嗎? OS: Ubuntu 18.10 kernel: 4.19.5 - ...

ubuntu 18.10 vsftpd 架設

Anthony avatar
By Anthony
at 2018-11-28T10:14
抱歉,問一下, 我電腦當 Server 用 ubuntu 18.10 版灌 vsftpd, Server 這台本機開網頁打 ftp://192.168.yy.xxx 連 ftp ok. 用另外一台電腦開網頁打 ftp://192.168.yy.xxx 連不上 ftp, 有點無助, 請幫忙我看一下, 我在 ...