質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

88.37%

バッチファイルを使用して、テキストファイルの文字列置換処理

解決済

回答 3

投稿 編集

  • 評価
  • クリップ 1
  • VIEW 5,803

durasama

score 18

前提・実現したいこと

バッチファイル(.bat)でのfor文の中での文字列置換を行う際
CSVファイルの後ろから16文字取得し、
9文字目と10文字目の間にスペースを入れる処理を行いたいです
置換前) "abcdefghijklmn"
置換後) "abcdefgh ijklmn"

そして可能であれば、以下のnew.csvに書き込むのではなく
old.csvで直接文字列を探して置換できるようにしたいと思っています。

該当のソースコード

@echo off
SETLOCAL enabledelayedexpansion

set filepath=C:\hoge\old.csv

for /f %%d in (%filepath%) do (
 set line=%%d
 set before=!line:~-16!
 set after=!line:~-16,9! !line:~-7!

 echo !line:%before%=%after%! >> new.csv

試したこと

現状、上記コードでバッチを実行するとnew.csvline:= という文字がfor文の回数入るだけでした
遅延環境変数の問題かと思い、
beforeafter%before%じゃなく!before!にしたりも試しましたが
結果はbeforeafterと入力されるだけでした

もう一つ試したこととして片方だけ環境変数では文字列

echo !line:"abcdefghijklmn"=%after%! >> new.csv 


こちらと!after!で試してみましたが
%%だと文字列が消えて!!だとafterという文字列が入力されました

バッチファイル作成は初めてで至らないところが多いですが知恵をお貸しいただければ幸いです。

補足情報(FW/ツールのバージョンなど)

Windows10
PostgreSQL11

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

質問への追記・修正、ベストアンサー選択の依頼

  • m6u

    2019/02/22 17:49

    そこまでかしこまらなくていいよ。(個人的にはこういう案件だとすぐPHPとかPerlとかやり慣れてる言語でやっちゃうので、お力になれず。)

    キャンセル

  • durasama

    2019/02/25 09:55

    質問の追記が遅くなって申し訳ございません。
    現在、`before`と`after`にはスペース無し、スペース有りの入れるべき情報が入っているので
    それをどのようにして`line`の中で`before`を探し出して`after`に置き換えるか、という部分で
    詰まっております。

    キャンセル

  • durasama

    2019/02/25 10:02

    現状、置換の部分を
    echo !line:!!before!!=!!after! >> new.csv
    としているのですが、バッチを実行するとnew.csvに
    "abcdefghijklmn""abcdefgh ijklmn"
    という文字列が表示されるという状態です。

    キャンセル

回答 3

checkベストアンサー

+2

提示されたコードを見て混乱していますが、質問文を正とすると、old.csvのある1行がabcdefghijklmnとして、それに対応するnew.csvの行は`abcdefgh ijklmn'となっていればいいんですよね?

だとすると、バッチ特有の面倒さはいろいろあるものの、難しく考えすぎのような気がします。 9文字より短い場合どうするかとか考えなければ、for文の内側はこれだけで良いです。

rem 加工用に1行取得
set line=%%d
rem 最後の16文字
set line=!line:~-16!

rem 取得した16文字の先頭9文字+空白+続きの7文字
echo !line:~0,8! !line:~9,7!>>new.csv

結果をnew.csvではなく、old.csvへ直接書き戻すのは厳しいので、処理後にリネームまたはコピーするのが落としどころだと思います。エラー処理を考えなければこうでしょうか。

rem 元ファイルを待避
rename old.csv old.bak

rem 新ファイルを元ファイル名にリネーム
rename new.csv old.csv


old.csvが不要ならば、deleteでも良いと思います。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2019/02/25 09:44 編集

    お忙しいところ回答ありがとうございます。
    遅くなってしまい申し訳ございません。
    質問の説明不足でした。。。

    old.csvにはほかのデータも入っているため上記のアルファベット文字列だけを切り取るのではなく
    全行の最後にアルファベット文字列があるのでそこから9文字目と10文字目の間にスペースを入れたいと思っております。
    なので、アルファベット文字列を抽出して書き出すというよりは
    old.csvの中の行末にあるアルファベット文字列を探して
    スペースを含めた状態の物と文字列置換したいというイメージです。
    説明不足でお手数をお掛けして申し訳ございません。

    キャンセル

  • 2019/02/25 10:46

    >old.csvの中の行末にあるアルファベット文字列を探して
    > スペースを含めた状態の物と文字列置換したいというイメージです。
    これ、具体的に3行ぐらいでいいですので、前と後を提示出来ませんか?

    キャンセル

  • 2019/02/25 10:52

    aki.aki様
    "1","28234235","abcdefghijklmn"
    "1","284532","abcdefghijklmn"
    "1","23456","abcdefghijklmn"
    このようなCSVファイルになっております。
    前にあるデータは文字数が変わってき、行末には必ずこのアルファベット文字列が入るため
    後ろから文字数を数える処理にしております。

    キャンセル

  • 2019/02/25 18:18

    durasamaさん

    解決できたようで良かったですね。なるほど、やっとやりたいことがわかりました。私も勉強させて頂きました。ありがとうございました。

    キャンセル

+1

@echo off
rem 9桁目のあとにスペースを入れる。
for /f %%d in (old.csv) do (
    set e=%%d
    set line=%e:~0,9% %e:~9,16%
    echo %line%>>new.csv
)


割と、バッチは好きです。(が、良くわからないところもあるのですが。)
パスを変数にしてらっしゃますが、そこは省略しました。

時間の関係もあり、説明は省略させて下さい。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2019/02/22 20:50

    提示されたコードでは、遅延展開もcallも使っていないですが、行毎の繰り返し処理は正しく動作しますか?全行同じ結果になってしまうのでは。

    バッチファイルは、たいしたことが出来ない中で何とかする感が、大昔のプログラム環境っぽくもあり、嫌いではないどころか好きな方だと思いますが、積極的に使いたいかというと、とってつけた感のある仕様がなんとも…。

    キャンセル

0

皆様、お忙しいところ時間を割いて回答を頂き、誠にありがとうございます。

たったいま、この質問内容に対する問題が解決しましたので記述させていただきます。

for /f %%d in (%filepath%) do (
 set line=%%d
 set before=!line:~-16!
 set after=!line:~-16,9! !line:~-7!

 call set line1=%%line:!before!=!after!%%

 echo !line1!>>new.csv


と、いった風に修正することにより行おうとしていた結果がnew.csvに記入され
ファイルが生成されました。

色々な知恵をお貸しいただいてありがとうございました。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 88.37%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る