2014-09-10 75 views
2

在我的脚本中,我有一个未知数的变量,其中包含要转换为矢量的角度。我创建了变量名,使得每个'参数'的角度的形式为: {parameter} _angle_ {lat/perp} 因此,每个参数都有一个'lat'和'perp'角度变量。 所以我想要做的是找到所有包含'_angle_lat'的变量,对这些变量的值做一些计算并创建一个包含'参数'名称的新变量。 例如:Bash:通过名称中包含模式的变量循环

export M0_angle_lat=4 
export M1_angle_lat=70 
export M1_angle_perp=8 
export M0_angle_perp=90 

# Now I want to use these values to calculate vectors 
for varname in *_angle_lat 
do 
    # at first iteration it will get for example "varname=M0_angle_lat" in which case 
    # I want it to do: 
    M0_X=$(($M0_angle_lat * $M0_angle_perp)) 
    # The second iteration in case would then give "varname=M1_angle_lat" in which case 
    # I want it to do: 
    M1_X=$(($M1_angle_lat * $M1_angle_perp)) 
done 

我希望这是清楚我的目标是在这里。谢谢您的帮助!

+2

'bash'确实不是这种类型的编程语言。它确实有数组,但它们不适合构建更复杂的数据结构,并且尝试使用动态生成的变量名编写代码的情况最容易出错。 (虽然'bash' 4.3引入了一个使它更简单的特性。) – chepner 2014-09-10 12:32:19

+0

我使用'bash'的原因是因为它是一个小型脚本,必须在集群上调用,所有周围的代码和脚本也都在bash中。 – Bjorn 2014-09-10 12:37:16

+0

如果您可以通过前缀重写您的变量以匹配,那么您可以使用'$ {!prefix *}'和'$ {!prefix @}'扩展来获取以'prefix'开头的变量名称。 – 2014-09-10 13:49:53

回答

6

你可以做的是使用env来获取所有变量的列表,然后遍历它们:

while IFS='=' read -r name value ; do 
    if [[ $name == *'_angle_lat'* ]]; then 
    echo "$name" ${!name} 
    prefix=${name%%_*} # delete longest match from back (everything after first _) 
    angle_lat="${prefix}_angle_lat" 
    angle_perp="${prefix}_angle_perp" 
    result="${prefix}_X" 
    declare "${result}=$((${!angle_lat} * ${!angle_perp}))"  
    fi 
done < <(env) 
+0

这似乎是一个巨大的进步。但在这种情况下,我有'$ var'是整个字符串:'M0_angle_lat = 4'。我如何从这里提取'参数'(在这种情况下是'M0')? 如何使用它来创建变量'M0_X = $(($ M0_angle_lat * $ M0_angle_perp))'? – Bjorn 2014-09-10 12:41:55

+0

我看到这段代码试图做什么,但它似乎并没有工作。我刚刚检查了群集上运行的bash的版本,结果发现它是旧的: 'GNU bash,版本3.2.25(1) - 发布(x86_64-redhat-linux-gnu) 版权所有(C)2005自由软件基金会,' – Bjorn 2014-09-10 13:28:33

+0

事实证明,我以前是没有问题的bash,但纯粹的代码中的一些小错误,现在已经修复! – Bjorn 2014-09-11 10:02:24

1

此代码需要bash 4.3,使用由declare -n创建命名引用。

这也需要你稍微重新命名你的变量。

angle_lat_M0=4 
angle_lat_M1=70 
angle_perp_M1=8 
angle_perp_M0=90 

# Now I want to use these values to calculate vectors 
for varname in ${!angle_lat*} 
do 
    # Ref to angle_lat_* 
    declare -n lat=$varname 
    # Ref to angle_perp_* 
    declare -n perp=${varname/_lat_/_perp_} 
    # Ref to X_* 
    declare -n x=${varname/angle_lat_/X_} 

    x=$((lat * perp)) 
done 

echo $X_M0 
echo $X_M1 

在4.3之前,你需要一些额外的技巧来工作varname。 (事实上​​,它不是那么糟糕,因为我认为这将是。)

angle_lat_M0=4 
angle_lat_M1=70 
angle_perp_M1=8 
angle_perp_M0=90 

# Now I want to use these values to calculate vectors 
for varname in ${!angle_lat*} 
do 
    tag=${varname#angle_lat_} # M0, M1, etc 
    lat=${!varname} 
    perp_name=angle_perp_$tag 
    perp=${!perp_name} 
    x=$((lat * perp)) 

    declare "X_$tag=$x" 
done 

echo $X_M0 
echo $X_M1 

就更简单了,它应该为bash所有版本(可能不包括 一些很老的版本,但是3.2至少支持) 。这很简单,主要是因为 它放弃尝试迭代一组相似的变量名称。

lats=($M0_angle_lat $M1_angle_lat) 
perps=($M0_angle_perp $M1_angle_lat) 
declare -a x 

for i in "${!lats[@]}"; do 
    x+=(${lats[i]} * ${perps[i]}) 
done 

M0_X=${x[0]} 
M1_X=${x[1]} 
# or 
for i in "${!x[@]}"; do 
    declare "M${i}_X"=${x[i]} 
done