我有一个文本文件,我正在尝试修改。我正在获取具有以下行形式的输入文件
(y+1/4,-x+1/2,z+3/4)
并尝试将其更改为
0 1 0 -1 0 0 0 0 1 1 / 4 1 / 2 3 / 4
我目前可以达到这一点
0 1 0 1/4 -1 0 0 1/2 0 0 1 3/4
用
#!bin/bash
filename="227.dat"
sed -i 's/(/ /g' $filename
sed -i 's/)//g' $filename
sed -i 's/,/ /g' $filename
sed -i 's/-x/-1 0 0/g' $filename
sed -i 's/x/ 1 0 0/g' $filename
sed -i 's/-y/ 0 -1 0/g' $filename
sed -i 's/y/ 0 1 0/g' $filename
sed -i 's/-z/ 0 0 -1/g' $filename
sed -i 's/z/ 0 0 1/g' $filename
sed -i '/+/! s/$/ 0 / 1 0 / 1 0 / 1/' $filename
while ((i++)); read -r line; do
if [[ $line == *[+]* ]]
then
sed -i 's/+/ /g' $filename
echo $i
fi
done < "$filename"
回显$i的原因是看到它正确地给出了行号,我想也许我可以将其用于这些特定行上的命令。我正在做这个转换,因为我们在创建晶体结构时使用的代码需要最后带有分数的向量符号,而不是 x,y,z 符号。我已经知道这不是"最漂亮"或最简单的解决方案,但我对所有这些都非常陌生,这是我迄今为止能够拼凑起来的。有什么建议吗?
这是一种可以简化解析的方法。使用设置为您不关心的所有可能的分隔符和字符的 IFS 将每一行读入数组:
while IFS=$'()+,' read -ra line; do
for i in 1 3 5; do
case "${line[$i]}" in
x) printf "%st%st%st" 1 0 0 ;;
y) printf "%st%st%st" 0 1 0 ;;
z) printf "%st%st%st" 0 0 1 ;;
-x) printf "%st%st%st" -1 0 0 ;;
-y) printf "%st%st%st" 0 -1 0 ;;
-z) printf "%st%st%st" 0 0 -1 ;;
esac
done
for i in 2 4 6; do
printf "%st" "${line[$i]}"
done
echo
done < "$filename"
#!/usr/bin/env bash
filename="227.dat"
re='[(]y[+]([[:digit:]/]+),-x[+]([[:digit:]/]+),z[+]([[:digit:]/]+)[)]';
while IFS= read -r line; do
if [[ $line =~ $re ]]; then
printf 't%s'
0 1 0
-1 0 0
0 0 1
"${BASH_REMATCH[1]}"
"${BASH_REMATCH[2]}"
"${BASH_REMATCH[3]}";
printf 'n';
else
echo "ERROR: $line does not match $re" 1>&2;
fi;
done <"$filename"
。给定,您的输入返回:
0 1 0 -1 0 0 0 0 1 1/4 1/2 3/4
。据我所知,这是正确的。
一种更复杂的方法,进行毫无根据的推断(考虑到问题本身缺乏细节和示例(,可能如下所示:
#!/usr/bin/env bash
while IFS='(),' read -a pieces; do
declare -A vars=( [x]=1 [y]=1 [z]=1 [x_sigil]='' [y_sigil]='' [z_sigil]='' )
for piece in "${pieces[@]}"; do
# 1 2 3 4
if [[ $piece =~ (-?)([xyz])([+]([[:digit:]/]+))? ]]; then
if [[ ${BASH_REMATCH[4]} ]]; then # only if there *are* digits
vars[${BASH_REMATCH[2]}]=${BASH_REMATCH[4]} # ...then store them.
fi
vars[${BASH_REMATCH[2]}_sigil]=${BASH_REMATCH[1]} # store - if applicable
fi
done
printf 't%s'
"0" "${vars[x_sigil]}1" 0
"${vars[y_sigil]}1" 0 0
0 0 "${vars[z_sigil]}1"
"${vars[y]}" "${vars[x]}" "${vars[z]}"
printf 'n'
done
给定此答案的注释中提供的示例输入,输出为:
0 1 0 1 0 0 0 0 1 1 1 1
0 1 0 1 0 0 0 0 1 1 1 1
0 1 0 1 0 0 0 0 1 1 1 1
0 1 0 1 0 0 0 0 -1 3/4 1/4 1/2
0 1 0 -1 0 0 0 0 1 1/2 3/4 1/4
0 -1 0 1 0 0 0 0 1 1/4 1/2 3/4
0 -1 0 -1 0 0 0 0 -1 1 1 1
0 -1 0 -1 0 0 0 0 -1 1 1 1
0 -1 0 -1 0 0 0 0 -1 1 1 1
0 -1 0 -1 0 0 0 0 1 1/4 3/4 1/2
0 -1 0 1 0 0 0 0 -1 1/2 1/4 3/4
0 1 0 -1 0 0 0 0 -1 3/4 1/2 1/4
0 -1 0 -1 0 0 0 0 1 1/4 3/4 1/2
0 -1 0 -1 0 0 0 0 1 1/4 3/4 1/2
0 -1 0 -1 0 0 0 0 1 1/4 3/4 1/2
0 -1 0 -1 0 0 0 0 -1 1 1 1
0 -1 0 1 0 0 0 0 1 1/4 1/2 3/4
0 1 0 -1 0 0 0 0 1 1/2 3/4 1/4
0 1 0 1 0 0 0 0 -1 3/4 1/4 1/2
0 1 0 1 0 0 0 0 -1 3/4 1/4 1/2
0 1 0 1 0 0 0 0 -1 3/4 1/4 1/2
0 1 0 1 0 0 0 0 1 1 1 1
0 1 0 -1 0 0 0 0 -1 3/4 1/2 1/4
0 -1 0 1 0 0 0 0 -1 1/2 1/4 3/4
0 -1 0 1 0 0 0 0 -1 1/2 1/4 3/4
0 -1 0 1 0 0 0 0 -1 1/2 1/4 3/4
0 -1 0 1 0 0 0 0 -1 1/2 1/4 3/4
0 -1 0 1 0 0 0 0 1 1/4 1/2 3/4