如何在文件夹层次结构中找到所有不同的文件扩展名?

问题描述 投票:192回答:14

在Linux机器上,我想遍历文件夹层次结构并获取其中所有不同文件扩展名的列表。

从shell实现这一目标的最佳方法是什么?

linux grep filesystems file-extension
14个回答
310
投票

试试这个(不确定它是否是最好的方法,但它有效):

find . -type f | perl -ne 'print $1 if m/\.([^.\/]+)$/' | sort -u

它的工作原理如下:

  • 查找当前文件夹中的所有文件
  • 打印文件的扩展名(如果有)
  • 制作一个独特的排序列表

3
投票

由于已经有另一种使用Perl的解决方案:

如果你安装了Python,你也可以(从shell):

python -c "import os;e=set();[[e.add(os.path.splitext(f)[-1]) for f in fn]for _,_,fn in os.walk('/home')];print '\n'.join(e)"

2
投票

到目前为止,所有回复都没有正确处理带有换行符的文件名(ChristopheD除外,因为我输入的内容刚刚进入)。以下不是shell单线程,但是有效并且速度相当快。

import os, sys

def names(roots):
    for root in roots:
        for a, b, basenames in os.walk(root):
            for basename in basenames:
                yield basename

sufs = set(os.path.splitext(x)[1] for x in names(sys.argv[1:]))
for suf in sufs:
    if suf:
        print suf

2
投票

我认为还没有提到这个:

find . -type f -exec sh -c 'echo "${0##*.}"' {} \; | sort | uniq -c

1
投票

我认为最简单直接的方法是

for f in *.*; do echo "${f##*.}"; done | sort -u

它是在ChristopheD的第三种方式上修改的。


0
投票

你也可以这样做

find . -type f -name "*.php" -exec PATHTOAPP {} +

45
投票

不需要管道到sort,awk可以做到这一切:

find . -type f | awk -F. '!a[$NF]++{print $NF}'

33
投票

递归版:

find . -type f | sed -e 's/.*\.//' | sed -e 's/.*\///' | sort -u

如果你想要总数(看到扩展的时间有多少):

find . -type f | sed -e 's/.*\.//' | sed -e 's/.*\///' | sort | uniq -c | sort -rn

非递归(单个文件夹):

for f in *.*; do printf "%s\n" "${f##*.}"; done | sort -u

我的基础是this forum post,信用应该去那里。


23
投票

电源外壳:

dir -recurse | select-object extension -unique

感谢http://kevin-berridge.blogspot.com/2007/11/windows-powershell.html


12
投票

使用点查找everythin并仅显示后缀。

find . -type f -name "*.*" | awk -F. '{print $NF}' | sort -u

如果你知道所有后缀都有3个字符

find . -type f -name "*.???" | awk -F. '{print $NF}' | sort -u

或者使用sed显示所有带有一到四个字符的后缀。将{1,4}更改为后缀中您期望的字符范围。

find . -type f | sed -n 's/.*\.\(.\{1,4\}\)$/\1/p'| sort -u

7
投票

添加我自己的变化混合。我认为这是最简单的,当效率不是一个大问题时可能很有用。

find . -type f | grep -o -E '\.[^\.]+$' | sort -u

6
投票

我的awk-less,sed-less,Perl-less,Python-less-POSIX兼容的替代方案:

find . -type f | rev | cut -d. -f1 | rev  | tr '[:upper:]' '[:lower:]' | sort | uniq --count | sort -rn

诀窍是它会反转线并在开始时切断扩展。 它还将扩展名转换为小写。

示例输出:

   3689 jpg
   1036 png
    610 mp4
     90 webm
     90 mkv
     57 mov
     12 avi
     10 txt
      3 zip
      2 ogv
      1 xcf
      1 trashinfo
      1 sh
      1 m4v
      1 jpeg
      1 ini
      1 gqv
      1 gcs
      1 dv

5
投票

在Python中使用生成器来处理非常大的目录,包括空白扩展,并获得每个扩展显示的次数:

import json
import collections
import itertools
import os

root = '/home/andres'
files = itertools.chain.from_iterable((
    files for _,_,files in os.walk(root)
    ))
counter = collections.Counter(
    (os.path.splitext(file_)[1] for file_ in files)
)
print json.dumps(counter, indent=2)

5
投票

我在这里尝试了一堆答案,甚至是“最好”的答案。他们都没有达到我特别追求的目标。因此,除了过去12个小时的多个程序的正则表达式代码以及阅读和测试这些答案之外,这就是我想出来的,它正如我想要的那样正常工作。

 find . -type f -name "*.*" | grep -o -E "\.[^\.]+$" | grep -o -E "[[:alpha:]]{2,16}" | awk '{print tolower($0)}' | sort -u
  • 查找可能具有扩展名的所有文件。
  • 仅限扩展名
  • 文件扩展名为2到16个字符的Greps(如果它们不符合您的需要,只需调整数字)。这有助于避免缓存文件和系统文件(系统文件位是搜索jail)。
  • awk以小写字母打印扩展名。
  • 排序并引入唯一值。最初我试图尝试awk答案,但它会使区分敏感性不同的打印项目翻倍。

如果您需要文件扩展名的计数,请使用以下代码

find . -type f -name "*.*" | grep -o -E "\.[^\.]+$" | grep -o -E "[[:alpha:]]{2,16}" | awk '{print tolower($0)}' | sort | uniq -c | sort -rn

虽然这些方法需要一些时间才能完成,而且可能不是解决问题的最佳方法,但它们可以正常工作。

更新:每@ alpha_989长文件扩展名将导致问题。这是由于原始的正则表达式“[[:alpha:]] {3,6}”。我更新了答案,包括正则表达式“[[:alpha:]] {2,16}”。但是,使用此代码的任何人都应该知道这些数字是最终输出允许扩展的最小值和最大值。超出该范围的任何内容都将在输出中分成多行。

注意:原帖显示为“ - 文件扩展名为3到6个字符的Greps(如果它们不符合您的需要,只需调整数字)。这有助于避免缓存文件和系统文件(系统文件位是搜索jail)。 “

想法:可用于通过以下方式查找特定长度的文件扩展名:

 find . -type f -name "*.*" | grep -o -E "\.[^\.]+$" | grep -o -E "[[:alpha:]]{4,}" | awk '{print tolower($0)}' | sort -u

其中4是要包含的文件扩展名长度,然后查找超出该长度的任何扩展名。

© www.soinside.com 2019 - 2024. All rights reserved.