✨三万字制作,关于C语言,你必须知道的这些知识点(高阶篇)✨

本文详细探讨了C语言的高级主题,包括数据的存储(数据类型、存储方式、大小端介绍)、指针的进阶用法(字符指针、指针数组、数组指针、函数指针、回调函数、指针与数组笔试题)、字符函数和字符串函数的原理与应用、自定义类型(结构体、枚举、联合)的使用,以及动态内存管理和文件操作的基础知识。通过实例和习题解析,帮助读者深入理解C语言的高级特性。
摘要由CSDN通过智能技术生成


目录

一,写在前面

二,数据的存储

1,数据类型介绍

2,类型的基本归类

3,整形在内存中的存储

4,浮点型在内存中的存储

三,指针的进阶

1,字符指针

2,指针数组

3,数组指针的使用

 4,函数指针

5,函数指针数组

6,回调函数

7,指针和数组笔试题

四,字符函数和字符串函数

1,strlen

2,strcpy

3,strcat

4,strcmp

5,strstr

6,memcpy

7,memmove

8,模拟实现上述内存函数与字符串函数

 五,自定义类型:结构体,枚举,联合

1,结构体

2,枚举

3,联合(共用体)

六,动态内存管理

1,为什么存在动态内存分配

2,动态内存函数的介绍

七,C语言文件操作


一,写在前面

说实话,我上一篇写的基础篇能上全站热搜榜一,真的是诚惶诚恐,觉得自己配不上这个榜一,分享内容其实没那么好,得到C站的朋友的认可,有点小开心,这会更加督促我提升自己的水平,提升自己博客的质量,对的起大家的认可。学习本篇之前可以学习我上一篇的博客,有利于本篇更好的理解和学习。点击标题即可跳转到相应博文哟。

❤️万字总结,C语言的这些万年坑你还在踩吗(基础篇)❤️

上一篇讲的是基础,这篇讲的是高阶版,需要多练习多揣摩,本篇文章是之前学习C语言的总结,制作主要是我复习用的,既然是知识,当然分享是很重要的,还是那句老话,如果你认为这篇博客写的不错的话,求评论,求收藏,求点赞,您的三连是我最大的制作动力,本文大约三万字,没有时间看完可以收藏抽时间看,部分内容我以链接形式展示,废话不多说,让我们学起来吧!!!


二,数据的存储

1,数据类型介绍

char        //字符数据类型
short       //短整型
int         //整形
long        //长整型
long long   //更长的整形
float       //单精度浮点数
double      //双精度浮点数

类型的意义:

使用这个类型开辟内存空间的大小(大小决定了使用范围)。

如何看待内存空间的视角。

2,类型的基本归类

整形家族

char
   unsigned char
   signed char
short
   unsigned short[int]
   signed short[int]
int
   unsigned int
   signed int
long
   unsigned long[int]
   signed long[int]

浮点数家族

float
double

构造类型

  数组类型
  结构体类型 struct
  枚举类型 enum
  联合类型 union
int main()
{
	unsigned char a = 200;
	unsigned char b = 100;
	unsigned char c = 0;
	c = a + b;
	printf("%d %d", a + b, c);
	return 0;
}

程序的执行结果为( )

A.300 300

B.44 44

C.300 44

D.44 300

说明:printf在传入参数的时候如果是整形会默认传入四字节,所以a+b的结果是用一个四字节的整数接收的,不会越界。而c已经在c = a + b这一步中丢弃了最高位的1,所以只能是300-256得到的44了。

※由于printf是可变参数的函数,所以后面参数的类型是未知的,所以甭管你传入的是什么类型,printf只会根据类型的不同将用两种不同的长度存储。其中8字节的只有long long、float和double(注意float会处理成double再传入),其他类型都是4字节。所以虽然a + b的类型是char,实际接收时还是用一个四字节整数接收的。另外,读取时,%lld、%llx等整型方式和%f、%lf等浮点型方式读8字节,其他读4字节。

3,整形在内存中的存储

原码、反码、补码

原码

直接将二进制按照正负数的形式翻译成二进制就可以。

反码

将原码的符号位不变,其他位依次按位取反就可以得到了。

补码

反码+1就得到补码。

想了解原反补码的计算和进制的可以看看我之前的博客,二进制的讲解

关于C语言二进制相关的内容+笔试习题,建议收藏

原码、反码、补码说法错误的是( )

A.一个数的原码是这个数直接转换成二进制

B.反码是原码的二进制符号位不变,其他位按位取反

C.补码是反码的二进制加1

D.原码、反码、补码的最高位是0表示负数,最高位是1表示正数

 ABC正确,D关于符号位的描述说反了

数据在内存的储存

#include<stdio.h>
int main()
{
	int a = 1;
	int b = -2;
	return 0;
}

 数据在内存中存储中有大小端之分

正数的原、反、补码都相同。

对于整形来说:数据存放内存中其实存放的是补码。

大小端介绍

大端(存储)模式,是指数据的低位保存在内存的高地址中,而数据的高位,保存在内存的低地址中;

小端(存储)模式,是指数据的低位保存在内存的低地址中,而数据的高位,,保存在内存的高地址中。

为什么有大端和小端

为什么会有大小端模式之分呢?这是因为在计算机系统中,我们是以字节为单位的,每个地址单元都对应着一 个字节,一个字节为8bit。但是在C语言中除了8bit的char之外,还有16bit的short型,32bit的long型(要看具 体的编译器),另外,对于位数大于8位的处理器,例如16位或者32位的处理器,由于寄存器宽度大于一个字 节,那么必然存在着一个如果将多个字节安排的问题。因此就导致了大端存储模式和小端存储模式。 例如一个 16bit 的 short 型 x ,在内存中的地址为 0x0010 , x 的值为 0x1122 ,那么 0x11 为高字节, 0x22 为低字节。对于大端模式,就将 0x11 放在低地址中,即 0x0010 中, 0x22 放在高地址中,即 0x0011 中。小 端模式,刚好相反。我们常用的 X86 结构是小端模式,而 KEIL C51 则为大端模式。很多的ARM,DSP都为小 端模式。有些ARM处理器还可以由硬件来选择是大端模式还是小端模式。

unsigned int a = 0x1234; 
unsigned char b = *(unsigned char*)&a;

在32位大端模式处理器上变量b等于( )

大端序中,低地址到高地址的四字节十六进制排列分别为00 00 12 34,其中第一个字节的内容为00,故选A

关于大小端字节序的描述正确的是( )

A.大小端字节序指的是数据在电脑上存储的二进制位顺序

B.大小端字节序指的是数据在电脑上存储的字节顺序

C.大端字节序是把数据的高字节内容存放到高地址,低字节内容存放在低地址处

D.小端字节序是把数据的高字节内容存放到低地址,低字节内容存放在高地址处

小端字节序: 低位放在低地址

大端字节序:高位放在低地址

下面代码的结果是( )

int main()
{
	char a[1000] = { 0 };
	int i = 0;
	for (i = 0; i < 1000; i++)
	{
		a[i] = -1 - i;
	}
	printf("%d", strlen(a));
	return 0;
}

a是字符型数组,strlen找的是第一次出现尾零(即值为0)的位置。考虑到a[i]其实是字符型,如果要为0,则需要-1-i的低八位要是全0,也就是问题简化成了“寻找当-1-i的结果第一次出现低八位全部为0的情况时,i的值”(因为字符数组下标为i时第一次出现了尾零,则字符串长度就是i)。只看低八位的话,此时-1相当于255,所以i==255的时候,-1-i(255-255)的低八位全部都是0,也就是当i为255的时候,a[i]第一次为0,所以a[i]的长度就是255了

4,浮点型在内存中的存储

常见的浮点数:

3.14159 1E10 浮点数家族包括: float、double、long double 类型。 浮点数表示的范围:float.h中定义

举个例子

int main()
{
	int n = 9;
	float* pFloat = (float*)&n;
	printf("n的值为:%d\n", n);
	printf("*pFloat的值为:%f\n", *pFloat);
	*pFloat = 9.0;
	printf("num的值为:%d\n", n);
	printf("*pFloat的值为:%f\n", *pFloat);
	return 0;
}

根据国际标准IEEE(电气和电子工程协会) 754,任意一个二进制浮点数V可以表示成下面的形式:

(-1)^S * M * 2^E

(-1)^s表示符号位,当s=0,V为正数;当s=1,V为负数。

M表示有效数字,大于等于1,小于2。

2^E表示指数位。

举例来说: 十进制的5.0,写成二进制是 101.0 ,相当于 1.01×2^2 。 那么,按照上面V的格式,可以得出s=0, M=1.01,E=2。

十进制的-5.0,写成二进制是 -101.0 ,相当于 -1.01×2^2 。那么,s=1,M=1.01,E=2。

IEEE 754对有效数字M和指数E,还有一些特别规定。

E不全为0或不全为1

这时,浮点数就采用下面的规则表示,即指数E的计算值减去127(或1023),得到真实值,再将有效数字M前 加上第一位的1。 比如: 0.5(1/2)的二进制形式为0.1,由于规定正数部分必须为1,即将小数点右移1位, 则为1.0*2^(-1),其阶码为-1+127=126,表示为01111110,而尾数1.0去掉整数部分为0,补齐0到23位 00000000000000000000000,则其二进制表示形式为:

0 01111110 00000000000000000000000

E全为0

这时,浮点数的指数E等于1-127(或者1-1023)即为真实值, 有效数字M不再加上第一位的1,而是还原为 0.xxxxxx的小数。这样做是为了表示±0,以及接近于0的很小的数字。

E全为1

这时,如果有效数字M全为0,表示±无穷大(正负取决于符号位s);

解释前面的题目:

下面,让我们回到一开始的问题:为什么 0x00000009 还原成浮点数,就成了 0.000000 ? 首先,将 0x00000009 拆 分,得到第一位符号位s=0,后面8位的指数 E=00000000 ,最后23位的有效数字M=000 0000 0000 0000 0000 1001。

9 -> 0000 0000 0000 0000 0000 0000 0000 1001

由于指数E全为0,所以符合上一节的第二种情况。因此,浮点数V就写成: V=(-1)^0 × 0.00000000000000000001001×2^(-126)=1.001×2^(-146) 显然,V是一个很小的接近于0的正数,所以用十进制小 数表示就是0.000000。

再看例题的第二部分。 请问浮点数9.0,如何用二进制表示?还原成十进制又是多少? 首先,浮点数9.0等于二进制 的1001.0,即1.001×2^3。

9.0 -> 1001.0 ->(-1) ^ 01.0012 ^ 3->s = 0, M = 1.001, E = 3 + 127 = 130

那么,第一位的符号位s=0,有效数字M等于001后面再加20个0,凑满23位,指数E等于3+127=130,即 10000010。 所以,写成二进制形式,应该是s+E+M,即

0 10000010 001 0000 0000 0000 0000 0000

这个32位的二进制数,还原成十进制,正是 1091567616 。


三,指针的进阶

1,字符指针

一般使用

int main()
{
	char ch = 'w';
	char* pc = &ch;
	*pc = 'w';
	return 0;
}

高阶使用

int main()
{
	char* pstr = "hello bit.";
	printf("%s\n", pstr);
	return 0;
}

下面练习一道题

#include <stdio.h>
int main()
{
	char str1[] = "hello word.";
	char str2[] = "hello word.";
	char* str3 = "hello word.";
	char* str4 = "hello word.";
	if (str1 == str2)
		printf("str1 and str2 are same\n");
	else
		printf("str1 and str2 are not same\n");

	if (str3 == str4)
		printf("str3 and str4 are same\n");
	else
		printf("str3 and str4 are not same\n");

	return 0;
}

这里str3和str4指向的是一个同一个常量字符串。C/C++会把常量字符串存储到单独的一个内存区域, 当几个指针。指向同一个字符串的时候,他们实际会指向同一块内存。但是用相同的常量字符串去初始 化不同的数组的时候就会开辟出不同的内存块。所以str1和str2不同,str3和str4不同。

下面关于"指针"的描述不正确的是:( )

A.当使用free释放掉一个指针内容后,指针变量的值被置为NULL

B.32位系统下任何类型指针的长度都是4个字节

C.指针的数据类型声明的是指针实际指向内容的数据类型

D.野指针是指向未分配或者已经释放的内存地址

Afree不会更改指针的指向。

B选项强调了32位系统,所以没问题。

CD选项是定义本身。

所以排除法也可以确定是A

关于下面代码描述正确的是:( )

char* p = "hello word";

A.把字符串hello bit存放在p变量中

B.把字符串hello bit的第一个字符存放在p变量中

C.把字符串hello bit的第一个字符的地址存放在p变量中

D.*p等价于hello bit

双引号引起来的这一段是一个常量字符串,本质是一个常量字符数组类型,赋给一个指针,相当于把一个数组的首地址赋给指针,即第一个元素h的地址。

只有选项C提到了第一个字符的地址,故选C

2,指针数组

定义

  • 539
    点赞
  • 2118
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 39
    评论
评论 39
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

-孤单又灿烂的神-

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值