目录树是这样的:
.
├── A_123
│ └── 123.txt
├── A_456
│ ├── tmp
│ └── tmp.log
└── A_789
└── 789.txt
有3个目录(A_123,A_456,A_789)。
目录名称的模式为:A_{numbers}
,我感兴趣的文件是{numbers}.txt
。
我想知道是否有办法获取其中没有A_{numbers}
文件的目录{numbers}.txt
。例如,上面这个脚本应该返回:
./A_456
由于A_456
在其文件夹中没有456.txt
,但A_123
和A_789
在相关文件夹中有{numbers}.txt
个文件。
任何人都有这方面的想法?谢谢!
答案 0 :(得分:4)
这是一种方法:
for dir in *;do
if [ $(find "$dir" -maxdepth 1 -regex '.*/[0-9][0-9]*\.txt' | wc -l) = 0 ]; then
echo $dir
fi
done
答案 1 :(得分:1)
由于A_ [0-9]目录未嵌套,因此您可以使用循环中的glob轻松完成此操作。这个实现是纯粹的bash,不会在外部实用程序中产生:
for d in A_[0-9]*/; do # the trailing / causes only directories to be matched
files=("$d"/[0-9]*.txt) # populate an array with matching text files
((!${#files})) && echo "$d" # echo $d if the array is empty
done
此实施存在一些问题。它将匹配诸如“12ab.txt”之类的文件,并且需要将目录的所有文件名加载到数组中。
以下是bash中另一种执行更准确的文件名匹配的方法:
re='^[0-9]+[.]txt$'
for d in A_[0-9]*/; do
for f in *; do
if [[ -f $f && $f =~ $re ]]; then
echo "$d"
break
fi
done
done
答案 2 :(得分:1)
其他几个答案略有不同:使用bash extended pattern matching:
shopt -s extglob nullglob
for dir in A_+([0-9]); do
files=($dir/+([0-9]).txt)
(( ${#files[@]} == 0 )) && echo $dir
done
答案 3 :(得分:0)
for file in *; do
if [[ "$file" =~ ^A_([0-9]+)$ && ! -f "$file/${BASH_REMATCH[1]}.txt" ]]; then
echo $file;
fi
done
工作原理:
=~
)检查文件/文件夹的名称是“A_”,后跟数字。${BASH_REMATCH[1]}
{number}.txt
。