จะเปลี่ยนชื่อด้วยคำนำหน้า / คำต่อท้ายได้อย่างไร?


98

ฉันจะmv original.filename new.original.filenameไม่พิมพ์ชื่อไฟล์เดิมได้อย่างไร

ฉันจะจินตนาการว่าสามารถทำอะไรบางอย่างเช่นmv -p=new. original.filenameหรือบางทีmv original.filename new.~หรืออะไรก็ได้ - แต่ฉันไม่เห็นอะไรแบบนี้หลังจากดูที่man mv/ info mvหน้า

แน่นอนฉันสามารถเขียนเชลล์สคริปต์เพื่อทำสิ่งนี้ได้ แต่ไม่มีคำสั่ง / แฟล็กที่มีอยู่แล้วใช่หรือไม่

คำตอบ:


125

ในทุบตีและ zsh คุณสามารถทำเช่นนี้กับการขยายตัวรั้ง นี่เป็นเพียงการขยายรายการในวงเล็บปีกกา ตัวอย่างเช่น:

# echo {vanilla,chocolate,strawberry}-ice-cream
vanilla-ice-cream chocolate-ice-cream strawberry-ice-cream

ดังนั้นคุณสามารถเปลี่ยนชื่อของคุณได้ดังนี้:

mv {,new.}original.filename

เมื่อสิ่งนี้ขยายเป็น:

mv original.filename new.original.filename

ขอบคุณนี่ดูเหมือนจะเป็นทางออกที่ดีที่สุด - ใช้ได้กับ ksh ด้วย
Peter Boughton

29
สิ่งนี้ใช้ไม่ได้กับไวลด์การ์ด นั่นคือในไดเร็กทอรีที่มีชื่อไฟล์ a, b, c โดยใช้ "echo {, new.} *" ให้รายการ "abc new. *" (เนื่องจากการสร้างชื่อไฟล์ผ่านการขยายวงเล็บปีกกาเกิดขึ้นก่อนที่จะขยายอักขระไวด์การ์ด - อย่างน้อยที่สุด โดยการอนุมาน).
Jonathan Leffler

ตั้งแต่เดือนที่แล้วฉันกำลังมองหาวิธีแก้ปัญหาดังกล่าวโดยใช้ฟังก์ชันเชลล์พื้นฐานเท่านั้น ... ฉันจะไม่คิดอะไรง่ายๆขนาดนั้นขอบคุณมาก
flonk

ขอบคุณสำหรับการอ้างอิงถึง Brace Expansion :-) ตรงกับสิ่งที่ฉันกำลังมองหา!
พิมหมอกบรูค

1
ฉันชอบตัวอย่าง ตอนนี้ฉันรู้สึกอยากไปบาสกิ้นร็อบบินส์ในช่วงพักเที่ยง
Sridhar Sarnobat

158

คุณสามารถใช้rename(1)คำสั่ง:

rename 's/(.*)$/new.$1/' original.filename

แก้ไข:หากrenameไม่พร้อมใช้งานและคุณต้องเปลี่ยนชื่อไฟล์มากกว่าหนึ่งไฟล์เชลล์สคริปต์อาจสั้นและเรียบง่ายสำหรับสิ่งนี้ ตัวอย่างเช่นในการเปลี่ยนชื่อทั้งหมด*.jpgเป็นprefix_*.jpgในไดเร็กทอรีปัจจุบัน:

for filename in *.jpg; do mv "$filename" "prefix_$filename"; done;

ติดตั้งกราไฟท์ในสุดสัปดาห์นี้และมีขั้นตอนที่เป็นพื้นฐาน ("คัดลอกไฟล์ * .example ทั้งหมดลงในโฟลเดอร์ conf โดยลบคำต่อท้ายตัวอย่าง") ขี้เกียจจริง ๆ ในส่วนของพวกเขา แต่rename 's/(.*).example/$1/' *.exampleช่วยฉันจากความเบื่อหน่าย
คอนราดดีน

2
นี่คือคำตอบที่ดีที่สุดจนถึงตอนนี้
lifeisfoo

1
ฉันไม่คิดว่าคุณต้องการสิ่งแรก$ในสถานการณ์นี้ เพียงแค่'s/(.*)/new.$1/'
wisbucky

1
renameคำสั่งไม่ทำงานบนเครื่อง RHEL 6 ของฉัน คำสั่งนี้มีเวอร์ชันต่างๆหรือไม่? for ... doneทำงานให้ฉัน
robsch

25

คุณสามารถเปลี่ยนชื่อไฟล์หลายไฟล์ที่เข้ากันได้กับ unix (โดยใช้สัญลักษณ์แทน) โดยการสร้าง for loop:

for file in *; do
  mv $file new.${file%%}
done

7
ความแตกต่างระหว่างสิ่งนี้กับสิ่งที่ Simon Lehmann ตอบไปแล้วคืออะไร?
Peter Boughton

4
ใช้งานได้ดี โด${file%%}ทำอะไร? ดูเหมือนจะคล้ายกับแค่ $ {file}
Sidmitra

ฉันสงสัยเกี่ยวกับ $ {file %%} เช่นกัน
Lucas Bustamante

10

ฉันเคยเห็นคนพูดถึงrenameคำสั่ง แต่ไม่สามารถใช้งานได้เป็นประจำในระบบ Unix (ซึ่งตรงข้ามกับระบบ Linux พูดหรือ Cygwin ซึ่งทั้งสองอย่างนี้การเปลี่ยนชื่อเป็นไฟล์ปฏิบัติการแทนสคริปต์) เวอร์ชันrenameนั้นมีฟังก์ชันที่ค่อนข้าง จำกัด :

rename from to file ...

มันแทนที่จากส่วนหนึ่งของชื่อไฟล์ด้วยtoและตัวอย่างที่ระบุใน man page คือ:

rename foo foo0 foo? foo??

สิ่งนี้เปลี่ยนชื่อ foo1 เป็น foo01 และ foo10 เป็น foo010 เป็นต้น

ฉันใช้สคริปต์ Perl ที่เรียกว่าrenameซึ่งเดิมฉันขุดมาจากหนังสือ Camel ฉบับพิมพ์ครั้งแรกประมาณปี 2535 แล้วขยายเพื่อเปลี่ยนชื่อไฟล์

#!/bin/perl -w
#
# @(#)$Id: rename.pl,v 1.7 2008/02/16 07:53:08 jleffler Exp $
#
# Rename files using a Perl substitute or transliterate command

use strict;
use Getopt::Std;

my(%opts);
my($usage) = "Usage: $0 [-fnxV] perlexpr [filenames]\n";
my($force) = 0;
my($noexc) = 0;
my($trace) = 0;

die $usage unless getopts('fnxV', \%opts);

if ($opts{V})
{
    printf "%s\n", q'RENAME Version $Revision: 1.7 $ ($Date: 2008/02/16 07:53:08 $)';
    exit 0;
}
$force = 1 if ($opts{f});
$noexc = 1 if ($opts{n});
$trace = 1 if ($opts{x});

my($op) = shift;
die $usage unless defined $op;

if (!@ARGV) {
    @ARGV = <STDIN>;
    chop(@ARGV);
}

for (@ARGV)
{
    if (-e $_ || -l $_)
    {
        my($was) = $_;
        eval $op;
        die $@ if $@;
        next if ($was eq $_);
        if ($force == 0 && -f $_)
        {
            print STDERR "rename failed: $was - $_ exists\n";
        }
        else
        {
            print "+ $was --> $_\n" if $trace;
            print STDERR "rename failed: $was - $!\n"
                unless ($noexc || rename($was, $_));
        }
    }
    else
    {
        print STDERR "$_ - $!\n";
    }
}

สิ่งนี้ช่วยให้คุณสามารถเขียนคำสั่ง Perl แทนหรือคำสั่งทับศัพท์เพื่อแม็พชื่อไฟล์ ในตัวอย่างเฉพาะที่ร้องขอคุณจะใช้:

rename 's/^/new./' original.filename

ขอบคุณถ้าฉันสามารถยอมรับคำตอบที่สองได้ก็จะเป็นเช่นนั้น ฉันต้องการคำสั่งมาตรฐานเป็นสิ่งที่ใช้งานได้กับทุกสิ่งที่ฉันอาจใช้ แต่ฉันสามารถใส่สคริปต์นี้ลงในเครื่องที่ฉันใช้บ่อยและมันจะยังมีประโยชน์
Peter Boughton

เจมส์วงศ์ตั้งข้อสังเกตว่าสำหรับ Arch Linux คุณก็สามารถดาวน์โหลดและติดตั้งสคริปต์นี้ผ่านการใช้:pacman pacman -Sy perl-renameจากนั้นเพิ่มนามแฝงใน bashrc ของคุณ:alias rename='perl-rename'
Jonathan Leffler

จะเกิดอะไรขึ้นถ้า "จาก" หรือ "ถึง" เป็นสตริงว่างกล่าวคือคุณต้องการต่อท้ายชื่อไฟล์แต่ละชื่อหรือลบบางอย่างออกทั้งหมด
Troyseph

1
@Troyseph - เพื่อต่อท้าย'.htmlชื่อไฟล์ให้ใช้'s/$/.html/'- เพื่อลบออกtmpจากทุกที่ในชื่อไฟล์ให้ใช้'s/tmp//g'(ค้นหาtmpและแทนที่โดยไม่มีอะไรเลย)
Jonathan Leffler

4

วิธีที่ง่ายที่สุดในการเปลี่ยนชื่อไฟล์จำนวนมากในไดเรกทอรีคือ:

ls | xargs -I fileName mv fileName fileName.suffix

2

หากเปิดให้มีการแก้ไขคุณสามารถใช้คำต่อท้ายแทนคำนำหน้าได้ จากนั้นคุณสามารถใช้การเติมแท็บเพื่อรับชื่อไฟล์ต้นฉบับและเพิ่มคำต่อท้าย

มิฉะนั้นนี่ไม่ใช่สิ่งที่รองรับโดยคำสั่ง mv เชลล์สคริปต์ธรรมดาสามารถรับมือได้


บ่อยครั้งที่ฉันต้องการทั้งสองอย่าง แต่ส่วนใหญ่เป็นคำนำหน้าอย่างที่คุณพูดการเติมแท็บสามารถจัดการส่วนต่อท้ายได้
Peter Boughton

2

ฉันรู้ว่ามีคำตอบที่ดีที่นี่ แต่ฉันไม่พบการอ้างอิงถึงการจัดการนามสกุลไฟล์เมื่อเพิ่มคำต่อท้าย

ฉันต้องการเพิ่มคำต่อท้าย '_en' ให้กับwavไฟล์ทั้งหมดในโฟลเดอร์ก่อนนามสกุลไฟล์

ความมหัศจรรย์อยู่ที่นี่: %.*

for filename in *.wav; do mv $filename ${filename%.*}_en.wav; done;

หากคุณต้องการที่จะจัดการกับนามสกุลไฟล์ที่แตกต่างกันตรวจนี้คำตอบ ใช้งานง่ายน้อยกว่าเล็กน้อย


1

ในกรณีของฉันฉันมีกลุ่มไฟล์ที่ต้องเปลี่ยนชื่อก่อนจึงจะสามารถใช้งานได้ แต่ละไฟล์มีบทบาทของตัวเองในกลุ่มและมีรูปแบบของตัวเอง

ด้วยเหตุนี้ฉันจึงมีรายการคำสั่งเปลี่ยนชื่อดังนี้:

f=`ls *canctn[0-9]*`   ;  mv $f  CNLC.$f
f=`ls *acustb[0-9]*`   ;  mv $f  CATB.$f
f=`ls *accusgtb[0-9]*` ;  mv $f  CATB.$f
f=`ls *acus[0-9]*`     ;  mv $f  CAUS.$f

ลองสิ่งนี้ด้วย:

f=MyFileName; mv $f {pref1,pref2}$f{suf1,suf2}

สิ่งนี้จะสร้างชุดค่าผสมทั้งหมดที่มีคำนำหน้าและคำต่อท้าย:

pref1.MyFileName.suf1
...
pref2.MyFileName.suf2

อีกวิธีหนึ่งในการแก้ปัญหาเดียวกันคือการสร้างอาร์เรย์การแมปและเพิ่มคำนำหน้า corespondent สำหรับไฟล์แต่ละประเภทดังที่แสดงด้านล่าง:

#!/bin/bash
unset masks
typeset -A masks
masks[ip[0-9]]=ip
masks[iaf_usg[0-9]]=ip_usg
masks[ipusg[0-9]]=ip_usg
...
for fileMask in ${!masks[*]}; 
do  
registryEntry="${masks[$fileMask]}";
fileName=*${fileMask}*
[ -e ${fileName} ] &&  mv ${fileName} ${registryEntry}.${fileName}  
done

1

สคริปต์ทุบตีเปลี่ยนชื่อไฟล์จำนวนมาก

#!/bin/bash
# USAGE: cd FILESDIRECTORY; RENAMERFILEPATH/MultipleFileRenamer.sh FILENAMEPREFIX INITNUMBER
# USAGE EXAMPLE: cd PHOTOS; /home/Desktop/MultipleFileRenamer.sh 2016_
# VERSION: 2016.03.05.
# COPYRIGHT: Harkály Gergő | mangoRDI (https://wwww.mangordi.com/) 

# check isset INITNUMBER argument, if not, set 1 | INITNUMBER is the first number after renaming
if [ -z "$2" ]
    then i=1;
else
    i=$2;
fi

# counts the files to set leading zeros before number | max 1000 files
count=$(ls -l * | wc -l)
if [ $count -lt 10 ]
    then zeros=1;
else
    if [ $count -lt 100 ]
        then zeros=2;
    else
        zeros=3
    fi
fi

# rename script
for file in *
do
    mv $file $1_$(printf %0"$zeros"d.%s ${i%.*} ${file##*.})
    let i="$i+1"
done
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.