下面是一个大脚本中的shell脚本片段。它从变量持有的字符串中删除引号。我正在使用sed,但它是有效的吗?如果不是,那么什么是有效的方法?
#!/bin/sh
opt="\"html\\test\\\""
temp=`echo $opt | sed 's/.\(.*\)/\1/' | sed 's/\(.*\)./\1/'`
echo $temp
下面是一个大脚本中的shell脚本片段。它从变量持有的字符串中删除引号。我正在使用sed,但它是有效的吗?如果不是,那么什么是有效的方法?
#!/bin/sh
opt="\"html\\test\\\""
temp=`echo $opt | sed 's/.\(.*\)/\1/' | sed 's/\(.*\)./\1/'`
echo $temp
当前回答
我的版本
strip_quotes() {
while [[ $# -gt 0 ]]; do
local value=${!1}
local len=${#value}
[[ ${value:0:1} == \" && ${value:$len-1:1} == \" ]] && declare -g $1="${value:1:$len-2}"
shift
done
}
该函数接受变量名并去掉引号。它只删除一对匹配的开头和结尾引号。它不检查末尾的引号是否转义(前面有\,但\本身没有转义)。
根据我的经验,像这样的通用字符串实用函数(我有一个库)在直接操作字符串时是最有效的,不使用任何模式匹配,特别是不创建任何子shell,或调用任何外部工具,如sed、awk或grep。
var1="\"test \\ \" end \""
var2=test
var3=\"test
var4=test\"
echo before:
for i in var{1,2,3,4}; do
echo $i="${!i}"
done
strip_quotes var{1,2,3,4}
echo
echo after:
for i in var{1,2,3,4}; do
echo $i="${!i}"
done
其他回答
使用tr删除“:
echo "$opt" | tr -d '"'
注意:这并没有完全回答问题,删除了所有双引号,而不仅仅是开头和结尾。请看下面的其他答案。
使用xargs有一个简单的方法:
> echo '"quoted"' | xargs
quoted
如果没有提供命令,Xargs使用echo作为默认命令,并从输入中删除引号,参见这里的例子。但是请注意,这只在字符串不包含附加引号的情况下才有效。在这种情况下,它要么失败(引号数量不均匀),要么删除所有引号。
在Bash中,你可以使用下面的一行代码:
[[ "${var}" == \"*\" || "${var}" == \'*\' ]] && var="${var:1:-1}"
这将从存储在var中的字符串中删除周围的引号(单引号和双引号),同时保持字符串中的引号字符完整。同样,如果只有一个前引号或只有一个后引号,或者在开始/结束处有混合引号字符,这将不起任何作用。
包装在函数中:
#!/usr/bin/env bash
# Strip surrounding quotes from string [$1: variable name]
function strip_quotes() {
local -n var="$1"
[[ "${var}" == \"*\" || "${var}" == \'*\' ]] && var="${var:1:-1}"
}
str="'hello world'"
echo "Before: ${str}"
strip_quotes str
echo "After: ${str}"
我知道这是一个非常古老的问题,但这里有另一个sed变体,它可能对某些人有用。不像其他的一些,它只替换开头或结尾的双引号…
echo "$opt" | sed -r 's/^"|"$//g'
如果您需要匹配单引号或双引号,并且只匹配正确引用的字符串。你可以使用这个稍微复杂一点的正则表达式…
echo $opt | sed -E "s|^(['\"])(.*)\1$|\2|g"
这使用了反向引用,以确保结尾的引用与开头相同。
我的版本
strip_quotes() {
while [[ $# -gt 0 ]]; do
local value=${!1}
local len=${#value}
[[ ${value:0:1} == \" && ${value:$len-1:1} == \" ]] && declare -g $1="${value:1:$len-2}"
shift
done
}
该函数接受变量名并去掉引号。它只删除一对匹配的开头和结尾引号。它不检查末尾的引号是否转义(前面有\,但\本身没有转义)。
根据我的经验,像这样的通用字符串实用函数(我有一个库)在直接操作字符串时是最有效的,不使用任何模式匹配,特别是不创建任何子shell,或调用任何外部工具,如sed、awk或grep。
var1="\"test \\ \" end \""
var2=test
var3=\"test
var4=test\"
echo before:
for i in var{1,2,3,4}; do
echo $i="${!i}"
done
strip_quotes var{1,2,3,4}
echo
echo after:
for i in var{1,2,3,4}; do
echo $i="${!i}"
done