
我最近需要使用 bash 產生和驗證 NMEA 0183 校驗和,但我找不到太多可以完成我所需要的內容。
NMEA 0183 句子以 $ 開頭,以 * 結尾,以及兩個字符,這兩個字符是 $ 和 * 之間所有位元組的十六進制異或。例子:
$INGGA,230501.547,2118.97946,N,15752.60495,W,2,08,1.1,5.17,M,,,0,0001*02
該實用程式會將字串轉換為十六進制並對其進行異或。它可用於驗證已經存在的校驗和,或為您正在生成的 NMEA 句子末尾生成校驗和(它會從您提供的字串中刪除 $ 和 *..)。
#!/bin/bash
# =========================================================
# Reads a NMEA 0183 sentence and calculates the proper
# XOR checksum for the end.
# Will accept a string with or without a checksum on
# the end or $ on the front and calculate what the checksum
# should be.
# Sentence can be read as an argument but must be single quoted
# or preceded by a \ or the shell will try to interpret the
# talker as a variable and the result will be incorrect.
# Examples:
# xor '$INHDT,207.7,T*27'
# xor \$INHDT,207.7,T*27
# xor INHDT,207.7,T
# If run with no arguments, will prompt user for data. No
# quotes or backslash is needed then.
# Depends: xxd sed
# ===T.Young 09/2016=======================================
set -o pipefail
set -o errexit
set -o nounset
# Functions
# =========
depcheck() { # Checks that necessary external commands are present
# and executable
local DEPENDS="sed xxd"
for PROG in $DEPENDS; do
[[ -x "$(command -v $PROG)" ]] || {
echo "$PROG MISSING! Exiting."
exit 0
}
done
}
x_or() { # Here is where the magic happens
# The next two lines strip out $ characters, or an
# * and anything after it (checksum)
HEAD="${SENTENCE%\**}"
TAIL="${HEAD//\$}"
# Convert ASCII string into hex and read into an array.
# Each element in the array gets preceded by "0x"
HEXVAL="$(xxd -pu <<< ${TAIL})"
HEXARRAY=($(printf '%s' "${HEXVAL%0a}" | sed -e 's/../0x& /g'))
# Loop through the array and do the xor, initially start $XOR at 0
for (( x=0; x<"${#HEXARRAY[@]}"; x++ )); do
XOR=0x$(printf '%02x' "$(( ${XOR:-0} ^ ${HEXARRAY[$x]} ))")
done
# Strip off the 0x from the result
CLEAN=${XOR#0x}
printf '%s\n' "${CLEAN^^}"
}
main() {
case "${1:-}" in
"") # No input specified, read from stdin
depcheck
read -r SENTENCE
x_or
;;
*) # Input was provided, use that
depcheck
SENTENCE="$1"
x_or
;;
esac
}
# Main
# ====
main "$@"
當編寫 shell 腳本時,我總是試圖找到方法來消除外部程式的使用,甚至是像 sed 或 xxd 這樣常見的程式。如果有人知道僅使用 shell 內建函數執行上述操作的方法,請插話。
更新:這是一個考慮佐藤方法的新函數。它允許完全消除外部程式呼叫以及上面關聯的 depcheck 函數。
x_or() { # Create a hex XOR checksum of all the bytes
# Clean the line of $ character and anything before it
TAIL="${SENTENCE##*$}"
HEAD=${TAIL%\**}
LEN=${#HEAD}
# Loop through the string and do the xor
# initially start $XOR at 0
XOR=0
for (( x=0; x<$LEN; x++ )); do
(( XOR^=$(printf '%d' "'${HEAD:$x:1}'") ))
done
printf '%02X\n' "${XOR}"
}
使用“LC_CTYPE=C”呼叫該函數。這裡可能還可以做更多的事情,但這相當簡潔。
答案1
就我個人而言,我會這樣做:
#! /usr/bin/env bash
log() {
{
printf '%s: ' "${0##*/}"
printf "$@"
printf '\n'
} >&2
}
cksum() {
tot=${#1}
let len=tot-4
let res=0
while [ $len -gt 0 ]; do
let res^=$( LC_CTYPE=C printf '%d' "'${1:$len:1}'" )
let len--
done
let ptr=tot-2
if [ x"$( printf '%s' "${1:$ptr}" | tr a-f A-F )" != x"$( printf '%02X' $res )" ]; then
log '%s: invalid checksum (found %02X)' "$1" $res
fi
}
check () {
if expr "$2" : '\$.*\*[0-9a-fA-F][0-9a-fA-F]$' >/dev/null; then
cksum "$2"
else
log 'invalid input on line %d: %s' "$1" "$2"
fi
}
let cnt=0
if [ $# -ne 0 ]; then
while [ $# -gt 0 ]; do
let cnt++
check $cnt "$1"
shift
done
else
while read -r str; do
let cnt++
check $cnt "$str"
done
fi
shebang 線聲稱bash
,但它應該仍然適用於ksh93r
和zsh
。不依賴於xxd
.也沒有聲稱是可遵循的腳本風格的範例。 :)