分享

c/c 中的一些基础但必须熟记的知识

 cntagu 2016-02-28

1)、const 与#define
 const在C++中包含了更丰富的含义,而在C语言中仅意味着:“只能读的普通变量,”或“不能改变的变量”,故在编译阶段需要的常数仍然只能以#DEFIEN宏定义!故在C语言中如下程序时非法的:
 const int SIZE = 10;
 char a[SIZE];错误:SIZE不是常数!


(2)、static变量初始化的问题

看下面代码:

#include


int main(void)

{

    int i = 0;

    for(i=0; i<5;>

    {

        static int a = 10;

        printf('static a is [%d]\n',a);

        a++;

    }

    return 0;

该代码打印出如下内容:
static a is 10
static a is 11
static a is 12
static a is 13
static a is 14
该代码说明变量在定义为static变量后,初始化只进行一次。

(3)、externC问题
1.被extern “C”限定的函数或变量是extern类型的

 2.被extern “C”修饰的函数或变量是按照C语言编译和链接的
C++和C在编译时的区别:
C++支持函数重载,而C不支持
例如:函数 viod foo(int x, int y);在C++中编译后在符号库中的名字是_foo_int_int,而C编译后生成的名字是_foo.
一句话概括extern “C”的目的:
实现C++和C及其他语言的混合编程。
具体用法:
1.在C++中引用C语言中的函数和变量,在包含C语言头文件是,需进行下列处理:
extern 'C'
{
    #include 'cExample.h'
}

2.在C中引用C++语言中的函数和变量,C++头文件中需添加extern “C”,而在C语言中不能直接引用声明了extern “C”的头文件,而应该仅将C文件中在C++中定义的extern “C”函数声明为extern类型
例如:
C++头文件
#ifndef CPP_EXAMPLE_H
#define CPP_EXAMPLE_H
extern 'C' int add(int x, int y);
#endif
C++实现文件cpp_Example.cpp
#include 'cppExample.h'
int add(int x, int y)
{
    return x + y;
}

C 实现函数Cfile.c
extern int add(int x, int y)
int main(int argc, char* argv[])
{
    add(2, 3);return 0;;
}

(4)、什么是宏定义

1.宏定义“像”函数;
2.宏定义不是函数,因而需要括上所有参数;
3.宏定义可能产生副作用;
例如:
#define MIN(a,b) ((a) < (b)="" (a)="" :="">
(5)、void和void 指针深层探讨
规则:在C语言中,凡是不加返回值类型限定的函数,就会被编译器作为返回整形值处理,但很多程序员却误认为其为void类型。
1.任何类型的指针都可以直接赋值给void*类型的指针,无需进行强制类型转换。但是void*指针却不可以不进行强制类型转换而直接就赋值给其他类型的指针。
2.如果函数没有返回值,应声明为void类型。
3.如果函数无参数,那么应声明其参数为void类型。在C语言中可以给无参数的函数传送任意类型的参数,但是在C++中不能向无参数的函数传送任何参数,错误提示:function does not take 1 parameters.所以无论是C还是C++,若函数不接受任何参数,应将其声明为void类型。
4按照ANSI(American National Standards Institute)标准,不能对void指针进行算法操作,这是因为ANSI标准认定:进行算法操作的指针必须是确定知道其指向类型大小的。例如:
int *ptr;
ptr++;
ptr++的结果是使其增大sizeof(int).
但是GNU则不这么认定,它指定void*的算法操作和char*一致。
5.如果函数的参数可以是任意类型指针,那么应将其声明为void*类型。
典型的如内存操作函数:
void * memcpy(void *dest, const void *src, size_t len);
void * memset(void *buffer, int c,size_t num);
这样,任何类型的指针都可以传入memcpy和memset中,这也真实的体现了内存操作函数的意义,因为它操作的对象仅仅是一片内存,而无论这片内存是什么类型!
6.void不能代表一个真实的变量
如void a;错误

(6)、内存分配方式
内存分配方式有3中:
1. 从静态存储区域分配。内存在程序编译的时候就已经分配好,这块内存在程序的整个运行期间都存在,例如全局变量,static变量。
2.在栈上创建,在执行函数时,函数内部的局部变量的存储单元都是可以在栈上创建的。函数执行结束时这些存储单元自动被释放,栈内存分配运算内置于处理器的指令集中,效率很高,但是分配的内存容量有限。
3。在堆上分配,亦称动态内存分配,程序在运行的时候用malloc或new申请任意多少的内存,程序员自己负责在何时用free或delete释放内存,动态内存的生存期由我们决定,使用灵活,但是容易出错。

(7)、内存操作注意事项
1.用malloc或new申请内存之后,应该立即检查指针值是否为NULL,防止使用指针值为NULL的内存。
2.不要忘记为数组和动态内存赋初值,防止将未被初始化的内存作为右值使用。
3。避免数组或指针的下标越界,特别要当心发生多1或者少1的操作。
4.动态内存的申请和释放必须配对,防止内存泄露。
5.用free或delete释放了内存之后,立即将指针设置为NULL,防止产生'野指针'.
'野指针'不是NULL指针,是指向”垃圾“内存的指针。

(8)、如何判断大小端格式
编写一个C函数,若处理器是big_endian的,则返回0,若是little_endian的,则返回1;
int checkCPU(void)
{
    {
     union w
    { 
        int a;
        char b;
    }c;
    c.a = 0x1234;
    return(c.b == 0x34);  
    }
}
同样的功能:linux操作系统中的相关源代码是这么做的:
static union{char c[4];unsigned long I;}enddian_test = {{'l','?','?','b'}};
#define ENDIANNESS ((char)endian_test.I)
如果ENDIANNESS = l,则为小端格式,反之;

总结:
在C C++程序的编写中,当多个基本数据类型或复合数据结构要占用同一片内存时,我们要使用联合体;当多种类型,多个对象,多个事物只取其一时,我们也可以使用联合体来发挥长处!





    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多