C语言【自定义数据类型、typedef、动态内存分配】
C语言【自定义数据类型、typedef、动态内存分配】
一、自定义数据类型。
关于下面讲到的所有自定义数据类型(enum、struct、union),有一点要说的是:定义类型不是声明变量,做这步操作时不分配内存,也不能在定义类型时赋值(枚举那个不是赋值,是做一个限定,赋值时赋限定之外的值也不报错。)。
1、typedef (给类型起别名的关键字)
// C语言中给数据类型起别名的同时不能声明变量。
// 一个错误的示范:// typedef int Integer i; // 不能在这里声明i。自定义数据类型同理。
// 给指针类型起别名
typedef int* intptr;
typedef char* String;
// 给数组类型起别名
typedef int fiveInts[5]; // 有一丢丢不一样
// 使用数组类型的别名声明变量并初始化
fiveInts a = {1, 2, 3, 4, 5}; // 记一点,这种大括号形式的初始化只能声明变量时这样使用,否则报错。前面有记错的地方记得改正。
// 给数组指针类型起别名
typedef int (* IntArrayPointer)[5];
// 数组指针类型的使用
int a[5] = {1, 2, 3, 4, 5};
IntArrayPointer p = &a;
2、枚举
可以看作是一个限定取离散值范围的类型。
枚举类型的定义。这个类型一般定义为全大写,因为里面的元素全都是常量。
// 定义枚举类型。
enum WEEKDAY{
MON, // 默认为0
TUE, // 上面是0,这个默认就是1;如果上面定义了2,这个就是3
WED,
THU = 5, // 告知 THU为5
FRI,
SAT = 1, // 可与上面的重复,但不建议。 不可为浮点数。
SUN
};
// 使用上面定义的枚举类型。声明并赋值。
void main(){
// enum WEEKDAY 是类型; wd是变量名; 值可以是枚举类型之外的数,比如100,但不建议。
enum WEEKDAY wd = TUE;
}
// 隐式定义枚举并声明出变量。
enum {
// 不管是不是隐式定义,这个大括号中不能没有内容,否则报错。
A,
B
} day;
枚举没有特殊的遍历方法,也就是说枚举的元素如果值是错乱的,一般就无法完成遍历了。
// 枚举变量的内存大小
// 用上面定义的枚举实验。一个枚举变量就是一个元素的值,整型为4。
// 定义枚举时不分配空间。声明变量时,那个变量存的就是枚举中的一个元素。想一想java中的枚举类。
sizeof(enum WEEKDAY); // 4
// 枚举与switch...case的搭配。
switch(wd){ // 借用上面声明的wd变量
case MON:
// ...
break;
case TUE:
// ...
break;
// ...
default:
// ...
break;
}
3、结构体
// 结构体类型的定义
struct Student{
int id;
int age;
// char arr[]; // 会报错
char *name; // 直接赋字符串字面值可以,字面值也算是有过空间分配。如果拿它接收个用户输入就会报错。直接指向有空间的值当然也没问题。
};
// 使用
void main(){
struct Student stu1;
stu1.id = 1001;
// stu1[0] = 1002; // 没有这种写法的。
}
// 结构体指针-----指向结构体类型变量的指针
struct Struct *ptr = &stu1; // 这个结构体类型上面定义了。stu1上面声明了。
// 结构体指针的使用
(*ptr).id = 1003;
ptr->id = 1004;
结构体中关于空间有一个对齐的问题。两点要求:1、结构体中某一成员的起始地址为该成员所占字节的整数倍;2、结构体整个空间大小要求为其中最大成员的整数倍。
4、共用体
共用体内的成员共同使用同一段空间。
共用体所占内存空间为其内部成员中最大的那个空间。
应用场景:根据条件在字段内定义不同类型的值。
// 共用体类型定义
union Score{
int score1;
double score 2;
}
// 使用
union Score a; // 声明变量
union Score b = {.score2 = 1}; // 声明变量并赋值。
a.score1 = 10; // 赋值
5、手动动态分配内存
内存的自动动态分配是系统在栈空间完成的。
*void ** :C99允许定义一个类型为void的指针变量。这个(void*)类型的指针变量可以指向一块地址,但是这个指针变量除了输出首地址外,其余操作均无意义,这个指针变量的++操作移动一个地址,即1Byte。 这个指针变量可以强转为任何指针类型(如强转为int,就可以一次移动4Byte), 也可以被任何指针强转成这个指针类型。 下面的几个手动动态分配内存的函数返回值都为void * , 可以强转为自己需要的指针类型,即使你需要char * 也建议转过去,而不是用void *。
以下是一些在堆空间手动分配内存的几个函数及其使用。需要引入头文件 <stdlib.h>
// malloc函数
// 内存分配成功返回一个void*,指针指向新分配内存的起始地址;分配失败返回NULL
void * malloc(size_t size);
// malloc函数的使用
int *p = NULL;
if(p==NULL){
p = (int *) malloc(sizeof(int)); // 分配一个int类型的空间
}
*p = 100; // 给分配的这个int赋值
free(p); // 释放这个内存
我们可以通过指针修改const声明的常量的值。但是const这种常量根本就不能作为数组的长度。
如果想使用变量指定数组长度,除了动态分配我想不出别的方法。
// maloc实现数组的动态分配
int n = 5;
int *p = (int *) malloc(n*sizeof(int));
p[0] = 10; // 因为转成的(int*)类型,所以p[0]即前四个Byte所表示的数组,可赋值或修改。
// calloc函数 (自带初始化为0的功能)
// 第一个参数为要分配的元素个数,第二个参数为要分配给每个元素的字节数。
void * calloc(size_t numElements, size_t sizeofElement);
// calloc函数的使用
int *p = (int *) calloc(5, sizeof(int)); // 给p指针分配了5个int堆内存
free(p); //释放
// realloc函数
// 第一个参数是要重新分配堆内存的指针,第二个参数是新分配内存的大小。
// 返回一个指向重新分配内存块的指针,一般分配的地址还是原地址,且一般数据不会改
void * realloc(void *ptr, size_t size);
// realloc函数的使用
int *p = (int *) malloc(5*sizeof(int));
p = realloc(p, 5*sizeof(int));
free(p);
在全局声明的指针,不能再全局分配内存。
给指针动态分配好内存后,它的初始值是随机的。
calloc函数有初始化0的功能。