30

我想计算给定目录中的所有 *bin 文件。最初我使用的是for-loop

var=0
for i in *ls *bin
do
   perform computations on $i ....
   var+=1
done
echo $var

但是,在某些目录中,文件过多会导致错误:Argument list too long

因此,我正在尝试使用管道while-loop

var=0
ls *.bin | while read i;
do
  perform computations on $i
  var+=1
done
echo $var

现在的问题是使用管道创建了子壳。因此,echo $var返回0
我该如何处理这个问题?
原代码:

#!/bin/bash

function entropyImpl {
    if [[ -n "$1" ]]
    then
        if [[ -e "$1" ]]
        then
            echo "scale = 4; $(gzip -c ${1} | wc -c) / $(cat ${1} | wc -c)" | bc
        else
            echo "file ($1) not found"
        fi
    else
        datafile="$(mktemp entropy.XXXXX)"
        cat - > "$datafile"
        entropy "$datafile"
        rm "$datafile"
    fi

    return 1
}
declare acc_entropy=0
declare count=0

ls *.bin | while read i ;
do  
    echo "Computing $i"  | tee -a entropy.txt
    curr_entropy=`entropyImpl $i`
    curr_entropy=`echo $curr_entropy | bc`  
    echo -e "\tEntropy: $curr_entropy"  | tee -a entropy.txt
    acc_entropy=`echo $acc_entropy + $curr_entropy | bc`
    let count+=1
done

echo "Out of function: $count | $acc_entropy"
acc_entropy=`echo "scale=4; $acc_entropy / $count" | bc`

echo -e "===================================================\n" | tee -a entropy.txt
echo -e "Accumulated Entropy:\t$acc_entropy ($count files processed)\n" | tee -a entropy.txt
4

4 回答 4

78

问题是while循环是在子shell中执行的。在while循环终止后,子shell的副本var被丢弃,而var父(其值不变)的原始副本被回显。

解决此问题的一种方法是使用流程替换,如下所示:

var=0
while read i;
do
  # perform computations on $i
  ((var++))
done < <(find . -type f -name "*.bin" -maxdepth 1)

查看BashFAQ/024以了解其他解决方法。

请注意,我也已替换ls为,因为parsefind不是一个好习惯。ls

于 2012-12-05T15:51:21.880 回答
16

符合 POSIX 的解决方案是使用管道(p 文件)。这个解决方案非常好,可移植,而且是 POSIX,但是在硬盘上写了一些东西。

mkfifo mypipe
find . -type f -name "*.bin" -maxdepth 1 > mypipe &
while read line
do
    # action
done < mypipe
rm mypipe

您的管道是硬盘上的一个文件。如果您想避免拥有无用的文件,请不要忘记将其删除。

于 2016-09-18T15:45:29.837 回答
4

因此,研究通用问题,将变量从子壳 while 循环传递给父级。我发现的一种解决方案是使用here-string。因为那是 bash-ish,而且我更喜欢 POSIX 解决方案,所以我发现 here-string 实际上只是 here-document 的快捷方式。有了这些知识,我想出了以下方法,避免使用子外壳;从而允许在循环中设置变量。

#!/bin/sh

set -eu

passwd="username,password,uid,gid
root,admin,0,0
john,appleseed,1,1
jane,doe,2,2"

main()
{
    while IFS="," read -r _user _pass _uid _gid; do
        if [ "${_user}" = "${1:-}" ]; then
            password="${_pass}"
        fi
    done <<-EOT
        ${passwd}
    EOT

    if [ -z "${password:-}" ]; then
        echo "No password found."
        exit 1
    fi

    echo "The password is '${password}'."
}

main "${@}"

exit 0

所有复制粘贴的一个重要注意事项是,here-document 是使用连字符设置的,表示要忽略选项卡。这是为了保持布局有点好。重要的是要注意,因为 stackoverflow 不会在“代码”中呈现选项卡,而是用空格替换它们。Grmbl。所以,不要破坏我的代码,只是因为你们喜欢空格而不是制表符,在这种情况下无关紧要!

This probably breaks on different editor(settings) and what not. So the alternative would be to have it as:

    done <<-EOT
${passwd}
EOT
于 2019-08-21T18:25:26.820 回答
-2

这也可以通过 for 循环来完成:

var=0;
for file in `find . -type f -name "*.bin" -maxdepth 1`; do 
    # perform computations on "$i"
    ((var++))
done 
echo $var
于 2016-05-12T09:21:04.343 回答