Как вы рекурсивно распаковываете архивы в каталоге и его подкаталогах из командной строки Unix? - PullRequest
55 голосов
/ 20 сентября 2008

Команда unzip не имеет возможности рекурсивно разархивировать архивы.

Если у меня есть следующая структура каталогов и архивов:

/Mother/Loving.zip
/Scurvy/Sea Dogs.zip
/Scurvy/Cures/Limes.zip

И я хочу разархивировать все архивы в каталоги с тем же именем, что и у каждого архива:

/Mother/Loving/1.txt
/Mother/Loving.zip
/Scurvy/Sea Dogs/2.txt
/Scurvy/Sea Dogs.zip
/Scurvy/Cures/Limes/3.txt
/Scurvy/Cures/Limes.zip

Какую команду или команды я бы дал?

Важно, чтобы это не давило имена файлов, в которых есть пробелы.

Ответы [ 9 ]

86 голосов
/ 23 февраля 2010

Если вы хотите извлечь файлы в соответствующую папку, вы можете попробовать это

find . -name "*.zip" | while read filename; do unzip -o -d "`dirname "$filename"`" "$filename"; done;

Мультипроцессорная версия для систем, способных обрабатывать высокий уровень ввода-вывода:

find . -name "*.zip" | xargs -P 5 -I fileName sh -c 'unzip -o -d "$(dirname "fileName")/$(basename -s .zip "fileName")" "fileName"'
30 голосов
/ 20 сентября 2008

Вот одно решение, которое включает команду find и цикл while:

find . -name "*.zip" | while read filename; do unzip -o -d "`basename -s .zip "$filename"`" "$filename"; done;
15 голосов
/ 13 марта 2014

Решение, которое правильно обрабатывает все имена файлов (включая переводы строк) и извлекает их в каталог, который находится в том же месте, что и файл, только с удаленным расширением:

find . -name '*.zip' -exec sh -c 'unzip -o -d "${0%.*}" "$0"' '{}' ';'

Обратите внимание, что вы можете легко заставить его обрабатывать больше типов файлов (например, .jar), добавив их, используя -o, например ::

find . '(' -name '*.zip' -o -name '*.jar' ')' -exec ...
5 голосов
/ 20 сентября 2008

Вы можете использовать find вместе с флагом -exec в одной командной строке для выполнения работы

find . -name "*.zip" -exec unzip {} \;
2 голосов
/ 20 сентября 2008

Что-то вроде gunzip с использованием флага -r? ....

Рекурсивно перемещаться по структуре каталогов. Если какое-либо из имен файлов, указанных в командной строке, является каталогом, gzip спустится в каталог и сожмет все найденные файлы (или распакует их в случае gunzip).

http://www.computerhope.com/unix/gzip.htm

1 голос
/ 29 марта 2018

Это работает отлично, как мы хотим:

Распаковать файлы:

find . -name "*.zip" | xargs -P 5 -I FILENAME sh -c 'unzip -o -d "$(dirname "FILENAME")" "FILENAME"'

Команда выше не создает дубликаты каталогов.

Удалить все zip-файлы:

find . -depth -name '*.zip' -exec rm {} \;
0 голосов
/ 28 мая 2014

Другое интересное решение будет:

DESTINY=[Give the output that you intend]

# Don't forget to change from .ZIP to .zip.
# In my case the files were in .ZIP.
# The echo were for debug purpose.

find . -name "*.ZIP" | while read filename; do
ADDRESS=$filename
#echo "Address: $ADDRESS"
BASENAME=`basename $filename .ZIP`
#echo "Basename: $BASENAME"
unzip -d "$DESTINY$BASENAME" "$ADDRESS";
done;
0 голосов
/ 14 мая 2013

Я понимаю, что это очень давно, но это был один из первых обращений в Google, когда я искал решение для чего-то подобного, поэтому я опубликую то, что я сделал здесь. Мой сценарий немного отличается, так как я просто хотел полностью разобрать банку вместе со всеми банками, содержащимися в ней, поэтому я написал следующие функции bash:

function explode {
    local target="$1"
    echo "Exploding $target."
    if [ -f "$target" ] ; then
        explodeFile "$target"
    elif [ -d "$target" ] ; then
        while [ "$(find "$target" -type f -regextype posix-egrep -iregex ".*\.(zip|jar|ear|war|sar)")" != "" ] ; do
            find "$target" -type f -regextype posix-egrep -iregex ".*\.(zip|jar|ear|war|sar)" -exec bash -c 'source "<file-where-this-function-is-stored>" ; explode "{}"' \;
        done
    else
        echo "Could not find $target."
    fi
}

function explodeFile {
    local target="$1"
    echo "Exploding file $target."
    mv "$target" "$target.tmp"
    unzip -q "$target.tmp" -d "$target"
    rm "$target.tmp"
}

Обратите внимание на <file-where-this-function-is-stored>, который необходим, если вы сохраняете это в файле, который не читается для неинтерактивной оболочки, как это случилось у меня. Если вы храните функции в файле, загруженном в неинтерактивные оболочки (например, .bashrc, я полагаю), вы можете отбросить весь оператор source. Надеюсь, это кому-нибудь поможет.

Небольшое предупреждение - explodeFile также удаляет заархивированный файл, вы, конечно, можете изменить это, закомментировав последнюю строку.

0 голосов
/ 11 апреля 2009

Если вы используете cygwin, синтаксис немного отличается для команды basename.

find . -name "*.zip" | while read filename; do unzip -o -d "`basename "$filename" .zip`" "$filename"; done;
...