下面的概述是参考的这篇文章:http://blog.csdn.net/bingxx11/article/details/7771437

c语言编程中也有,也需要头文件,

头文件不只是C++的类才需要!

比如: c中的string.h,  内存操作的头文件 #include <mem.h>

即是: c语言中, 函数/变量的声明和实现, 也可以像c++一样,

头文件中, 哪些函数/变量需要使用extern来说明?

  c语言有一个约定: 凡是在对应的.c文件中, 有那个函数的实现的, 就不加extern, 凡是没有对应实现的, 就需要加extern.

extern的解决,相当于一个口头承诺, 一个口头约定, 口头答应,  在linker连接时, 由连接器(链接器)自动去找.不用我们去管的!!

---------------------------------------------

exten的头文件如何区分"定义声明" - "引用声明"?

(引用自: http://blog.csdn.net/bingxx11/article/details/7771437)

两种方式:

  1. 顶层声明中,存在初始化语句是,表示这个声明是定义声明,其他声明是引用声明。C语言的所有文件之中,只能有一个定义声明。按照这个模型,我们可以在first.h中定义如下TPYE G_test=1;那么就确定在first中的是定义声明,在其他的所有声明都是引用声明。

2、省略存储类型说明
  在这个模型中,所有引用声明要显式的包括存储类extern,而每个外部变量的唯一定义声明中省略存储类说明符。
 
c语言中 数组并没有和char*  指针完全等同起来:
int G_glob[100];
 在另一个文件中引用声明如下:
int * G_glob; // 这个错误!
在vc中,是可以编译通过的,这种情况大家都比较模糊并且需要注意,数组与指针类似,但并不等于说对数组的声明起变量就是指针。上面所说的的程序在运行时发现了问题,在引用声明的那个文件中,使用这个指针时总是提示内存访问错误,原来我们的连接程序并不把指针与数组等同,连接时,也不把他们当做同一个定义,而是认为是不相关的两个定义,当然会出现错误。正确的使用方法是在引用声明中声明如下:
 int G_glob[100];
 并且最好再加上一个extern,更加明了。
 extern int G_glob[100]; 或者: extern int G_glob[];  引用声明不需要分配内存, 所以不需要指明数组大小.

 

--------------------------------------------

C语言程序在内存中的存储区域:

  代码区;

  常量区;

  全局/静态(变量)区;

  堆区/自由存储区: malloc, free | new, delete

  栈区(是一段公共内存区, 公共的: 是指所有的函数运行时,都是使用的这个区域, 这个区域被反复的使用,

      前一个函数使用后, 退出时, "函数内的栈区被释放", 什么叫释放, 并没有内存清零这个动作,只是栈区

      的栈顶指针退回到栈顶,下一次,另外一个函数执行时,仍然使用这个内存区, 而且,局部变量的值可能是上一次

      函数运行时留下的值, 这个值的类型是各种各样的...但是由于字节数的长度不同, 所以组合起来的值,完全

      不可控, 所以是随机的, 需要对其进行初始化)

  关于函数的返回值问题?

    返回值无非有两种: 一是返回地址的, 比如数组地址, 指针, 对象的地址, 也就是, 一般"非内部数据类型"

      的返回值,最好用返回引用的方式;

      另一种是返回拷贝传值类型的, 一般, 返回的是"内部/基本"数据类型的,用这个传值, 不要用传引用.

那么, 对于传值的函数 的返回值, 函数退出时, 其内存被回收, 也就是,程序无法引用得到, 那么这个值怎么传出来呢?

  网上的说法, 我比较相信的一种是: "函数退出 时,将返回值拷贝到" eax" 这个cpu内的寄存器中, 后面的函数,你要

  这个返回值, 你就去取, 你不要就算了, 但是过后的话,你就取不到,因为这个寄存器随时都可能被覆盖.

const 修饰函数?

  修饰函数的返回值 const one_class_type * foo(..) , 表示这个函数的返回值是一个常量指针, 因此它的

    返回值只能付给 同类型的const 指针变量, 不能付给非const的.. 因为要强制保证const变量的值不被

      直接/甚至修改!

  const修饰类的成员函数的动作: 这时const要放在函数的最后: type foo(...) const {....} , 它表示, foo函数不能修改它之外的成员变量, 但可以引用/调用外部的成员变量, 但是不能调用非const的成员函数.

------------------------------------------------

sizeof 和 strlen 的列出几个重要的区别:

1.sizeof是算符,strlen是函数。

2.sizeof可以用类型做参数(必须加括号)/普通变量,strlen只能用char*做参数,且必须是以''/0''结尾的。

sizeof还可以用函数做参数,比如:
short f();
printf("%d/n", sizeof(f()));
输出的结果是sizeof(short),即2。

3.strlen计算的是字符串的长度,sizeof计算的是变量使用的内存大小,不受里面存储的内容改变

4.strlen的结果要在运行的时候才能计算出来,时用来计算字符串的长度,不是类型占内存的大小。

5.sizeof后如果是类型必须加括弧,如果是变量名可以不加括弧。这是因为sizeof是个操作符不是个函数。

6.当适用了于一个结构类型时或变量, sizeof 返回实际的大小, 当适用一静态地空间数组, sizeof归还全部数组的尺寸。 sizeof 操作符不能返回动态地被分派了的数组或外部的数组的尺寸 。

6.数组作为参数传给函数时传的是指针而不是数组,传递的是数组的首地址,如:

fun(char [8])
fun(char [])
都等价于 fun(char *)
在C++里参数传递数组永远都是传递指向数组首元素的指针,编译器不知道数组的大小
如果想在函数内知道数组的大小, 需要这样做:
进入函数后用memcpy拷贝出来,长度由另一个形参传进去
fun(unsiged char *p1, int len)
{
  unsigned char* buf = new unsigned char[len+1]
  memcpy(buf, p1, len);
}

我们能常在用到 sizeof 和 strlen 的时候,通常是计算字符串数组的长度

c语言的头文件-不是c++类的头文件?的更多相关文章

  1. 在类的头文件里尽量少引入其它头文件 &lt;&lt;Effective Objective-C&gt;&gt;

    与C 和C++ 一样,Objective-C 也使用"头文件"(header file) 与"实现文件"(implementation file)来区隔代码.用 ...

  2. c++ 一个cpp文件如何调用另一个cpp文件已经定义的类?我不想重复定义

    文件test1.cpp有类class A;文件test2.cpp有类class B.如在test2.cpp中想用A:#include "test1.cpp" 当然一般的做法是将类的 ...

  3. C++ 中的模板类声明头文件和实现文件分离后,如何能实现正常编译?

    C++ 中的模板类声明头文件和实现文件分离后,如何能实现正常编译? 这个feature叫做Export Template,即外名模板,它的作用在于使得模板代码可依照C/C++语言习惯,将模板声明和实现 ...

  4. ndk学习之c++语言基础复习----C++容器、类型转换、异常与文件流操作

    继续来复习C++,比较枯燥,但是这是扎实掌握NDK开发的必经之路,不容小觑. 容器: 容器,就是用来存放东西的盒子. 常用的数据结构包括:数组array, 链表list, 树tree, 栈stack, ...

  5. 配置apue的头文件apue.h和unp的头文件anp.h

    配置apue的头文件apue.h和unp的头文件anp.h 如果要使用gcc -g 来生成可调试文件一定要修改Make.defines.linux文件中的CFLAGS变量 修改为:CFLAGS=-an ...

  6. Effective Objective-C 2.0 — 第二条:类的头文件中尽量少引入其他头文件

    第二条:类的头文件中尽量少引入其他头文件 使用向前声明(forward declaring) @class EOCEmployer 1, 将引入头文件的实际尽量延后,只在确有需要时才引入,这样就可以减 ...

  7. VS 2010不显示头文件源文件和所有以前分类的文件夹,*.h 和*.cpp都显示在同一个文件

    打开VS后不显示头文件源文件和所有以前分类的文件夹,*.h 和*.cpp都显示在同一个文件 点击右图红色指示显示所有文件夹按钮,就能恢复.

  8. 用javah 导出类的头文件, 常见的错误及正确的使用方法

    ******************************************************************************** 用javah 导出类的头文件, 常见的 ...

  9. 于用cocoapods添加第三方库,并且cocoapods添加成功,但是却在任何一个文件上都导入不了头文件

    关于用cocoapods添加第三方库,并且cocoapods添加成功,但是却在任何一个文件上都导入不了头文件,而且根本没有提示,即使手动打#import "xxxx.h"也报错xx ...

随机推荐

  1. 关于多个EditText的OnTextChange事件陷入死循环的处理

    需求:ListView的Item上面有三个EditText控件,分别为 数量 ,单价,总价,要求输入数量跟单价时候 总价跟着计算变化,当输入总价时候 数量不变,改变单价. 实现:首先肯定想到的是对Ed ...

  2. C#脚本引擎 CS-Script 之(二)——性能评测

    以下以一个简单的HelloWord程序为例,来分析csscript脚本引擎的性能. class HelloWorld { public void SayHello() { Console.WriteL ...

  3. C#脚本引擎 CS-Script 之(一)——初识

    最近在做新产品,这个产品需要满足不同项目对于系统的定制性数据处理需求,比如有的要统计一段时间内某开关打开关闭了多少次,有的要统计一段时间内空调的使用率,有的希望根据温度来控制空调的开还是关,有的则是希 ...

  4. LeetCode:Jump Game I II

    Jump Game Given an array of non-negative integers, you are initially positioned at the first index o ...

  5. JavaScript实现MVVM之我就是想监测一个普通对象的变化

    http://hcysun.me/2016/04/28/JavaScript%E5%AE%9E%E7%8E%B0MVVM%E4%B9%8B%E6%88%91%E5%B0%B1%E6%98%AF%E6% ...

  6. "互联网思维"背后的谎言

    互联网公司/思维是什么鬼,说来惭愧上学的时候还因为知道www(World Wide Web)的中文名自豪了好久,之后在”高等学府“里学习软件工程,还愚蠢的以为自己步入了互联网之门. internet嘛 ...

  7. poj 3669 线段树成段更新+区间合并

    添加 lsum[ ] , rsum[ ] , msum[ ] 来记录从左到右的区间,从右到左的区间和最大的区间: #include<stdio.h> #define lson l,m,rt ...

  8. Java设计模式-抽象工厂模式(Abstract Factory )

    工厂方法模式有一个问题就是,类的创建依赖工厂类,也就是说,如果想要拓展程序,必须对工厂类进行修改,这违背了闭包原则,所以,从设计角度考虑,有一定的问题,如何解决?就用到抽象工厂模式,创建多个工厂类,这 ...

  9. <supports-screens>的用法

    <supports-screens android:resizeable=["true"| "false"] android:smallScreens=[ ...

  10. 【Gym 100712A】Who Is The Winner?

    题 题意 解题数目越多越排前,解题数目相同罚时越少越排前,求排第一的队伍名字. 分析 用结构体排序. 代码 #include<cstdio> #include<algorithm&g ...