C语言深度剖析--不定期更新的第四弹

2024-09-08 02:12

本文主要是介绍C语言深度剖析--不定期更新的第四弹,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在这里插入图片描述
哈哈哈哈哈哈,今天一天两更!

void关键字

void关键字不能用来定义变量,原因是void本身就被编译器解释为空类型,编译器强制地不允许定义变量
在这里插入图片描述
定义变量的本质是:开辟空间

而void 作为空类型,理论上不应该开辟空间(针对编译器而言),即使开辟了空间,也只是作为一个占位符看待(针对Linux来说)

所以,既然无法开辟空间,也无法作为正常变量使用,既然无法使用,干脆编译器不让它编译变量

void修饰函数返回值和参数

需要注意的点是:C语言中函数可以不带返回值,默认的返回值是int

但是我们平常在编写函数相关的代码的时候还是得带上函数的返回值类型,否则人家会在猜测究竟是默认?还是忘了没有写返回值?

所以在前面用void函数修饰的作用是起到一个提醒和占位的作用

void修饰函数返回值:1.占位符,让用户明确不需要返回值2.告知编译器,这个返回值无法被接受

void充当函数的形参列表:告知用户或编译器,该函数不需要传参

结论:如果一个函数没有参数,将参数列表设置为void,是一个不错的习惯,因为可以将错误提前发现

void指针

void指针可以创建变量,原因在于void*是指针,是指针,空间大小就能明确出来

void*可以被任何类型的指针接受,void * 可以接受任意类型指针(常用)

进一步来说就是库,系统接口的设计上,尽量设计成通用接口

如这样的:
在这里插入图片描述
例子如下:

#include <stdio.h>
int main()
{void*p=NULL;int*x=NULL;double*y=NULL;p=x;//虽然类型不同,但是编译器不会报错p=y;//同上x=p;y=p;//编译器也不会报错return 0;
}

这里产生了一个问题:void类型的指针是否可以计算呢?

在不同的平台上是不一样的,在VS的环境下,是不可以的,但是在Linux的环境下是可以的,主要原因出现在两个平台对于void大小的理解,VS认为void大小为0,但是Linux认为是sizeof(void)

void*指针不可以解引用,虽然void *可以接受任意类型,但是还是不可以解引用

return关键字

两个问题的区别:C语言有没有字符串类型VS C语言有没有字符串

C语言有字符串,但是C语言没有字符串类型

注意点:求字符串长度是不包括‘\0’的,求字符串容量是包括’\0’

计算机中是否真的需要将所有的数据清零?

计算机中清空数据,只需要设置该数据无效即可。

这句话的意思其实不太准确,只是因为我们所学的知识还没有这么多而已,打个比方,一个10GB的文件,可能只需要十个比特位大小,一个比特位代表1GB

接下来看如何正确理解下面的代码:

#include <stdio.h>
char*show()
{char str[]="hello cosmic love";return str;
}
int main()
{char*s=show();printf("%s\n",s);return 0;
}

在这里插入图片描述
打印结果是一串乱码

这里我们需要懂得函数栈帧相关的知识。

调用函数,形成栈帧;函数返回,释放栈帧

在这里插入图片描述
在这里插入图片描述
但是在调试的时候,s指向的值还在

从12行调试到13行的时候发生了变化
在这里插入图片描述
原因有下面几点:

1.计算机并不清空数据

2.printf也是函数,也要遵守这些规则,所以就二次覆盖了show的栈区

补充2个点:

1.怎样保证栈帧申请的空间是够的?

因为编译器会根据关键字大小预估充足的空间大小

2.栈帧的结构是怎样的?

可以联想我们之前学过的递归的概念,栈帧的创建也是一个不断向下创建的过程

有个问题:临时变量为什么具有临时性?

因为临时变量在函数栈帧中创建,栈帧结构在函数调用完毕之后要被释放

书写规范上的注意:

return语句不可返回指向“栈内存”的指针,因为该内存在函数体结束的时候会被销毁

可以看下面几行代码:

int GetData()
{int x=0x11223344;printf("run get data!\n");return x;
}
int main()
{int y=GetData();printf("ret:%x\n",y);
}

来看运行结果:

在这里插入图片描述
貌似跟前面有点悖论

这里拿到的不是X,拿到的是里面的内容

看一下里面的反汇编代码:
得到了下面的结论:

在这里插入图片描述
在上面的代码做一点小小的修改:

int GetData()
{int x=0x11223344;printf("run get data!\n");return x;
}
int main()
{GetData();printf("ret:%x\n",y);
}

在这里插入图片描述
需要注意一个概念:函数的返回值具有常性

结论

一个函数如何返回给外部调用方,本质是通过寄存器;

当我们返回,没有对应的接收时,调用return 会生成同等汇编语言,如果对应的接收方,就会继续往下走

这篇关于C语言深度剖析--不定期更新的第四弹的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1146816

相关文章

使用SQL语言查询多个Excel表格的操作方法

《使用SQL语言查询多个Excel表格的操作方法》本文介绍了如何使用SQL语言查询多个Excel表格,通过将所有Excel表格放入一个.xlsx文件中,并使用pandas和pandasql库进行读取和... 目录如何用SQL语言查询多个Excel表格如何使用sql查询excel内容1. 简介2. 实现思路3

Go语言实现将中文转化为拼音功能

《Go语言实现将中文转化为拼音功能》这篇文章主要为大家详细介绍了Go语言中如何实现将中文转化为拼音功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 有这么一个需求:新用户入职 创建一系列账号比较麻烦,打算通过接口传入姓名进行初始化。想把姓名转化成拼音。因为有些账号即需要中文也需要英

Go语言使用Buffer实现高性能处理字节和字符

《Go语言使用Buffer实现高性能处理字节和字符》在Go中,bytes.Buffer是一个非常高效的类型,用于处理字节数据的读写操作,本文将详细介绍一下如何使用Buffer实现高性能处理字节和... 目录1. bytes.Buffer 的基本用法1.1. 创建和初始化 Buffer1.2. 使用 Writ

深入理解C语言的void*

《深入理解C语言的void*》本文主要介绍了C语言的void*,包括它的任意性、编译器对void*的类型检查以及需要显式类型转换的规则,具有一定的参考价值,感兴趣的可以了解一下... 目录一、void* 的类型任意性二、编译器对 void* 的类型检查三、需要显式类型转换占用的字节四、总结一、void* 的

Redis缓存问题与缓存更新机制详解

《Redis缓存问题与缓存更新机制详解》本文主要介绍了缓存问题及其解决方案,包括缓存穿透、缓存击穿、缓存雪崩等问题的成因以及相应的预防和解决方法,同时,还详细探讨了缓存更新机制,包括不同情况下的缓存更... 目录一、缓存问题1.1 缓存穿透1.1.1 问题来源1.1.2 解决方案1.2 缓存击穿1.2.1

五大特性引领创新! 深度操作系统 deepin 25 Preview预览版发布

《五大特性引领创新!深度操作系统deepin25Preview预览版发布》今日,深度操作系统正式推出deepin25Preview版本,该版本集成了五大核心特性:磐石系统、全新DDE、Tr... 深度操作系统今日发布了 deepin 25 Preview,新版本囊括五大特性:磐石系统、全新 DDE、Tree

Linux Mint Xia 22.1重磅发布: 重要更新一览

《LinuxMintXia22.1重磅发布:重要更新一览》Beta版LinuxMint“Xia”22.1发布,新版本基于Ubuntu24.04,内核版本为Linux6.8,这... linux Mint 22.1「Xia」正式发布啦!这次更新带来了诸多优化和改进,进一步巩固了 Mint 在 Linux 桌面

SpringCloud配置动态更新原理解析

《SpringCloud配置动态更新原理解析》在微服务架构的浩瀚星海中,服务配置的动态更新如同魔法一般,能够让应用在不重启的情况下,实时响应配置的变更,SpringCloud作为微服务架构中的佼佼者,... 目录一、SpringBoot、Cloud配置的读取二、SpringCloud配置动态刷新三、更新@R

Node.js 中 http 模块的深度剖析与实战应用小结

《Node.js中http模块的深度剖析与实战应用小结》本文详细介绍了Node.js中的http模块,从创建HTTP服务器、处理请求与响应,到获取请求参数,每个环节都通过代码示例进行解析,旨在帮... 目录Node.js 中 http 模块的深度剖析与实战应用一、引言二、创建 HTTP 服务器:基石搭建(一

C语言线程池的常见实现方式详解

《C语言线程池的常见实现方式详解》本文介绍了如何使用C语言实现一个基本的线程池,线程池的实现包括工作线程、任务队列、任务调度、线程池的初始化、任务添加、销毁等步骤,感兴趣的朋友跟随小编一起看看吧... 目录1. 线程池的基本结构2. 线程池的实现步骤3. 线程池的核心数据结构4. 线程池的详细实现4.1 初