编写程序,将一个.cpp文件中的注释,即以"/*......*/":括起来的文本和以“//”开始的

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了编写程序,将一个.cpp文件中的注释,即以"/*......*/":括起来的文本和以“//”开始的相关的知识,希望对你有一定的参考价值。

编写程序,将一个.cpp文件中的注释,即以"/*......*/":括起来的文本和以“//”开始的文本,全部删除,并另存为一个新.cpp文件。 .cpp文件请自行选择。

做一个状态机

下面的状态表示:
0: 正在分析
1:读到第一个/
2:读完第二个/,"//...
3:读到 "/*...
4:读到 "/*...*
5:读到第一个"
6:读到字符串里的转义符 "...\\
7:找到注释
0和7是等价的状态,但是可以用7状态做些特别的事情,比如这个时候删除刚找到的注释等。最后的程序如下,输入输出还是用C的文件比较方便:

#include "stdio.h"
#include "string"
char fsm[8][128];
void initfsm()

const int line_len=sizeof(char)*128;
memset(fsm[0],0,line_len);
memset(fsm[1],0,line_len);
memset(fsm[2],2,line_len);
memset(fsm[3],3,line_len);
memset(fsm[4],3,line_len);
memset(fsm[5],5,line_len);
memset(fsm[6],5,line_len);
memset(fsm[7],0,line_len);
fsm[0][\'/\']=1;
fsm[0][\'"\']=5;
fsm[1][\'/\']=2;
fsm[1][\'*\']=3;
fsm[1][\'"\']=5;
fsm[2][\'\\n\']=7;
fsm[3][\'*\']=4;
fsm[4][\'/\']=7;
fsm[4][\'*\']=4;
fsm[5][\'"\']=0;
fsm[5][\'\\\\\']=6;
fsm[7][\'/\']=1;
fsm[7][\'"\']=5;

int main()

int state=0;
char c;
std::string s;
FILE *fin=fopen("e:\\\\in.txt","r");
FILE *fout=fopen("e:\\\\out.txt","w");
initfsm();
while(fscanf(fin,"%c",&c)!=EOF)

state=fsm[state][c];
s+=c;
switch(state)

case 0:
fprintf(fout,"%s",s.c_str());
s="";
break;
case 7:
s="";
if(c==\'\\n\')

fputc(c,fout);

break;


fclose(fin);
fclose(fout);
return 0;
参考技术A 一、标准文件的读写1.文件的打开fopen()文件的打开操作表示将给用户指定的文件在内存分配一个FILE结构区,并将该结构的指针返回给用户程序,以后用户程序就可用此FILE指针来实现对指定文件的存取操作了。当使用打开函数时,必须给出文件名、文件操作方式(读、写或读写),如果该文件名不存在,就意味着建立(只对写文件而言,对读文件则出错),并将文件指针指向文件开头。若已有一个同名文件存在,则删除该文件,若无同名文件,则建立该文件,并将文件指针指向文件开头。fopen(char*filename,char*type);其中*filename是要打开文件的文件名指针,一般用双引号括起来的文件名表示,也可使用双反斜杠隔开的路径名。而*type参数表示了对打开文件的操作方式。其可采用的操作方式如下:方式含义"r"打开,只读;"w"打开,文件指针指到头,只写;"a"打开,指向文件尾,在已存在文件中追加;"rb"打开一个二进制文件,只读;"wb"打开一个二进制文件,只写;"ab"打开一个二进制文件,进行追加;"r+"以读/写方式打开一个已存在的文件;"w+"以读/写方式建立一个新的文本文件;"a+"以读/写方式打开一个文件文件进行追加;"rb+"以读/写方式打开一个二进制文件;"wb+"以读/写方式建立一个新的二进制文件;"ab+"以读/写方式打开一个二进制文件进行追加;当用fopen()成功的打开一个文件时,该函数将返回一个FILE指针,如果文件打开失败,将返回一个NULL指针。如想打开test文件,进行写:[cpp]viewplaincopyFILE*fp;if((fp=fopen("test","w"))==NULL)printf("Filecannotbeopened/n");exit();elseprintf("Fileopenedforwriting/n");……fclose(fp);DOS操作系统对同时打开的文件数目是有限制的,缺省值为5,可以通过修改CONFIG.SYS文件改变这个设置。2.关闭文件函数fclose()文件操作完成后,必须要用fclose()函数进行关闭,这是因为对打开的文件进行写入时,若文件缓冲区的空间未被写入的内容填满,这些内容不会写到打开的文件中去而丢失。只有对打开的文件进行关闭操作时,停留在文件缓冲区的内容才能写到该文件中去,从而使文件完整。再者一旦关闭了文件,该文件对应的FILE结构将被释放,从而使关闭的文件得到保护,因为这时对该文件的存取操作将不会进行。文件的关闭也意味着释放了该文件的缓冲区。intfclose(FILE*stream);它表示该函数将关闭FILE指针对应的文件,并返回一个整数值。若成功地关闭了文件,则返回一个0值,否则返回一个非0值。常用以下方法进行测试:[cpp]viewplaincopyif(fclose(fp)!=0)printf("Filecannotbeclosed/n");exit(1);elseprintf("Fileisnowclosed/n");当打开多个文件进行操作,而又要同时关闭时,可采用fcloseall()函数,它将关闭所有在程序中打开的文件。intfcloseall();该函数将关闭所有已打开的文件,将各文件缓冲区未装满的内容写到相应的文件中去,接着释放这些缓冲区,并返回关闭文件的数目。如关闭了4个文件,则当执行:n=fcloseall();时,n应为4。3.文件的读写(1).读写文件中字符的函数(一次只读写文件中的一个字符):intfgetc(FILE*stream);intgetchar(void);intfputc(intch,FILE*stream);intputchar(intch);intgetc(FILE*stream);intputc(intch,FILE*stream);其中fgetc()函数将把由流指针指向的文件中的一个字符读出,例如:ch=fgetc(fp);将把流指针fp指向的文件中的一个字符读出,并赋给ch,当执行fgetc()函数时,若当时文件指针指到文件尾,即遇到文件结束标志EOF(其对应值为-1),该函数返回一个-1给ch,在程序中常用检查该函数返回值是否为-1来判断是否已读到文件尾,从而决定是否继续。[cpp]viewplaincopy#include"stdio.h"#includeintmain()FILE*fp;charch;if((fp=fopen("myfile.txt","r"))==NULL)printf("filecannotbeopened/n");exit(1);while((ch=fgetc(fp))!=EOF)fputc(ch,stdout);fclose(fp);该程序以只读方式打开myfile.txt文件,在执行while循环时,文件指针每循环一次后移一个字符位置。用fgetc()函数将文件指针指定的字符读到ch变量中,然后用fputc()函数在屏幕上显示,当读到文件结束标志EOF时,关闭该文件。上面的程序用到了fputc()函数,该函数将字符变量ch的值写到流指针指定的文件中去,由于流指针用的是标准输出(显示器)的FILE指针stdout,故读出的字符将在显示器上显示。又比如:fputc(ch,fp);该函数执行结构,将把ch表示的字符送到流指针fp指向的文件中去。在TC中,putc()等价于fputc(),getc()等价于fgetc()。putchar(c)相当于fputc(c,stdout);getchar()相当于fgetc(stdin)。注意,这里使用charch,其实是不科学的,因为最后判断结束标志时,是看ch!=EOF,而EOF的值为-1,这显然和char是不能比较的。所以,某些使用,我们都定义成intch。(2).读写文件中字符串的函数char*fgets(char*string,intn,FILE*stream);char*gets(char*s);intfprintf(FILE*stream,char*format,variable-list);intfputs(char*string,FILE*stream);intfscanf(FILE*stream,char*format,variable-list);其中fgets()函数将把由流指针指定的文件中n-1个字符,读到由指针string指向的字符数组中去,例如:fgets(buffer,9,fp);将把fp指向的文件中的8个字符读到buffer内存区,buffer可以是定义的字符数组,也可以是动态分配的内存区。注意,fgets()函数读到'/n'就停止,而不管是否达到数目要求。同时在读取字符串的最后加上'/0'。fgets()函数执行完以后,返回一个指向该串的指针。如果读到文件尾或出错,则均返回一个空指针NULL,所以长用feof()函数来测定是否到了文件尾或者是ferror()函数来测试是否出错,例如下面的程序用fgets()函数读test.txt文件中的第一行并显示出来:[cpp]viewplaincopy#include"stdio.h"intmain()FILE*fp;charstr[128];if((fp=fopen("test.txt","r"))==NULL)printf("cannotopenfile/n");exit(1);while(!feof(fp))if(fgets(str,128,fp)!=NULL)printf("%s",str);fclose(fp);gets()函数执行时,只要未遇到换行符或文件结束标志,将一直读下去。因此读到什么时候为止,需要用户进行控制,否则可能造成存储区的溢出。fputs()函数想指定文件写入一个由string指向的字符串,'/0'不写入文件。fprintf()和fscanf()同printf()和scanf()函数类似,不同之处就是printf()函数是想显示器输出,fprintf()则是向流指针指向的文件输出;fscanf()是从文件输入。下面程序是向文件test.dat里输入一些字符:[cpp]viewplaincopy#include#includeintmain()char*s="That'sgoodnews";inti=617;FILE*fp;fp=fopen("test.dat","w");/*建立一个文字文件只写*/fputs("YourscoreofTOEFLis",fp);/*向所建文件写入一串字符*/fputc(':',fp);/*向所建文件写冒号:*/fprintf(fp,"%d/n",i);/*向所建文件写一整型数*/fprintf(fp,"%s",s);/*向所建文件写一字符串*/fclose(fp);用DOS的TYPE命令显示TEST.DAT的内容如下所示:屏幕显示YourscoreofTOEFLis:617That'sgoodnews下面的程序是把上面的文件test.dat里的内容在屏幕上显示出来:[cpp]viewplaincopy#includeintmain()chars[24],m[20];inti;FILE*fp;fp=fopen("test.dat","r");/*打开文字文件只读*/fgets(s,24,fp);/*从文件中读取23个字符*/printf("%s",s);fscanf(fp,"%d",&i);/*读取整型数*/printf("%d",i);putchar(fgetc(fp));/*读取一个字符同时输出*/fgets(m,17,fp);/*读取16个字符*/puts(m);/*输出所读字符串*/fclose(fp);运行后屏幕显示:YourscoreofTOEFLis:617That'sgoodnews4.清除和设置文件缓冲区(1).清除文件缓冲区函数:intfflush(FILE*stream);intflushall();fflush()函数将清除由stream指向的文件缓冲区里的内容,常用于写完一些数据后,立即用该函数清除缓冲区,以免误操作时,破坏原来的数据。flushall()将清除所有打开文件所对应的文件缓冲区。(2).设置文件缓冲区函数voidsetbuf(FILE*stream,char*buf);voidsetvbuf(FILE*stream,char*buf,inttype,unsignedsize);这两个函数将使得打开文件后,用户可建立自己的文件缓冲区,而不使用fopen()函数打开文件设定的默认缓冲区。对于setbuf()函数,buf指出的缓冲区长度由头文件stdio.h中定义的宏BUFSIZE的值决定,缺省值为512字节。当选定buf为空时,setbuf函数将使的文件I/O不带缓冲。而对setvbuf函数,则由malloc函数来分配缓冲区。参数size指明了缓冲区的长度(必须大于0),而参数type则表示了缓冲的类型,其值可以取如下值:type值含义_IOFBF文件全部缓冲,即缓冲区装满后,才能对文件读写_IOLBF文件行缓冲,即缓冲区接收到一个换行符时,才能对文件读写_IONBF文件不缓冲,此时忽略buf,size的值,直接读写文件,不再经过文件缓冲区缓冲。5.文件的随机读写函数前面介绍的文件的字符/字符串读写,均是进行文件的顺序读写,即总是从文件的开头开始进行读写。这显然不能满足我们的要求,C语言提供了移动文件指针和随机读写的函数,它们是:(1).移动文件指针函数:longftell(FILE*stream);intrewind(FILE*stream);fseek(FILE*stream,longoffset,intorigin);函数ftell()用来得到文件指针离文件开头的偏移量。当返回值是-1时表示出错。rewind()函数用于文件指针移到文件的开头,当移动成功时,返回0,否则返回一个非0值。fseek()函数用于把文件指针以origin为起点移动offset个字节,其中origin指出的位置可有以下几种:origin数值代表的具体位置SEEK_SET0文件开头SEEK_CUR1文件指针当前位置SEEK_END2文件尾例如:fseek(fp,10L,0);把文件指针从文件开头移到第10字节处,由于offset参数要求是长整型数,故其数后带L。fseek(fp,-15L,2);把文件指针从文件尾向前移动15字节。(2).文件随机读写函数intfread(void*ptr,intsize,intnitems,FILE*stream);intfwrite(void*ptr,intsize,intnitems,FILE*stream);fread()函数从流指针指定的文件中读取nitems个数据项,每个数据项的长度为size个字节,读取的nitems数据项存入由ptr指针指向的内存缓冲区中,在执行fread()函数时,文件指针随着读取的字节数而向后移动,最后移动结束的位置等于实际读出的字节数。该函数执行结束后,将返回实际读出的数据项数,这个数据项数不一定等于设置的nitems,因为若文件中没有足够的数据项,或读中间出错,都会导致返回的数据项数少于设置的nitems。当返回数不等于nitems时,可以用feof()或ferror()函数进行检查。fwrite()函数从ptr指向的缓冲区中取出长度为size字节的nitems个数据项,写入到流指针stream指向的文件中,执行该操作后,文件指针将向后移动,移动的字节数等于写入文件的字节数目。该函数操作完成后,也将返回写入的数据项数。二、非标准文件的读写这类函数最早用于UNIX操作系统,ANSI标准未定义,但有时也经常用到,DOS3.0以上版本支持这些函数。它们的头文件为io.h。由于我们不常用这些函数,所以在这里就简单说一下。1.文件的打开和关闭open()函数的作用是打开文件,其调用格式为:intopen(char*filename,intaccess);该函数表示按access的要求打开名为filename的文件,返回值为文件描述字,其中access有两部分内容:基本模式和修饰符,两者用""("或")方式连接。修饰符可以有多个,但基本模式只能有一个。access的规定--------------------------------------------------------基本模式含义修饰符含义--------------------------------------------------------O_RDONLY只读O_APPEND文件指针指向末尾O_WRONLY只写O_CREAT文件不存在时创建文件,属性按基本模式属性O_RDWR读写O_TRUNC若文件存在,将其长度缩为0,属性不变O_BINARY打开一个二进制文件O_TEXT打开一个文字文件---------------------------------------------------------open()函数打开成功,返回值就是文件描述字的值(非负值),否则返回-1。close()函数的作用是关闭由open()函数打开的文件,其调用格式为:intclose(inthandle);该函数关闭文件描述字handle相连的文件。2.读写函数intread(inthandle,void*buf,intcount);read()函数从handle(文件描述字)相连的文件中,读取count个字节放到buf所指的缓冲区中,返回值为实际所读字节数,返回-1表示出错。返回0表示文件结束。write()函数的调用格式为:intwrite(inthandle,void*buf,intcount);write()函数把count个字节从buf指向的缓冲区写入与handle相连的文件中,返回值为实际写入的字节数。3.随机定位函数lseek()函数的调用格式为:intlseek(inthandle,longoffset,intfromwhere);该函数对与handle相连的文件位置指针进行定位,功能和用法与fseek()函数相同。tell()函数的调用格式为:longtell(inthandle);该函数返回与handle相连的文件现生位置指针,功能和用法与ftell()相同5.read函数和write函数来源:蚂蚁的C/C++标准编程作者:antigloss1.read#includessize_tread(intfiledes,void*buf,size_tnbytes);返回值:读取到的字节数;0(读到EOF);-1(出错)read函数从filedes指定的已打开文件中读取nbytes字节到buf中。以下几种情况会导致读取到的字节数小于nbytes:A.读取普通文件时,读到文件末尾还不够nbytes字节。例如:如果文件只有30字节,而我们想读取100字节,那么实际读到的只有30字节,read函数返回30。此时再使用read函数作用于这个文件会导致read返回0。B.从终端设备(terminaldevice)读取时,一般情况下每次只能读取一行。C.从网络读取时,网络缓存可能导致读取的字节数小于nbytes字节。D.读取pipe或者FIFO时,pipe或FIFO里的字节数可能小于nbytes。E.从面向记录(record-oriented)的设备读取时,某些面向记录的设备(如磁带)每次最多只能返回一个记录。F.在读取了部分数据时被信号中断。读操作始于cfo。在成功返回之前,cfo增加,增量为实际读取到的字节数。2.write#includessize_twrite(intfiledes,constvoid*buf,size_tnbytes);返回值:写入文件的字节数(成功);-1(出错)write函数向filedes中写入nbytes字节数据,数据来源为buf。返回值一般总是等于nbytes,否则就是出错了。常见的出错原因是磁盘空间满了或者超过了文件大小限制。对于普通文件,写操作始于cfo。如果打开文件时使用了O_APPEND,则每次写操作都将数据写入文件末尾。成功写入后,cfo增加,增量为实际写入的字节数。

使用带有头文件的 cpp 文件编写 C++ 程序

【中文标题】使用带有头文件的 cpp 文件编写 C++ 程序【英文标题】:writing a c++ program using a cpp file with header file 【发布时间】:2014-07-26 14:15:20 【问题描述】:

只是想为我的大学项目编写一个基本的 c++ 程序

我声明了一个名为TestClass 的示例类,并将它放在header files 文件夹中的testclass.h 文件中。

class TestClass

private:
    int an;
public:
    TestClass(void);
    int a[];
    void print();
    void compute();
    void get();
    ~TestClass(void);
;

然后到implement上面这个类,我在Source Files文件夹下新建了一个文件,然后写了这段代码。

#include <iostream>
#include <conio.h>
#include "TestClass.h"



    TestClass:: vinayak(void)
    

    

    void TestClass:: print()
    

    
    void TestClass:: compute()
    

    
    void TestClass:: get()
    

    

    TestClass:: ~TestClass(void)
    

    

然后我创建了一个名为Source.cpp 的新文件,然后编写了这段代码。

#include "TestClass.h"
#include "TestClass.cpp"

void main()

    TestClass TObject = TestClass();
    TObject.get();


我收到如下错误:

public: void __thiscall TestClass::compute(void)" already defined in Source.obj

有什么想法吗?

编辑:

根据@Marco 的评论,我能够编译该程序。 另外,我必须删除header file 中的constructordestructor 声明。

这使程序运行良好:)

【问题讨论】:

int a[]; 在类定义中是非法的 如果您使用的是 C++Builder,那么将变量命名为 TObject 以外的名称是明智的,因为它将该名称用于基本对象。例如使用以小写字母开头的变量名。 @MattMcNabb:谢谢 :) 我从来不知道.. 【参考方案1】:

每次写作

#include "something"

这就像将“某物”文件的内容复制粘贴到特定位置。

您正在编译一个“TestClass”翻译单元和一个“Source”翻译单元,它们都包含您的compute 方法的定义。

删除 Source.cpp 中包含的 cpp:

#include "TestClass.h"
// #include "TestClass.cpp" <- Remove this

void main()

    TestClass TObject = TestClass();
    TObject.get();


并且Source 翻译单元会知道TestClass 有一个compute 方法,但是ODR(阅读它)会得到满足。


您需要阅读的内容摘要:

声明/定义的区别 什么是 TU - 翻译单元 编译和链接的区别 什么是 ODR 及其重要性 #include 指令有什么作用

【讨论】:

答:谢谢马可。我删除了redefinition,一切都很好。再次感谢:)【参考方案2】:

必须防止头文件被多次包含。

在您的头文件顶部添加#pragma once,或将其更改为:

#ifndef TEST_CLASS_INCLUDE
#define TEST_CLASS_INCLUDE
class TestClass

...
;
#endif

此外,您永远不应包含 .cpp 文件。只需包含来自 Source.cpp 的 TestClass.h 并确保 TestClass.cpp 已编译且 TestClass.o 已链接。

【讨论】:

谢谢。正如你所说,我已经包含了#pragma once。再次感谢。 upvote 的声誉不够 :( 但再次感谢【参考方案3】:

你需要阅读链接

这一行

#include "TestClass.cpp"

不需要

【讨论】:

【参考方案4】:

你包含TestClass.cpp,你只需要包含头文件

【讨论】:

以上是关于编写程序,将一个.cpp文件中的注释,即以"/*......*/":括起来的文本和以“//”开始的的主要内容,如果未能解决你的问题,请参考以下文章

Linux Shell脚本去掉几类常见文件中的注释

用Delphi编写数据报存储控件

使用带有头文件的 cpp 文件编写 C++ 程序

# codeblocks中运行多个.cpp文件

你如何防止 yaml-cpp 解析器删除所有注释?

在 C++ 程序中使用多个 .cpp 文件?