[問題] 請教mac sed裡面\數量的問題

看板RegExp (正規表示式 Regular Expression)作者 (Bad_To_The_Bone)時間11年前 (2014/03/24 15:47), 10年前編輯推噓1(1021)
留言22則, 3人參與, 最新討論串1/1
環境 mac osx 使用 sed filename內容 ----------------- I\ am\ John\ you\ are\ Mary I\\ am\\ John\\ you\\ are\\ Mary ----------------- 請問一下 我用 sed -i "" s/"I\\\ am\\\ John"/Oh/g filename ----------------- 與 sed -i "" s/"I\\\\\ am\\\\\ John"/Oh/g filename --------------------- 都能將第二行取代成 Oh\ you\ are\ Mary 第一行指令標色 sed -i "" s/"I\\\ am\\\ John"/Oh/g filename ----------------- 對應I\ am\ John\ you\ are\ Mary 這個我可以理解 第二行我原本是要拿來取代兩個斜線的 I\\ am\\ John\\ you\\ are\\ Mary的 但是實際上測試要取代兩個斜線的得下指令 sed -i "" s/"I\\\\\\\ am\\\\\\\ John"/Oh/g filename ------------------------ 想請教一下 "\\\\\ " 與 "\\\\\\\ " -------- ---------- 在這邊是怎麼對應的呢 感謝大家 -- ※ 發信站: 批踢踢實業坊(ptt.cc) ◆ From: 210.59.147.226 ※ 編輯: donkeychen 來自: 210.59.147.226 (03/24 15:47) ※ 編輯: donkeychen 來自: 210.59.147.226 (03/24 15:52)

03/24 16:55, , 1F
基本上這些多餘的 \ 都是給 shell 看的, regexp 並沒有對 \
03/24 16:55, 1F

03/24 16:56, , 2F
有特殊的處理; 空白前面加 \ 也是為了告訴 shell "這個空白
03/24 16:56, 2F

03/24 16:56, , 3F
是我的參數的一部份, 不是分隔參數用的"
03/24 16:56, 3F

03/24 16:57, , 4F
regexp 裡的 \ 只有在後面跟著一些特殊字元時 (如小括號)
03/24 16:57, 4F

03/24 16:57, , 5F
才有用, 所以後者其實送給 sed 的是三個 \
03/24 16:57, 5F
配合fjm31714大大說的我才看懂 感謝LPH66大大

03/24 16:58, , 6F
就是為了要把第一個 \ 的這個意義 escape 掉
03/24 16:58, 6F

03/24 16:59, , 7F
(以上是照印象啦, 沒時間查說明書有錯還請指正)
03/24 16:59, 7F

03/24 17:04, , 8F
喔喔 LPH66大的意思是 指令還先被shell拆一層
03/24 17:04, 8F

03/24 17:05, , 9F
哎呀我試到都眼花了 一直在那邊數 PRINT出來 @.@;
03/24 17:05, 9F

03/24 17:17, , 10F
用 sed -i "" 's/\\\\/Oh/g' filename 就簡單的取代兩
03/24 17:17, 10F

03/24 17:21, , 11F
個\ 只是遇到要取代'的就又出現瓶頸><
03/24 17:21, 11F

03/24 17:22, , 12F
如果要取代' 我是用 s/"\'"/Oh/g
03/24 17:22, 12F

03/26 17:40, , 13F
為什麼我看不太懂你的 sed 指令 Orz 好多 "
03/26 17:40, 13F

03/26 17:41, , 14F
字串會先被 shell 拆過沒錯,用其他語言也是這樣
03/26 17:41, 14F

03/26 17:42, , 15F
像是 python 的 re module 有說明 http://goo.gl/2g6TRc
03/26 17:42, 15F

03/26 17:47, , 16F
我是寫像這樣@@ sed -i "s/I\\\ am\\\ John/Oh/" file
03/26 17:47, 16F
這邊我補充一下 -i "" 是說直接取代檔案 不產生備分檔 如果原檔名是 lalala.txt 取代時下指令 -i try regexp lalala.txt 會把還沒改過的東西存到lalala.txttry裡面 與正規表示無關 所以最前面的-i "" 沒有mac的可以不用特別去管他 我把我原文正規的部分用紅線標記 希望對閱讀有幫助^^;

03/26 17:47, , 17F
然後你可以印出取代指令 echo "s/I\\\ am\\\ John/Oh/"
03/26 17:47, 17F
這邊的結果是印出 s/I\\ am\\ John/Oh/ 我猜您的意思是說 因為整個指令會先被shell接收處理後才給sed程式 所以先把"正規表示式" 丟給echo 看shell如何處理"正規表示式" 這邊看起來是\\ 被變成\了

03/26 17:49, , 18F
就知道 regex 長怎樣 另外 "" 裡的空白原本是不用 escape
03/26 17:49, 18F

03/26 17:50, , 19F
可是如果這樣寫 "s/I\\ am\\ John/Oh/" sed 拿到會變
03/26 17:50, 19F

03/26 17:51, , 20F
"s/I\ am\ John/Oh/" 在 regex 裡 \ 用來逃脫東西如果沒
03/26 17:51, 20F

03/26 17:51, , 21F
有這個逃脫他就會找原本那個字元這邊就會是空白
03/26 17:51, 21F
如同大大說明的 我將問題在這邊整理一下 原檔案 ======================================================= I\ am\ John\ you\ are\ Mary I\\ am\\ John\\ you\\ are\\ Mary ======================================================= 以""包裹正規表示式 指令行1個\: sed -i "" s/"I\ am\ John"/Oh/g filename ------------- shell拆過給sed: I\ am\ John 對sed來說\ 沒用 要比對的對象是: I am John 比對結果: no matched ======================================================= 指令行2個\: sed -i "" s/"I\\ am\\ John"/Oh/g filename --------------- shell拆過給sed: I\ am\ John 對sed來說\ 沒用 要比對的對象是: I am John 比對結果: no matched ======================================================= 指令行3個\: sed -i "" s/"I\\\ am\\\ John"/Oh/g filename ----------------- shell拆過給sed: I\\ am\\ John 要比對的對象是: I\ am\ John 比對結果: line1 matched ======================================================= 指令行4個\: sed -i "" s/"I\\\\ am\\\\ John"/Oh/g filename ------------------- shell拆過給sed: I\\ am\\ John 要比對的對象是: I\ am\ John 比對結果: line1 matched ======================================================= 指令行5個\: sed -i "" s/"I\\\\\ am\\\\\ John"/Oh/g filename --------------------- shell拆過給sed: I\\\ am\\\ John 對sed來說\ 沒用 要比對的對象是: I\ am\ John 比對結果: line1 matched ======================================================= 指令行6個\: sed -i "" s/"I\\\\\\ am\\\\\\ John"/Oh/g filename ----------------------- shell拆過給sed: I\\\ am\\\ John 對sed來說\ 沒用 要比對的對象是: I\ am\ John 比對結果: line1 matched ======================================================= 指令行7個\: sed -i "" s/"I\\\\\\\ am\\\\\\\ John"/Oh/g filename ------------------------- shell拆過給sed: I\\\\ am\\\\ John 要比對的對象是: I\\ am\\ John 比對結果: line2 matched ======================================================= 指令行8個\: sed -i "" s/"I\\\\\\\\ am\\\\\\\\ John"/Oh/g filename --------------------------- shell拆過給sed: I\\\\ am\\\\ John 要比對的對象是: I\\ am\\ John 比對結果: line2 matched 整理上面的指令 發現shell會把兩個\\ 拆成一個\ 然後單獨的\後面接著東西也會被當作單獨\來看 (如果單獨的\後面沒接東西 就會變成多行指令當作一行進入下一行等你輸入) 於是在這個情況使用""包裹正規表示式 最正確的指令是 要比對內容一個\ 在指令行雙引號內要有4個\ 所以我原本用5個\ 跟7個\ 都是歪打正著 應該是用4個\ 跟8個\ 此外如果用''來包裹正規表示式的情況如下 ======================================================= 指令行1個\: sed -i "" s/'I\ am\ John'/Oh/g filename ------------- shell拆過給sed: I\ am\ John 對sed來說\ 沒用 要比對的對象是: I am John 比對結果: no matched ======================================================= 指令行2個\: sed -i "" s/'I\\ am\\ John'/Oh/g filename --------------- shell拆過給sed: I\\ am\\ John 要比對的對象是: I\ am\ John 比對結果: line1 matched ======================================================= 指令行3個\: sed -i "" s/'I\\\ am\\\ John'/Oh/g filename ----------------- shell拆過給sed: I\\\ am\\\ John 對sed來說\ 沒用 要比對的對象是: I\ am\ John 比對結果: line1 matched ======================================================= 指令行4個\: sed -i "" s/'I\\\\ am\\\\ John'/Oh/g filename ------------------- shell拆過給sed: I\\\\ am\\\\ John 要比對的對象是: I\\ am\\ John 比對結果: line2 matched

03/26 17:52, , 22F
以上是我的看法,有錯請指教 ><
03/26 17:52, 22F
感謝大大 您說的用echo來看解決我的疑惑 包括了" 與 ' 的問題 處理 " echo "\"" echo '"' 均可 處理 ' echo "'" 才可以 ※ 編輯: donkeychen (210.59.147.226), 04/14/2014 11:00:38
文章代碼(AID): #1JB-Bxs_ (RegExp)
文章代碼(AID): #1JB-Bxs_ (RegExp)