免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
楼主: listenxu
打印 上一主题 下一主题

[C] 请教一个编译问题,不用include头文件也可以编译运行成功是怎么回事? [复制链接]

论坛徽章:
0
11 [报告]
发表于 2014-04-27 17:22 |只看该作者
本帖最后由 listenxu 于 2014-04-27 17:50 编辑
cxm240 发表于 2014-04-27 17:03
所以说,头文件是不一定要包含的吗? 但是有一个问题是,为什么有时候不包含某些头文件又会报错呢?
----- ...


谢谢, 请问这种说法有出处吗?

我找到一片文章,说头文件主要是做类型检查,而不是为了编译是否可以通过,觉得有一定道理

http://blog.chinaunix.net/uid-24774106-id-3291005.html

但是貌似有些情况下,不包含头文件确实会报error; 按照你的说法,我对结构体做了测试;
======================================
创建文件helper.h, 声明结构体
void msg1(void);

typedef struct dint{
int a;
int b;
}dint;

然后再hello.c中定义结构体的实例dintx,并赋值:
#include <stdio.h>
//#include "helper.h"
int main()
{
  struct dint dintx;
  dintx.a = 1;
  dintx.b = 2;
  printf("Hello,Linuxprogramming world!\n");
  msg1();
  msg2();  
  return 0;
}
编译后报错:
error: storage size of ‘dintx’ isn’t known

将 //#include "helper.h"  注释“//”去掉,编译通过;
======================================


然后又做了如下测试:
======================================

添加文件helper2.h,加入一个变量:
int i;
void msg2(void);
在hello.c 中,对变量赋值
#include <stdio.h>
int main()
{
  i = 1;
  printf("Hello,Linuxprogramming world!\n");
  msg1();
  msg2();
  printf("i=%d\n",i);
  return 0;
}
编译,报如下错误
hello.c:6:3: error: ‘i’ undeclared (first use in this function)
然后将hello.c中include helper2.h,变成如下:
#include <stdio.h>
#include "helper2.h"
int main()
{
  i = 1;
  printf("Hello,Linuxprogramming world!\n");
  msg1();
  msg2();
  printf("i=%d\n",i);
  return 0;
}
编译即可通过;
=================================================

所以,综合起来看,结论如下

1, 如果仅引用了头文件中的函数,可以不include头文件,但是有风险,编译器不会对函数的参数做类型检查;
2, 如果引用了头文件中声明的变量,一定要include头文件
3,   如果引用了头文件中定义的非内部类型,如struct,一定要include头文件



论坛徽章:
0
12 [报告]
发表于 2014-04-27 17:41 |只看该作者
函数没有声明,自动会当成未定义符号,在链接的时候会在其他目标文件中查找。介绍编译链接原理的书中都会有相关解释的

论坛徽章:
0
13 [报告]
发表于 2014-04-27 17:48 |只看该作者
cxm240 发表于 2014-04-27 17:41
函数没有声明,自动会当成未定义符号,在链接的时候会在其他目标文件中查找。介绍编译链接原理的书中都会有 ...


谢谢哦,我做了一些测试,确实对函数的引用是不用include头文件的;
我根据测试结果更新了上一封回复;

因为不是计算机专业科班出生,所以编译原理之类的没有什么概念,做一步学一步

论坛徽章:
95
程序设计版块每日发帖之星
日期:2015-09-05 06:20:00程序设计版块每日发帖之星
日期:2015-09-17 06:20:00程序设计版块每日发帖之星
日期:2015-09-18 06:20:002015亚冠之阿尔艾因
日期:2015-09-18 10:35:08月度论坛发贴之星
日期:2015-09-30 22:25:002015亚冠之阿尔沙巴布
日期:2015-10-03 08:57:39程序设计版块每日发帖之星
日期:2015-10-05 06:20:00每日论坛发贴之星
日期:2015-10-05 06:20:002015年亚冠纪念徽章
日期:2015-10-06 10:06:482015亚冠之塔什干棉农
日期:2015-10-19 19:43:35程序设计版块每日发帖之星
日期:2015-10-21 06:20:00每日论坛发贴之星
日期:2015-09-14 06:20:00
14 [报告]
发表于 2014-04-27 18:31 |只看该作者
本帖最后由 MMMIX 于 2014-04-27 18:33 编辑
listenxu 发表于 2014-04-27 17:48

因为不是计算机专业科班出生,所以编译原理之类的没有什么概念,做一步学一步

对这些不了解就规规矩矩来,该包含的头文件你就老老实实的包含,不要老想着投机取巧,也没必要胡乱找些文章把自己看的一头雾水。最容易出问题的就是一知半解还自作聪明。

喷完了这种不靠谱的行为,来点头文件的介绍。

C 头文件:

1、谁处理?

C 的预处理器会查找、包含、并处理头文件。

2、头文件通常有些什么内容?

a. 常数定义(通过宏)
b. 宏定义
c. 类型定义 (struct, typedef 等)
d. 函数原型、变量声明
e. inline 函数、变量定义
f. 其他

3. 不包含需要的头文件会有什么问题?

a. 编译出错,例如类型未定义
b. 编译有警告但是可通过,不过运行会出错
c. 编译有警告但可通过,当时运行也没问题,但是换个环境(例如换了编译器,或现有编译器、依赖库升级,或换了操作系统),结果程序就运行出错了。

在你搞不清楚不包含头文件会有什么问题的时候,就老老实实的按照要求包含需要的头文件

论坛徽章:
0
15 [报告]
发表于 2014-04-27 18:43 |只看该作者
回复 14# MMMIX


    再声明一次,不是为了投机取巧,实在是本人遇到问题之后想要搞清楚背后的原因;
  
    不管怎样,很感谢你的耐心而且专业的回复

论坛徽章:
95
程序设计版块每日发帖之星
日期:2015-09-05 06:20:00程序设计版块每日发帖之星
日期:2015-09-17 06:20:00程序设计版块每日发帖之星
日期:2015-09-18 06:20:002015亚冠之阿尔艾因
日期:2015-09-18 10:35:08月度论坛发贴之星
日期:2015-09-30 22:25:002015亚冠之阿尔沙巴布
日期:2015-10-03 08:57:39程序设计版块每日发帖之星
日期:2015-10-05 06:20:00每日论坛发贴之星
日期:2015-10-05 06:20:002015年亚冠纪念徽章
日期:2015-10-06 10:06:482015亚冠之塔什干棉农
日期:2015-10-19 19:43:35程序设计版块每日发帖之星
日期:2015-10-21 06:20:00每日论坛发贴之星
日期:2015-09-14 06:20:00
16 [报告]
发表于 2014-04-27 18:49 |只看该作者
回复 15# listenxu


    如果你想要以编程为主业,那就去把缺的基础知识补上;如果你只是想通过编程来解决问题,那就把精力集中在你需要解决的问题上。至于编程,你按规矩来就是了,至于不按规矩来发生的各种奇奇怪怪的结果,就不要费神去关注了,也没有必要弄清楚它们的原因或是原理。

论坛徽章:
95
程序设计版块每日发帖之星
日期:2015-09-05 06:20:00程序设计版块每日发帖之星
日期:2015-09-17 06:20:00程序设计版块每日发帖之星
日期:2015-09-18 06:20:002015亚冠之阿尔艾因
日期:2015-09-18 10:35:08月度论坛发贴之星
日期:2015-09-30 22:25:002015亚冠之阿尔沙巴布
日期:2015-10-03 08:57:39程序设计版块每日发帖之星
日期:2015-10-05 06:20:00每日论坛发贴之星
日期:2015-10-05 06:20:002015年亚冠纪念徽章
日期:2015-10-06 10:06:482015亚冠之塔什干棉农
日期:2015-10-19 19:43:35程序设计版块每日发帖之星
日期:2015-10-21 06:20:00每日论坛发贴之星
日期:2015-09-14 06:20:00
17 [报告]
发表于 2014-04-27 18:55 |只看该作者
listenxu 发表于 2014-04-27 18:43
回复 14# MMMIX

实在是本人遇到问题之后想要搞清楚背后的原因;


你现在唯一的问题就是不按规矩来。

明明需要的函数声明你为什么不提供?明明需要的头文件你为什么要想着不包含?

编译器之所以允许这种行为,不过是为了兼容以前的老代码,可是你现在写代码为什么还要用这种古老、过时的方式?

论坛徽章:
0
18 [报告]
发表于 2014-04-27 19:05 |只看该作者
回复 17# MMMIX


    我想,并不是我一定要用这种古老的方式来编程,应该说,没有养成好的编程习惯;
    像你说的,我平时是用编程来解决问题,所以,平时不怎么关注程序的编译和运行原理,只要解决问题就OK;
    但是,这样造成的一个后果是,遇到由深层次的原因(比如编译)导致的问题时,往往需要花很长时间来解决,因为不了解;  所以,有些时候,想尝试着研究一下,我说这时候是好奇驱动,你信吗?

论坛徽章:
95
程序设计版块每日发帖之星
日期:2015-09-05 06:20:00程序设计版块每日发帖之星
日期:2015-09-17 06:20:00程序设计版块每日发帖之星
日期:2015-09-18 06:20:002015亚冠之阿尔艾因
日期:2015-09-18 10:35:08月度论坛发贴之星
日期:2015-09-30 22:25:002015亚冠之阿尔沙巴布
日期:2015-10-03 08:57:39程序设计版块每日发帖之星
日期:2015-10-05 06:20:00每日论坛发贴之星
日期:2015-10-05 06:20:002015年亚冠纪念徽章
日期:2015-10-06 10:06:482015亚冠之塔什干棉农
日期:2015-10-19 19:43:35程序设计版块每日发帖之星
日期:2015-10-21 06:20:00每日论坛发贴之星
日期:2015-09-14 06:20:00
19 [报告]
发表于 2014-04-27 21:10 |只看该作者
回复 18# listenxu


    你这头痛医脚所为何来?明明是编程风格/习惯的问题,这你也说了,可是你不去想办法养成好的风格、习惯,却反而想要研究编译系统是如何处理那些有问题的代码?你这不是南辕北辙、缘木求鱼么?

论坛徽章:
1
巨蟹座
日期:2014-03-18 23:44:30
20 [报告]
发表于 2014-04-28 09:17 |只看该作者
如果你使用了第三方库,即便你只是用它的函数,你也编译过不了的!
至于不包含头文件能编译过,只是 编译器的 一直自定义行为而已!

您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP