【问题标题】:How to rename files in zip archive without extracting and recompressing them?如何重命名 zip 存档中的文件而不提取和重新压缩它们?
【发布时间】:2015-12-26 02:23:07
【问题描述】:

我需要将 zip 文件中的所有文件从 AAAAA-filename.txt 重命名为 BBBBB-filename.txt,我想知道是否可以自动执行此任务,而无需提取所有文件、重命名然后再次压缩。一次解压一个,重命名并再次压缩是可以接受的。

我现在拥有的是:

for file in *.zip
do
    unzip $file
    rename_txt_files.sh
    zip *.txt $file
done;

但我不知道是否有更好的版本,我不必使用所有额外的磁盘空间。

【问题讨论】:

  • 使用zipnote的重命名功能? computerhope.com/unix/zipnote.htm
  • 我没有权限在我需要执行此操作的机器上安装软件@bishop。
  • 好吧,在旧名称和新名称之间创建一个映射,然后遍历该映射。如果存档中存在旧名称,请将其从存档中提取到磁盘,在磁盘上重命名,将新重命名的名称添加到存档中。如果一切成功,请从存档中删除旧名称。可以在 bash 中完成,但我发现 bash 数组很麻烦,所以我可能会使用 P* 语言。
  • 请展示您的编码工作。
  • @Cyrus 我目前有一个可行的解决方案,但我想知道这是否可以在不提取所有文件的情况下完成。

标签: linux bash


【解决方案1】:

计划

  • 用字符串查找文件名的偏移量
  • 使用 dd 覆盖新名称(注意仅适用于相同的文件名长度)。否则还必须找到并覆盖 文件名长度字段..

在尝试之前备份您的压缩文件

zip_rename.sh

#!/bin/bash

strings -t d test.zip | \
grep '^\s\+[[:digit:]]\+\sAAAAA-\w\+\.txt' | \
sed 's/^\s\+\([[:digit:]]\+\)\s\(AAAAA\)\(-\w\+\.txt\).*$/\1 \2\3 BBBBB\3/g' | \
while read -a line; do
  line_nbr=${line[0]};
  fname=${line[1]};
  new_name=${line[2]};
  len=${#fname};
#  printf "line: "$line_nbr"\nfile: "$fname"\nnew_name: "$new_name"\nlen: "$len"\n";
  dd if=<(printf $new_name"\n") of=test.zip bs=1 seek=$line_nbr count=$len conv=notrunc  
done;

输出

$ ls
AAAAA-apple.txt  AAAAA-orange.txt  zip_rename.sh
$ zip test.zip AAAAA-apple.txt AAAAA-orange.txt 
  adding: AAAAA-apple.txt (stored 0%)
  adding: AAAAA-orange.txt (stored 0%)
$ ls
AAAAA-apple.txt  AAAAA-orange.txt  test.zip  zip_rename.sh
$ ./zip_rename.sh 
15+0 records in
15+0 records out
15 bytes (15 B) copied, 0.000107971 s, 139 kB/s
16+0 records in
16+0 records out
16 bytes (16 B) copied, 0.000109581 s, 146 kB/s
15+0 records in
15+0 records out
15 bytes (15 B) copied, 0.000150529 s, 99.6 kB/s
16+0 records in
16+0 records out
16 bytes (16 B) copied, 0.000101685 s, 157 kB/s
$ unzip test.zip 
Archive:  test.zip
 extracting: BBBBB-apple.txt         
 extracting: BBBBB-orange.txt        
$ ls
AAAAA-apple.txt   BBBBB-apple.txt   test.zip
AAAAA-orange.txt  BBBBB-orange.txt  zip_rename.sh
$ diff -qs AAAAA-apple.txt BBBBB-apple.txt 
Files AAAAA-apple.txt and BBBBB-apple.txt are identical
$ diff -qs AAAAA-orange.txt BBBBB-orange.txt 
Files AAAAA-orange.txt and BBBBB-orange.txt are identical

【讨论】:

  • WARINIG 副作用:如果文件未压缩存储,它还可以更新 cmets 和文件内容。
【解决方案2】:

我们用python来

  1. 解压xyz.zip文件到xyz_renamed/文件夹
  2. renameFile 函数中重命名所需文件(我们只重命名 jpeg 和 png 图像)(我们将名称更改为更低)
  3. 将重命名的内容再次打包到xyz_renamed.zip
  4. 删除xyz_renamed/文件夹和xyz.zip文件
  5. xyz_renamed.zip 重命名为xyz.zip
import os
from zipfile import ZipFile
import shutil

# rename the individual file and return the absolute path of the renamed file
def renameFile(root, fileName):
    toks = fileName.split('.')
    newName = toks[0].lower()+'.'+toks[1]
    renamedPath =  os.path.join(root,newName)
    os.rename(os.path.join(root,fileName),renamedPath)
    return renamedPath

def renameFilesInZip(filename):
    # Create <filename_without_extension>_renamed folder to extract contents into
    head, tail = os.path.split(filename)
    newFolder = tail.split('.')[0]
    newFolder += '_renamed'
    newpath = os.path.join(head, newFolder)
    if(not os.path.exists(newpath)):
        os.mkdir(newpath)

    # extracting contents into newly created folder
    print("Extracting files\n")
    try:
        with ZipFile(filename, 'r', allowZip64=True) as zip_ref:
            zip_ref.extractall(newpath)
        zip_ref.close()
    except ResponseError as err:
        print(err)
        return -1

    # track the files that need to be repackaged in the zip with renamed files
    filesToPackage = []

    for r, d, f in os.walk(newpath):
        for item in f:
            # filter the file types that need to be renamed
            if item.lower().endswith(('.jpg', '.png')):
                # renaming file
                renamedPath = renameFile(r, item)
                filesToPackage.append(renamedPath)
            else:
                filesToPackage.append(os.path.join(r,item))


    # creating new zip file
    print("Writing renamed file\n")

    zipObj = ZipFile(os.path.join(head, newFolder)+'.zip', 'w', allowZip64 = True)
    for file in filesToPackage:
        zipObj.write(file, os.path.relpath(file, newpath))
    zipObj.close()

    # removing extracted contents and origianl zip file
    print('Cleaning up \n')
    shutil.rmtree(newpath)
    os.remove(filename)

    # renaming zipfile with renamed contents to original zipfile name
    os.rename(os.path.join(head, newFolder)+'.zip', filename)

使用以下代码调用多个zip文件的重命名函数


if __name__ == '__main__':

    # zipfiles with contents to be renamed
    zipFiles = ['/usr/app/data/mydata.zip', '/usr/app/data/mydata2.zip']
    
    # do the renaming for all files one by one
    for _zip in zipFiles:
        renameFilesInZip(_zip)

【讨论】:

    猜你喜欢
    • 2019-02-25
    • 1970-01-01
    • 2023-02-05
    • 2021-12-03
    • 2013-03-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2021-06-23
    相关资源
    最近更新 更多