Basti's Scratchpad on the Internet

批处理之坑爹的感叹号和变量延迟扩展

这是我在博客园的博客中的文章。

下面是原文(未大改,稍作了一些格式上的调整):


先来说说变量延迟扩展吧。当然,放狗一搜,就能看到满天飞的关于变量延迟扩展的文章,所以,我这里就简单介绍一下。先来看一段批处理:

set str=test

if %str%==test (
    set str=another test
    echo %str%
)

上面的代码段极其简单,给str赋值,判断其值是否为test,如果是,重新赋值为another test,再显示str的值。

作为正常人的思维,这里显示的肯定是another test了,但其实不是,其显示的仍然是test,这是为什么?因为:windows在解释执行此代码段时,在遇到if语句后的括号后,只把它当一条语句处理而不是两条语句,所以,在第二条语句中的%str%会被替换成它目前的值test,上面的代码相当于下面的代码的效果:

set str=test

if %str%==test (
    set str=another test
    echo test    ::注意这里
)

所以,输出自然是test了。

这样编程的灵活性就大大降低了,于是,M$就想了一个workground的方法,那就是变量延迟,很简单,看如下代码:

@echo off
setlocal enabledelayedexpansion    ::注意这里

set str=test

if %str%==test (
    set str=another test
    echo !str!      ::注意这里
    echo %str%  ::区别
)

现在会输出什么呢?试一下就知道,第一行输出another test,第二行输出test。

现在解释一下, setlocal enabledelayedexpansion 用于开启变量延迟,这是告诉解释器,在遇到复合语句的时候,不要将其作为一条语句同时处理,而仍然一条一条地去解释。但是这时必须用!str!来引用变量,如果仍然用%str%引用是不起作用的。

好了,变量延迟扩展解释完了,至少这就是我知道的变量延迟扩展。


今天变量延迟扩展和感叹号让我蛋疼了半天,所以现在也要来扯扯它们的蛋。

试试以下代码段:

@echo off

setlocal enabledelayedexpansion

set str=test!!!

echo %str%

开启变量延迟,给str赋值,输出str,于是,输出自然是test!!!了。但其实,windows告诉我们,又错了,输出是test,感叹号被windows吃了。

当然如果注释掉开启变量延迟那一行,这几行是可以正常工作的,所以说是开启变量延迟影响了我们正常的工作,但我至今还没搞清楚为什么会这样,这只有M$知道。

于是,关闭变量延迟吧,可是程序又需要使用变量延迟来实现一些逻辑,怎么办?两个办法:

  1. 暂时关闭变量延迟:
    @echo off
    
    setlocal enabledelayedexpansion
    
    :: do something here
    
    setlocal disabledelayedexpansion    ::关闭变量延迟
    
    set str=test!!!
    
    echo %str%
    
    setlocal enabledelayedexpansion
    
    :: continue...
    

    在需要处理感叹号的地方,暂时关闭变量延迟,处理完后再打开。这时能正常输出感叹号。

  2. 在开启变量延迟之前set:
    @echo off
    
    set str=test!!!
    
    setlocal enabledelayedexpansion
    
    echo !str!    ::注意这里
    

    在开启变量延迟之前设置好变量,但是要注意,在使用变量的时候,需要使用感叹号引用。这时感叹号也可以正常输出。


最后贴一段为什么让我蛋疼的代码(变量延迟自然是打开的):

set server=%~1
set username=%~2
set password=%~3
echo %date%, %time% [INFO]   Report server address: [%server%], username: [%username%], password: [*******]

rs.exe -i "PublishReports.rss" -s "%server%" -u "%username%" -p "%password%" -l 600

为了不回显password,我输出了一大串*。在我机器上测试时,我的password字符都很常规,所以通过。但是,QA的password里面有感叹号。。。

关键的是,这里调用的是M$的Report Server的rs.exe来上传模板,它抛出一个Could not connect的exception,于是,我自然地想到是report server的service出了问题,但查了好久也没发现report server有什么问题。。。谁TMD会想到是别人的password里有感叹号,谁TMD又会想到是跟所谓延迟变量的冲突导致的,更有谁会想到,我自己自作聪明地为了不显示password而显示出了一大串hard code的星号导致看不到password的值。。。

这么多巧合在一起,谁能不蛋疼。。。

Other posts
comments powered by Disqus