标题: 两个文本跳行输出(难题)
[打印本页]
作者: bat-zw
时间: 2008-4-1 03:42
标题: 两个文本跳行输出(难题)
===============================================
虽然目前还没有人达到开始的要求主(可能是一个for镶嵌根本做不到),但还是有了比较完美的方案=======>见14楼,同时本人在17楼对这段代码进行了粗略的分析。
===================================
本人苦思多日,终于想到了基本在一个大循环中解决的方案,==>见25楼
文本1.txt
aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
ccccccccccccccccccccccccccccccccccccccc
eeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee
ggggggggggggggggggggggggggggggggggggggg
wwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwww
zzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzz
文本2.txt
hhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhh
iiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiii
jjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjj
nnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnn
要求用批处理输出为
aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
hhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhh
ccccccccccccccccccccccccccccccccccccccc
iiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiii
eeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee
jjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjj
ggggggggggggggggggggggggggggggggggggggg
nnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnn
wwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwww
zzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzz
我写下的如下代码:
@echo off
for /f "tokens=*" %%i in ('findstr /n .* 1.txt') do (
for /f "tokens=*" %%j in ('findstr /n .* 2.txt') do (
set var=%%i
set str=%%j
setlocal enabledelayedexpansion
if "!var:~,1!"=="!str:~,1!" echo !var:~2,50!&echo !str:~2,50!
endlocal
)
)
pause
输出为:
aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
hhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhh
ccccccccccccccccccccccccccccccccccccccc
iiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiiii
eeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee
jjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjj
ggggggggggggggggggggggggggggggggggggggg
nnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnnn
丢了1.txt最后两行,搞了好久都没解决,只好请教大家了(注,要在一个循环中解决)。
[
Last edited by zw19750516 on 2008-4-2 at 09:54 PM ]
作者: 26933062
时间: 2008-4-1 04:12
呵呵,
为何非要在一个循环中完成,你的那个代码,效率是个大问题,1.txt有多少行,就要把2.txt循环多少遍。。。
怎么才算是一个循环啊?我觉得你的这个代码就应该不叫一个循环。所谓一个循环,应该是说只把一个文档循环一次。
另外,1.txt和2.txt的行数不一样,最后1.txt多的行怎么处理?
你给的标准输出,好像也有错误啊,wwwwwwww那一行去哪啦?
:
@echo off
for /f "tokens=1* delims=:" %%a in ('findstr /n .* 1.txt') do (
set __%%a=%%b&set max=%%a
)
for /f "tokens=1* delims=:" %%a in ('findstr /n .* 2.txt') do (
set ++%%a=%%b&set str=%%a
)
if %str% gtr %max% set max=%str%
setlocal enabledelayedexpansion
for /l %%a in (1 1 %max%) do (
if defined __%%a echo !__%%a!
if defined ++%%a echo !++%%a!
)
pause
:
@echo off
set max=-1
for /f "delims=" %%a in (1.txt) do (
echo %%a&set /a max+=1
call :loop %%max%%
)
pause
:loop
if %max% equ 0 (set skip=) else set "skip=skip=%max%"
for /f "%skip% delims=" %%i in (2.txt) do echo %%i&goto :eof
goto :eof
[
Last edited by 26933062 on 2008-4-1 at 04:28 AM ]
作者: bat-zw
时间: 2008-4-1 11:04
Quote: |
Originally posted by 26933062 at 2008-4-1 04:12:
呵呵,
为何非要在一个循环中完成,你的那个代码,效率是个大问题,1.txt有多少行,就要把2.txt循环多少遍。。。
怎么才算是一个循环啊?我觉得 ... |
|
我的意思是在一个for镶嵌中解决这个问题,不然就谈不上难了,呵呵。
作者: huahua0919
时间: 2008-4-1 11:52
@ECHO OFF
findstr /n .* a.txt>>c.txt
echo.>>c.txt
findstr /n .* b.txt>>c.txt
for /l %%i in (1 1 6) do (
findstr %%i: c.txt >>d.txt
)
for /f "delims=: tokens=1*" %%a in (d.txt) do (
echo %%b>>e.txt
)
del c.txt
del d.txt
作者: bat-zw
时间: 2008-4-1 12:09
标题: 不在一个循环中的解决方案:
还是没达到我最初的想法,但通用性还好,2楼、四楼的代码通用性还不行:
1 实际上我们不可能每次都知道到底1.txt和2.txt哪个的行数多(针对2楼的)
2 我们也不可能每次都知道1.txt和2.txt都有多少行(针对4楼的)
如意见不对,请指正,目的只有一个为了提高。
@echo off
for /f "tokens=*" %%i in ('findstr /n .* 1.txt') do (
for /f "tokens=*" %%j in ('findstr /n .* 2.txt') do (
set var=%%i
set str=%%j
setlocal enabledelayedexpansion
if "!var:~,2!"=="!str:~,2!" echo !var:~2,50!&echo !str:~2,50!
endlocal
)
)
for /f "delims=: tokens=1" %%i in ('findstr /n .* 1.txt') do set n=%%i
for /f "delims=: tokens=1" %%j in ('findstr /n .* 2.txt') do set m=%%j
if %n% equ %m% pause&goto :eof
if %n% lss %m% goto 2
:1
set /a n=%m%
for /f "skip=%n%" %%i in (1.txt) do echo %%i
pause&goto :eof
:2
set /a n=%n%
for /f "skip=%n%" %%j in (2.txt) do echo %%j
pause&goto :eof
[
Last edited by zw19750516 on 2008-4-1 at 04:33 PM ]
作者: huahua0919
时间: 2008-4-1 12:57
想知道多少行还不简单
列出两个行后,进行比较不就行了?
作者: abcd
时间: 2008-4-1 13:25
P不容易就换别的嘛。
作者: ansipeter
时间: 2008-4-1 13:52
虽然操作有些麻烦还是可以勉强实现的,不过代码多了点而已,唉
@echo off&setlocal
for /f "delims=: tokens=1*" %%a in ('findstr /n .* 1.txt') do call:ansi %%a "%%b"
goto:eof
:ansi
echo %~2
for /f "delims=: tokens=1*" %%y in ('findstr /n .* 2.txt^|findstr "^%1:.*"') do echo %%z
作者: huahua0919
时间: 2008-4-1 14:07
@ECHO OFF
findstr /n .* a.txt>>c.txt
for /f "delims=: tokens=1" %%i in (c.txt) do set max1=%%i
echo.>>c.txt
findstr /n .* b.txt>>c.txt
for /f "delims=: tokens=1" %%i in (c.txt) do set max2=%%i
set/a max=(max2-max1)-max1
if %max% leq 0 (set/a max=max1) else (set/a max=max2-max1)
for /l %%a in (1 1 %max%) do (
findstr %%a: c.txt >>d.txt
)
for /f "delims=: tokens=1*" %%a in (d.txt) do (
echo %%b>>e.txt
)
这个总应该行
作者: abcd
时间: 2008-4-1 14:14
Quote: |
Originally posted by ansipeter at 2008-4-1 01:52 PM:
虽然操作有些麻烦还是可以勉强实现的,不过代码多了点而已,唉
[code]
@echo off&setlocal
for /f "delims=: tokens=1*" %%a in ('findstr /n .* 1.txt') do call ... |
|
说说怎么讨厌9527?
作者: bat-zw
时间: 2008-4-1 14:31
Quote: |
Originally posted by huahua0919 at 2008-4-1 14:07:
[code]
@ECHO OFF
findstr /n .* a.txt>>c.txt
for /f "delims=: tokens=1" %%i in (c.txt) do set max1=%%i
echo.>>c.txt
findstr /n .* b.txt>>c.txt
for /f "delims=: ... |
|
代码有问题,测试后e.txt内容如下:
11
[
Last edited by zw19750516 on 2008-4-1 at 02:33 PM ]
作者: bat-zw
时间: 2008-4-1 14:33
Quote: |
Originally posted by ansipeter at 2008-4-1 13:52:
虽然操作有些麻烦还是可以勉强实现的,不过代码多了点而已,唉
[code]
@echo off&setlocal
for /f "delims=: tokens=1*" %%a in ('findstr /n .* 1.txt') do call ... |
|
你的和我一楼的代码效果一样,会丢掉行数多的txt中多出的行。
作者: ansipeter
时间: 2008-4-1 14:51
楼上的,你测试了吗?对,我的代码是有些通用性不好,必须在for /f "delims=: tokens=1*" %%a in ('findstr /n .* 1.txt') do call:ansi %%a "%%b"这句放置行多的那个文件,我这里测试通过,如果要是怎么放都可以的话,那添加个判断就可以很好的通用了,唉
作者: terse
时间: 2008-4-1 16:35
我来一个 虽然多一FOR 但有效 怎么多个%%I
@echo off
for /f "delims=" %%i in (a.txt) do >>c.txt echo %%i&call :loop
for /f "delims=" %%i in (b.txt) do if not defined %%i echo %%i>>c.txt
pause
goto :eof
:loop
for /f "delims=" %%i in (b.txt) do if not defined %%i set %%i=A &>>c.txt echo %%i&goto :eof
上面处理不了b.txt重复行 所以改一下:
@echo off
for /f "delims=" %%i in (a.txt) do >>c.txt echo %%i&call :loop
for /f "%n% delims=" %%i in (b.txt) do >>c.txt echo %%i
pause
goto :eof
:loop
for /f "%n% tokens=1* delims=:" %%i in ('findstr /n .* "b.txt"') do set n=skip=%%i &>>c.txt echo %%j&goto :eof
[
Last edited by terse on 2008-4-2 at 12:53 PM ]
作者: huahua0919
时间: 2008-4-1 16:35
楼主可嘉
@ECHO OFF
findstr /n .* a.txt>>c.txt
for /f "delims=: tokens=1" %%i in (c.txt) do set max1=%%i
echo.>>c.txt
findstr /n .* b.txt>>c.txt
findstr /n .* b.txt>>m.txt
for /f "delims=: tokens=1" %%i in (c.txt) do set max2=%%i
set /a max=max1-max2
if %max% leq 0 (set/a max=max2) else (set/a max=max1)
for /l %%a in (1 1 %max%) do (
findstr %%a: c.txt >>d.txt
)
for /f "delims=: tokens=1*" %%a in (d.txt) do (
echo %%b>>e.txt
)
del m.txt&&del c.txt&&del d.txt
作者: bat-zw
时间: 2008-4-1 20:38
Quote: |
Originally posted by terse at 2008-4-1 16:35:
我来一个 虽然多一FOR 但有效 怎么多个%%I
@echo off
for /f "delims=" %%i in (a.txt) do >>c.txt echo %%i&call :loop
for /f "delims=" %%i in (b.txt) do i ... |
|
学习!!!!!!!!!!
作者: bat-zw
时间: 2008-4-1 21:37
Quote: |
@echo off
for /f "delims=" %%i in (a.txt) do >>c.txt echo %%i&call :loop
for /f "delims=" %%i in (b.txt) do if not defined %%i echo %%i>>c.txt
pause
goto :eof
:loop
for /f "delims=" %%i in (b.txt) do if not defined %%i set %%i=A &>>c.txt echo %%i&goto :eof |
|
不好意思,为了得到提高,本人虽然水平有限,还是试着解释这段代码如下:
第一个for循环表示从a.txt中依次读取每行,每读取一行就把这行添加输入到c.txt中然后跳转到第二个for循环中;
第二个for循环的意思是依次读取b.txt中每一行,每读一行就把存在于循环中%%i变量(第二个for循环的%%i),在b.txt中这行中查找一次,如果没发现就把这一行改变变量名为A并添加输入到c.txt中,然后结束再开始第一个for循环,这样后面的每次循环就不会查找这一行了(其变量名为A而不是%%i);
而当第一个大循环也就是第一和第二个for循环结束后,便开始了第三个for语句的循环,在这个for循环中就是一行行地对b.txt查找在当前循环中是否有变量%%i(同样行,其余的行的变量名都变成A了)的存在,如没有就添加输入到c.txt中。
真的好难解释啊!!!!!!!!!!!!
总结:trse巧妙的将三个for循环中的变量都命名为%%i,同时反复运用 if (not) defined语句进行查找识别,成功的实现了对a.txt和b.txt跳行并不丢失行的输出,实在是高,佩服!!!
作者: 26933062
时间: 2008-4-1 21:40
原来是在不知道哪个文件行数多的情况下
如此把2楼代码稍作修改即可达到楼主的要求
:
@echo off&cd.>c.txt
set max=-1
for /f "delims=" %%a in (1.txt) do (
>>c.txt echo %%a&set /a max+=1
call :loop %%max%%
)
set /a max+=1
set flag=a
:loop
if %max% equ 0 (set skip=) else set "skip=skip=%max%"
for /f "%skip% delims=" %%i in (2.txt) do (
>>c.txt echo %%i
if not defined flag goto :eof
)
if not defined flag (goto :eof) else start c.txt&exit
另外: 2 楼的第一个代码到底哪里不行?怎么不通用?
[
Last edited by 26933062 on 2008-4-1 at 09:43 PM ]
作者: bat-zw
时间: 2008-4-1 21:46
标题: 感谢兄弟对本贴的多次关注:
Quote: |
Originally posted by 26933062 at 2008-4-1 21:40:
原来是在不知道哪个文件行数多的情况下
如此把2楼代码稍作修改即可达到楼主的要求
:[code]
@echo off&cd.>c.txt
set max=-1
for /f "delims=" %%a i ... |
|
但你的代码还是有问题,你可测试下看。
作者: 26933062
时间: 2008-4-1 21:51
代码有问题?
我测试没问题啊?
你测试有什么问题?说说啊?
作者: balinger
时间: 2008-4-1 21:58
14楼的思路的确奇妙,令人佩服!
不过还有个小问题,如果 b.txt 中有相同的行,就会出错。
作者: topmcs
时间: 2008-4-2 11:58
高手!!!!!!!!!11
作者: terse
时间: 2008-4-2 12:50
Quote: |
Originally posted by balinger at 2008-4-1 21:58:
14楼的思路的确奇妙,令人佩服!
不过还有个小问题,如果 b.txt 中有相同的行,就会出错。 |
|
确实如此 改这样
@echo off
for /f "delims=" %%i in (a.txt) do echo %%i&call :loop
for /f "%n% delims=" %%i in (b.txt) do echo %%i
pause
goto :eof
:loop
for /f "%n% tokens=1* delims=:" %%i in ('findstr /n .* "b.txt"') do set n=skip=%%i &echo %%j&goto :eof
作者: bat-zw
时间: 2008-4-2 13:51
标题: 高!!!!!!!!!!!!!
for /f "%n% tokens=1* delims=:" %%i in ('findstr /n .* "b.txt"') do set n=skip=%%i &echo %%j&goto :eof
使用"delims=:" "findstr /n" "%%n=skip=%%i" 把%%i变量变为行号,思路的确是太广了。
[
Last edited by zw19750516 on 2008-4-2 at 01:55 PM ]
作者: bat-zw
时间: 2008-4-2 21:26
标题: 基本在一个大循环解决问题的方案:
通用范围为1-999行
@echo off
set n=-1
for /f "delims=" %%i in ('findstr /n .* a.txt') do (
set a=%%i
set /a n+=1
set /a v=n+1
setlocal enabledelayedexpansion
set a=!a:*:=!&echo !a!&call :lp
endlocal
)
for /f "skip=%n% delims=" %%i in ('findstr .* b.txt') do echo %%i
pause&goto :eof
:lp
set m=skip=!n!
if "!m!"=="skip=0" set m=
for /f "delims= %m%" %%i in ('findstr /n .* b.txt') do (
set b=%%i
if "!b:~,1!"=="!v!" set b=!b:*:=!&echo !b!&goto :eof
if "!b:~,2!"=="!v!" set b=!b:*:=!&echo !b!&goto :eof
if "!b:~,3!"=="!v!" set b=!b:*:=!&echo !b!&goto :eof
)
[
Last edited by zw19750516 on 2008-4-2 at 11:47 PM ]