C/C++学习之若干问题

时间:2023-02-26 12:03:31
在书本上学到好多东西,但是那也仅仅是学习,转眼就给忘了。再实际的工作中基本上也都想不起来了,或者记得知识点太零碎了,很难串起来,但是遇到自己觉得不会的,或者有疑问的,一定要去求证,并做好记录,再附上自己的思考,那么自己就会越来越鸟了。

问题一、堆/栈问题

在C++中,内存分成5个区,他们分别是堆、栈、*存储区、全局/静态 存储区和常量存储区。

  • 栈,就是那些由编译器在需要的时候分配,在不需要的时候自动清楚的变量 的存储区。里面的变量通常是局部变量、函数参数等。
  • 堆,就是那些由new分配的内存块,他们的释放编译器不去管,由我们的应 用程序去控制,一般一个new就要对应一个delete。如果程序员没有释放掉, 那么在程序结束后,操作系统会自动回收。
  • *存储区,就是那些由malloc等分配的内存块,他和堆是十分相似的,不过它是用free来结束自己的生命的。
  • 全局/静态存储区,全局变量和静态变量被分配到同一块内存中,在以前的C语言中,全局变量又分为初始化的和未初始化的(初始化的全局变量和静态变量在一块区域,未初始化的全局变量与静态变量在相邻的另一块区域,同时未被初始化的对象存储区可以通过void*来访问和操纵,程序结束后由系统自行释 放),在C++里面没有这个区分了,他们共同占用同一块内存区。
  • 常量存储区,这是一块比较特殊的存储区,他们里面存放的是常量,不允许 修改(当然,你要通过非正当手段也可以修改,而且方法很多)明确区分堆与栈

下面这两行代码中,两个变量和两个值所在的内存类型

    char char_array[] = "array";
char *char_ptr = "pointer";

这里一定要搞清楚,变量是变量,值是值。char_array和char_ptr 是变量,如果是在函数中定义的那么他们被放在栈上,其中的值是内存地址,指向所代表的值得地址。首先说明,在C++中,内存分成5个区,他们分别是堆、栈、*存储区、全局/静态存储区和常量存储区。
那么值“array”存放在哪里?如果在函数内定义的,那么它存放在栈上,是数组得各个分量,即char_array[0]=’a’, char_array[1]=’r’, …, char_array[5]=0。那么char_array[5]为啥等于0呢,那是因为”array”表示的是{‘a’, ‘r’, ‘r’, ‘a’,’y’,0},也就是char_array的数组大小为6,char_arrayi可以被重新赋值。
值”pointer”存放在常量存储区,他的大小为8,其中的值不能够被修改。

问题二、fgets和fread的区别

fgets的一个调用如下:

fgets(buf, MAX, fp);

其中buf是一个char数组的名称,MAX是字符串的最大长度,fp是一个FILE指针。fgets()函数读取到它遇到的第一个换行符的后面,或者读取比字符串的最大长度少一个的字符,或者读取到文件结尾。然后fgets()函数向末尾添加一个空字符以构成一个字符串。所以,字符串的最大长度代表字符的最大数目加上一个空字符。如果fgets()函数在达到字符最大数目之前读完了一整行,它将在字符串的空字符之前添加一个换行符以标识一行结束。这是fgets()函数和gets()函数的不同之处,后者读取换行符后将其丢弃。
与gets()类似,fgets()遇到EOF的时候会返回NULL值,可以据此检查文件结尾。否则返回传递给他的地址值。

fread()函数的原型是:

size_t fread(void * restrict ptr, size_t size, size_t nmemb, FILE * restrict fp);

其中ptr为读入文件数据的内存存储地址,fp指定要读取的文件。例如:

double earnings[10];
fread(earnings, sizeof(double), 10, fp);

该调用将10个double值复制到earnings数组中。
fread()函数返回成功读入的项目数。正常情况下,它与nmemb相等;不过如果有读取错误的话,返回值就会小于nmemb。