传智播客旗下高端IT在线教育平台|咨询热线:010-56288220

返回顶部 返回列表
218 0

[人工智能专区] C/C++中的一些基础但必须熟记的知识

[复制链接]

9

主题

10

帖子

61

积分

注册会员

双鱼座

Rank: 2

积分
61
2180 双鱼座 发表于 2017-6-24 09:51:01
本帖最后由 双鱼座 于 2017-6-24 09:56 编辑

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


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


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

看下面代码:
#include<stdio.h>
int main(void){
   int i = 0;
   for(i=0; i<5; i++){
       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++头文件中需添加externC”,而在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) : (b))


(5)voidvoid 指针深层探讨

规则:在C语言中,凡是不加返回值类型限定的函数,就会被编译器作为返回整形值处理,但很多程序员却误认为其为void类型。
1.任何类型的指针都可以直接赋值给void*类型的指针,无需进行强制类型转换。但是void*指针却不可以不进行强制类型转换而直接就赋值给其他类型的指针。
2.如果函数没有返回值,应声明为void类型。
3.如果函数无参数,那么应声明其参数为void类型。在C语言中可以给无参数的函数传送任意类型的参数,但是在C++中不能向无参数的函数传送任何参数,错误提示:function does not take 1 parameters.所以无论是C还是C++,若函数不接受任何参数,应将其声明为void类型。
4按照ANSI(AmericanNational 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_tnum);
这样,任何类型的指针都可以传入memcpymemset中,这也真实的体现了内存操作函数的意义,因为它操作的对象仅仅是一片内存,而无论这片内存是什么类型!
6.void不能代表一个真实的变量
void a;错误


(6)内存分配方式

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


(7)内存操作注意事项

1.mallocnew申请内存之后,应该立即检查指针值是否为NULL,防止使用指针值为NULL的内存。
2.不要忘记为数组和动态内存赋初值,防止将未被初始化的内存作为右值使用。
3。避免数组或指针的下标越界,特别要当心发生多1或者少1的操作。
4.动态内存的申请和释放必须配对,防止内存泄露。
5.freedelete释放了内存之后,立即将指针设置为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 longI;}enddian_test = {{'l','?','?','b'}};
#define ENDIANNESS ((char)endian_test.I)
如果ENDIANNESS = l,则为小端格式,反之;

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

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