什么Bat 這是dos下面的批處理文件,可以批量執(zhí)行Dos命令。
下面我們來講一下bat文件中的For 語句的用法:
我們先來看下for的幫助在 cmd下輸入 for /? 就可以打印出來
對一組文件中的每一個文件執(zhí)行某個特定命令。
FOR %variable IN (set) DO command [command-parameters]
%variable 指定一個單一字母可替換的參數(shù)。
(set) 指定一個或一組文件。可以使用通配符。
command 指定對每個文件執(zhí)行的命令。
command-parameters
為特定命令指定參數(shù)或命令行開關(guān)。
在批處理程序中使用 FOR 命令時,指定變量請使用 %%variable
而不要用 %variable。變量名稱是區(qū)分大小寫的,所以 %i 不同于 %I.
如果啟用命令擴(kuò)展,則會支持下列 FOR 命令的其他格式:
FOR /D %variable IN (set) DO command [command-parameters]
如果集中包含通配符,則指定與目錄名匹配,而不與文件名匹配。
FOR /R [[drive:]path] %variable IN (set) DO command [command-parameters]
檢查以 [drive:]path 為根的目錄樹,指向每個目錄中的 FOR 語句。
如果在 /R 后沒有指定目錄規(guī)范,則使用當(dāng)前目錄。如果集僅為一個單點(.)字符,
則枚舉該目錄樹。
FOR /L %variable IN (start,step,end) DO command [command-parameters]
該集表示以增量形式從開始到結(jié)束的一個數(shù)字序列。因此,(1,1,5)將產(chǎn)生序列
1 2 3 4 5,(5,-1,1)將產(chǎn)生序列(5 4 3 2 1)
FOR /F ["options"] %variable IN (file-set) DO command [command-parameters]
FOR /F ["options"] %variable IN ("string") DO command [command-parameters]
FOR /F ["options"] %variable IN ('command') DO command [command-parameters]
或者,如果有 usebackq 選項:
FOR /F ["options"] %variable IN (file-set) DO command [command-parameters]
FOR /F ["options"] %variable IN ("string") DO command [command-parameters]
FOR /F ["options"] %variable IN ('command') DO command [command-parameters]
fileset 為一個或多個文件名。繼續(xù)到 fileset 中的下一個文件之前,
每份文件都被打開、讀取并經(jīng)過處理。處理包括讀取文件,將其分成一行行的文字,
然后將每行解析成零或更多的符號。然后用已找到的符號字符串變量值調(diào)用 For 循環(huán)。
以默認(rèn)方式,/F 通過每個文件的每一行中分開的第一個空白符號。跳過空白行。
你可通過指定可選 "options" 參數(shù)替代默認(rèn)解析操作。這個帶引號的字符串包括一個
或多個指定不同解析選項的關(guān)鍵字。這些關(guān)鍵字為:
eol=c - 指一個行注釋字符的結(jié)尾(就一個)
skip=n - 指在文件開始時忽略的行數(shù)。
delims=xxx - 指分隔符集。這個替換了空格和制表符的
默認(rèn)分隔符集。
tokens=x,y,m-n - 指每行的哪一個符號被傳遞到每個迭代
的 for 本身。這會導(dǎo)致額外變量名稱的分配。m-n
格式為一個范圍。通過 nth 符號指定 mth。如果
符號字符串中的最后一個字符星號,
那么額外的變量將在最后一個符號解析之后
分配并接受行的保留文本。
usebackq - 指定新語法已在下類情況中使用:
在作為命令執(zhí)行一個后引號的字符串并且一個單
引號字符為文字字符串命令并允許在 file-set
中使用雙引號擴(kuò)起文件名稱。
某些范例可能有助:
FOR /F "eol=; tokens=2,3* delims=, " %i in (myfile.txt) do @echo %i %j %k
會分析 myfile.txt 中的每一行,忽略以分號打頭的那些行,將
每行中的第二個和第三個符號傳遞給 for 函數(shù)體,用逗號和/或
空格分隔符號。請注意,此 for 函數(shù)體的語句引用 %i 來
獲得第二個符號,引用 %j 來獲得第三個符號,引用 %k
來獲得第三個符號后的所有剩余符號。對于帶有空格的文件
名,你需要用雙引號將文件名括起來。為了用這種方式來使
用雙引號,還需要使用 usebackq 選項,否則,雙引號會
被理解成是用作定義某個要分析的字符串的。
%i 在 for 語句中顯式聲明,%j 和 %k 是通過
tokens= 選項隱式聲明的?梢酝ㄟ^ tokens= 一行
指定最多 26 個符號,只要不試圖聲明一個高于字母 "z" 或
"Z" 的變量。請記住,F(xiàn)OR 變量是單一字母、分大小寫和全局的變量;
而且,不能同時使用超過 52 個。
還可以在相鄰字符串上使用 FOR /F 分析邏輯,方法是,
用單引號將括號之間的 file-set 括起來。這樣,該字符
串會被當(dāng)作一個文件中的一個單一輸入行進(jìn)行解析。
最后,可以用 FOR /F 命令來分析命令的輸出。方法是,將
括號之間的 file-set 變成一個反括字符串。該字符串會
被當(dāng)作命令行,傳遞到一個子 CMD.EXE,其輸出會被捕獲到
內(nèi)存中,并被當(dāng)作文件分析。如以下例子所示:
FOR /F "usebackq delims==" %i IN (`set`) DO @echo %i
會枚舉當(dāng)前環(huán)境中的環(huán)境變量名稱。
另外,F(xiàn)OR 變量參照的替換已被增強(qiáng)。你現(xiàn)在可以使用下列
選項語法:
%~I - 刪除任何引號("),擴(kuò)展 %I
%~fI - 將 %I 擴(kuò)展到一個完全合格的路徑名
%~dI - 僅將 %I 擴(kuò)展到一個驅(qū)動器號
%~pI - 僅將 %I 擴(kuò)展到一個路徑
%~nI - 僅將 %I 擴(kuò)展到一個文件名
%~xI - 僅將 %I 擴(kuò)展到一個文件擴(kuò)展名
%~sI - 擴(kuò)展的路徑只含有短名
%~aI - 將 %I 擴(kuò)展到文件的文件屬性
%~tI - 將 %I 擴(kuò)展到文件的日期/時間
%~zI - 將 %I 擴(kuò)展到文件的大小
%~$PATH:I - 查找列在路徑環(huán)境變量的目錄,并將 %I 擴(kuò)展
到找到的第一個完全合格的名稱。如果環(huán)境變量名
未被定義,或者沒有找到文件,此組合鍵會擴(kuò)展到
空字符串
可以組合修飾符來得到多重結(jié)果:
%~dpI - 僅將 %I 擴(kuò)展到一個驅(qū)動器號和路徑
%~nxI - 僅將 %I 擴(kuò)展到一個文件名和擴(kuò)展名
%~fsI - 僅將 %I 擴(kuò)展到一個帶有短名的完整路徑名
%~dp$PATH:I - 搜索列在路徑環(huán)境變量的目錄,并將 %I 擴(kuò)展
到找到的第一個驅(qū)動器號和路徑。
%~ftzaI - 將 %I 擴(kuò)展到類似輸出線路的 DIR
在以上例子中,%I 和 PATH 可用其他有效數(shù)值代替。%~ 語法
用一個有效的 FOR 變量名終止。選取類似 %I 的大寫變量名
比較易讀,而且避免與不分大小寫的組合鍵混淆。
上面是官方的解釋下面針對具體的實列來說明
注意 直接在 cmd窗口中運(yùn)行 for 是不行的,你要把文件保存為 .bat 雙擊來執(zhí)行
1.沒有任何參數(shù)的for格式為: for %%i in (set) do command
%%i 為變量set為一個文件或者一組文件,其實就是一個集合?捎猛ㄅ浞热*.txt。
command 是要執(zhí)行的命令。
看個例子:
有個文件夾,有如下文件:
a.txt
b.mp3
c.mp3
c.wma
a.rm
e.bat
你想顯示里面的擴(kuò)展名為mp3的文件,你會用什么命令呢?當(dāng)然是dir了,dir *.mp3。
同樣的,for也可以來實現(xiàn),如下:for %%i in (*.mp3) do echo %%i
這如何理解?for會先從括號里面執(zhí)行,因為括號里面是*.mp3,所以for會先得到當(dāng)前目錄下所有的擴(kuò)展名是mp3的文件,并把它們作為一個集合,而每個文件名就是一個元素,像這樣{b.mp3,c.mp3},然后用%%i依次代替每個元素,之后做do后面的命令。
具體過程如下:
用%%i代替b.mp3,然后執(zhí)行echo %%i,顯示b.mp3
用%%i代替c.mp3,再執(zhí)行echo %%i,顯示c.mp3
沒有元素了,for就停止了。
當(dāng)然,()里面不局限于一個文件或者通配符,可以有多個,比如:for %%i in (*.mp3,*.wma) do echo %%i 也是可以的。
注意:do 后面可以跟任何命令,不要局限于echo,這里只是演示。
2. 含有/L的for格式為:for /L %%i in (start,step,end) do command
這個表示以增量形式從start到end的一個數(shù)字序列,步長是step,就是每次的增量是step
比如:(1,1,5) 將產(chǎn)生序列 1 2 3 4 5,(1,2,9) 將產(chǎn)生1 3 5 7 9序列。
這有什么用呢?
舉個例子,你想同時打開3個記事本程序,可以這樣:for /L %%i in (1,1,3) do notpad.exe
如何理解?
還是從括號先執(zhí)行,產(chǎn)生集合{1,2,3},然后用%%i依次代替集合中的每個元素,之后執(zhí)行do后面的命令。
當(dāng)然這里每次%%i替換集合里面的元素是沒有意義的,因為我們并沒有在do后面的命令中用到%%i,但是你不能阻止%%i代替每個元素。
因為有3個元素,所以do后面的命令執(zhí)行3次就停止了。
再舉一個例子,加入你要顯示1到20之間的奇數(shù),怎么辦,可以如下:for /L %%i in (1,2,20) do echo %%i
同樣是先產(chǎn)生集合{1,3,5,7,9,11,13,15,17,19},然后用%%i依次代替每個元素,之后執(zhí)行do后面的echo %%i
這個例子就體現(xiàn)出變量%%i會始終代替集合中的每個元素。
3.含有/F的for格式:
FOR /F ["options"] %%i IN (file) DO command
FOR /F ["options"] %%i IN ("string") DO command
FOR /F ["options"] %%i IN ('command') DO command
這個可能是最常用的,也是最強(qiáng)的命令,主要用來處理文件和一些命令的輸出結(jié)果。
file代表一個或多個文件
string 代表字符串
command代表命令
["options"] 可選
對于FOR /F %%i IN (file) DO command
file為文件名,按照官方的說法是,for會依次將file中的文件打開,并且在進(jìn)行到下一個文件之前將每個文件讀取到內(nèi)存,按照每一行分成一個一個的元素,忽略空白的行,看個例子。
假如文件a.txt中有如下內(nèi)容:
第1行第1列 第1行第2列 第1行第3列
第2行第1列 第2行第2列 第2行第3列
第3行第1列 第3行第2列 第3行第3列
你想顯示a.txt中的內(nèi)容,會用什么命令呢?當(dāng)然是type,type a.txtfor也可以完成同樣的命令:for /f %%i in (a.txt) do echo %%i
還是先從括號執(zhí)行,因為含有參數(shù)/f,所以for會先打開a.txt,然后讀出a.txt里面的所有內(nèi)容,把它作為一個集合,并且以每一行作為一個元素,所以會產(chǎn)生這樣的集合,
{“第1行第1列 第1行第2列 第1行第3列”, rem 第一個元素“第2行第1列 第2行第2列 第2行第3列”, rem 第二個元素“第3行第1列 第3行第2列 第3行第3列”} rem 第三個元素集合中只有3個元素,同樣用%%i依次代替每個元素,然后執(zhí)行do后面的命令。
具體過程:
用%%i代替“第1行第1列 第1行第2列 第1行第3列”,執(zhí)行do后面的echo %%i,顯示“第1行第1列 第1行第2列 第1行第3列”,
用%%i代替“第2行第1列 第2行第2列 第2行第3列”,執(zhí)行echo %%i,顯示“第2行第1列 第2行第2列 第2行第3列”,
依次,直到每個元素都代替完為止。
為了加強(qiáng)理解/f的作用,請執(zhí)行一下兩個命令,對比即可明白:for /f %%i in (a.txt) do echo %%i rem 這個會顯示a.txt里面的內(nèi)容,因為/f的作用,會讀出a.txt中 的內(nèi)容。for %%i in (a.txt) do echo %%i rem 而這個只會顯示a.txt這個名字,并不會讀取其中的內(nèi)容。通過上面的學(xué)習(xí),我們發(fā)現(xiàn)for /f會默認(rèn)以每一行來作為一個元素,但是如果我們還想把每一行再分解更小的內(nèi)容,該怎么辦呢?不用擔(dān)心,for命令還為我們提供了更詳細(xì)的參數(shù),使我們將每一行分為更小的元素成為可能。
它們就是:delims和tokens
delims 用來告訴for每一行應(yīng)該拿什么作為分隔符,默認(rèn)的分隔符是空格和tab鍵
比如,還是上面的文件,我們執(zhí)行下面的命令:for /f "delims= " %%i in (a.txt) do echo %%i
顯示的結(jié)果是:
第1行第1列
第2行第1列
第3行第1列
為什么是這樣的呢。因為這里有了delims這個參數(shù),=后面有一個空格,意思是再將每個元素以空格分割,默認(rèn)是只取分割之后的第一個元素。
執(zhí)行過程是:
將第一個元素“第1行第1列 第1行第2列 第1行第3列”分成三個元素:“第1行第1列” “第1行第2列” “第1行第3列”,它默認(rèn)只取第一個,即“第1行第1列”,然后執(zhí)行do后面的命令,依次類推。
但是這樣還是有局限的,如果我們想要每一行的第二列元素,那又如何呢?
這時候,tokens跳出來說,我能做到。
它的作用就是當(dāng)你通過delims將每一行分為更小的元素時,由它來控制要取哪一個或哪幾個。
還是上面的例子,執(zhí)行如下命令:for /f "tokens=2 delims= " %%i in (a.txt) do echo %%i
執(zhí)行結(jié)果:
第1行第2列
第2行第2列
第3行第2列
如果要顯示第三列,那就換成tokens=3。
同時tokens支持通配符*,以及限定范圍。
如果要顯示第二列和第三列,則換成tokens=2,3或tokens=2-3,如果還有更多的則為:tokens=2-10之類的。
此時的命令為:for /f "tokens=2,3 delims= " %%i in (a.txt) do echo %%i %%j
怎么多出一個%%j?
這是因為你的tokens后面要取每一行的兩列,用%%i來替換第二列,用%%j來替換第三列。
并且必須是按照英文字母順序排列的,%%j不能換成%%k,因為i后面是j
執(zhí)行結(jié)果為:
第1行第2列 第1行第3列
第2行第2列 第2行第3列
第3行第2列 第3行第3列
對以通配符*,就是把這一行全部或者這一行的剩余部分當(dāng)作一個元素了。
比如:for /f "tokens=* delims= " %%i in (a.txt) do echo %%i
執(zhí)行結(jié)果為:
第1行第1列 第1行第2列 第1行第3列
第2行第1列 第2行第2列 第2行第3列
第3行第1列 第3行第2列 第3行第3列
其實就跟for /f %%i in (a.txt) do echo %%i的執(zhí)行結(jié)果是一樣的。
再如:for /f "tokens=2,* delims= " %%i in (a.txt) do echo %%i %%j
執(zhí)行結(jié)果為:
第1行第2列 第1行第3列
第2行第2列 第2行第3列
第3行第2列 第3行第3列
用%%i代替第二列,用%%j代替剩余的所有
最后還有skip合eol,這倆個簡單,skip就是要忽略文件的前多少行,而eol用來指定當(dāng)一行以什么符號開始時,就忽略它。
比如:for /f "skip=2 tokens=*" %%i in (a.txt) do echo %%i
結(jié)果為:
第3行第1列 第3行第2列 第3行第3列
用skip來告訴for跳過前兩行。
如果不加tokens=*的話,執(zhí)行結(jié)果為:
第3行第1列
不知道怎么回事。
再如,當(dāng)a.txt內(nèi)容變成:
.第1行第1列 第1行第2列 第1行第3列
.第2行第1列 第2行第2列 第2行第3列
第3行第1列 第3行第2列 第3行第3列
執(zhí)行for /f "eol=. tokens=*" %%i in (a.txt) do echo %%i結(jié)果是:
第3行第1列 第3行第2列 第3行第3列
用eol來告訴for忽略以“.”開頭的行。
同樣也必須加tokens=*,否則只會顯示“第3行第1列”,還是不知道怎么回事。
好了,關(guān)于for的/f參數(shù)中的file就先說到這,有時間再寫其他的,如果有什么疑問,可以留言交流。
for 詳細(xì)的講解
for 語句的基本用法 :
最復(fù)雜的for 語句,也有其基本形態(tài),它的模樣是這樣的:
在cmd 窗口中:for %I in (command1) do command2
在批處理文件中:for %%I in (command1) do command2
之所以要區(qū)分 cmd 窗口和批處理文件兩種環(huán)境,是因為在這兩種環(huán)境下,命令語句表現(xiàn)出來的行為雖然基本一樣,但是在細(xì)節(jié)上還是稍有不同,最明顯的一個差異就是:在cmd 窗口中,for 之后的形式變量I 必須使用單百分號引用,即%I;而在批處理文件中,引用形式變量I 必須使用雙百分號,即%%I。為了方便起見,若不是特別強(qiáng)調(diào),以下的講解都以批處理文件環(huán)境為例。
我們先來看一下for 語句的基本要素都有些什么:
1、for、in 和do 是 for 語句的關(guān)鍵字,它們?nèi)齻缺一不可;
2、%%I 是for 語句中對形式變量的引用,就算它在do 后的語句中沒有參與語句的執(zhí)行,也是必須出現(xiàn)的;
3、in 之后,do 之前的括號不能省略;
4、command1 表示字符串或變量,command2 表示字符串、變量或命令語句;
5、for /d 注意:for 后 可以 加 /f /r /l /d 這四個 參數(shù),現(xiàn)在 先 介紹 /d 參數(shù)的 作用,一句話 /d 用于 搜索 目錄 或 文件夾,但 不會搜索文件,也不搜索 子目錄;
現(xiàn)在,你可能已經(jīng)會寫一個簡單的for 語句了,比如:
在 一個 cmd.bat 文件里 寫,寫 如下 兩條 命令:
for /d %%i in (*) do echo %%i
pause
@echo off
for %%I in (bbs.bathome.cn) do echo %%I
pause
保存為批處理文件并執(zhí)行,將會在彈出的批處理窗口中看到這樣的信息:
[result1]
bbs.bathome.cn
請按任意鍵繼續(xù)...
先來分析一下 for 語句的一些注意事項,之后,再讓大家看看更為強(qiáng)大的for 語句實例。
1、for 語句的形式變量 I,可以換成 26 個字母中的任意一個,這些字母會區(qū)分大小寫,也就是說,%%I 和%%i 會被認(rèn)為不是同一個變量;形式變量 I還可以換成其他的字符,但是,為了不與批處理中的%0~%9 這10 個形式變量發(fā)生沖突,請不要隨意把%%I 替換為%%0 ~%%9 中的任意一個;
2、in 和do 之間的command1 表示的字符串或變量可以是一個,也可以是多個,每一個字符串或變量,我們稱之為一個元素,每個元素之間,用空格鍵、跳格鍵、逗號、分號或等號分隔;
3、for 語句依次提取 command1 中的每一個元素,把它的值賦予形式變量I,帶到do 后的command2 中參與命令的執(zhí)行;并且每次只提取一個元素,然后執(zhí)行一次 do 后的命令語句,而無論這個元素是否被帶到 command2 中參與了command2 的運(yùn)行;當(dāng)執(zhí)行完一次do 后的語句之后,再提取command1 中的下一個元素,再執(zhí)行一次 command2,如此循環(huán),直到 command1 中的所有元素都已經(jīng)被提取完畢,該for 語句才宣告執(zhí)行結(jié)束;
其中,第3 點是最為關(guān)鍵的,它描述了for 語句的執(zhí)行過程,是for 語句的精髓所在,大家一定要牢記這一條,才能深刻理解更為復(fù)雜的for 流程。
有了以上的基礎(chǔ),我們再來看一個例子,這個例子修改了code1 的部分內(nèi)容,
結(jié)果將大不一樣:
@echo off
for %%I in (bbs,bathome,cn) do echo %%I
pause
和code1 的執(zhí)行結(jié)果相比,code2 的執(zhí)行結(jié)果發(fā)生了如下變化:
1、顯示結(jié)果分成了3 行(不算最后一行中文提示);
2、每一行都從逗號處被切分;
如果把 bbs.bathome.cn 這個字符串中的點號換為 空格、跳格 或 等號,執(zhí)行結(jié)果將和code2 的執(zhí)行結(jié)果別無二致。
以下這段代碼的 功能是 :檢測當(dāng)前硬盤都有哪些分區(qū)
@echo off
set str=c d e f g h i j k l m n o p q r s t u v w x y z
echo 當(dāng)前硬盤的分區(qū)有:
for %%i in (%str%) do if exist %%i: echo %%i:
pause
這段代碼能檢測硬盤都有哪些分區(qū),包括 U 盤和移動硬盤的分區(qū),但是,
當(dāng)光驅(qū)中有盤的時候,也會被列出來,這是本代碼的一個缺憾,在以后的講解中,
我將向大家講述如何消除這個瑕疵,敬請關(guān)注本系列的后續(xù)章節(jié)。
想知道當(dāng)前目錄下都有哪些文件,請用下面的代碼:
@echo off
for %%i in (*.*) do echo "%%i"
pause
第二句 可以 用 for %%i in (*) do echo "%%i" 代替;
想列出當(dāng)前目錄下所有的文本文件嗎?請用下面的代碼:
@echo off
for %%i in (*.txt) do echo "%%i"
pause
想列出只用兩個字符作為文件名的文本文件嗎?請用下面的代碼:
@echo off
for %%i in (??.txt) do echo "%%i"
pause
===============================
題外話:
1、列出當(dāng)前目錄下各種文件的方法,最簡單的還是用dir 命令,但是,從以上代碼中,各位可以加深對for 語句執(zhí)行流程的理解(用到了通配符*和?);
2、注意:以上代碼不能列出含有隱藏或系統(tǒng)屬性的文件;
-----------------------------------------
三、文本解析顯神威:for /f 用法詳解
前言
for /f 是個十分強(qiáng)大的家伙。
如果說,for 語句是批處理中最強(qiáng)大的語句的話,那么,for /f 就是精華中的精華。
for /f 的強(qiáng)大,和它擁有眾多的開關(guān)密切相關(guān)。因為開關(guān)眾多,所以用法復(fù)雜,本章將分成若干小節(jié),為大家逐一介紹強(qiáng)大的 for /f 語句。
(一) 為解析文本而生:f or /f 的基本用法
所有的對象,無論是文件、窗體、還是控件,在所有的非機(jī)器語言看來,無外乎都是形如"c:\test.txt"、"CWnd"之類的文本信息;而所有的對象,具體的如ini
文件中的某條配置信息、注冊表中的某個鍵值、數(shù)據(jù)庫中的某條記錄„„都只有轉(zhuǎn)化為具有一定格式的文本信息,方可被代碼識別、操控?梢哉f,編程的很大
一部分工作,都是在絞盡腦汁想方設(shè)法如何提取這些文本信息。
而提取文本信息,則是for /f 的拿手好戲:讀取文件內(nèi)容;提取某幾行字符;截取某個字符片段;對提取到的內(nèi)容再切分、打亂、雜糅„„只要你所能想到的花樣,for /f 都會想方設(shè)法幫你辦到,因為,for /f 就是被設(shè)計成專門用于解析文本的。
先來看個例子。
假如有個文本文件test.txt,內(nèi)容如下:
[txt1]
論壇的目標(biāo)是:不求最大,但求最好,做最實用的批處理論壇。
論壇地址:bbs.bathome.cn。
這里是:新手晉級的福地,高手論劍的天堂。
那么,將如下代碼保存為test.cmd,并放在test.txt 同一目錄下運(yùn)行,將會在屏幕上原樣顯示test.txt 的內(nèi)容:
@echo off
for /f %%i in (test.txt) do echo %%i
pause
這段代碼,主要是讓你樹立這樣一種觀念:讀取文本文件的內(nèi)容,請使用 for /f 語句!
進(jìn)階話題:for /f 語句是把整個test.txt 一次性顯示出來的?
在這段代碼中,雖然執(zhí)行結(jié)果是把 test.txt 中的所有內(nèi)容都顯示出來了,貌似 for /f 語句是把整個test.txt 一次性顯示到屏幕上,實際上并非如此。
無論for 語句做何種變化,它的執(zhí)行過程仍然遵循基本的for 流程:依次處理每個元素,直到所有的元素都被處理為止。只不過在for /f 語句中,這里的元素是指文件中的每一行,也就是說,for /f 語句是以行為單位處理文本文件的。這是一條極為重要的規(guī)則,在上一章中也強(qiáng)調(diào)過它的重要性,希望在接下來的學(xué)習(xí)過程中,你能時刻牢記這一原則,那么,很多問題將會迎刃而解。以下是驗證這一說法的演示代碼(在[code4]的基礎(chǔ)上添加了&pause 語句):
@echo off
for /f %%i in (test.txt) do echo %%i&pause
pause
(二) 切分字符串的利器:delims=
也許你對[code4]這段代碼不屑一顧:不就是把test.txt 的內(nèi)容顯示出來了么?
好像用處不大啊。
好吧,我們來玩?zhèn)魔術(shù)。
還是[txt1]這段文本,把[code4]改造一下:
[code6]
@echo off
for /f "delims=," %%i in (test.txt) do echo %%i
pause
再次運(yùn)行test.cmd,看到什么變化了嗎?!
論壇的目標(biāo)是:不求最大
論壇地址:bbs.bathome.cn。
這里是:新手晉級的福地
請按任意鍵繼續(xù)...
結(jié)果,你驚奇地發(fā)現(xiàn),每行第一個逗號之后的所有內(nèi)容都不見了(如果有不存在逗號的行,則保留原樣),也就說,你成功地提取到了每行第一個逗號之前的所有內(nèi)容!
試想一下,這段代碼會有什么用呢?
如果別人給了你一個軟件清單,每行都是"英文軟件名(逗號)中文軟件名"的格式,而你卻只想保留英文名的時候,這段代碼將是多么有用啊!再假設(shè),有這么一個IP 文件,第一列是數(shù)字格式的IP 地址,第二列是具體的空間地址,列與列之間用逗號分隔,而你想提取其中數(shù)字格式的 IP,呵呵,我不說你也知道該怎么辦了吧?
要是文本內(nèi)容不是以逗號分隔,而是以其他符號分隔,那么,把"delims=,"的逗號換成相應(yīng)的符號就可以了。
在這里,我們引入了一個新的開關(guān):"delims=,",它的含義是:以逗號作為被處理的字符串的分隔符號。
在批處理中,指定分隔符號的方法是:添加一個形如 "delims=符號列表" 的開關(guān),這樣,被處理的每行字符串都會被符號列表中羅列出來的符號切分開來。
需要注意的是:如果沒有指定"delims=符號列表"這個開關(guān),那么,for /f 語句默認(rèn)以空格鍵或跳格鍵作為分隔符號。請把[txt1]中不同位置上的標(biāo)點符號改為空格或跳格,再運(yùn)行[code4]試試。
=============================================
進(jìn)階話題:如果我要指定的符號不止一個,該怎么辦?
在上面的講解中,我提到了指定分隔符號的方法:添加一個形如"delims=符
號列表"的開關(guān)。不知道你注意到?jīng)]有,我的說法是"符號列表"而非"符號",這是
大有講究的,因為,你可以一次性指定多個分隔符號!
還是以[txt1]為例,把[code6]再改造一下:
@echo off
for /f "delims=.," %%i in (test.txt) do echo %%i
pause
結(jié)果顯示:
[result3]
論壇的目標(biāo)是:不求最大
論壇地址:bbs
這里是:新手晉級的福地
請按任意鍵繼續(xù)...
這樣,第一個點號或第一個逗號之前的內(nèi)容都被提取出來了。
[code7]的執(zhí)行過程是:逐行讀取 test.txt 中的內(nèi)容,以點號和逗號切分每一
行的內(nèi)容(不存在點號和逗號的行,則不再切分,為了描述的方便,我們把被點
號或逗號切分的一個一個的字符串片段,稱之為節(jié)),然后,for /f 會提取第一節(jié)
的內(nèi)容作為最終結(jié)果,顯示在屏幕上。需要注意的是,在這里,所有行的字符串
被切分成了兩個以上的節(jié),但是,[code7]的代碼只會提取第一節(jié)字符串的內(nèi)容,
因為 for /f 語句默認(rèn)只提取第一節(jié)的符串。
(三) 定點提取:tokens=
上一節(jié)在講解 delims= 的時候,我一再強(qiáng)調(diào) for /f 默認(rèn)只能提取到第一節(jié)
的內(nèi)容,現(xiàn)在我們來思考一個問題:如果我要提取的內(nèi)容不在第一節(jié)上,那怎么
辦?
這回,就該輪到 tokens= 出馬了。
tokens= 后面一般跟的是數(shù)字,如 tokens=2,也可以跟多個,但是每個數(shù)字
之間用逗號分隔,如 tokens=3,5,8,它們的含義分別是:提取第2 節(jié)字符串、提
取第3、第5 和第8 節(jié)字符串。注意,這里所說的“節(jié)”,是由 delims= 這一開
關(guān)劃分的,它的內(nèi)容并不是一成不變的。
下面來看一個例子:
尺有所短,寸有所長,學(xué)好批處理沒商量,考慮問題復(fù)雜化,解決問題簡潔化。
對[txt2]這段文本,假設(shè)它們保存在文件 test.txt 中,如果我想提取“學(xué)好批
處理沒商量”這句話,該如何寫代碼呢?
我們稍微觀察一下[txt2]就會發(fā)現(xiàn),如果以逗號作為切分符號,就正好可以
把“學(xué)好批處理沒商量”化為單獨(dú)的一“節(jié)”,結(jié)合上一節(jié)的講解,我們知道,
"delims=," 這個開關(guān)是不可缺少的,而要提取的內(nèi)容在以逗號切分的第3 節(jié)上,
那么,tokens= 后面的數(shù)字就應(yīng)該是3 了,最終的代碼如下:
@echo off
for /f "delims=, tokens=3" %%i in (test.txt) do echo %%i
pause
如果我們現(xiàn)在要提取的不只一個“節(jié)”,而是多個,那又怎么辦呢?比如,
要提取以逗號切分的第2 節(jié)和第5 節(jié)字符串,是寫成這樣嗎?
[code9]
@echo off
for /f "delims=, tokens=2,5" %%i in (test.txt) do echo %%i
pause
運(yùn)行批處理后發(fā)現(xiàn),執(zhí)行結(jié)果只顯示了第2 節(jié)的內(nèi)容。
原來,echo 后面的 %%i 只接收到了 tokens=2,5 中第一個數(shù)值2 所代表的
那個字符串,而第二個數(shù)值5 所代表的字符串因為沒有變量來接收,所以就無法
在執(zhí)行結(jié)果中顯示出來了。
那么,要如何接收 tokens= 后面多個數(shù)值所指代的內(nèi)容呢?
for /f 語句對這種情況做如下規(guī)定:
如果 tokens= 后面指定了多個數(shù)字,如果形式變量為%%i,那么,第一個
數(shù)字指代的內(nèi)容用第一個形式變量%%i 來接收,第二個數(shù)字指代的內(nèi)容用第二
個形式變量%%j 來接收,第三個數(shù)字指代的內(nèi)容用第三個形式變量%%k 來接
收„„第N 個數(shù)字指代的內(nèi)容用第N 個形式變量來接收,其中,形式變量遵循
字母的排序,第 N 個形式變量具體是什么符號,由第一個形式變量來決定:如
果第一個形式變量是%%i,那么,第二個形式變量就是%%j;如果第一個形式
變量用的是%%x,那么,第二個形式變量就是%%y。
現(xiàn)在回頭去看[code9],你應(yīng)該知道如何修改才能滿足題目的要求了吧?修改
結(jié)果如下:
@echo off
for /f "delims=, tokens=2,5" %%i in (test.txt) do echo %%i %%j
pause
如果有這樣一個要求:顯示[txt2]中的內(nèi)容,但是逗號要替換成空格,如何
編寫代碼?
結(jié)合上面所學(xué)的內(nèi)容,稍加思索,你可能很快就得出了答案:
@echo off
for /f "delims= , tokens=1,2,3,4,5" %%i in (test.txt) do
echo %%i %%j %%k %%l %%m
pause
寫完之后,你可能意識到這樣一個問題:假如要提取的“節(jié)”數(shù)不是5,而
是10,或者20,或者更多,難道我也得從1 寫到10、20 或者更多嗎?有沒有更
簡潔的寫法呢?
答案是有的,那就是:如果要提取的內(nèi)容是連續(xù)的多“節(jié)”的話,那么,連
續(xù)的數(shù)字可以只寫最小值和最大值,中間用短橫連接起來即可,比如
tokens=1,2,3,4,5 可以簡寫為 tokens=1-5 。
還可以把這個表達(dá)式寫得更復(fù)雜一點:tokens=1,2-5 ,tokens=1-3,4,5 ,
tokens=1-4,5„„怎么方便就怎么寫吧。
大家可能還看到一種比較怪異的寫法:
@echo off
for /f "delims=, tokens=1,*" %%i in (test.txt) do echo %%i %%j
pause
結(jié)果,第一個逗號不見了,取代它的是一個空格符號,其余部分保持不變。
其中奧妙就在這個星號上面。
tokens=后面所接的星號具備這樣的功能:字符串從左往右被切分成緊跟在*
之前的數(shù)值所表示的節(jié)數(shù)之后,字符串的其余部分保持不變,整體被*所表示的
一個變量接收。
理論講解是比較枯燥的,特別是為了嚴(yán)密起見,還使用了很多限定性的修飾
詞,導(dǎo)致句子很長,增加了理解的難度,我們還是結(jié)合[code12]來講解一下吧。
[txt2] 的內(nèi)容被切分,切分符號為逗號,當(dāng)切分完第一節(jié)之后,切分動作不
再繼續(xù)下去,因為 tokens=1,* 中,星號前面緊跟的是數(shù)字 1;第一節(jié)字符串被
切分完之后,其余部分字符串不做任何切分,整體作為第二節(jié)字符串,這樣,[txt2]
就被切分成了兩節(jié),分別被變量%%i 和變量%%j 接收。
以上幾種切分方式可以結(jié)合在一起使用。不知道下面這段代碼的含義你是否
看得懂,如果看不懂的話,那就運(yùn)行一下代碼,然后反復(fù)揣摩,你一定會更加深
刻地理解本節(jié)所講解的內(nèi)容的:
[code13]
@echo off
for /f "delims=, tokens=1,3-4,*" %%i in (test.txt) do echo %%i %%j %%k %%l
pause
(四) 跳過無關(guān)內(nèi)容,直奔主題:skip=n
很多時候,有用的信息并不是貫穿文本內(nèi)容的始終,而是位于第 N 行之后
的行內(nèi),為了提高文本處理的效率,或者不受多余信息的干擾,for /f 允許你跳
過這些無用的行,直接從第N+1 行開始處理,這個時候,就需要使用參數(shù) skip=n,
其中,n 是一個正整數(shù),表示要跳過的行數(shù)。例如:
[code14]
@echo off
for /f "skip=2" %%i in (test.txt) do echo %%i
pause
這段代碼將跳過頭兩行內(nèi)容,從第3 行起顯示test.txt 中的信息。
(五) 忽略以指定字符打頭的行:eol=
在cmd 窗口中敲入:for /?,相關(guān)的解釋為:
[quote]
eol=c - 指一個行注釋字符的結(jié)尾(就一個)
[/quote]
[quote]
FOR /F "eol=; tokens=2,3* delims=, " %i in (myfile.txt) do @echo %i %j %k
會分析 myfile.txt 中的每一行,忽略以分號打頭的那些行„„
[/quote]
第一條解釋狗屁不通,頗為費(fèi)解:行注釋字符的結(jié)尾是什么意思?“(就一
個)”怎么回事?結(jié)合第二條解釋,才知道eol 有忽略指定行的功能。但是,這兩
條解釋是互相矛盾的:到底是忽略以指定字符打頭的行,還是忽略以指定字符結(jié)
尾的行?
實踐是檢驗真理的唯一標(biāo)準(zhǔn),還是用代碼來檢驗一下eol 的作用吧:
[code15]
@echo off
for /f "eol=;" %%i in (test.txt) do echo %%i
pause
結(jié)果,那些以分號打頭的行沒有顯示出來。
由此可見,第二條解釋是正確的,eol= 的準(zhǔn)確含義是:忽略以指定字符打
頭的行。而第一條的“結(jié)尾”純屬微軟在信口開河。
那么,“(就一個)”又作何解釋呢?
試試這個代碼:
[code16]
@echo off
for /f "eol=,;" %%i in (test.txt) do echo %%i
pause
此時,屏幕上出現(xiàn) 此時不應(yīng)有 ;"。 的報錯信息?梢,在指定字符的時
候,只能指定1 個——在很多時候,我對這樣的設(shè)計頗有微詞而又無可奈何:為
什么只能指定1 個而不是多個?要忽略多個還得又是 if 又是 findstr 加管道來多
次過濾,那效率實在太低下了——能用到的功能基本上都提供,但是卻又做不到
更好,批處理,你的功能為什么那么弱?
不知道大家注意到?jīng)]有,如果 test.txt 中有以分號打頭的行,那么,這些行
在代碼[code14]的執(zhí)行結(jié)果中將憑空消失。
原來,for /f 語句是默認(rèn)忽略以分號打頭的行內(nèi)容的,正如它默認(rèn)以空格鍵
或跳格鍵作為字符串的切分字符一樣。
很多時候,我們可以充分利用這個特點,比如,在設(shè)計即將用for 讀取的配
置文件的時候,可以在注釋文字的行首加上分號,例如在編寫病毒文件查殺代碼
的時候,可以通過 for 語句來讀取病毒文件列表,那么,病毒文件列表.ini 這個
配置文件可以這樣寫:
;以下是常見的病毒文件,請見一個殺一個^_^
;copyleft:沒有
qq.exe
msn.exe
iexplore.exe
如果要取消這個默認(rèn)設(shè)置,可選擇的辦法是:
1、為eol=指定另外一個字符;
2、使用 for /f "eol=" 語句,也就是說,強(qiáng)制指定字符為空,就像對付delims=
一樣。
(六)如何決定該使用 f or /f 的哪種句式?(兼談usebackq 的使用)
for /f %%i in („„) do („„) 語句有好幾種變形語句,不同之處在于第一個
括號里的內(nèi)容:有的是用單引號括起來,有的是用雙引號包住,有的不用任何符
號包裹,具體格式為:
1、for /f %%i in (文件名) do („„)
2、for /f %%i in ('命令語句') do („„)
3、for /f %%i in ("字符串") do („„)
看到這里,我想很多人可能已經(jīng)開始犯了迷糊了:如果要解決一個具體問題,
面對這么多的選擇,如何決定該使用哪一條呢?
實際上,當(dāng)我在上面羅列這些語句的時候,已經(jīng)有所提示了,不知道你是否
注意到了。
如果你一時無法參透其中奧妙,那也無妨,請聽我一一道來便是。
1、當(dāng)你希望讀取文本文件中的內(nèi)容的話,第一個括號中不用任何符號包裹,
應(yīng)該使用的是第1 條語句;例如:你想顯示test.txt 中的內(nèi)容,那么,就使用 for
/f %%i in (test.txt) do echo %%i;
2、當(dāng)你讀取的是命令語句執(zhí)行結(jié)果中的內(nèi)容的話,第一個括號中的命令語
句必須使用單引號包裹,應(yīng)該使用的是第2 條語句;例如:你想顯示當(dāng)前目錄下
文件名中含有test 字符串的文本文件的時候,應(yīng)該使用 for /f %%i in ('dir /a-d
/b *test*.txt') do echo %%i 這樣的語句;
3、當(dāng)你要處理的是一個字符串的時候,第一個括號中的內(nèi)容必須用雙引號
括起來,應(yīng)該是用的是第 3 條語句;例如:當(dāng)你想把 bbs.bathome.cn 這串字符
中的點號換為短橫線并顯示出來的話,可以使用 for /f "delims=.
tokens=1-3" %%i in ("bbs.bathome.cn") do echo %%i-%%j-%%k 這樣的語
句。
很顯然,第一個括號里是否需要用符號包裹起來,以及使用什么樣的符號包
裹,取決于要處理的對象屬于什么類型:如果是文件,則無需包裹;如果是命令
語句,則用單引號包裹;如果是字符串,則使用雙引號括起來。
當(dāng)然,事情并不是絕對如此,如果細(xì)心的你想到了批處理中難纏的特殊字符,
你肯定會頭大如斗。
或許你頭腦中靈光一閃,已經(jīng)想到了一個十分頭痛的問題:在第1 條語句中,
如果文件名中含有空格或&,該怎么辦?
照舊嗎?
拿個叫 test 1.txt 的文件來試試。
你很快寫好了代碼,新建文件-->碼字-->保存為批處理,前后費(fèi)時不到 1 分
鐘:
[code17]
@echo off
for /f %%i in (test 1.txt) do echo %%i
pause
你興沖沖地雙擊批處理,運(yùn)行后,屏幕上出現(xiàn)了可恥的報錯信息:系統(tǒng)找不
到文件 test 。
當(dāng)你把 test 1.txt 換成 test&1.txt 后,更怪異的事情發(fā)生了:CMD 窗口在你
眼前一閃而過,然后,優(yōu)雅地消失了。
你可能覺得自己的代碼寫錯了某些符號,你再仔細(xì)的檢查了一次,確認(rèn)沒有
筆誤,然后,你再次雙擊批處理,結(jié)果問題照舊;你開始懷疑其他程序?qū)λ赡?/p>
有影響,于是關(guān)掉其他窗口,再運(yùn)行了一次,問題依舊;你不服氣地連續(xù)運(yùn)行了
好幾次,還是同樣的結(jié)果。
怪哉!
你一拍大腿,猛然想起了一件事:當(dāng)路徑中含有特殊字符的時候,應(yīng)該使用
引號把路徑括起來。對,就是它了!
但是,當(dāng)你把代碼寫出來之后,你很快就焉了:for /f %%i in ("test 1.txt") do
echo %%i,這不就是上面提到的第3 條 for /f 命令的格式嗎?批處理會把 test
1.txt 這個文件名識別為字符串!
你百無聊賴地在CMD 窗口中輸入 for /? ,并重重地敲下了回車,漫無目的
地在幫助信息中尋找,希望能找到點什么。
結(jié)果還真讓你到了點什么。
你看到了這樣的描述:
指定新語法已在下類情況中使用:
在作為命令執(zhí)行一個后引號的字符串并且一個單
引號字符為文字字符串命令并允許在 filenameset
中使用雙引號擴(kuò)起文件名稱。
但是,通讀一遍之后,你卻如墜五里霧中,不知所云。
還好,下面有個例子,并配有簡單的說明:
FOR /F "usebackq delims==" %i IN (`set`) DO @echo %i
會枚舉當(dāng)前環(huán)境中的環(huán)境變量名稱。
你仔細(xì)對比了 for /f 語句使用usebackq 和不使用usebackq 時在寫法上的差
別,很快就找到了答案:當(dāng)使用了 usebackq 之后,如果第一個括號中是一條命
令語句,那么,就要把單引號'改成后引號`(鍵盤左上角esc 鍵下面的那個按鍵,
與~在同一鍵位上)。
回過頭去再看那段關(guān)于 usebackq 的描述,字斟句酌,反復(fù)揣摩,終于被你
破譯了天機(jī):usebackq 是一個增強(qiáng)型參數(shù),當(dāng)使用了這個參數(shù)之后,原來的for
語句中第一個括號內(nèi)的寫法要做如下變動:如果第一個括號里的對象是一條命
令語句的話,原來的單引號'要改為后引號`;如果第一個括號里的對象是字符串
的話,原來的雙引號"要改為單引號';如果第一個括號里的對象是文件名的話,
要用雙引號"括起來。
驗證一下,把[code17]改寫成如下代碼:
[code18]
@echo off
for /f "usebackq" %%i in ("test 1.txt") do echo %%i
pause
測試通過!
此時,你很可能會仰天長嘆:Shit,微軟這該死的機(jī)器翻譯!
至于把[code17]代碼中的空格換成&后,CMD 窗口會直接退出,那是因為&
是復(fù)合語句的連接符,CMD 在預(yù)處理的時候,會優(yōu)先把&前后兩部分作為兩條
語句來解析,而不是大家想象中的一條完整的for 語句,從而產(chǎn)生了嚴(yán)重的語法
錯誤。因為牽涉到預(yù)處理機(jī)制問題,不屬于本節(jié)要討論的內(nèi)容,在此不做詳細(xì)講解
這個時候,我們會吃驚地發(fā)現(xiàn),區(qū)區(qū)一條for 語句,竟然有多達(dá)6 種句型:
1、for /f %%i in (文件名) do („„)
2、for /f %%i in ('命令語句') do („„)
3、for /f %%i in ("字符串") do („„)
4、for /f "usebackq" %%i in ("文件名") do („„)
5、for /f "usebackq" %%i in (`命令語句`) do („„)
6、for /f "usebackq" %%i in ('字符串') do („„)
其中,4、5、6 由1、2、3 發(fā)展而來,他們有這樣的對應(yīng)關(guān)系:1-->4、2-->5、
3-->6。
好在后3 種情形并不常用,所以,牢牢掌握好前三種句型的適用情形就可以
了,否則,要在這么多句型中確定選擇哪一條語句來使用,還真有點讓人頭腦發(fā)
至于 for /f 為什么要增加usebacq 參數(shù),我只為第4 條語句找到了合理的解
釋:為了兼容文件名中所帶的空格或&。它在第 5、6 條語句中為什么還有存在
的必要,我也不是很明白,這有待于各位去慢慢發(fā)現(xiàn)。
一些常用的dos命令
pause
暫停命令
find
搜索文件或文件內(nèi)指定的字符串
prompt
設(shè)置命令提式符
ren
批量重名
用法:ren 1.txt 2.bat
效果:1.txt就被改為2.bat
subst s: \\*.*.*.*\c$ (這個是遠(yuǎn)程映射)
\\*.*.*.*將c盤下windows映視到s盤下
刪除的話是
這個是使用例子:subst s: /d
regedit /s
用法:regedit /s 1.reg
在批處理中運(yùn)行所有注冊表文件而不顯示
assoc
修改文件關(guān)聯(lián)
dir
查看本目錄文件
dir /s /p "*.txt"
顯示當(dāng)前所在盤所有txt文件
rd
刪除目錄
"rd /s" 是刪除目錄下所有東西
rd /s 123 刪除123目錄下的全都東西
del
刪除文件
例子:del /f F:\xx
效果:刪除F盤下的xx文件
type
查看文件內(nèi)容命令 比如*.txt
attrib
更改文件屬性命令
move
剪切命令 "move 1.bat 123" 1.bat文件剪切到123目錄
format
格式化命令 "format d: \q"
chkdsk
檢查磁盤
path
path=c:\dos;c:\win
搜索當(dāng)前可執(zhí)行程序
copy
復(fù)制文件命令 比如: "copy 1.txt 2" 就是1的文件 復(fù)制到2文件夾里面 copy不能復(fù)制目錄
copy cmd.exe \\192.168.1.102\c$
將cmd復(fù)制到192.168.1.102的盤下
xcopy
高級復(fù)制命令 "xcopy aaa a123 /e" 將aaa目錄復(fù)制到123目錄
title
更改cmd標(biāo)題命令 如 "title 腳本之家" 標(biāo)題就改成腳本之家了.
color
更改背景參數(shù)如 "color 07" 就是黑色背景白色字 "color 12" 就是藍(lán)色背景綠色字 詳細(xì)見color /?
defrag
磁盤碎片整理 "defrag c: -a" 對C盤的碎片進(jìn)行分析不處理 "defrag c: -f" 進(jìn)行處理
ipconfig
查看本機(jī)ip地址命令
CLS
清屏
tree
顯示當(dāng)前所以文件文件夾和結(jié)構(gòu)
call
運(yùn)行另一個批處理 不結(jié)束父進(jìn)程
mstsc
遠(yuǎn)程桌面連接
ping 計算機(jī)名字
計算機(jī)名稱得到ip
ping的返回ttl值確認(rèn)操作系統(tǒng)
xp/2000的ttl值128
windwos98為64
unix為255
ping /n 3 127.0.0.0>nul
向ip發(fā)送3次,沿遲6秒
date
顯示當(dāng)前日期 沒現(xiàn)在時間
TIME
顯示當(dāng)前時間 當(dāng)不會更新時間
taskkill
結(jié)束進(jìn)程
tasklist
查看進(jìn)程
echo 再見 & pause>nul
結(jié)束的時候不是請安任何鍵結(jié)束 而是再見 可以在再見里面隨意寫 同時也可以去除“echo 再見 &”就什么都沒顯示了
shutdown
關(guān)機(jī)命令