Мне нужно переименовать файлы с регулярным выражением i.e.
s/123/onetwothree/g
Я помню, я могу использовать awk и sed с регулярным выражением, но не мог понять, как свести их вместе для желаемого вывода.
Мне нужно переименовать файлы с регулярным выражением i.e.
s/123/onetwothree/g
Я помню, я могу использовать awk и sed с регулярным выражением, но не мог понять, как свести их вместе для желаемого вывода.
Эффективным способом выполнения операции переименования является создание команд переименования в конвейере sed и их передача в оболочку.
ls |
sed -n 's/\(.*\)\(123\)\(.*\)/mv "\1\2\3" "\1onetwothree\2"/p' |
sh
Вы можете установить служебную программу переименования perl:
brew install rename
и просто используйте его как:
rename 's/123/onetwothree/g' *
если вы хотите протестировать свое регулярное выражение без переименования каких-либо файлов, просто добавьте -n switch
files = "*"
for f in $files; do
newname='echo "$f" | sed 's/123/onetwothree/g''
mv "$f" "$newname"
done
Я использую дружественный рекурсивный метод переименования имен файлов regex, который по умолчанию только эмулирует замену и показывает, какие будут в результате имена файлов.
Используйте -w
для фактической записи изменений, когда вы удовлетворены результатом пробного запуска, -s
для подавления отображения несоответствующих файлов; -h
или --help
будут показывать заметки об использовании.
Простейшее использование:
# replace all occurences of 'foo' with 'bar'
# "foo-foo.txt" >> "bar-bar.txt"
ren.py . 'foo' 'bar' -s
# only replace 'foo' at the beginning of the filename
# "foo-foo.txt" >> "bar-foo.txt"
ren.py . '^foo' 'bar' -s
Также поддерживаются соответствующие группы (например, \1
, \2
и т.д.):
# rename "spam.txt" to "spam-spam-spam.py"
ren.py . '(.+)\.txt' '\1-\1-\1.py' -s
# rename "12-lovely-spam.txt" to "lovely-spam-12.txt"
# (assuming two digits at the beginning and a 3 character extension
ren.py . '^(\d{2})-(.+)\.(.{3})' '\2-\1.\3' -s
ПРИМЕЧАНИЕ. Не забудьте добавить
-w
, когда вы проверили результаты и хотите действительно внести изменения.
Работает как с Python 2.x, так и с Python 3.x.
#!/usr/bin/python
# -*- coding: utf-8 -*-
from __future__ import print_function
import argparse
import os
import fnmatch
import sys
import shutil
import re
def rename_files(args):
pattern_old = re.compile(args.search_for)
for path, dirs, files in os.walk(os.path.abspath(args.root_folder)):
for filename in fnmatch.filter(files, "*.*"):
if pattern_old.findall(filename):
new_name = pattern_old.sub(args.replace_with, filename)
filepath_old = os.path.join(path, filename)
filepath_new = os.path.join(path, new_name)
if not new_name:
print('Replacement regex {} returns empty value! Skipping'.format(args.replace_with))
continue
print(new_name)
if args.write_changes:
shutil.move(filepath_old, filepath_new)
else:
if not args.suppress_non_matching:
print('Name [{}] does not match search regex [{}]'.format(filename, args.search_for))
if __name__ == '__main__':
parser = argparse.ArgumentParser(description='Recursive file name renaming with regex support')
parser.add_argument('root_folder',
help='Top folder for the replacement operation',
nargs='?',
action='store',
default='.')
parser.add_argument('search_for',
help='string to search for',
action='store')
parser.add_argument('replace_with',
help='string to replace with',
action='store')
parser.add_argument('-w', '--write-changes',
action='store_true',
help='Write changes to files (otherwise just simulate the operation)',
default=False)
parser.add_argument('-s', '--suppress-non-matching',
action='store_true',
help='Hide files that do not match',
default=False)
args = parser.parse_args(sys.argv[1:])
print(args)
rename_files(args)
Namechanger супер приятно. Он поддерживает регулярные выражения для поиска и замены: учтите, что я делаю супер сложное переименование со следующим регулярным выражением:
\.sync-conflict-.*\.
это спасатель жизни.