2

我有一个由以下内容组成的 YAML 文件:

acceleration_matrix:
  # 1ere row: x
  - [20, 0, 0, 15, 15]
  # 2eme row: y
  - [0, 15, 0, 0, 0]
  # 3eme row: z
  - [0, 0, 30, 15, -15]
  # 4eme row: repere de l'acceleration
  - [0, 0, 0, 0, 0]
  # 5eme row: loadcase_id
  - [1, 2, 3, 4, 5]

我想提取最后一行的最后一个值(在这种情况下,5)取消 shell 脚本。使用这篇文章,我编码:

#!/bin/bash

function parse_yaml {
   local prefix=$2
   local s='[[:space:]]*' w='[a-zA-Z0-9_]*' fs=$(echo @|tr @ '\034')
   sed -ne "s|^\($s\):|\1|" \
        -e "s|^\($s\)\($w\)$s:$s[\"']\(.*\)[\"']$s\$|\1$fs\2$fs\3|p" \
        -e "s|^\($s\)\($w\)$s:$s\(.*\)$s\$|\1$fs\2$fs\3|p"  $1 |
   awk -F$fs '{
      indent = length($1)/2;
      vname[indent] = $2;
      for (i in vname) {if (i > indent) {delete vname[i]}}
      if (length($3) > 0) {
         vn=""; for (i=0; i<indent; i++) {vn=(vn)(vname[i])("_")}
         printf("%s%s%s=\"%s\"\n", "'$prefix'",vn, $2, $3);
      }
   }'
}

parse_yaml nastran_acceleration_matrix.yml

echo $acceleration_matrix

但由于我是 shell 的新手,我找不到如何提取这些值。我想要的只是:

set number_load_case = ### acceleration_matrix[-1,-1] ### (I'm more used to coding in python thus the -1 aka last row/column)

有人可以帮帮我吗?

提前致谢!

4

3 回答 3

2

安装 shyaml

 $ pip install shyaml      ## installation

conda env create -f environment.yml -n cat test.yaml | shyaml get-value conda_env_name

您可以在安装 syaml 后尝试上述命令。

如果这对您不起作用,那么您可以在 bash 脚本中尝试'jq''yq',这会将您的 yaml 解析为 json,然后您可以从 JSON 读取值,之后您可以从 bash 本身运行命令.

于 2021-08-23T12:19:52.803 回答
1

我建议使用适当的解析器,例如yq。你的任务(最后一个元素的最后一个值acceleration_matrix)然后变得像

yq eval '.acceleration_matrix[-1][-1]' nastran_acceleration_matrix.yml
于 2021-07-21T15:44:06.970 回答
1

基于 OP 的示例输入的一个awk想法:

$ awk -F'[] ]' '{lastid=$(NF-1)} END {print lastid}' test.yaml
5

在哪里:

  • -F'[] ]'- 将输入字段分隔符设置为<space>]
  • lastid=$(NF-1)- 跟踪最后一个字段
  • END {print lastid}- 打印我们保存的最后一个'next to last field'
  • 最终结果:扫描整个文件,在我们进行时保存在最后一个字段旁边,并在文件末尾打印/输出我们捕获的最后一个“倒数第二个”字段

要存储在bash变量中:

$ number_load_case=$(awk -F'[] ]' '{lastid=$(NF-1)} END {print lastid}' test.yaml)
$ echo "${number_load_case}"
5

笔记:

  • 如果输入的格式与 OPs 示例中的格式不完全相同,这显然会产生错误的结果(例如,如果最后一行为空白,awk则将失败并出现错误fatal: attempt to access field -1:)
  • 正如已经提到的那样,类似的东西yq会更适合这个问题,特别是如果格式不像 OP 示例中那样“漂亮”
  • 对于不太“漂亮”的格式,以及无法使用类似的东西yq,可以编写更健壮的解析器(例如,在awk
于 2021-07-21T16:26:32.917 回答