6

我有以下在 shell 脚本中运行的代码

foo=`seq 1 1 100`
for i in $foo; do
  echo "input$i\_now"
done

这是我的问题:在某些情况下,输出会打印input1_now,而其他时候会打印input1\_now. 我确定有些不同,但我不知道是什么让它以一种或另一种方式打印。如果我的代码是

for i in $foo; do
   echo "input$i_now"
done

我总是会input忽略该行的其余部分。

我知道我可以改用input${i}_now它并让它每次都正确打印,但我最感兴趣的是理解为什么在看似相同的条件下输出不同。

更新:

在以下示例中,第一部分正确格式化变量和文本,以便将\_替换为_。但是,最后一部分要求我将变量放在大括号中,以便正确格式化它们。

echo "Enter Simulation #: "
read sim

g.mapset results

for i in `seq 1 1 100`; do

file=sim$sim\_run$i\_sum
g.copy $file\@expSim$sim\_$i,$file

file=sim$sim\_run$i\_average
g.copy $file\@expSim$sim\_$i,$file

for year in `seq 2004 1 2006`; do   

    file=sim$sim\_$year\_run$i\_sum
    g.copy $file\@expSim$sim\_$i,$file

    file=sim$sim\_$year\_run$i\_average
    g.copy $file\@expSim$sim\_$i,$file

done

years="2004 2005 2006"
times=`seq -w 1 16 365`
runs=`seq 1 1 100`

for year in $years; do
for ptime in $times; do
    for i in $runs; do
        if [ $i -eq 1 ]; then
            g.copy  vect=sim${sim}_pts_${year}_${ptime}_run${i}@expSim${sim}_${i},sim${sim}_pts_${year}_${ptime}
        fi
        if [ $i -gt  1 ]; then  
        v.patch input=sim${sim}_pts_${year}_${ptime}_run${i}@expSim${sim}_${i} output=sim${sim}_pts_${year}_${ptime} -e -a --o
        fi
    done
done
done
4

2 回答 2

8

应该是_一个占位符,有时是一个不同的字符?

bash"input$i\_now"用实际_总会产生input1\_now。在双引号内,仅当 a后跟 a 、 a 、 a 、 a或换行符bash时才删除它。请参阅Bash 参考手册中的“双引号”。这是 POSIX 标准行为;请参阅Shell 命令语言中的“双引号”\$`"\

更新

如果你写"input$i_now"bash只会打印input。它不会打印input1input1_now。它这样做是因为它_是一个有效的参数名称字符,因此bash认为您正在询问i_now参数的值。除非你设置i_now为非空字符串,bash否则会扩展$i_now为空字符串,从而"input$i_now"变成input.

更新 2

现在您已经发布了真实的代码,我们可以看到发生了什么。

首先,在您发布的真实代码中,您从未在参数扩展周围使用双引号。这会有所不同。

在双引号之外, a\总是被删除。请参阅Bash 参考手册中的“引号删除”。因此input$i\_now没有周围的双引号)扩展为input1_now.

但是,正如我在第一次更新中解释的那样,_是参数名称字符。请参阅Shell 命令语言中的“名称”。所以当bashsees时input$i_now,它i_now作为参数名称。

无论您是否使用双引号,都必须将参数名称与后面的字符分开,否则bash会将后面的字符视为参数名称的一部分。您可以通过将参数名称放在\后面来执行此操作,也可以通过将参数名称放在{...}.

始终使用 更安全{...},因为(正如您所发现的那样?)\根据它是否在双引号内,处理方式不同。如果您稍后返回并添加双引号,并且您已使用,则无论如何\您都需要将 更改\为。{...}

更新 3

下面是\, {...}, 和双引号的效果的演示。首先,我们设置一些变量:

$ year=2004 ptime=1 i=1 sim=123

以下是在不引用任何内容的情况下发生的情况:

$ echo vect=sim$sim_pts_$year_$ptime_run$i@expSim$sim_$i,sim$sim_pts_$year_$ptime
vect=sim1@expSim1,sim1

如果我们{...}不使用双引号,会发生以下情况:

$ echo vect=sim${sim}_pts_${year}_${ptime}_run${i}@expSim${sim}_${i},sim${sim}_pts_${year}_${ptime}
vect=sim123_pts_2004_1_run1@expSim123_1,sim123_pts_2004_1

如果我们添加双引号,它们将不起作用:

$ echo "vect=sim${sim}_pts_${year}_${ptime}_run${i}@expSim${sim}_${i},sim${sim}_pts_${year}_${ptime}"
vect=sim123_pts_2004_1_run1@expSim123_1,sim123_pts_2004_1

如果我们只使用 会发生以下情况\

$ echo vect=sim$sim\_pts\_$year\_$ptime\_run$i@expSim$sim\_$i,sim$sim\_pts\_$year\_$ptime
vect=sim123_pts_2004_1_run1@expSim123_1,sim123_pts_2004_1

请注意,每个\都已删除。\如果没有引用,shell 会删除它。

如果我们添加双引号,它们会阻止 shell 删除每个\

$ echo "vect=sim$sim\_pts\_$year\_$ptime\_run$i@expSim$sim\_$i,sim$sim\_pts\_$year\_$ptime"
vect=sim123\_pts\_2004\_1\_run1@expSim123\_1,sim123\_pts\_2004\_1
于 2013-04-16T04:23:40.950 回答
6

下划线是 bash 变量名中的有效字符,因此您需要保护变量名不受周围字符的影响。

foo=`seq 1 1 100`
for i in $foo; do
  echo "input${i}_now"
done

区别:

input$i_now    # "input" + $i_now
input${i}_now  # "input" + $i + "_now"
input"$i"_now  # input + "$i" + now
input$i\_now   # input + $1 + _now - \_ "escapes" the underscore

https://man7.org/linux/man-pages/man1/bash.1.html

名称仅由字母数字字符和下划线组成并以字母字符 或下划线开头的单词。也称为标识符。

您必须通过、 with或转义参数名称来保护参数名称,这些字符在参数名称中是有效的。所有“工作”,但我更喜欢大括号,这也是文档所指的内容:PREFIX${braces}SUFFIXPREFIX"$quotes"SUFFIX\

要扩展的参数名称或符号可以用大括号括起来,这是可选的,但用于保护要扩展的变量免受紧随其后的字符的影响,这些字符可以解释为名称的一部分。

于 2013-04-16T07:14:27.447 回答