Windows批处理:同时迭代多个文件行

时间:2012-08-23 09:49:08

标签: file cmd batch-file

是否可以同时迭代多个文件的行?我需要通过连接两个不同文件中的行来输出文件(并添加一些额外的单词)。

我知道CMD命令:

FOR %variable IN (set) DO command [command-parameters]

但如果你需要同时迭代两个文件,这没有任何帮助。 更确切地说,我需要以下内容:

While the first file is not ended
   line1 <- READ a line from the first file
   line2 <- READ a line from the second file
   WRITE on a third file line1 + line2

我正在寻找一种方法来完成我之前在 dos batch 文件中描述的内容。谢谢!

4 个答案:

答案 0 :(得分:8)

我知道3种基本技术,可以从中衍生出大量的杂种。

选项1 (此选项与wmz解决方案非常相似)

使用FOR / F读取文件1,使用SKIP选项读取带有第二个FOR / F的文件2。必须在CALLed子例程中读取文件2,以便可以在不中断文件1循环的情况下中断循环。

限制:

  • 空行将导致行不同步。空白行包含在跳过计数中,但不会被FOR / F读取。
  • 由于默认的EOL选项,将无法读取以;开头的行。如有必要,可以通过将EOL设置为新行字符来解决此问题。请参阅HOW TO: FOR /F Disabling EOF or using a quote as delim
  • 行的最大长度限制为8191个字节。

由于CALL,此选项很慢,因为每行必须读取第二个文件一次。

编辑 - 如果第二个文件提前结束,则修复的代码仍会输出一行

@echo off
setlocal disableDelayedExpansion
set "file1=a.txt"
set "file2=b.txt"
set "out=out.txt"

set /a cnt=0
set "skip="

>"%out%" (
  for /f "usebackq delims=" %%A in ("%file1%") do (
    set "found="
    call :readFile2
    if not defined found (echo %%A - )
  )
)
type "%out%"
exit /b

:readFile2
for /f "usebackq %skip% delims=" %%B in ("%file2%") do (
  set found=1
  echo %%A - %%B"
  goto :break
)
:break
set /a cnt+=1
set "skip=skip=%cnt%"
exit /b


选项2

此选项通过使用FINDSTR为每行添加行号后跟冒号来解决空行问题。 FINDSTR用于只读取文件2中的第n行,因此不需要突破第二个循环。

限制:

  • 领先的冒号将被剥夺。使用额外的代码可以消除这种限制,但这会使它变得更复杂和更慢。
  • 行的最大长度限制为8191个字节。

此选项甚至比选项1

编辑 - 如果第二个文件提前结束,则修复的代码仍会输出一行

@echo off
setlocal disableDelayedExpansion
set "file1=a.txt"
set "file2=b.txt"
set "out=out.txt"

>"%file2%.tmp" findstr /n "^" "%file2%"
>"%out%" (
  for /f "tokens=1* delims=:" %%A in ('findstr /n "^" "%file1%"') do (
    set "found="
    for /f "tokens=1* delims=:" %%a in ('findstr "^%%A:" "%file2%.tmp"') do (
      set found=1
      echo %%B - %%b
    )
    if not defined found (echo %%B - )
  )
)
del "%file2%.tmp"
type "%out%"


选项3

使用SET / P读取这两个文件。 FIND用于获取文件1中行数的计数,因为SET / P无法区分空行和文件结尾。

此选项消除了许多限制和复杂性,但引入了自身的局限性。

限制:

  • 行必须使用<CR><LF>的Windows样式行终止符。 Unix风格<LF>不起作用。
  • 行限制为1021字节
  • 从每一行中删除尾随控制字符。

此选项是迄今为止最快的选项。只要限制是可以接受的,这是优选的。

@echo off
setlocal enableDelayedExpansion
set "file1=a.txt"
set "file2=b.txt"
set "out=out.txt"

for /f %%N in ('type "%file1%"^|find /c /v ""') do set "cnt=%%N"
>"%out%" 9<"%file1%" <"%file2%" (
  for /l %%N in (1 1 %cnt%) do (
    set "ln1="
    set "ln2="
    <&9 set /p "ln1="
    set /p "ln2="
    echo !ln1! - !ln2!
  )
)
type "%out%"

答案 1 :(得分:2)

现在我必须交付: - )

@echo off
setlocal disabledelayedexpansion
set count=0
for /f "delims=" %%F in (outer.txt) do (
  set fline=%%F
  call :inner %%count%%
  call echo %%fline%%
  set /a count+=1  

)
goto :eof

:inner
if %1 neq 0 set skipline=skip=%1 
  for /f "%skipline% delims=" %%S  in (inner.txt) do (
    set fline=%%F%%S
    goto :eof
  )

请注意,某些类型的输入可能会失败,例如。当行包含&或其他一些由shell专门处理的字符时

答案 2 :(得分:0)

追加文件内容:

echo FOO > foo.txt
echo BAR > bar.txt
type foo.txt > both.txt
echo "Both" >> both.txt
type bar.txt >> both.txt

答案 3 :(得分:0)

我必须为2个字符串执行此操作(可能会为其他人派上用场)

@echo off
setlocal enableextensions enabledelayedexpansion

set string1=token1,token2,token3
set string2=2ndtoken1,2ndtoken2,2ndtoken3

for %%A in (%string1%) do (
    echo From String1: %%A

    set str2=
    for %%S in (!string2!) do (
        if not defined str2 (
            set str2=%%S

            echo !str2!%>tempfile.txt
            FOR %%? IN (tempfile.txt) DO ( SET /A strlength=%%~z? - 1 )

            REM echo StrLen=!strlength!
            echo From String2: !str2!

            call set string2=%%string2:~!strlength!%%
            REM echo String2: !string2!
        )
    )
)