C语言在Linux系统下的多文件编程过程

2023-09-13 08:18:13
 随着学习的深入,程序的业务逻辑越来越复杂,代码量越来越多,就需要多人组成团队协同开发,那么就必须把任务拆分成若干个文件。
一般的拆分方案:
main.c 只当作程序的入口,不实现业务逻辑代码。
    用于实现程序具体的业务逻辑代码
    模块名.h 用说明.c文件中有哪些函数、全局变量,也就是函数声明、全局变量声明。
    模块名.c 具体的函数实现,全局变量定义。

项目中常用的、通用的工具,宏函数、函数
tools.h
tools.c

只用于类型设计
type.h 结构体、联合、枚举、宏常量、宏函数

头文件中可以写什么:
由于头文件可能会被若干个.c文件包含,那么每包含一次.c文件中就会有一份头文件的内容,所以头文件中的内容必须可重复,因此我们只适合在头文件中实现以下内容:

1、头文件卫士
2、#include 语句
3、宏常量、宏函数
4、全局变量的声明(变量的声明可以有多份,但定义只能有一份)。
5、函数声明
6、结构、联合、枚举复合的类型设计
7、类型重定义

头文件互相包含、递归包含:
注意:头文件卫士只能解决重复包含的问题,但无法解决相互包含、递归包含的问题。
互相包含:
a.h #include “b.h”
b.h #include “a.h”

递归包含:
a.h #include “b.h”
b.h #include “c.h”
c.h #include “a.h”

解决这种问题的文件,再设计一个.h文件,把他们共用的内存,实现在新的.h文件中,被他们共同包含即可。

多文件编译过程:
1、gcc xxx.h 检查头文件是否有语法错误,如果没有语法错误会生成xxx.h.gch,检查完毕后该文件要立即删除。
2、gcc -c xxx.c 把.c文件编译成二进制目标文件
3、gcc *.o 把所有目标文件合成可执行文件,也可以使用-o 设置可执行文件的名字。

接下来以用户管理系统的项目为例子,把用户管理模块拆分成多文件格式。
一、先创建各个模块的. c文件和 .h文件:main.c (只当作程序的入口,不实现业务逻辑代码)、tools.h( 项目中常用的、通用的工具,宏函数、函数)、tools.c(代码主题)、manager_user.h、manager_user.c。

二、把相应的内容放到合适的文件中:
tools.h:
在这里插入图片描述

manager_user.h:
在这里插入图片描述
在声明完函数后进行编译:gcc -c tools.h/manager_user,值得注意的是编译完后会生成tools.h.gch,这个文件一定要删除,如果这个文件存在编译器会优先使用这个文件,之后在tools.h修改的内容会无效,因为编译器会继续使tools.h.gch。

tools.c:

#include"tools.h"#include<assert.h>#include<getch.h>#include<string.h>#include<stdbool.h>charget_cmd(char start,char end){assert(start<= end);puts("-------------------");printf("请输入指令:");for(;;){char cmd=getch();if(start<= cmd&& cmd<= end){printf("%c\n",cmd);return cmd;}}}voidprint_sec(constchar* msg,float sec){printf("\33[01;32m %s \33[00m\n",msg);usleep(1000000*sec);}voidanykey_continue(void){puts("按任意键继续...");getch();}char*get_str(char* str,size_t size){assert(NULL!=str&& size>1);// 计算fgets读取了多少个字符size_t len=strlen(fgets(str,size,stdin));// 如果最后一个字符是'\n',则把它改为'\0'if('\n'== str[len-1])
		str[len-1]='\0';else// 如果最后一个字符不是'\n',则说明缓冲区中有垃圾数据,则需要清理输入缓冲区while('\n'!=getchar());return str;}char*get_passwd(char* passwd,size_t size){int i=0;while(i< size-1){
		passwd[i]=getch();// 读取到退格键if(127== passwd[i]){// 数组中有已输入密码if(i>0){// 删除一位密码
				i--;printf("\b \b");}continue;}
		
		i++;printf("*");}
	passwd[size-1]='\0';printf("\n");return passwd;}

boolyes_or_no(void){printf("是否确认此操作(y/n)?");for(;;){char cmd=getch();if('y'== cmd||'Y'== cmd||'n'== cmd||'N'== cmd){printf("%c\n",cmd);return'y'==cmd||'Y'==cmd;}}}

manager_user.c:

#include"manger_user.h"#include<string.h>#include<stdlib.h>

User user[USER_MAX];size_t cnt;int login_index=-1;

boollogoff_user(void){if(yes_or_no()){// 把最后一个用户移动给要删除的用户位置
		user[login_index]= user[cnt-1];// 用户数量减1
		cnt--;// 还原用户的登录状态
		login_index=-1;print_sec("注销用户成功!",0.75);return true;}else{print_sec("取消注销操作!",0.75);return false;}}voidmodify_passwd(void){char passwd[7];printf("请输入旧密码:");get_passwd(passwd,sizeof(passwd));if(strcmp(passwd,user[login_index].passwd)){print_sec("旧密码错误,无法修改密码!",0.75);return;}printf("请输入新密码:");get_passwd(passwd,sizeof(passwd));char repasswd[7];printf("请确认新密码:");get_passwd(repasswd,sizeof(repasswd));if(0==strcmp(passwd,repasswd)){strcpy(user[login_index].passwd,passwd);print_sec("修改密码成功!",0.75);}else{print_sec("两次输入的密码不符,修改失败!",0.75);}}voidmodify_user(void){printf("请输入用户电话:");get_str(user[login_index].phone,sizeof(user[login_index].phone));print_sec("修改用户信息成功!",0.75);}voidsub_menu(void){for(;;){system("clear");puts("*****用户信息管理模块*****");puts("1、注销用户");puts("2、修改密码");puts("3、修改信息");puts("4、退出登录");switch(get_cmd('1','4')){case'1':if(logoff_user())return;elsebreak;case'2':modify_passwd();break;case'3':modify_user();break;case'4': login_index=-1;return;}}}// 以下是一级菜单的功能函数voidmenu(void){system("clear");puts("*****用户管理模块*****");puts("1、注册用户");puts("2、用户登录");puts("3、遍历用户");puts("4、退出系统");}intquery_user(constchar* name){for(int i=0; i<cnt; i++){if(0==strcmp(user[i].name,name))return i;}return-1;}voidregister_user(void){// 判断用户是否已满if(cnt>= USER_MAX){print_sec("用户数量已满,无法添加!",0.75);return;}// 输入用户名printf("请输入用户名:");get_str(user[cnt].name,sizeof(user[cnt].name));// 检查用户名是否被占用if(-1<query_user(user[cnt].name)){print_sec("该用户名已被占用,无法注册!",0.75);return;}// 输入密码printf("请输入密码:");get_passwd(user[cnt].passwd,sizeof(user[cnt].passwd));// 再一次输入密码char repasswd[7];printf("请输入再一次输入密码:");get_passwd(repasswd,sizeof(repasswd));// 确认密码if(strcmp(user[cnt].passwd,repasswd)){print_sec("两次输入的密码不符,注册失败!",0.75);return;}// 输入电话号printf("请输入用于找回密码的手机号:");get_str(user[cnt].phone,sizeof(user[cnt].phone));// 初始化锁定标记
	user[cnt].lock=0;// 用户数量+1//cnt++;
	FILE* fp=fopen("data.txt","a");if(NULL==fp)return;fprintf(fp,"%s %s %s %hhd\n",user[cnt].name,user[cnt].passwd,user[cnt].phone,user[cnt].lock);fclose(fp);
	fp=NULL;
    cnt++;}voidlogin_user(void){// 输入用户名char name[20];printf("请输入用户名:");get_str(name,sizeof(name));// 查询用户是否存在int index=query_user(name);if(-1== index){print_sec("该用户不存在,登录失败!",0.75);return;}// 判断是否被锁if(user[index].lock>=3){print_sec("该用户已经被锁定,请联系大师兄!",0.75);return;}// 输入密码char passwd[7];printf("请输入密码:");get_passwd(passwd,sizeof(passwd));// 比较密码if(0==strcmp(user[index].passwd,passwd)){// 初始化锁定标志
		user[index].lock=0;// 记录登录成功的用户下标
		login_index= index;// 登录成功,进入子菜单sub_menu();return;}// 锁定标志+1switch(++user[index].lock){case1:print_sec("密码错误,你还有2次机会!",0.75);break;case2:print_sec("密码错误,你还有1次机会!",0.75);break;case3:print_sec("三次密码错误,账号被锁定,请联系大师兄!",0.75);break;}}voidshow_user(void){for(int i=0; i<cnt; i++){printf("%s %s %s %hhd\n",
			user[i].name,
			user[i].passwd,
			user[i].phone,
			user[i].lock);}anykey_continue();}

最后把入口函数移动过去
main.c:

#include"manger_user.h"intmain(int argc,constchar* argv[]){for(;;){menu();switch(get_cmd('1','4')){case'1':register_user();break;case'2':login_user();break;case'3':show_user();break;case'4':return0;}}}

三、进行整合
先进行编译gcc -c main.c,然后把所以生成的.o文件整合在一起:gcc main.o manager_user.o tools.o;最后一步可以使用脚本Makefile,脚本代码:

CC=gcc
STD=-std=gnu99
FLAG=-Wall-Werror
TARGE=manger
OBJECT=main.o tools.o manger_user.o

$(TARGE):$(OBJECT)
    $(CC) $(OBJECT)-o $(TARGE) 

main.o:%.o:%.c manger_user.h
    $(CC) $(STD) $(FLAG)-c $<

tools.o:%.o:%.c tools.c tools.h
    $(CC) $(STD) $(FLAG)-c $<

manager_user.o:%.o:%.c manger_user.c manger_user.h tools.h
    $(CC) $(STD) $(FLAG)-c $<

clean:
    rm-rf $(OBJECT) $(TARGE)
    rm-rf.h.gch

使用这个脚本的可以:
1、节约时间
2、记录文件之间依赖关系
3、自动化执行编译过程

  • 作者:冷笑话 冷笑话
  • 原文链接:https://blog.csdn.net/low1234567/article/details/125864625
    更新时间:2023-09-13 08:18:13