C语言注释与C++注释的相互转换

做此项目的经历主要的收获是熟悉了状态机这一方法的使用,还有就是对每实现一个功能就尽量封装一个函数这一概念把握的更为精到。

状态机:关于状态机的一个极度确切的描述是它是一个有向图形,由一组节点和一组相应的转移函数组成。状态机通过响应一系列事件而“运行”。每个事件都在属于“当前” 节点的转移函数的控制范围内,其中函数的范围是节点的一个子集。函数返回“下一个”(也许是同一个)节点。这些节点中至少有一个必须是终态。当到达终态, 状态机停止。(百度摘抄)

首先说明:
C语言注释是以斜杠星开始到第一次遇到星斜杠为止的一个字符串
CPP注释是以双斜杠开始,直到遇到回车为止的一个字符串

1、放声明的头文件

#ifndef __COMMENTCONVERT_HEADFILE__     //预编译指令
#define __COMMENTCONVERT_HEADFILE__#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <stdlib.h>
#define MAX_NUMBER_OF_CHAR 50typedef enum Convert_state    //状态机
{NULL_STATE,     //无状态,即不做任何处理C_STATE,        //C注释状态CPP_STATE,      //CPP注释状态STRING_STATE,   //字符串状态 END_STATE       //结束状态
}Con_state;//void CommentConvert();
//void C_Convert_To_CPP(FILE *fread, FILE* fwrite);
//void CPP_Convert_To_C(FILE *fread, FILE* fwrite);
//void Null_State_of_C_Con_CPP(FILE *fread, FILE* fwrite);
//void C_State_of_C_Con_CPP(FILE *fread, FILE* fwrite);
//void Cpp_State_of_C_Con_CPP(FILE *fread, FILE* fwrite);
//void Null_State_of_CPP_Con_C(FILE *fread, FILE* fwrite);
//void C_State_of_CPP_Con_C(FILE *fread, FILE* fwrite);
//void Cpp_State_of_CPP_Con_C(FILE *fread, FILE* fwrite);
//void String_Convert(FILE *fread, FILE* fwrite);#endif 

2、实现声明的.c文件

#include "CommentConvert.h"Con_state state = NULL_STATE;    //声明一个全局的状态机变量void menu()
{printf("*******************************************\n");printf("                   注释转换                \n");printf("            请选择程序要实现的功能         \n");printf("               [1] C----->>CPP             \n");printf("               [2] CPP----->>C             \n");printf("               [0]    EXIT                 \n");printf("*******************************************\n");printf("请选择>");
}void Null_State_of_C_Con_CPP(FILE *fread, FILE* fwrite)
{int ch = fgetc(fread);switch (ch)     //遇到不同的情况,进入不同的状态{int next_ch = 0;case '/':next_ch = fgetc(fread);if (next_ch == '*'){fputc(ch, fwrite);fputc('/', fwrite);state = C_STATE;}else if (ch == '/'){fputc(ch, fwrite);fputc(next_ch, fwrite);state = CPP_STATE;}else{fputc(ch, fwrite);fputc(next_ch, fwrite);}break;case '"':fputc(ch, fwrite);state = STRING_STATE;break;case EOF:fputc(ch, fwrite);state = END_STATE;break;default:fputc(ch, fwrite);break;}
}void C_State_of_C_Con_CPP(FILE *fread, FILE* fwrite)
{int ch = fgetc(fread);switch (ch){int next_ch = 0;case '*':next_ch = fgetc(fread);if (next_ch == '/'){char frith_ch = fgetc(fread);if (('\n' != frith_ch) && (EOF != frith_ch)){fputc('\n', fwrite);  //输出回车是因为要是在C语言注释里遇到“*/”,说明C语言注释已经结束了,后面的数据//不再是注释的一部分,而此时本行数据已经被“//”修饰为注释内容,要是不换行,“*/”//后面的数据也会被当做是注释的一部分,即使“*/”后面仍为注释,那么也最好输出回车,//因为这是两块相互独立的注释,不换行,就会被认为是一条注释fseek(fread, -1, SEEK_CUR);}else{fputc(frith_ch, fwrite);}state = NULL_STATE;}else if (next_ch == '*'){fputc(ch, fwrite);fseek(fread, -1, SEEK_CUR);//在此需要回退一个自己的指针,是因为碰到**之后可能接下来的字符时是/,那么会和它的上一个字符*组成一个*/,成为C语言注释的结束标志,所以需要回退一个字符,判断此时连续读取的两个字符会不会是*/}else{fputc(ch, fwrite);fputc(next_ch, fwrite);}break;case EOF:fputc(ch, fwrite);state = END_STATE;break;case '\n':fputc(ch, fwrite);fputc('/', fwrite);fputc('/', fwrite);break;default:fputc(ch, fwrite);break;}
}void Cpp_State_of_C_Con_CPP(FILE *fread, FILE* fwrite)
{int ch = fgetc(fread);switch (ch){case '\n':fputc(ch, fwrite);state = NULL_STATE;break;case EOF:fputc(ch, fwrite);state = END_STATE;break;default:fputc(ch, fwrite);break;}
}void Null_State_of_CPP_Con_C(FILE *fread, FILE* fwrite)
{int ch = fgetc(fread);switch (ch){int next_ch = 0;case '/':next_ch = fgetc(fread);if (next_ch == '*'){fputc(ch, fwrite);fputc(next_ch, fwrite);state = C_STATE;}else if (ch == '/'){fputc(ch, fwrite);fputc('*', fwrite);state = CPP_STATE;}else{fputc(ch, fwrite);fputc(next_ch, fwrite);}break;case '"':fputc(ch, fwrite);state = STRING_STATE;break;case EOF:fputc(ch, fwrite);state = END_STATE;break;default:fputc(ch, fwrite);break;}
}void C_State_of_CPP_Con_C(FILE *fread, FILE* fwrite)
{int ch = fgetc(fread);switch (ch){int next_ch = 0;case '*':next_ch = fgetc(fread);if (next_ch == '/'){fputc(ch, fwrite);fputc(next_ch, fwrite);state = NULL_STATE;}else if (next_ch == '*'){fputc(ch, fwrite);fseek(fread, -1, SEEK_CUR);}else{fputc(ch, fwrite);fputc(next_ch, fwrite);}break;case EOF:fputc(ch, fwrite);state = END_STATE;break;default:fputc(ch, fwrite);break;}
}void Cpp_State_of_CPP_Con_C(FILE *fread, FILE* fwrite)
{int ch = fgetc(fread);switch (ch){case '\n':fputc('*', fwrite);fputc('/', fwrite);fputc(ch, fwrite);state = NULL_STATE;break;case EOF:fputc(ch, fwrite);state = END_STATE;break;default:fputc(ch, fwrite);break;}
}void String_Convert(FILE *fread, FILE* fwrite)
{int ch = fgetc(fread);switch (ch){case '"':fputc(ch, fwrite);state = NULL_STATE;break;case EOF:fputc(ch, fwrite);state = END_STATE;break;default:fputc(ch, fwrite);break;}
}void C_Convert_To_CPP(FILE *fread, FILE* fwrite)
{while (state != END_STATE){switch (state)      //遇到不同的状态,进入不同的函数{case NULL_STATE:Null_State_of_C_Con_CPP(fread, fwrite);break;case C_STATE:C_State_of_C_Con_CPP(fread, fwrite);break;case CPP_STATE:Cpp_State_of_C_Con_CPP(fread, fwrite);break;case STRING_STATE:String_Convert(fread, fwrite);}}fclose(fread);fclose(fwrite);
}void CPP_Convert_To_C(FILE *fread, FILE* fwrite)
{while (state != END_STATE){switch (state){case NULL_STATE:Null_State_of_CPP_Con_C(fread, fwrite);break;case C_STATE:C_State_of_CPP_Con_C(fread, fwrite);break;case CPP_STATE:Cpp_State_of_CPP_Con_C(fread, fwrite);break;case STRING_STATE:String_Convert(fread, fwrite);}}fclose(fread);fclose(fwrite);
}void CommentConvert()
{int input = 1;char select = 'y';char input_file[MAX_NUMBER_OF_CHAR] = { 0 };char output_file[MAX_NUMBER_OF_CHAR] = { 0 };FILE *fread = NULL;FILE *fwrite = NULL;while (input){menu();scanf("%d", &input);switch (input){case 1:printf("请输入转换文件的路径及文件名:");scanf("%s", input_file);fread = fopen(input_file, "r");if (NULL == fread){perror("open file for read");exit("EXIT_FAILURE");}printf("请输入输出文件的路径及文件名:");scanf("%s", output_file);fwrite = fopen(output_file, "w");if (NULL == fwrite){fclose(fread);perror("open file for write");exit("EXIT_FAILURE");}C_Convert_To_CPP(fread, fwrite);printf("转换成功!\n");printf("请选择是否继续转换其他文件,输入:y -->>继续转换,输入:n结束转换\n");printf("请输入>");fflush(stdin);         //因为scanf()函数是以回车结束的,而回车不会被统计到上一次的输入而被读入,//而是残留在缓冲区,所以此处直接用scanf()函数读取一个字符的话,读到的就是'\n',//因此这里要用fflush()清空缓冲区scanf("%c", &select);if (select == 'y'){system("cls");}else{input = 0;}break;case 2:printf("请输入转换文件的路径及文件名:");scanf("%s", input_file);fread = fopen(input_file, "r");if (NULL == fread){perror("open file for read");exit("EXIT_FAILURE");}printf("请输入输出文件的路径及文件名:");scanf("%s", output_file);fwrite = fopen(output_file, "w");if (NULL == fwrite){fclose(fread);perror("open file for write");exit("EXIT_FAILURE");}CPP_Convert_To_C(fread, fwrite);printf("转换成功!\n");printf("请选择是否继续转换其他文件,输入:y -->>继续转换,输入:n结束转换\n");printf("请输入>");fflush(stdin);scanf("%c", &select);if (select == 'y'){system("cls");}else{input = 0;}break;break;case 0:exit(0);break;default:printf("输入有误,请重新输入!");system("cls");break;}}
}

3、用于测试函数功能的测试文件

int main()
{CommentConvert();system("pause");return 0;
}

程序实现的靠几个主要的函数,每个函数的作用几乎都能见名知其义(虽然名字起的很丑很丑很丑),程序逻辑只要从测试函数一点一点开始看起,都能看懂

接下来给出结果:
这里写图片描述

这里写图片描述

这里写图片描述

以上就是程序运行及得到的结果(图片有些模糊),在此只做了C语言注释到CPP注释的转换,CPP注释到C语言注释的转换与之相似

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/526205.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

运算符优先级与结合性

在编程中经常会有遇到运算符的地方&#xff0c;而每个运算符都有不同的优先级&#xff0c;错误的使用运算符&#xff0c;可能会发生难以预料而又不易发现的错误&#xff0c;因此我们需要掌握运算符的优先级。 优先级的序号越小&#xff0c;优先级越高。

switch()语句块的出口:break;

switch()语句块里的case:相当于一个开关&#xff0c;只要满足case的条件&#xff0c;开关就打开&#xff0c;从而执行case语句块的内容&#xff0c;而break就相当于一个出口&#xff0c;只要碰到了break&#xff0c;就跳出switch语句块&#xff0c;否则就继续执行下一条语句&am…

函数声明

对C缺陷与陷阱里第二章第一节里面的函数声明的一些段落的理解

字符变量存放多个字符

在我们的认知中&#xff0c;不会有字符变量中存放多个字符的情况&#xff08;不要说\n、\t等&#xff0c;这些本来就被C语言标准定义为是一个字符&#xff09;&#xff0c;因为一个字符变量就只占1个字节空间&#xff0c;只能存放一个字符&#xff0c;但往下看&#xff0c;你会…

不同类型的变量与零值比较的方法

在if()、while()等语句块中&#xff0c;我们经常会遇到变量与零值比较的情况&#xff0c;然而不同的类型的变量与零值的比较方法是有一定标准的&#xff0c;编程时最好遵守这些标准&#xff0c;否则会发生一些难以预料的错误。

测试机器大小端的方法

首先&#xff0c;给出一些大小端相关概念。 大小端的由来&#xff1a; 在乔纳森斯威夫特的著名讽刺小说《格列夫游记》中&#xff0c;小人国内部分裂成Big-endian和Little-endian两派&#xff0c;区别在于一派要求从鸡蛋的大头把鸡蛋打破&#xff0c;另一派要求从鸡蛋的小头把…

main()函数参数

很多人学了很长时间的C语言&#xff0c;可能还不知道mian()函数也是有参数的&#xff0c;甚至会说&#xff0c;main()函数哪来的参数&#xff0c;我从来都没见到过&#xff0c;更没有使用过&#xff0c;然而&#xff0c;不得不说&#xff0c;main()函数确实是有参数的。 main函…

运算符求值顺序

&&与||两个运算符的求值顺序是非常重要的&#xff0c;因为很多判断语句都是基于它们拥有一定的求值顺序才能正确进行的。 eg: if((num1 ! 0) && (num2 / num1 \ > num2 - num1)) 上面的if()判断语句合理的运用了&&运算符&#xff0c;使之不会出现…

CentOS设置

我的CentOS是在VMware Workstation里面装的虚拟机&#xff0c;但操作都是一样的 1、设置系统语言 许多人在安装CentOS系统时&#xff0c;可能在开始阶段选择区域及语言时选择了非中文&#xff0c;而却想在中文环境下使用系统&#xff0c;却苦于不知道如何切换到中文系统&…

printf函数的格式修饰符

本文介绍printf函数的一些特殊的格式控制修饰符 1、域宽修饰符 –> 数字 对所有格式控制符都有效&#xff0c;即便是%%也不例外 宽度修饰符出现在%和格式控制符之间的数字&#xff0c;使数据在固定区域打印。如果待打印的数值不能填满位置&#xff0c;它的左侧会被天上空…

线性表之顺序表与单链表的区别及优缺点

这里比较的是基于C语言实现的顺序表与单链表&#xff0c;与其他语言的实现可能会有差异&#xff0c;但我相信语言是相通的&#xff0c;它们的实现机制应该也差不多。 1、What 什么是顺序表和单链表 ①顺序表&#xff1a; 顺序表是在计算机内存中以数组的形式保存的线性表&a…

前置++与后置++之一道简单的题目引发的思考

引言 昨晚一时兴起&#xff0c;我脑子就问自己下面的代码会输出什么&#xff0c;也不知道我脑子为什么有这个代码模型&#xff0c;只是模糊的有些印象&#xff1a; #include <stdio.h> #include <stdlib.h>int main(int argc,char** argv) {int i3,j;j(i)(i)(i);…

有趣的for循环

#include <stdio.h> int main() {int i 0;for(i 0, printf(" First %d", i);i<10, printf(" Second %d", i);i, printf(" Third %d", i)){printf(" Fourth %d", i);}system("pause");return 0; } 这段代码会…

重载函数编译后的新名字

我们都知道很多语言都支持函数重载&#xff0c;那么编译器是怎么处理编译后它们的命名冲突的呢&#xff1f; 1、先给出几个正确的重载函数&#xff1a; #include <iostream> using namespace std;int Add(int a, int b) {return a b; }double Add(double a, double b)…

乱入的'\0'

看这个题之前&#xff0c;先来回忆一下strlen函数的工作机制&#xff1a; strlen所作的仅仅是一个计数器的工作&#xff0c;它从内存的某个位置&#xff08;可以是字符串开头&#xff0c;中间某个位置&#xff0c;甚至是某个不确定的内存区域&#xff09;开始扫描&#xff0c;…

初识C++之剖析const与#define

1、 编译器处理方式不同   #define是一个宏定义命令&#xff0c;它是在预处理阶段就进行了替换&#xff1b;   const修饰的是一个编译时常量&#xff0c;它是在编译阶段处理的。 2、 类型和安全检查不同   #define定义的标识符仅仅是产生文本替换&#xff0c;不管内容…

初识C++之运算符重载

C里面有一个叫作运算符重载的特性&#xff0c;它其实是基于函数实现的&#xff0c;下面就来介绍一下运算符重载。 1、What  C中预定义的运算符的操作对象只能是基本数据类型。但实际上&#xff0c;对于许多用户自定义类型&#xff08;例如类&#xff09;&#xff0c;也需要类…

对堆栈的认识

什么是堆和栈&#xff0c;它们在哪儿&#xff1f; 问题描述 编程语言书籍中经常解释值类型被创建在栈上&#xff0c;引用类型被创建在堆上&#xff0c;但是并没有本质上解释这堆和栈是什么。我仅有高级语言编程经验&#xff0c;没有看过对此更清晰的解释。我的意思是我理解什…

Waiting For Debugger

最近楼主在使用手机上的邮政银行时&#xff0c;总是打不开&#xff0c;要强制关闭&#xff0c;给我一个“Waiting For Debugger”的提示&#xff0c;相信朋友们应该遇到过类似的问题&#xff0c;当然这里不一定就是邮政银行出问题&#xff0c;可能是你手机里的任意一款软件&…

初识C++之继承

1、何为继承 C中所谓继承&#xff0c;就是在一个已存在类的基础上创建一个新的类&#xff0c;新类获得已存在类的部分特性&#xff08;为什么是部分特性&#xff0c;后面会讲到&#xff09;。已存在类被称为基类&#xff08;Base Class&#xff09;或父类&#xff08;Father Cl…