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…

分号不是个“随便的人”

分号是个很不起眼的东西&#xff0c;但在C语言编程里面&#xff0c;它的作用不可忽视&#xff0c;多写与漏写都会造成不可预料的错误&#xff0c;下面就列举一些常见的错误。 1、多谢了分号 if或while语句块内如果只有一条语句&#xff0c;那么可以省略大括号的书写&#xff…

函数声明

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

字符变量存放多个字符

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

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

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

悬挂“else”

else匹配规则&#xff1a;与它上面的距离它最近的且在同一级语句块的一个未与else匹配的if语句匹配 初接触C语言的同志可能会犯下面这种错误 if(x 0if(y 0){printf("这里有两个if语句\n");} else {printf("只有一个else语句&#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;使之不会出现…

跳出循环的三种方式

break、continue、goto三个关键字都可以用于跳出循环&#xff0c;但三者跳出循环的情况却有所不同 braek&#xff1a;跳出本层循环&#xff0c;执行本层循环下面的语句。 continue&#xff1a;终止本次循环&#xff0c;进入下一次循环。 goto&#xff1a;调到标号所在位置&a…

C语言实现静态顺序表

1、用于存放声明的头文件 #ifndef __SEP__LIST__ #define __SEP__LIST__#define _CRT_SECURE_NO_WARNINGS 1 #include <stdio.h> #include <assert.h>#define MAX_SIZE 10 //使顺序表中可以插入的数据是可以很方便调整的 typedef int DataType;typede…

C语言实现动态顺序表

1、用于存放声明的头文件 #pragma once //避免重复引入头文件&#xff0c;作用于预编译指令几乎相同#define _CRT_SECURE_NO_WARNINGS 1 #include <stdio.h> #include <assert.h> #include <malloc.h>typedef int DataType; typedef struct SepList_d {Dat…

CentOS设置

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

VS Code无法远程ubuntu

①windows下cmd中执行命令&#xff1a; ssh-keygen -t rsa -C "xxxxxx.com" ②ubuntu 下bash 命令 ssh-keygen -t rsa -C "xxxxxx.com" ③cat id_rsa.pub >> ~/.ssh/authorized_keys ④ C:\Users\Administrator\AppData\Roaming\Code\User\set…

C语言实现单链表

1、用于存放声明的头文件 #define _CRT_SECURE_NO_WARNINGS #pragma once #include <stdio.h> #include <assert.h> #include <malloc.h>typedef int DataType;typedef struct SListNode {DataType data; //链表中存放的数据struct SLis…

vim复制、删除和粘贴一行

声明&#xff1a;下面的操作的都是在基本模式&#xff08;Esc&#xff09;&#xff0c;否则无法操作。 1、复制命令&#xff1a; y1 复制光标处的一个字符到缓冲区   yw 复制当前单词到缓冲区   yy 复制当前行到缓冲区   nyy / yny 复制当前行及其下面n-1行的内容到缓…

printf函数的格式修饰符

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

C语言实现单链表面试题汇总

这篇博客只有针对单链表的不同面试题的不同函数&#xff0c;没有对单链表的具体实现方法的介绍。 单链表的具体实现方法&#xff08;创建&#xff0c;初始化&#xff0c;前插&#xff0c;后插&#xff0c;删除&#xff0c;插入&#xff0c;销毁等&#xff09;&#xff0c;可以…

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

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