CStdioFile  的 Readstring是一行行往下读的,现在需要从最后一行开始读,往上只要读几十行左右的数据就可以。
(文件可能很大,可能有几十万行,所以从上往下读太费劲)该如何实现呢??? 求指点

解决方案 »

  1.   

    仅供参考:#include <stdio.h>
    #include <io.h>
    #include <fcntl.h>
    #include <share.h>
    #include <conio.h>
    int fh;
    int c,r;
    __int64 offset,offset1,i64,n64,l64,r64;
    char ln[16000+1];
    int main(int argc,char **argv) {
        if (argc<2) {
            fprintf(stderr,"Usage:%s filename.txt [10]\nto reverse lines of file [default the last 10 lines,-1 for all lines].\n",argv[0]);
            return 1;
        }
        if (argc==2) n64=10i64;
        if (argc>2) sscanf(argv[2],"%I64d",&n64);
        fh=_sopen(argv[1],_O_BINARY|_O_RDONLY|_O_RANDOM,_SH_DENYNO);
        if (-1==fh) {
            fprintf(stderr,"Can not _sopen file %s!\n",argv[1]);
            return 2;
        }
        l64=_filelengthi64(fh);if (l64==0i64) {_close(fh);return 0;}
        offset=l64;
        offset1=_lseeki64(fh,-1i64,SEEK_END);
        if (offset1==-1i64) {_close(fh);return 3;}
        i64=0i64;
        while (1) {
            if (_read(fh,&c,1)<1) break;//
            if (('\n'==c && offset1<l64-1i64) || offset1==0i64) {
                if (offset1!=0i64) offset1++;
                if (_lseeki64(fh,offset1,SEEK_SET)==-1i64) break;//
                r64=offset-offset1;
                if (r64<=16000i64) {
                    r=_read(fh,ln,(unsigned int)r64);
                    if (r<=0) break;//
                } else {
                    r=16000;
                    while (1) {
                        if (r64>(__int64)r) {
                            r64-=(__int64)r;
                            if (1i64==r64 || 2i64==r64) {
                                r=8000;
                                r64+=(__int64)r;
                            }
                        } else {
                            r=(int)r64;
                            r64=0i64;
                        }
                        r=_read(fh,ln,r);
                        if (r<=0) break;//
                        if (0i64==r64) break;//
                        ln[r]=0;
                        printf("%s",ln);
                        r=16000;
                    }
                    if (r<=0) break;//
                }
                ln[r]=0;
                if (r>=2 && ln[r-1]=='\n' && ln[r-2]=='\r') {ln[r-2]='\n';ln[r-1]=0;r--;}//避免输出重定向到文件时行尾多出'\r'
                if (i64==0I64) {
                    if ('\n'!=ln[r-1]) printf("%s\n",ln);
                    else               printf("%s",ln);
                } else {
                    printf("%s",ln);
                }
                i64++;
                if ((i64%10000)==0) cprintf("\r%I64d/%I64d",offset1,l64);
                if (n64>0i64 && i64>=n64) break;//
                offset=offset1;
                _lseeki64(fh,offset1-2i64,SEEK_SET);
                offset1-=2i64;
            } else {
                _lseeki64(fh,-2i64,SEEK_CUR);
                offset1--;
            }
        }
        _close(fh);
        return 0;
    }
      

  2.   

    定义位置变量P1,P2
    文件指针移动到文件结尾,
    读出段数据固定大小数据,P2指向末尾
    寻找\r\n 为P1
    从P1处读取到P2
    令P2=P1,继续读前一行块没有数据,再读出一块,注意要读出的块要适当的重叠
      

  3.   

    获取文件大小 字节,
    用win32 api 文件操作。设置读取指针SetFilePointer  文件大小 -你需要读取的尾部数据大小。
    读出尾部块。
      

  4.   

    赵老师,能不能讲讲思路,我看您一上来大段大段的代码,新人接受起来好难。代码功能归根结底不是别人帮自己看或讲解或注释出来的;而是被自己静下心来花足够长的时间和精力亲自动手单步或设断点或对执行到某步获得的中间结果显示或写到日志文件中一步一步分析出来的。
    提醒:再牛×的老师也无法代替学生自己领悟和上厕所!
    单步调试和设断点调试(VS IDE中编译连接通过以后,按F10或F11键单步执行,按Shift+F11退出当前函数;在某行按F9设断点后按F5执行停在该断点处。)是程序员必须掌握的技能之一。查MSDN是Windows程序员必须掌握的技能之一。在我眼里,凡是不超过1000行的代码,写注释或用文字/口头解释都是多余!
      

  5.   

    命令行解释器概述
    命令行解释器是一个单独的软件程序,它可以在用户和操作系统之间提供直接的通讯。非图形命令行解释器用户界面提供运行基于字符的应用程序和实用程序的环境。通过使用类似于 MS-DOS 命令解释程序 Command.com 的单独字符,命令行解释器执行程序并在屏幕上显示其输出。Windows 服务器操作系统命令行解释器使用命令解释程序 Cmd.exe(该程序加载应用程序并指导应用程序之间的信息流动)将用户输入转换为操作系统可理解的形式。
    可以使用命令行解释器创建和编辑可自动执行常规任务的批处理文件(也称作脚本)。例如,可以使用脚本自动管理用户帐户或夜间备份。还可以使用 Windows 脚本宿主 CScript 的命令行版本在命令行解释器中运行更高级的脚本。详细信息,请参阅运行 Windows 脚本宿主。通过使用批处理文件来执行操作,可比使用用户界面更有效率。批处理文件接受命令行上可用的所有命令。有关批处理文件和脚本的详细信息,请参阅使用批处理文件。
    您可以自定义命令提示符窗口以易于查看并增强对程序运行方式的控制。有关自定义命令提示符窗口的详细信息,请参阅配置命令提示符。使用命令语法
    语法按命令及所带参数必须遵循的键入顺序出现。下面的 xcopy 命令示例显示了各种语法文本格式:xcopy Source [Destination] [/w] [/p] [/c] [/v] [/q] [/f] [/l] [/g] [/d[:MM-DD-YYYY]] [/u] [/i] [/s [/e]] [/t] [/k] [/r] [/h] [{/a | /m}] [/n] [/o] [/x] [/exclude:File1[+[File2]][+[File3]] [{/y | /-y}] [/z]下表说明如何解释不同的文本格式。格式化图例
    格式                                             含义
    斜体                                             用户必须提供的信息
    粗体                                             用户必须准确键入的要显示的元素
    省略号(...)                                      在命令行中可多次重复的参数
    中括号([])                                       可选项
    大括号({});选项用竖线(|)分隔。例如:{even|odd}  用户必须从选项集合中选择一个
    Courier字体                                      代码或程序输出使用多个命令和条件处理符号
    使用条件处理符号可以在单个命令行或脚本中运行多个命令。通过条件处理符号运行多个命令时,条件处理符号右边的命令根据条件处理符号左边命令的执行结果来发挥作用。例如,只有在前一个命令失败的情况下才可能需要运行一个新命令。或者,只有在前一个命令成功时才可能需要运行一个新命令。
    可以使用下表列出的特殊字符来传递多个命令。字符       语法                            定义
    & [...]    Command1 & Command2             用来分隔一个命令行中的多个命令。Cmd.exe 运行第一个命令,然后运行第二个命令。
    && [...]   Command1 && Command2            只有在符号 && 前面的命令成功时,才运行该符号后面的命令。Cmd.exe 运行第一个命令,然后只有在第一个命令运行成功时才运行第二个命令。
    || [...]   Command1 || Command2            只有在符号 || 前面的命令失败时,才运行符号 || 后面的命令。Cmd.exe 运行第一个命令,然后只有在第一个命令未能运行成功(接收到大于零的错误代码)时才运行第二个命令。
    ( ) [...]  (Command1 & Command2)           用来分组或嵌套多个命令。
    ;或者,     Command1 Parameter1;Parameter2  用来分隔命令参数。注意
    “与”符号 (&)、管道符号 (|) 以及括号 () 是特殊字符,将它们作为参数传递时,必须在其前面加上转义字符 (^) 或引号。
    如果某个命令成功完成操作,该命令就返回零 (0) 退出代码或不返回任何退出代码。有关退出代码的详细信息,请参阅 Microsoft Windows 部署和资源工具包。嵌套命令行解释器
    通过在命令提示符下打开新的 Cmd.exe 实例,可以在 Cmd.exe 内嵌套命令行解释器。默认情况下,Cmd.exe 的每个实例继承其父 Cmd.exe 应用程序的环境。通过嵌套 Cmd.exe 的实例,可以更改局部环境,而不会影响 Cmd.exe 的父应用程序。这使您能够保留 Cmd.exe 的原始环境,并在终止嵌套的命令行解释器之后返回到原始环境。但是在嵌套的命令行解释器中所做的更改将不会被保存。
    要嵌套命令行解释器,请在命令提示符下键入:cmd出现类似于下面内容的消息:Microsoft (R) Windows Server 2003 Standard Edition (TM)
    (C) 版权所有 1985-2002 Microsoft Corp.
    要关闭嵌套的命令行解释器,请键入 exit。使用 setlocal 和 endlocal 命令,可以在 Cmd.exe 的实例中(或在脚本中)进一步将更改局部化。Setlocal 创建局部作用范围,而 endlocal 终止局部作用范围。在 setlocal 和 endlocal 作用范围内所做的更改将会被放弃,从而保持原始环境不变。这两个命令的嵌套最高可达到 32 级。有关 setlocal 和 endlocal 命令的详细信息,请参阅 Setlocal 和 Endlocal。将环境变量与 Cmd.exe 一起使用
    Cmd.exe 命令行解释器环境由确定命令行解释器和操作系统行为的变量进行定义。可以使用两种类型的环境变量(系统和局部)来定义命令行解释器环境或整个操作系统环境的行为。系统环境变量定义全局操作系统环境的行为。局部环境变量定义 Cmd.exe 当前实例环境的行为。
    系统环境变量预置于操作系统之中,并可用于所有 Windows 服务器操作系统进程。只有具有管理凭据的用户才可以更改系统变量。这些变量最常用于登录脚本。
    局部环境变量只有在创建变量时针对的目标用户登录到计算机时才有效。HKEY_CURRENT_USER 配置单元中设置的局部变量只对当前用户有效,但它们可定义全局操作系统环境的行为。下表按优先顺序的降序描述变量的各种类型:
    1.内置系统变量
    2.在 HKEY_LOCAL_MACHINE 配置单元中找到的系统变量
    3.在 HKEY_CURRENT_USER 配置单元中找到的局部变量
    4.在 Autoexec.bat 文件中设置的所有环境变量和路径
    5.在登录脚本(如果有的话)中设置的所有环境变量
    6.在脚本或批处理文件中交互使用的变量在命令行解释器中,Cmd.exe 的每个实例都继承其父应用程序的环境。因此,可以在不影响父应用程序环境的情况下更改新的 Cmd.exe 环境中的变量。
    下表列出 Windows server operating system 的系统和本地环境变量。变量                     类型       描述
    %ALLUSERSPROFILE%        本地       返回“所有用户”配置文件的位置。
    %APPDATA%                本地       返回默认情况下应用程序存储数据的位置。
    %CD%                     本地       返回当前目录字符串。
    %CMDCMDLINE%             本地       返回用来启动当前的Cmd.exe的准确命令行。
    %CMDEXTVERSION%          系统       返回当前的“命令处理程序扩展”的版本号。
    %COMPUTERNAME%           系统       返回计算机的名称。
    %COMSPEC%                系统       返回命令行解释器可执行程序的准确路径。
    %DATE%                   系统       返回当前日期。使用与date/t命令相同的格式。由Cmd.exe生成。有关date命令的详细信息,请参阅Date。
    %ERRORLEVEL%             系统       返回上一条命令的错误代码。通常用非零值表示错误。
    %HOMEDRIVE%              系统       返回连接到用户主目录的本地工作站驱动器号。基于主目录值而设置。用户主目录是在“本地用户和组”中指定的。
    %HOMEPATH%               系统       返回用户主目录的完整路径。基于主目录值而设置。用户主目录是在“本地用户和组”中指定的。
    %HOMESHARE%              系统       返回用户的共享主目录的网络路径。基于主目录值而设置。用户主目录是在“本地用户和组”中指定的。
    %LOGONSERVER%            本地       返回验证当前登录会话的域控制器的名称。
    %NUMBER_OF_PROCESSORS%   系统       指定安装在计算机上的处理器的数目。
    %OS%                     系统       返回操作系统名称。Windows2000显示其操作系统为Windows_NT。
    %PATH%                   系统       指定可执行文件的搜索路径。
    %PATHEXT%                系统       返回操作系统认为可执行的文件扩展名的列表。
    %PROCESSOR_ARCHITECTURE% 系统       返回处理器的芯片体系结构。值:x86或IA64(基于Itanium)。
    %PROCESSOR_IDENTFIER%    系统       返回处理器说明。
    %PROCESSOR_LEVEL%        系统       返回计算机上安装的处理器的型号。
    %PROCESSOR_REVISION%     系统       返回处理器的版本号。
    %PROMPT%                 本地       返回当前解释程序的命令提示符设置。由Cmd.exe生成。
    %RANDOM%                 系统       返回0到32767之间的任意十进制数字。由Cmd.exe生成。
    %SYSTEMDRIVE%            系统       返回包含Windowsserveroperatingsystem根目录(即系统根目录)的驱动器。
    %SYSTEMROOT%             系统       返回Windowsserveroperatingsystem根目录的位置。
    %TEMP%和%TMP%            系统和用户 返回对当前登录用户可用的应用程序所使用的默认临时目录。有些应用程序需要 TEMP,而其他应用程序则需要 TMP。
    %TIME%                   系统       返回当前时间。使用与time/t命令相同的格式。由Cmd.exe生成。有关time命令的详细信息,请参阅Time。
    %USERDOMAIN%             本地       返回包含用户帐户的域的名称。
    %USERNAME%               本地       返回当前登录的用户的名称。
    %USERPROFILE%            本地       返回当前用户的配置文件的位置。
    %WINDIR%                 系统       返回操作系统目录的位置。设置环境变量
    使用 set 命令创建、更改、删除或显示环境变量。set 命令只更改当前解释器环境中的变量。要查看变量,请在命令提示符下键入:set VariableName要添加变量,请在命令提示符下键入:set variablename=Value要删除变量,请在命令提示符下键入:set VariableName=可以将大多数字符用作变量值,其中包括空格。如果使用特殊字符 <、>、|、& 或 ^,则必须在它们前面加上转义字符 (^) 或引号。如果使用引号,则必须将引号作为值的组成部分,因为等号后面的任何内容都会被视为值。请考虑下列示例:要创建变量值 new&name,请键入:
    set varname=new^&name要创建变量值 "new&name",请键入:
    set varname="new&name"如果在命令提示符下键入 set varname=new&name,就会出现与下面内容类似的错误消息:
    “'name' 不是内部或外部文件,也不是可运行的程序或批处理文件。”
    变量名不区分大小写。但是,set 显示的变量与您键入的完全相同。可以在变量名中结合使用大写字母和小写字母,这样可以使代码更具有可读性(例如,UserName)。注意单个环境变量的最大大小为 8192 字节。
    所有环境变量的大小总和(包括变量名和等号)最大为 65,536 KB。
    替换环境变量值
    要在命令行或脚本中启用变量值替换,请将变量名包括在百分号之中(即,%VariableName%)。使用百分号可以确保 Cmd.exe 引用变量值,而不是进行文字比较。为变量名定义变量值之后,请将变量名包括在百分号之中。Cmd.exe 搜索该变量名的所有实例,并用定义的变量值进行替换。例如,如果创建包含不同值(例如,用户名)的脚本,并且想要用这些值为每个用户定义 USERNAME 环境变量,可以使用包括在百分号之中的 USERNAME 来编写代码。运行此脚本时,Cmd.exe 将用变量值替换 %USERNAME%,这样就消除了为每个用户手动执行此任务的必要。变量替换是不可递归的。Cmd.exe 检查变量一次。有关变量替换的详细信息,请参阅 For 和 Call。
      

  6.   

    现在的码农竟然99%都不会在cmd窗口中输入cd命令设置当前目录为程序所在目录,输入程序名运行程序了!