我已经查过了,找不到答案,如果以前有人问过,我提前道歉。
我在 FreeBSD () 上使用 shell ,我/bin/sh
想将所有以. 我能得到的最接近的是,但它只转储多行变量的第一行(有些是多行的,我需要它们全部),并且在病理边缘情况下可能容易出错,例如包含“ myvar ”的字符串碰巧在那部分之前的换行符。_myvar_
set | grep '^_myvar_'
如果我可以只列出变量名称(而不是值),那么我可以在 a 中过滤do...read...while
并一次只为具有匹配名称的 var 获取一个值,但我找不到这样做的方法。我也无法过滤完整的输出,因为没有确定性的方法来判断输出行是否包含延续或新变量,包含_myvar_
, =
或newline (\n)
字符的字符串不存在边缘情况问题,或者可能是尾随空格。我不想修改环境,因为代码包含在其他代码中,环境必须是稳定的。
如果有帮助,输出/列表包含任何匹配的环境变量都不是问题(如果存在的话——它们极不可能)
有没有办法做到这一点?
由于
set
FreeBSD 的内置sh
输出格式适合重新输入到 shell,您可以执行以下操作:这是
set
with输出的每一行的前缀,"out "
并将其评估为 shell 代码(其中out
是一个函数,它打印其第一个参数的子字符串直到第一个=
)。sed
也将插入"out "
多行变量的内容,但这仍将包含在我们的out
函数接收的参数中并超过 first=
,因此在我们不显示的部分中。例如,在
set
如下输出上:我们将评估:
但这仍然很好,
out
因为这 3 个变量只调用了 3 次。打印变量名和值:
请注意,它只输出变量,而不是其他类型的参数
$-
,如位置参数......该方法仅适用于仅输出标量
sh
变量的实现(不适用于数组或关联数组或复合变量,其中会出现语法错误)。那些具有其他变量类型的 shell 通常也具有更好的自省功能。set
out var=(x)
在
zsh
:或仅用于名称
在
bash
:我最终对@StéphaneChazelas 解决方案进行了以下简化,因为我不需要更多。把它贴在下面,以防它帮助任何人。但是 Stéphane 找到了答案,这只是它的修改版本,仅此而已。
该
eval
调用为每一行添加前缀,但由于set
输出其值的方式,它实际上为每个变量创建了一个命令get_vars VARNAME=VALUE
。评估它会执行命令。然后每次调用它时,get_vars
检查当前行中捕获的变量是否与所需的正则表达式匹配,如果是,"get_vars "
则从任何第二行和后面的行中删除前缀,并显示结果。