本人在写学生信息管理系统时遇到一个很头疼的错误——error LNK2005重复定义错误,苦思冥想百度谷歌bing之后都没能解决问题,于一清早刹那间觉得知道问题出在哪儿了,于是乎起床、开机、修改代码一气呵成,终于0 error(s)\0 warning(s)。
error LNK2005错误分为好几种,我以下分析的是重复定义外部函数,如果是因为重复定义全局变量、头文件的重复包含、或者使用第三方库原因导致的error LNK2005请移步http://www.cnblogs.com/MuyouSome/p/3332699.html
一:问题描述
我的系统分为3个文件(stuheader.h、stufun.c、stuims.c)
- stuheader.h:该文件中包含头文件、结构体定义以及函数声明等;
- stufun.c:该文件是系统中除main函数外的其他自定义函数的实现和相互调用;
- stuims.c:该文件是主函数main调用其他函数组装的整个软件系统。
#include<stdio.h> //I/O函数
#include<stdlib.h> //标准库函数
#include<string.h> //字符串函数
#include<ctype.h> //字符操作函数
#define M 50 //定义常数表示记录数 typedef struct
{
char no[]; //学号
char name[]; //姓名
char sex[]; //性别
int age; //年龄
}STUDENTS; //以下是函数原型
int menu_select(); //主菜单函数
int enter(STUDENTS t[]); //输入记录
void list(STUDENTS t[],int n); //显示记录
void search(STUDENTS t[],int n); //按姓名查找显示记录
int del(STUDENTS t[],int n); //删除记录
int add(STUDENTS t[],int n); //插入记录
void save(STUDENTS t[],int n); //记录保存为文件
int load(STUDENTS t[]); //从文件中读记录
void display(STUDENTS t[],int n); //按序号查找显示记录
void sort(STUDENTS t[],int n); //按姓名排序
void copy(); //文件复制
void print(STUDENTS temp); //显示单条记录
int find_name(STUDENTS t[],int n,char *s); //按姓名查找函数
int find_no(STUDENTS t[],int n,char *no); //按学号查找
void modify(STUDENTS t[],int n); //修改记录
stuheader.h代码
#include "stuheader.h"
//菜单函数,返回值为整数,代表所选的菜单项
int menu_select()
{
} int enter(STUDENTS t[])
{
} //显示记录,参数为记录数组和记录条数
void list(STUDENTS t[],int n)
{
} //查找记录
void search(STUDENTS t[],int n)
{
} //删除函数,参数为记录数组和记录条数
static int del(STUDENTS t[],int n)
{
return ;
} //插入记录函数,参数为结构体数组和记录数
int add(STUDENTS t[],int n)
{
return ;
} //保存函数,参数为结构体数组和记录数
void save(STUDENTS t[],int n)
{
} //读入函数,参数为结构体数组
int load(STUDENTS t[])
{
return ;
} //按序号显示记录函数
void display(STUDENTS t[],int n)
{
} //按姓名排序函数
void sort(STUDENTS t[],int n)
{
} //复制文件
void copy()
{
} //显示指定的一条记录
void print(STUDENTS temp)
{
} //按姓名查找函数,参数为记录数组和记录条数以及姓名s
int find_name(STUDENTS t[],int n,char *s)
{
return ;
} //按学号查找函数,参数为记录数组和记录条数以及学号no
int find_no(STUDENTS t[],int n,char *no)
{
return ;
} //修改函数,按照输入学号修改
void modify(STUDENTS t[],int n)
{
}
stufun.c 代码(其中函数体已省略)
stuims.c 文件中代码如下:
#include "stufun.c" //stufun.c中已经包含了stufun.h
void main()
{
STUDENTS stu[M]; //定义结构体数组
int length; //保存记录长度
for(;;) //无限循环
{
system("cls"); switch(menu_select())
{
case :length=enter(stu);break; //输入记录
case :list(stu,length);break; //显示全部记录
case :search(stu,length);break; //按姓名查找记录
case :length=del(stu,length);break;//按姓名删除记录
case :modify(stu,length);break; //按学号修改记录
case :length=add(stu,length);break;//插入记录
case :save(stu,length);break; //保存文件
case :length=load(stu);break; //加载文件到内存
case :display(stu,length);break; //按序号显示记录
case :sort(stu,length);break; //按姓名排序
case :copy();break; //复制文件到目标文件
case :exit(); //程序结束
}
printf("按回车键回主菜单...\n");
getchar();
}
}
我的基本思路是用stufun.c文件包含stuheader.h文件,然后用stuims.c包含stufun.c文件,本觉得万无一失,boom~boom~boom,error LNK2005来的如此突然、如此猛烈、瞬间呆若木鸡。
二:原因分析
首先我们看向stufun.c文件中的函数头,没有加static、extern等关键字,所以所有的自定义函数都默认为外部函数(int menu_select、int enter等等);
接下来我们再来分析#include:文件包含预处理是指在文件编译之前将源文件的全部内容包含进来(简单的说就是将源文件的所有代码copy过来代替该#include语句);
然后我们分析代码:
- 在stuims.c 有语句:#include<stufun.c>。
- 被包含文件(stufun.c)中含有全局变量或外部函数(int menu_select、int enter等等)
这样的话就导致项目中stufun.c有自定义函数的定义,而stuims.c中也有着一模一样的自定义函数的定义,所以就出现了error LNK2005(重复定义错误)
所以我们该怎么改呢?
三:代码修改
知道问题所在就简单了,我存在的问题是项目中有多个外部函数定义导致重复定义错误,所以我可以有两种解决方法:
- 在stufun.c中的所有自定义函数头处加extern关键字明确为外部函数,然后将#include "stufun.c"语句替换成#include "stuheader.h",No Error;
- 将stufun.c中的所有自定义函数头部加static关键字明确其为内部函数,问题解决!
四:问题总结
出现这样的问题在于做项目经验太少,定义函数时没有想到去添加其作用范围,以后再定义全局变量和外部函数时一定谨慎谨慎再谨慎,一定要明确自己所定义的变量及函数的作用范围,不然在软件扩展时会出现意料之外的Bug。话已至此,还是非常感谢Bug2005,所以我决定:我——AboutSange和error2005在2016.04.09结为异性兄弟,一起去找寻成神路上尚未碰面的error!