Bash文本文件编辑/修改

时间:2017-11-10 01:15:30

标签: bash shell

我有一个我试图修改的文本文件。我正在使用具有

形式的行的输入文件
(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"

echo $ i的原因是看到它正确地给出了行号,我想也许我可以将它用于那些特定行上的命令。我正在进行这种转换,因为我们在创建晶体结构时使用的代码需要在末尾使用分数的矢量符号,而不是x,y,z符号。我已经知道这不是最漂亮的&#34;或者最简单的解决方案,但我对所有这些都是新手,而且到目前为止我能够拼凑起来。有什么建议吗?

2 个答案:

答案 0 :(得分:2)

这是一种可以简化解析的方法。使用设置为所有可能的分隔符和您不关心的字符的IFS将每行读入数组:

System.out.println(driver.findElement(By.xpath("//div[@id='header_cart']/a/em[1]")).getText());

答案 1 :(得分:0)

#!/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