下面是一个大脚本中的shell脚本片段。它从变量持有的字符串中删除引号。我正在使用sed,但它是有效的吗?如果不是,那么什么是有效的方法?

#!/bin/sh

opt="\"html\\test\\\""
temp=`echo $opt | sed 's/.\(.*\)/\1/' | sed 's/\(.*\)./\1/'`
echo $temp

当前回答

Bash中最简单的解决方案:

$ s='"abc"'
$ echo $s
"abc"
$ echo "${s:1:-1}"
abc

这被称为子字符串展开(参见Gnu Bash手册并搜索${parameter:offset:length})。在这个例子中,它从s获取子字符串,从位置1开始,到最后第二个位置结束。这是因为,如果length为负值,则它将被解释为从参数末尾开始的向后运行偏移。

其他回答

Bash中最简单的解决方案:

$ s='"abc"'
$ echo $s
"abc"
$ echo "${s:1:-1}"
abc

这被称为子字符串展开(参见Gnu Bash手册并搜索${parameter:offset:length})。在这个例子中,它从s获取子字符串,从位置1开始,到最后第二个位置结束。这是因为,如果length为负值,则它将被解释为从参数末尾开始的向后运行偏移。

我知道这是一个非常古老的问题,但这里有另一个sed变体,它可能对某些人有用。不像其他的一些,它只替换开头或结尾的双引号…

echo "$opt" | sed -r 's/^"|"$//g'

如果您需要匹配单引号或双引号,并且只匹配正确引用的字符串。你可以使用这个稍微复杂一点的正则表达式…

echo $opt | sed -E "s|^(['\"])(.*)\1$|\2|g"

这使用了反向引用,以确保结尾的引用与开头相同。

你只需要调用一次sed就可以做到:

$ echo "\"html\\test\\\"" | sed 's/^"\(.*\)"$/\1/'
html\test\

我使用这个正则表达式,避免从没有正确引用的字符串中删除引号,这里根据输入显示不同的输出,只有一个开始-结束引号受到影响:

echo '"only first' | sed 's/^"\(.*\)"$/\1/'

输出:>"only first<

echo 'only last"' | sed 's/^"\(.*\)"$/\1/'

输出:>"only last"<

echo '"both"' | sed 's/^"\(.*\)"$/\1/'

输出:> <

echo '"space after" ' | sed 's/^"\(.*\)"$/\1/'

输出:>"space after" <

echo ' "space before"' | sed 's/^"\(.*\)"$/\1/'

输出:> "space before"<

使用tr删除“:

 echo "$opt" | tr -d '"'

注意:这并没有完全回答问题,删除了所有双引号,而不仅仅是开头和结尾。请看下面的其他答案。