当前位置: 代码迷 >> Ruby/Rails >> Error LNK2005 从敌人到友人
  详细解决方案

Error LNK2005 从敌人到友人

热度:533   发布时间:2016-04-29 02:06:39.0
Error LNK2005 从敌人到朋友

本人在写学生信息管理系统时遇到一个很头疼的错误——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[20];                            //学号    char name[20];                            //姓名    char sex[5];                            //性别    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 0;}//插入记录函数,参数为结构体数组和记录数int add(STUDENTS t[],int n){    return 0;}//保存函数,参数为结构体数组和记录数void save(STUDENTS t[],int n){}//读入函数,参数为结构体数组int load(STUDENTS t[]){    return 0;}//按序号显示记录函数void display(STUDENTS t[],int n){}//按姓名排序函数void sort(STUDENTS t[],int n){}//复制文件void copy(){}//显示指定的一条记录void print(STUDENTS temp){}//按姓名查找函数,参数为记录数组和记录条数以及姓名sint find_name(STUDENTS t[],int n,char *s){    return 0;}//按学号查找函数,参数为记录数组和记录条数以及学号noint find_no(STUDENTS t[],int n,char *no){    return 0;}//修改函数,按照输入学号修改void modify(STUDENTS t[],int n){}
stufun.c 代码(其中函数体已省略)

 stuims.c 文件中代码如下:

#include "stufun.c"        //stufun.c中已经包含了stufun.hvoid main(){    STUDENTS stu[M];    //定义结构体数组    int length;            //保存记录长度    for(;;)                //无限循环    {        system("cls");                switch(menu_select())        {        case 0:length=enter(stu);break;        //输入记录        case 1:list(stu,length);break;        //显示全部记录        case 2:search(stu,length);break;    //按姓名查找记录        case 3:length=del(stu,length);break;//按姓名删除记录        case 4:modify(stu,length);break;    //按学号修改记录        case 5:length=add(stu,length);break;//插入记录        case 6:save(stu,length);break;        //保存文件        case 7:length=load(stu);break;        //加载文件到内存                    case 8:display(stu,length);break;    //按序号显示记录        case 9:sort(stu,length);break;        //按姓名排序        case 10:copy();break;                //复制文件到目标文件        case 11:exit(0);                    //程序结束        }        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语句);

然后我们分析代码:

  1. 在stuims.c 有语句:#include<stufun.c>。
  2. 被包含文件(stufun.c)中含有全局变量或外部函数(int menu_select、int enter等等)

这样的话就导致项目中stufun.c有自定义函数的定义,而stuims.c中也有着一模一样的自定义函数的定义,所以就出现了error LNK2005(重复定义错误)

所以我们该怎么改呢?

三:代码修改

知道问题所在就简单了,我存在的问题是项目中有多个外部函数定义导致重复定义错误,所以我可以有两种解决方法:

  1. 在stufun.c中的所有自定义函数头处加extern关键字明确为外部函数,然后将#include "stufun.c"语句替换成#include "stuheader.h",No Error;
  2. 将stufun.c中的所有自定义函数头部加static关键字明确其为内部函数,问题解决!

四:问题总结

出现这样的问题在于做项目经验太少,定义函数时没有想到去添加其作用范围,以后再定义全局变量和外部函数时一定谨慎谨慎再谨慎,一定要明确自己所定义的变量及函数的作用范围,不然在软件扩展时会出现意料之外的Bug。话已至此,还是非常感谢Bug2005,所以我决定:我——AboutSange和error2005在2016.04.09结为异性兄弟,一起去找寻成神路上尚未碰面的error!

  相关解决方案