眾所周知,如果echo后面跟一個環(huán)境變量,但是該變量卻為空時,相當(dāng)于不加任何參數(shù)的echo,即輸出當(dāng)前echo是on還是off。很多文章或者教程給出的解決方案都是在echo后面加一個點(diǎn)號echo.,這樣就會輸出空行。
復(fù)制代碼 代碼如下:
@echo off
echo %demon.tw%
:: ECHO is off.
echo.%demon.tw%
pause據(jù)我所知,用echo輸出空行至少有十種方法:
復(fù)制代碼 代碼如下:
@echo off
echo=
echo,
echo;
echo+
echo/
echo[
echo]
echo:
echo.
echo\
pause
這十種方法可以分為三組,每組的效率依次遞減??杀氖牵切┍环顬榻?jīng)典的教程給出的卻是效率最低那組中的echo.
echo.不僅效率低下,而且還容易引發(fā)錯誤:
復(fù)制代碼 代碼如下:
@echo off
cd .>echo
echo.
pause
我知道你很難接受,但事實(shí)的確如此。
第一組中echo后面的=,;都是批處理中的分隔符,所以CMD可以正確地解析出echo命令,并把=,;作為echo命令的參數(shù)。是的,你沒有看錯,分隔符并不是用來分隔命令與參數(shù),它們通常是參數(shù)的一部分。既然是參數(shù),那么為什么不會被輸出?那是因?yàn)閑cho命令直接跳過了參數(shù)的第一個字符,從第二個字符開始輸出,而第二個字符是NUL,所以輸出了空行。
你可能又要問,那為什么用空格做分隔符卻不能輸出空行呢?那是因?yàn)樵谳敵鲋?,CMD要檢查echo命令的參數(shù)是不是on或者off,或者參數(shù)為空:首先跳過所有空白字符,如果跳過之后字符串就結(jié)束了,那么就認(rèn)為沒有加參數(shù),輸出echo是on還是off;如果字符串沒有結(jié)束,就調(diào)用wcsnicmp函數(shù)來判斷剩下的字符串是否為on或者off,進(jìn)而修改echo的狀態(tài)。
因此加上很多空格也是一樣的效果:
復(fù)制代碼 代碼如下:
@echo off
echo
echo on
echo
pause
而對于第二和第三組,事情就沒那么簡單了,由于echo后面跟的并不是分隔符,所以解析之后會被當(dāng)成一個整體,而echo+ echo/等等顯然又不是內(nèi)部命令,CMD會把它們當(dāng)做外部命令進(jìn)行搜索。嗯,你知道,搜索是很花時間的,這就是為什么它們的效率低于第一組。
可惜的是,CMD花了很大力氣搜索,卻仍然找不到這樣的外部命令,這時候它會嘗試著修復(fù)(Fix)命令,看看命令中是否有某些字符(如圖):
![](/d/20211017/b5ec44934c13ca813ab7eebbc7187f53.gif)
可以看到,CMD對:.\的處理跟+[]/不太一樣,如果是+[]/,CMD會直接把它們從命令中刪除并且添加到原有參數(shù)的前面;而如果是:.\并且CMD拓展是開啟的話,那么會多調(diào)用一次GetFileAttributes函數(shù)獲取文件屬性,多調(diào)用一次函數(shù)自然會多花一些時間,所以第三組的效率又稍稍比第二組的低些。 ![](/d/20211017/7a098b84acf4a82aad34003e13aad1d6.gif)
再來解釋一下為什么echo.有時候會引起錯誤。文件名中是不能出現(xiàn):.\的,理論上GetFileAttributes函數(shù)都應(yīng)該返回-1(INVALID_FILE_ATTRIBUTES),然而事實(shí)卻不是如此,我也不知道這算不算GetFileAttributes函數(shù)的BUG:
復(fù)制代碼 代碼如下:
#include stdio.h>
#include windows.h>
int main()
{
FILE *fp = fopen("echo", "wb");
fclose(fp);
printf("0x%x\n", GetFileAttributes("echo:"));
printf("0x%x\n", GetFileAttributes("echo."));
printf("0x%x\n", GetFileAttributes("echo/"));
return 0;
}
如果你測試一下上面的C程序,就會發(fā)現(xiàn)echo.那行返回的不是-1。
如果GetFileAttributes函數(shù)返回的不是-1(一般表示文件不存在),也不是0×10(表示文件是文件夾),那么命令還是會保持原來的樣子,當(dāng)成外部命令運(yùn)行。
復(fù)制代碼 代碼如下:
@echo off
cd .>echo
echo.
pause
‘echo.' is not recognized as an internal or external command, operable program or batch file.
復(fù)制代碼 代碼如下:
@echo off
cd .>echo
setlocal disableextensions
echo.
pause
關(guān)閉了CMD拓展,沒有問題。
復(fù)制代碼 代碼如下:
@echo off
md echo
echo.
pause
echo是文件夾而不是文件,沒有問題。
最后總結(jié)一下吧,在大部分情況下,你都應(yīng)該使用第一組的echo, echo; echo=來進(jìn)行輸出,它們的效率跟echo (空格)是一樣的,并且可以用來輸出on或者off,在變量為空時還能輸出空行。
但是echo, echo; echo=卻不能輸出以/?開頭的行,如果你需要,可以使用第二組的echo+ echo/ echo[ echo],它們的效率低一些,但能保證原樣輸出。
我不建議你使用第三組的echo: echo. echo\,如果你仍然要像垃圾教程里面那樣用,我也沒有辦法。
作者: Demon
鏈接: http://demon.tw/reverse/cmd-internal-echo.html
您可能感興趣的文章:- 批處理 Set 命令詳解 讓你理解set命令
- DOS批處理之DATE命令的使用方法詳解
- dos命令或批處理 發(fā)生系統(tǒng)錯誤5 拒絕訪問
- Windows下用命令行修改IP地址的方法詳解(附批處理文件)
- Reg命令使用詳解 批處理操作注冊表必備
- windows批處理命令教程
- 批處理命令Start的使用介紹
- 批處理命令詳解之目錄跳轉(zhuǎn):cd
- 批處理命令教學(xué)之tree命令
- 批處理命令教學(xué)之if語句
- 批處理命令教學(xué)之管道符號(|)
- 批處理命令教學(xué)之復(fù)合語句連接符(&、&&和||)
- 批處理命令教學(xué)之字符串排序(sort)
- 批處理命令教學(xué)之more命令