《C Primer Plus》读书笔记——存储类、链接和内存管理

简介: 背景距离上次写读书笔记的日子已有半个月了。这段时间一直在做摄像头直立平衡车,也把《C Primer Plus》的中级部分扫了一遍。现在做赛道算法识别遇到瓶颈了,就想把读书笔记补回来。

背景

距离上次写读书笔记的日子已有半个月了。这段时间一直在做摄像头直立平衡车,也把《C Primer Plus》的中级部分扫了一遍。现在做赛道算法识别遇到瓶颈了,就想把读书笔记补回来。原计划是写指针和数组2的。现在发现还不如直接开新篇,反正之前没写的变长数组之类的也会提及。所以就有这篇存储类、链接和内存管理。虽说这里部分的知识都有在平时码代码时用到,认识得深一点,但是还是有很多过目就忘的,遂烂笔头记之。

存储类

C为变量提供了5种不同的存储模型,或称存储类。还有基于指针的第6种存储模型。可按照一个变量的存储时期(storage duration)、作用域(scope)及链接(linkage)来描述它。

作用域

作用域描述了程序中可以访问一个标识符的一个或多个区域。可以是代码块作用域、函数原型作用域或者文件作用域。

  • 代码块:整个函数体或一个函数内任一复合语句。
    C99允许在一个代码块任意位置声明变量:
    for(int i = 0; i<10; i++) //合法

  • 函数原型:从变量定义处一直到原型声明的末尾。
    如变长数组:
    void use_a_VLA(int n, int m, ar[n][m]);

  • 文件:在所有函数之外定义的变量,即全局变量。

链接

一个C变量具有下列链接之一:外部链接(external linkage),内部链接(internal linkage)或空链接(no linkage)。

  • 具有代码块作用域或函数原型作用域的变量有空链接,意味着他们是由其定义所在代码块或函数原型所私有的。
  • 具有外部链接:可在多文件使用。
  • 具有内部链接:可在单文件使用。

存储时期

一个C变量有以下两种存储时期之一:静态存储时期(static storage duration)和自动存储时期(automatic storage duration)。

  • 具有文件作用域的变量具有静态存储时期,它在程序执行期间将一直存在。
    注意:对于具有文件作用域的变量,关键词static表明链接类型,并非存储时期。一个使用static声明了的文件作用域变量具有内部链接,而所有的文件作用域变量,无论它具有内部还是外部链接,都具有静态存储时期。
  • 具有代码块作用域的变量一般情况下具有自动存储时期。
  • 程序进入该代码块,将为这些变量分配内存,退出代码块时,将释放内存。

5种存储类

存储类和函数

函数也有存储类,可以时外部的、静态的或内联的。
若无static,默认是extern。

随机数函数与静态变量

/* 自写随机数函数,包含两文件 */

/* s_and_r.c */
static unsigned long int next = 1; //种子

int rand(void)
{
    /*产生伪随机数的公式*/
    next = next * 1103515245 + 12345;
    return (unsigned int)(next / 65536) % 32768;
}

void srand(unsigned int seed)
{
    next = seed;
}


/* r_drive.c */
#include <stdio.h>
extern void srand(unsigned int x);
extern int rand(void);

int main(void)
{
    int count;
    unsigned seed;
    printf("Please enter your choice for seed.\n");
    while(scanf("%u", &seed) == 1)
    {
        srand(seed); //重置种子
        for(count = 0; count < 5; count++)
        printf("%hd\n", rand() );
        printf("Please enter next seed (q to quit):\n"); 
        //实际上输入的不是无符号十进制数应该就可以退出
    }
    printf("Done.\n");
    return 0;
}

另外还有利用计算机掷出非现实的任意个任意面骰子的程序。

分配内存:malloc()和free()

函数malloc() 接受一个参数:所需内存字节数。然后malloc()找到可用内存中一个大小适合的块。内存是匿名的,即malloc()分配了内存,但没有为它指定名字。然而,它可以返回那块内存第一个字节的地址。因此,可以把那个地址赋值给一个指针变量,并使用该指针来访问那块内存。malloc()可指向char或void(通用指针)的指针,可返回数组指针、结构指针等等。若找不到所需的空间,它将返回空指针。

double * ptd; //声明指针来存放块在内存中的位置
ptd = (double *)malloc(30 * sizeof(double)); //使用malloc请求一个存储块,类型指派(double *)在C中可选,在C++必须。

这段代码请求30个double类型值的空间,并把ptd指向该空间所在位置。注意ptd是作为指向一个double类型值的指针声明的。数组的名字是它第一个元素的地址。可用ptd[n]来访问第n个元素。

现在,创建一个数组有三种方法:

  • 声明一个数组,声明时用常量表达式指定数组维数,然后可用数组名访问数组元素。
  • 声明一个变长数组,声明时用变量表达式指定数组维数,然后用数组名来访问数组元素。
  • 声明一个指针,调用malloc(),然后使用该指针来访问数组元素。

一般地,对应每个malloc(),应调用一次free()。free的参数是先前malloc返回的地址,它释放先前分配的内存。在函数末尾处调用free()可防止内存泄漏(memory leak)。

存储类与动态内存分配

理想的程序将其可用内存分为三个独立的部分:一个是具有外部、内部及空链接的静态变量的;一个是自动变量的,另一个是动态内存分配的。
- 声明不同存储类,变量使用内存时间不同,将这一部分内存处理为一个堆栈。这样的新变量在内存中创建时按顺序加入,消亡时按相反顺序移除。
- 动态分配的内存在调用malloc等时产生,调用free时释放。这样的内存可能是碎片状的。

目录
相关文章
|
5天前
|
存储 C语言
数据在内存中的存储方式
本文介绍了计算机中整数和浮点数的存储方式,包括整数的原码、反码、补码,以及浮点数的IEEE754标准存储格式。同时,探讨了大小端字节序的概念及其判断方法,通过实例代码展示了这些概念的实际应用。
13 1
|
9天前
|
存储
共用体在内存中如何存储数据
共用体(Union)在内存中为所有成员分配同一段内存空间,大小等于最大成员所需的空间。这意味着所有成员共享同一块内存,但同一时间只能存储其中一个成员的数据,无法同时保存多个成员的值。
|
14天前
|
存储 弹性计算 算法
前端大模型应用笔记(四):如何在资源受限例如1核和1G内存的端侧或ECS上运行一个合适的向量存储库及如何优化
本文探讨了在资源受限的嵌入式设备(如1核处理器和1GB内存)上实现高效向量存储和检索的方法,旨在支持端侧大模型应用。文章分析了Annoy、HNSWLib、NMSLib、FLANN、VP-Trees和Lshbox等向量存储库的特点与适用场景,推荐Annoy作为多数情况下的首选方案,并提出了数据预处理、索引优化、查询优化等策略以提升性能。通过这些方法,即使在资源受限的环境中也能实现高效的向量检索。
|
19天前
|
存储 编译器
数据在内存中的存储
数据在内存中的存储
35 4
|
17天前
|
存储 Java
JVM知识体系学习四:排序规范(happens-before原则)、对象创建过程、对象的内存中存储布局、对象的大小、对象头内容、对象如何定位、对象如何分配
这篇文章详细地介绍了Java对象的创建过程、内存布局、对象头的MarkWord、对象的定位方式以及对象的分配策略,并深入探讨了happens-before原则以确保多线程环境下的正确同步。
36 0
JVM知识体系学习四:排序规范(happens-before原则)、对象创建过程、对象的内存中存储布局、对象的大小、对象头内容、对象如何定位、对象如何分配
|
24天前
|
存储 机器学习/深度学习 人工智能
数据在内存中的存储
数据在内存中的存储
|
27天前
|
存储 编译器 C++
【C++】掌握C++类的六个默认成员函数:实现高效内存管理与对象操作(二)
【C++】掌握C++类的六个默认成员函数:实现高效内存管理与对象操作
|
19天前
|
存储 C语言
深入C语言内存:数据在内存中的存储
深入C语言内存:数据在内存中的存储
|
27天前
|
存储
整型在内存中的存储
本文详细解释了计算机中整型数据的三种二进制表示方法:原码、反码和补码,并展示了如何将正数和负数的原码转换为反码和补码。
35 0
|
27天前
|
存储 编译器 C++
【C++】掌握C++类的六个默认成员函数:实现高效内存管理与对象操作(三)
【C++】掌握C++类的六个默认成员函数:实现高效内存管理与对象操作