Bash 文本文件编辑/修改



我有一个文本文件,我正在尝试修改。我正在获取具有以下行形式的输入文件

(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

最新更新