Board logo

标题: 【练习-037】批处理去除文本行前后的空格 [打印本页]

作者: batman    时间: 2009-8-15 01:18     标题: 【练习-037】批处理去除文本行前后的空格

  因为论坛练习题系列中确实缺少37题(这是本人的过失),所以就
以本题来做为37题吧。有文本a.txt如下:
  1. amd cx qu      
  2.     k of woo is qmom                 
  3.       jm ko    ppp     oe    mie    tnnn           
  4.   sis              jb  
复制代码
  要求通过批处理去除文本行的前后空格(空格数是随机和末知的
,且都为半角空格),输出如下(行的前后请加上引号):
  1. "amd cx qu"
  2. "k of woo is qmom"
  3. "jm ko    ppp     oe    mie    tnnn"
  4. "sis              jb"
复制代码
  本题满分10分,原则上以思路为重,视情况加分。
       参考答案如下:
  1. @echo off
  2. for /f "delims=" %%a in (a.txt) do set "a="&call :lp "$%%a$"
  3. pause>nul&goto :eof
  4. :lp
  5. for /f "tokens=1* delims= " %%a in ("%a%%~1") do set "str=%%b"&set "var="
  6. for %%a in (%str: = # %) do call,set "var=%%a%%var%%"
  7. if "%~1" neq "" (set "a=%var:#= %"&call :lp) else echo "%var:#= %"
复制代码


[ 本帖最后由 batman 于 2009-8-15 18:44 编辑 ]
作者: netbenton    时间: 2009-8-15 08:13

以前在cn - dos见过这种方法:
  1. @echo off
  2. set "abc=                    33 ff cc                  "
  3. for /f "tokens=*" %%a in ("%abc%") do echo;"%%~nxa"
复制代码


[ 本帖最后由 batman 于 2009-8-15 18:45 编辑 ]
作者: batman    时间: 2009-8-15 11:48     标题: 回复 2楼 的帖子

&&震撼!学习了。想不到变量扩展能强到如此地步,
但就是不明白其中的处理机制,希望有高人能明示
一二。
   另请大家不要局限于现有的解决方案,把思路充
分打开,“八仙过海各显神通”,这也是本人不断出
题的初衷了。
作者: kevinpeng    时间: 2009-8-15 12:52     标题: 回复 2楼 的帖子

哪位兄台能否解释一下这样为什么也可以,%%~nxa不是指的文件名及扩展名么?
作者: keen    时间: 2009-8-15 13:46

用了5个FOR:
  1. @echo off&setlocal enabledelayedexpansion
  2. for /f "tokens=*" %%i in (a.txt) do (
  3.     set str=%%i
  4.     for %%a in (%%i) do set str1=%%a
  5.     for %%h in (!str1!) do (
  6.         for %%m in (!str1!#) do set hh=!str:%%h=%%m!
  7.         for /f "delims=#" %%x in ("!hh!") do echo "%%x"
  8.     )
  9. )
  10. pause
复制代码

作者: keen    时间: 2009-8-15 14:48

只能去掉末尾100个空格,要是大于100个空格,必须改动for /l中的数字,这也是此代码的不合理之处:
  1. @echo off&setlocal enabledelayedexpansion
  2. for /f "tokens=*" %%i in (a.txt) do (
  3.     set str=%%i
  4.     call :lp
  5. )
  6. pause&exit/b
  7. :lp
  8. for /l %%a in (-1 -1 -100) do (
  9.         set "str1=!str:~%%a,1!"
  10.         if not "!str1!"==" " (
  11.            set n=%%a&set /a n+=1
  12.            for %%i in (!n!) do set "str1=!str:~0,%%i!"
  13.            echo "!str1!"&goto :eof
  14.         )
  15. )
复制代码

作者: 基拉freedom    时间: 2009-8-15 15:49


测试了下 2L的
发现只需要用~n就好了(用x也许有特殊的情况吧)................................
  1. @echo off
  2. for /f "tokens=*" %%a in (a.txt) do (
  3. echo "%%~na"
  4. )
  5. pause
复制代码
%~nI        - 仅将 %I 扩充到一个文件名
自己想了想
原因也许是:
对于文件名(这里指的是文件夹) win规定了开头和末尾都不允许有空格.......于是有的就自动去掉了
..
用md也无法建立前头和末尾有空格的文件夹
作者: 基拉freedom    时间: 2009-8-15 16:13

对于以上 我的想法
想出了一种dir的方法 建立文件夹 然后删除
这个方法最要命的就是 一定要放在一个没有文件夹的目录下 否则就................
  1. @echo off
  2. for /f "tokens=*" %%a in (1.txt) do (
  3. md "%%a"
  4. for /f "tokens=*" %%t in ('dir /b *.') do echo "%%t"
  5. rd "%%a"
  6. )
  7. pause
复制代码
这道题目等多点人解答 我再放到chm里去^_^
作者: 基拉freedom    时间: 2009-8-15 16:17

以上的改进是建立双层的文件夹 呼呼
把思路提出来 以后再改了 做完才想到的
要下网了 8
作者: kevinpeng    时间: 2009-8-15 17:05

其实可以创建一个无字节的文件名,方法就是用截取字符串的办法可以把原有的文件名换没了!如:
%str:filename=%
作者: netbenton    时间: 2009-8-15 17:52

考虑了特殊情况:即存在":" "<>" "\" "."
  1. @echo off
  2. set "abc=   33 ff c:\ab\exe.ffg  00 0 0-00-><   "
  3. set "abc=%abc:0=-0-%"
  4. set "abc=%abc::=a0%"
  5. set "abc=%abc:\=b0%"
  6. for /f "tokens=*" %%a in ("%abc%") do set "abc=%%~nxa"
  7. set "abc=%abc:b0=\%"
  8. set "abc=%abc:a0=:%"
  9. set "abc=%abc:-0-=0%"
  10. echo;"%abc%"
复制代码

作者: terse    时间: 2009-8-15 18:32

  1. @echo off
  2. set "str=   33 ff c:\ab\exe.ffg  echo;00 0 0-00->55.txt<   "
  3. for /f "tokens=*" %%i in ("%str%") do set "str=%%i"
  4. :lp
  5. if "%str:~-1%"==" " set "str=%str:~0,-1%"&goto lp
  6. echo "%str%"
  7. pause
复制代码

作者: batman    时间: 2009-8-15 18:49     标题: 回复 12楼 的帖子

逐字法效率一直是问题,这点是无法避免的。。。
作者: terse    时间: 2009-8-15 20:30

这样效率提高点吧
  1. @echo off
  2. set "str=   33 ff c:\ab\exe.ffg  echo;00 0 0-00->55.txt<   "
  3. for /f "tokens=*" %%i in ("%str%") do (
  4.    for %%j in ("c:\%%i") do set "str=%%~pnxj"
  5. )
  6. echo "%str:~1%"
  7. pause
复制代码

作者: batman    时间: 2009-8-15 20:39     标题: 回复 5楼 的帖子

代码是否可以做如下修改(去掉两个for),但文本中如存在#字符这个代码就挂掉了:
  1. @echo off&setlocal enabledelayedexpansion
  2. for /f "tokens=*" %%i in (a.txt) do (
  3.     set str=%%i
  4.     for %%a in (%%i) do set str1=%%a
  5.     call, set hh=%%str:!str1!=!str1!#%%
  6.     for /f "delims=#" %%x in ("!hh!") do echo "%%x"
  7. )
  8. pause
复制代码

[ 本帖最后由 batman 于 2009-8-15 20:42 编辑 ]
作者: keen    时间: 2009-8-15 20:51     标题: 回复 15楼 的帖子

之所以不用call,是以为随风在一篇讨论效率的帖子中说过,如果出现这种方式(call, set hh=%%str:!str1!=!str1!#%%),还是用for的好。
具体参看:
http://bbs.bathome.net/viewthrea ... ght=%2B%CB%E6%B7%E7
作者: wxcute    时间: 2009-8-15 21:04     标题: 今天回到家了,我也来凑个数^_^

  1. @echo off
  2. for /f "tokens=* delims= " %%s in (a.txt) do call :str %%s
  3. pause
  4. :str
  5. echo "%*"
复制代码

作者: batman    时间: 2009-8-15 21:07

&&修改下我顶楼的代码,顶楼原来是想通用一系列的技巧来简化代码,
结果是付出了效率上的代价,现提高代码运行效率如下:
  1. @echo off&setlocal enabledelayedexpansion
  2. for /f "delims=" %%a in (a.txt) do (
  3.     for /f "tokens=*" %%a in ("%%a") do set "str=%%a"&set "str=!str: = # !"&set "var="
  4.     for %%a in (!str!) do set "var=%%a!var!"
  5.     set "var=!var:#= !"
  6.     for /f "tokens=*" %%a in ("!var!") do set "str=%%a"&set "str=!str: = # !"&set "var="
  7.     for %%a in (!str!) do set "var=%%a!var!"
  8.     echo "!var:#= !"
  9. )
  10. pause>nul
复制代码

作者: shqf    时间: 2009-8-15 21:47

综合一下,还可如下
@echo off
for /f "tokens=* delims= " %%s in (a.txt) do echo "%%~nxs"
pause
作者: Lumiere    时间: 2009-8-16 00:32

  1. @echo off&setlocal enabledelayedexpansion
  2. for /f "tokens=*" %%a in (a.txt) do set "var=%%a"&call :loop&echo "!var!">>b.txt
  3. goto :eof
  4. :loop
  5. set "check=!var:~-1!"&if "!check!"==" " set "var=!var:~0,-1!"&goto loop
复制代码

[ 本帖最后由 Lumiere 于 2009-8-16 00:36 编辑 ]
作者: Lumiere    时间: 2009-8-16 20:38

原帖由 wxcute 于 2009-8-15 21:04 发表
@echo off
for /f "tokens=* delims= " %%s in (a.txt) do call :str %%s
pause
:str
echo "%*"

你的delims= 似乎是多余的吧?tokens=*不是不会分节字符串的么?
还有啊,%*我现在还不是太明白,能给讲解一下么
作者: terse    时间: 2009-8-16 21:12

不考虑特殊字符的话 call出来 echo %* 效率可以
利用%%~nxi扩展的话 路径的处理上也有问题
我再用替换法  发现效率比不上echo %*
文本中有一个引号的话也出错 两引号之间没空格可以处理
  1. @echo off&setlocal enabledelayedexpansion
  2. for /f "tokens=*" %%i in (a.txt) do (
  3.     set var=%%i&set str=&set "k= "
  4.     set "var=!var: =" "!"
  5.     for %%a in ("!var!") do if %%a neq "" (
  6.         set "str=!str!!k!%%~a"&set "k= "
  7.       ) else set "k= !k!"
  8.         echo "!str:~1!"
  9.    )
  10. pause
复制代码

作者: leap    时间: 2010-10-28 14:22     标题: 回复 7楼 的帖子

  1. @echo on&setlocal enabledelayedexpansion
  2. for /f "tokens=*" %%i in (a.txt) do (
  3. md "d:\2\%%i")
  4. for /f "delims=" %%i in ('dir /b "d:\2\"') do echo "%%i">>2.txt
  5. rd/s/q d:\2\
  6. pause
复制代码

作者: Hello123World    时间: 2011-7-20 19:34

参照2楼的算法,但不知道原理。
  1. @echo off
  2. For /f "Tokens=*" %%i in (a.txt) do (For /f "tokens=*" %%a in ("%%i") do Echo "%%~nxa")
  3. pause
复制代码

作者: Seder    时间: 2011-9-3 13:16

有分加那么好??
  1. sed -e "s/^[ ]*//;s/[ ]*$//;s/\(.*\)/\"\1\"/" test.txt
复制代码

作者: zaixinxiangnian    时间: 2011-9-3 20:39

回复 5# keen


    你这个还好理解些,但到这里就想不通了。  为什么有个#号?如果有时间每句给解释下就有富了

for %%h in (!str1!) do (

        for %%m in (!str1!#) do set hh=!str:%%h=%%m!

        for /f "delims=#" %%x in ("!hh!") do echo "%%x"
作者: civen    时间: 2011-9-26 19:44

&&修改下我顶楼的代码,顶楼原来是想通用一系列的技巧来简化代码,
结果是付出了效率上的代价,现提高代码 ...
batman 发表于 2009-8-15 21:07



有点不完美. 空行带空格的还得输出一个空格
作者: yashuer    时间: 2012-7-5 12:29

  1. 发个sed的
  2. #!/bin/bash
  3. # Fri Jun 22 15:43:42 CST 2012
  4. # Sun Jul  1 19:52:32 CST 2012
  5. # The code is listed as follows
  6. echo >uu.txt
  7. sed 's/^\s*\(.*[a-z]\)\s*$/\1/g' temp > uu.txt
  8. exit
复制代码

作者: 踏沙行    时间: 2018-11-6 09:12

本帖最后由 踏沙行 于 2018-11-6 09:17 编辑

回复 2# netbenton
旧贴重提,不知道论坛是否有禁止性规定?
使用扩展变量%~nx,固定速度快,但是假如,文本中包含有路径符号 \ 的话,就悲剧了。
比如:1.txt内容如下:
   A11 !b11 H\c11 ^E
      A13 B13\22\123 #C13 E
作者: 踏沙行    时间: 2018-11-6 09:14

本帖最后由 踏沙行 于 2018-11-6 09:19 编辑
wxcute 发表于 2009-8-15 21:04

原贴代码:
  1. @echo off
  2. for /f "tokens=* delims= " %%s in (a.txt) do call :str %%s
  3. pause
  4. :str
  5. echo "%*"
复制代码
这个方法也很好,但是有个困惑:怎么取得返回值?
如果我想把每行首尾的空格去掉后,再将结果返回给主程序中的某个变量,该怎么实现?
作者: Batcher    时间: 2020-10-21 10:15

  1. @echo off
  2. (for /f "tokens=*" %%i in ('type "C:\1.txt"') do (
  3.     echo,%%~nxi
  4. ))>"D:\2.txt"
复制代码





欢迎光临 批处理之家 (http://www.bathome.net/) Powered by Discuz! 7.2