![ファイル内のテキストを検索し、バッチでファイル名を変更する](https://rvso.com/image/1638143/%E3%83%95%E3%82%A1%E3%82%A4%E3%83%AB%E5%86%85%E3%81%AE%E3%83%86%E3%82%AD%E3%82%B9%E3%83%88%E3%82%92%E6%A4%9C%E7%B4%A2%E3%81%97%E3%80%81%E3%83%90%E3%83%83%E3%83%81%E3%81%A7%E3%83%95%E3%82%A1%E3%82%A4%E3%83%AB%E5%90%8D%E3%82%92%E5%A4%89%E6%9B%B4%E3%81%99%E3%82%8B.png)
フォルダー内の各ファイルで特定のテキストを検索し、そのテキストの後の内容を取得して、それをファイルの名前として設定し、これらすべてをループ内で実行して、複数のファイルに対して同じことを実行するスクリプトを作成するときに問題が発生しています。
たとえば、file1、file2、file3を含むフォルダがあります
content in file1: sw_part = "ABC-DEF.GHJ";
content in file2: sw_part = "ABC-123.XYZ";
content in file3: sw_part = "ABCDE-FGHJ.KLMN";
結果は次のようになります。
file1 renamed to ABC-DEF.GHJ
file2 renamed to ABC-123.XYZ
file3 renamed to ABCDE-FGHJ.KLMN
私が作ったのはこれです:
for /F "tokens=2 delims==" %%a in ('findstr /I "sw_part" file1') do set "swpart=%%a"
echo sw part is: %swpart%
copy file1 "%swpart%"
これはうまく動作しますが、特定のファイルに対してのみ動作します(ループでは動作しません)。また、ファイルの名前が変更されます。;
最後に。何か助けはありますか?
バッチで実行する必要があります。PowerShell でも問題ありません。
ありがとう
答え1
次のスクリプトは必要なことを実行します。
cd C:\temp\swpart
$files = [System.Collections.ArrayList]@()
select-string -Path "*.*" -Pattern "sw_part" | ForEach-Object {
$old_filename = $_.Filename
$new_filename = $_.line -replace "^.*sw_part = ""","" -replace """.*$",""
# put everything in array, so the files get closed and can be renamed later.
$null = $files.Add(@{"old"=$old_filename;"new"=$new_filename})
}
$files |ForEach-Object{
$old_filename = $_.old
$new_filename = $_.new
rename-item $old_filename $new_filename -Force
}
最初の行を変更して、ファイルがある場所のパスに変更するようにしてください。
何らかの理由ですべてのファイルの名前が変更されていない場合は、スクリプトを再度実行してください。その場合、ファイルはまだ使用中でした。
答え2
- PowerShell を使用すると、次の 2 行だけでこれを行うこともできます。
ls 'F:\2020-SU\Q1605996\*.txt'|%{$new=(cat $_.FullName).split('\"*\"')[-2];ren $_.FullName -new $new -force}
- エイリアスを使用しない場合:
Get-ChildItem 'F:\2020-SU\Q1605996\*.txt' | ForEach {
$new=(Get-Content $_.FullName).split('\"*\"')[-2]
Rename-Item $_.FullName -new $new -force
}
- cmd を使って 1 行でこれを行うこともできます:
コマンドラインの場合:
for /f tokens^=1-3delims^=^:^"^" %a in ('findstr . *.txt')do @ren "%~fa" "%~c"
bat/cmd ファイル内:
@echo off
cd /d "C:\Path\To\Your\Folder"
for /f tokens^=1-3delims^=:^"^" %%a in ('findstr . *.txt')do @ren "%%~fa" "%%~c"
Findstr
コマンドと出力
>F:\Q1605996>findstr . *.txt
file1.txt:sw_part = "ABC-DEF.GHJ";
file2.txt:sw_part = "ABC-123.XYZ";
file3.txt:sw_part = "ABCDE-FGHJ.KLMN";
file4.txt:sw_part = "ABCDE-FGHJ.KXMN";more test/strings
file5.txt:this is a test; sw_part = "fea.geow.eaf.gae"; this is another test;
For
ループコマンドメカニック
set yours delimiters : + " + " ==>> delims^=:^"^"
set yours tokens "1" - "3" ==>> tokens^=1-3
: = ;
token = 1:token =2= token = 3;
file1.txt:sw_part = "ABC-DEF.GHJ" ;
file2.txt:sw_part = "ABC-123.XYZ" ;
file3.txt:sw_part = "ABCDE-FGHJ.KLMN";
file4.txt:sw_part = "ABCDE-FGHJ.KXMN";more test/strings
| | | |
ren "%f~a" "%~c"
ren "F:\Q1605996\file1.txt" "ABC-DEF.GHJ"
ren "F:\Q1605996\file2.txt" "ABC-123.XYZ"
ren "F:\Q1605996\file3.txt" "ABCDE-FGHJ.KLMN"
ren "F:\Q1605996\file4.txt" "ABCDE-FGHJ.KXMN"
rem :: The same in file5.txt
file5.txt:this is a test;sw_part = "fea.geow.eaf.gae"; this is another test;
| | | |
ren "%f~a" "%~c"
ren "F:\Q1605996\file5.txt" "fea.geow.eaf.gae"
追加リソース PowerShell:
Split()
Replace()
Substring()
Rename-Item
|ren
ForEach-Object
|%
|?
Get-Content
|type
|cat
Get-ChildItem
|ls
|dir
追加リソース cmd/batch: