[新手上路]批处理新手入门导读[视频教程]批处理基础视频教程[视频教程]VBS基础视频教程[批处理精品]批处理版照片整理器
[批处理精品]纯批处理备份&还原驱动[批处理精品]CMD命令50条不能说的秘密[在线下载]第三方命令行工具[在线帮助]VBScript / JScript 在线参考
返回列表 发帖

[原创] 批处理脚本高级编程技巧——变量嵌套

批处理脚本高级编程技巧——变量嵌套
一,为什么要使用变量嵌套?
  1,可提高代码的效率
  2,可以缩减代码
二,批处理变量的种类
  1, %0~%9 , %*:   批处理运行的命令行参数,在代码行预处理时扩展,可视为最高优先级扩展。

    如果存在变量:
12a=###
!a=$$$
a=aaa

echo;%12a%
得到结果为:第一个参量+2a

echo;%!a%!
得到结果为:$$$

2, %var%:  在代码行预处理时扩展,其实与%1~%9 ,%*是同时扩展取值的,
    但是%1%得到的只有可能是参量值,绝不可能得到变量值,所以视为次优先级扩展,

   ********** 预处理问题 **********
    要很好的理解%号变量,需要知道一点,预处理一次是读取一组来进行的  
      如:
  
        [1]
        set a=aa&echo %bb%

        [2]
        if aa==%bb% (
                echo 等
        )

        [3]
        (
                echo %aa%
                echo %bb%
        )
        上面就是所“组”的形式。

      还要知道:%号变量在预处理读取代码行时已经扩展了
        例如:
  1.         set kh=)
  2.         (echo %kh%&pause
  3.         echo bb
  4.         )
  5.         
复制代码
因为%kh%扩展了,其值中的 ")" 与前面的 "(" 形成了一组代码,其实只读取了一行,
        运行代码遇到pause暂停后,修改bat文件, 把echo bb改为:echo cc,并保存后,再按键继续可见到显示的是"cc"
   ********** 预处理问题 **********

  3, for()变量: %%a %%1 %%中 等形式均可使用,

    虽然 %%a%或者%%1 等形式得到的还是for 变量的值,而不是作为:%a% %1 扩展,
    为什么呢?
    因为预处理在处理%var%形式变量时,是从左到右以两个%号作为一组变量的,而以%+一个数字作为一组参量的,
    还有一点就是如果一组%号的两个%号间没有其它字符时,将留下一个%号,
    所以是要将for()变量视为第三优先级,
    另外变量%var% 及%1扩展后的值里面,如果含有有效的for ()变量%a的话,该值中的%a 会得到for()变量再次扩展
    更说明了for()变量要比%var% %0~%9,%*要低一级
    这个例子是变量的嵌套扩展的一种形式,
    其实for ()的变量名应该是:经预处理后的一个 %+一个字符,以cmd批处理脚本里面,可以是数字,字母,或字符和汉字都可以。
    for变量仅在该组for()内有效,并向下一级子for ()继承
    如果父级for()与子级for()有相同的有效变量时,得到的是当前级的扩展值
例如:
  1.   
  2. for /f "tokens=1,2,3" %%a in ("aa ab ac") do (
  3.    for /f "tokens=1,2,3" %%b in ("$a $b $c") do (
  4.       echo;%%a-%%b-%%c-%%d &rem 结果为:aa-$a-$b-$c
  5.      rem 而不是:aa-ab-ac-$c
  6.    )
  7.       echo;%%a %%b %%c %%d &rem 结果为:aa-ab-ac-
  8. )
复制代码
4,!var!:  变量延迟打开时有效,在命令运行时扩展,可视为四级优先级,
为什么要这么认为呢?
    如果for()变量扩展后,重新组成的表达式中,含有!号的话,命令运行时会再次进行!var!的取值扩展
这也是变量嵌套扩展的一种形式
     !var!变量,在一对setlocal ... 和endlocal内有效,并向下一级子环境继承

  5,set/a 运算命令变量,这可以说最低级的变量扩展了,就连!var!变量也扩展后,
     如果得到的是以非数字及非运行符开头的话,就会再次进行变量扩展之后才执行运算。

三、只有了解变量扩展的优先级,才可能理解变量嵌套的用法。  
    变量嵌套扩展的方式有:
*** 注意:要开启变量延迟,否则!var!变量无从谈起 ***
    1,先%1或%var% 再 !var!
只要%1或%var%扩展的值中含有或和旁边的字符组合后含有:!var!,!var:n=m!,!var:~2,2! 等形式
就会再次进行!号变量的扩展
        例如:
  1.         setlocal enabledelayedexpansion
  2.         set aa=bb
  3.         set abb=##
  4.         echo 程序文件名:%0
  5.         echo 嵌套扩展:!a%aa%!
  6.         pause
  7.         
复制代码
2,先%1或%var% 再 for ()%a
只要%1或%var%扩展后的值里面,含有有效的for ()变量%a的话,该值中的%a 会再次得到for()变量扩展
        例如:
  1.         set #=%%a
  2.         for %#% in (aa bb cc) do (echo;%#%)
  3.         pause
  4.         
复制代码
3,先for()%a 再!var!
只要for()%a扩展的值中含有或和旁边的字符组合后含有:!var!,!var:n=m!,!var:~2,2! 等形式
就会再次进行!号变量的扩展,这也就是为什么开启了变量延迟后,会丢失!号的原因。
        例如:
  1.         set #aa=$a
  2.         set #bb=$b
  3.         set #cc=$c
  4.         for %%a in (aa bb cc) do (echo;%%a : !#%%a!)
  5.         pause
  6.         
复制代码
4,先%1或%var% 再 for ()%a  再!var!
也就是第2+第3种用法的

        例如:
  1.         setlocal enabledelayedexpansion
  2.         set #=%%a
  3.         set aaa=##a
  4.         set abb=##b
  5.         set acc=##c
  6.         for %#% in (aa bb cc) do (echo %%a : a%#% : !a%#%!)
  7.         pause
  8.         
复制代码
5,多级嵌套:先%1或%var% 再 for ()%a  再!var! 再for ()%b 再!var!
主要是for ()%a 和!var!间的循环嵌套,看例子:
  1.   
  2. for %%a in (aaa_$a bbb_$b ccc_$c) do (
  3.     set str=%%a
  4. for /f "tokens=1,2 delims=_" %%b in ("!str!") do (         rem !str!作为for /f 的字符串表达式
  5.   echo;[2f] %%b+%%c
  6.   set %%c=%%a        &rem 第一次循环时:相当于set $a=aaa_$a
  7.   for /f "tokens=1,2 delims=_" %%1 in ("!%%c!") do ( &rem 相当于:!$a!作为for /f 的字符串表达式
  8.    echo;[3f] %%1=%%2
  9.   )  
  10. )
  11. )
  12. set $           &rem 显示变量,方便理解
复制代码
** 这点是变量嵌套的最巧用法,是提高批处理脚本的关键 **
    6,运算命令变量,就是在其它变量都扩展完了之后,还可以再进行扩展一次。
  1. set a=100
  2. set b=a
  3. set "c=^!b^!*2"
  4. set/a d=%c%+10
复制代码
代码中:%c% => !b!*2 => a*2 => 100*2 所以结果进行了以下运算:
  1. set /a d=100*2+10
复制代码
7,管道方式:
  1. set aa=d 100 &((echo;%%aa%%&echo;q)|debug)
复制代码
虽然是在同一行,也没有用到call ,但是%%a%%扩展成了d 100
原因是(echo;%%a%%&echo;q)是在批处理外运行的,并且通过管道把显示值传给后面的命令debug 作为输入行。

    8,效率较低级的call 方式变量嵌套,相当于从第1种方式到第5种方式再循环一次。(有所重复,就不详述)

四、变量嵌套扩展的实际应用。
    1,if 用法
  1.   
  2. set "str= help open quit run "
  3. set in=输入:
  4. if "!str: %in% =!" neq "!str!" goto :m_%in%
  5. :m_help
  6. :m_open
  7. :m_quit
  8. :m_run
复制代码
2,for 用法:
  1. set n1=姓名:!@n!     性别:!@xb!
  2. set n2=学历:!@xl!     年龄:!@g!
  3. set n3=民族:!@m!     体重:!@tz!
  4. ::前面先不要开变量延迟
  5. setlocal enabledelayedexpansion
  6. for /f "tokens=1-6" %%1 in ("benton 男 中专 18岁 汉 90kg") do (
  7. set @n=%%1&set @xb=%%2&set @xl=%%3&set @g=%%4&set @m=%%5&set @tz=%%6
  8. )
  9. for /l %%a in (1,1,3) do (
  10. for %%b in (!n%%a!) do (echo;%%b)
  11. )
  12. set n  &rem 查看变量方便理解
复制代码
3,其它补充
[请看本主题三楼]

如果,看了本贴,还是不理解变量延迟的话,请再看其它相关贴子:

交流贴:call变量延迟与!!变量延迟的优先级问题

浅谈for命令中的变量扩展

[分享]从一个有趣的结果来谈谈批处理变量嵌套

(待续。。。)


学习实践:
【练习-053】按要求补全变量嵌套中的代码

本人的文笔不好,水平有限,有不全和表达不清楚的地方,大家就当作可以开心一下吧。
本着共同学习的出发点,还请各位在一笑而过之后,作出批评指正。谢谢


鸣谢:

batman版主 yslyxqysl版主 zqz0012005版主

指正错误
3

评分人数

    • wankoilz: 经典!PB + 5
    • batman: 从字里行间体会到net的一片苦心!PB + 20
    • Seter: 辛苦了!膜拜NET神牛PB + 18

感谢版主的分享。提高P技术

TOP

还不错,大部分都能弄懂。得要耐心啊

TOP

有点难度,,多看几遍吧。
问题解决后,请在标题前面注明[已解决],并给回答者加分——化繁为简,提高工作效率!

TOP

你为什么不说清楚呢?
echo;%!a%!
得到结果为:$$$
就这一点,通过开启本地环境变量延迟扩展的先后顺序就能得出三种结果。
一旦讲出来就是通篇大论的,说清楚啊拜托。
一开始开启延迟扩展的话,!是无法作为变量名的啊,
也就是说永远无法取得!a的值,因为我们只是为a赋予了值。
你没有开启延迟扩展,便为!a赋值为$$$
之后再开启延迟扩展,才能得到这个$$$的结果。而!在开启延迟扩展的情况下,就是特殊符号,后面你无论跟多少!都无法被输出的。
而如果一直未开启延迟扩展,echo;%!a%!
结果则为 $$$!。
真是的。
踏实一些点.不要着急.你想要的时间都会给你.2

TOP

本帖最后由 shelluserwlb 于 2014-12-26 07:54 编辑

挺复杂的,不过认真看还是看得懂的。

TOP

真复杂啊,严重打击学习积极性

TOP

回复 3楼 的帖子

谢谢分享。
对了哈,我看每一门编程语言都有一本手册。
在网上还没见到过批处理手册,要是本论坛出一本批处理手册就好了。例子要浅显。
可以分批次出。最后在整理。最后是chm格式。

TOP

唉,真是强贴。确实啊,用好FOR,SET,IF真是P是精髓,现在回来看也不是那么容易看明白,真是经典

[ 本帖最后由 wc726842270 于 2011-3-4 15:26 编辑 ]

TOP

看来看明白还得用些时日啊。先收藏了

TOP

欢迎

表示欢迎!非常感谢!

TOP

看的有些云里雾里~~
消化中~~~

TOP

真是好教程

特别感谢楼主一片苦心。菜鸟再这里学习了!!!!!!!强烈支持!!!!

TOP

看了幾遍還是不明白, 但.....
還是再努力吧﹗
多謝netbenton的細心解釋和batman的鼓勵啦"" ^.^

TOP

受益匪浅,受益匪浅!

TOP

返回列表