代码丛中过,bug 不沾身。
目录
在进入今天的内容之前,我们先看个小故事:
张三有一个U盘,U盘里下了一部 2G 的电影,有一天张三想把这个电影拷贝到他的电脑上观看,张三花了 3分钟 的时间把电影拷贝完了,张三看完电影后,为了节省空间,于是就把电影删除了,可是张三删除电影却只花了 3秒钟 ,所以张三就百思不得其解呐,所以计算机中删除数据是否真的会将我们的数据全部清空?
首先肯定是不会的,在计算机中清除数据,只要设置该数据无效即可!现阶段只要了解这个概念即可,更深入的会放到后期的操作系统中讲解。
在进入 return 关键字的学习之前,我们要先看一段代码,最重要的是我们如何正确理解这段代码?
这里我们需要了解一个知识点:调用函数,形成栈帧;函数返回,释放栈帧。
一个函数,可以被其他函数调用,也可以调用其他函数,我们的 main 函数也是被其他函数调用的!(细节后期会讲),只要是函数调用,我们都会形成栈帧,而且在栈区上是先使用高地址后使用低地址的,我们可以看图解:

为什么会打印乱码呢?(详细内容后期函数栈帧的创建与销毁会详细讲解)
前面我们讲了,计算机中删除数据本质上是把数据设置成无效,所以说当我们结束了 show 函数之后,本质上里面的内容是没有改变的,只不过紧接着我们又使用了 printf 函数所以会覆盖掉之前被释放的空间!那么我们如何证明是如我们所说的这样呢?如下图:

所以为什么临时变量具有临时性?因为栈帧结构在函数调用完毕,需要被释放!
那我们接着来看一串代码:

首先我们这串代码肯定没问题,我们可以看到 x 是函数定义的变量,具有临时性,那么这个临时变量在函数退出的时候,应该被释放!所以当我 return x; 的时候 main 函数里的变量 ret 是如何拿到 x 的值呢?
我们将从汇编的角度带大家去看到底是如何把值带回来的:

结论:函数的返回值,是通过寄存器的方式,返回给函数调用方!通过查看汇编可以看到,对于一般内置类型,寄存器eax可以充当返回值的临时空间。
问题:一般函数的返回值都是返回给函数调用方了,那么 main 函数的返回值返回给谁了?
~ 这个我会放到讲解操作系统时讲解,感兴趣的小伙伴可以先自行研究一下哈。
有C语言基础的小伙伴都知道,const 修饰的变量不可以直接被修改!
const 可以放在类型之前,也可以放在类型之后,他们两个是等价的:
int main()
{
const int a = 0;//等价于-> int const a = 0;
return 0;
}
那么我们上面说,const 修饰的变量不能直接被修改,说明是可以被间接修改的!

既然 const 修饰的变量可以间接修改,那它的意义何在呢?
- 让编译器进行直接修改式检查
- 告诉其他程序员这个变量后面不要修改,也属于一种“自描述”含义
2.1 既然 const 修饰的变量不可被修改,那么它修饰的变量可以作为数组定义的一部分吗?
int main()
{
const int n = 10;
int arr[n];
return 0;
}
上边的代码,小伙伴们可以放到不同的平台去跑一跑,在 vs2019(标准C) 下直接报错了,但是在gcc(GNU扩展)下可以,在 C99 标准中新增了变长数组,感兴趣的小伙伴可以了解一下。
2.2 const 修饰的数组数组的内容可以被修改吗?

看来是不可以被修改的,所以如果想定义一个只读数组,就可以用 const 修饰。
2.3 const 修饰指针又有什么效果呢?
1. const 放在 * 的左边:
p 指向的对象不能通过 *p 直接修改,但是 p 变量本身的值是可以修改的
2. const 放在 * 的右边:
p 指向的对象是可以通过 *p 来修改的,但是不能直接修改 p 变量本身的值
3. * 的左边和右边都放上 const :
p 指向的对象的值和 p 变量本身的值都不能直接被修改
例:
int main() { int a = 0; int b = 0; const int* p1 = &a; //const int* p1 <=等价于=> int const* p1 p1 = &b;//ok *p1 = 20;//err int* const p2 = &a; p2 = &b;//err *p2 = 20;//ok const int* const p3 = &a; p3 = &b;//err *p3 = 20;//err return 0; }
2.4 const 修饰函数返回值是怎么一回事?
其实这里我们还要讲到 const 修饰函数参数,但是他的效果跟 const 修饰局部变量的效果一样,这里就不演示了,感兴趣的小伙伴可以自己下来试试。

volatile 翻译过来是易变,不稳定的意思。其实有很多人在学完 C 语言都没见过这个关键字,同时这个关键字相较于 C 语言其他关键字也是面试常考的关键字,也有很多程序员知道他的存在,但可能从来都没有使用过它。
volatile 关键字和 const 一样是一种类型修饰符,用它修饰的变量表示可以被某些编译器未知的因素更改,比如操作系统,硬件或者其他线程等。遇到这个关键字声明的变量,编译器对访问该变量的代码就不再进行优化,从而可以提供对特殊地址的稳定访问。
我们会从 Linux 平台下汇编的角度带小伙伴们看一下加 volatile 和不加 volatile 的区别:
我们有这么一串代码:
#include <stdio.h>
int pass = 1;
int main()
{
while(pass)
{
}
return 0;
}
Linux 查看汇编以及优化的部分操作:
[lqg@VM-0-3-centos code]$ gcc test.c -O2 -g //以O2级别进行代码优化
[lqg@VM-0-3-centos code]$ objdump -S -d a.out > a.s //对形成的a.out可执行程序进行优化
[lqg@VM-0-3-centos code]$ vim a.s //查看汇编代码
不加 volatile 汇编效果:

加上 volatile 汇编效果:

最终结论:volatile 忽略编译器的优化,保持内存可见性。
篮球哥还有一个小问题:
const volatile int a = 10;
这样一句代码能编过吗?答案是可以的:
在 vs 和 gcc 中都能编译通过
const 是在编译期间起效果
volatile 在编译期间主要影响编译器,形成不优化的代码,进而影响运行。
故:编译和运行都起效果。
const 要求你不要进行写入就可以。
volatile 意思是你读取的时候,每次都要从内存读。 两者并不冲突。
虽然volatile就叫做易变关键字,但这里仅仅是描述它修饰的变量可能会变化,要编译器注意,并不是它要求对应变量必须变化!这点要特别注意。

半山腰还不够高,你总得去山顶看看吧!
我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚
几个月前,我读了一篇关于rubygem的博客文章,它可以通过阅读代码本身来确定编程语言。对于我的生活,我不记得博客或gem的名称。谷歌搜索“ruby编程语言猜测”及其变体也无济于事。有人碰巧知道相关gem的名称吗? 最佳答案 是这个吗:http://github.com/chrislo/sourceclassifier/tree/master 关于ruby-寻找通过阅读代码确定编程语言的rubygem?,我们在StackOverflow上找到一个类似的问题:
?博客主页:https://xiaoy.blog.csdn.net?本文由呆呆敲代码的小Y原创,首发于CSDN??学习专栏推荐:Unity系统学习专栏?游戏制作专栏推荐:游戏制作?Unity实战100例专栏推荐:Unity实战100例教程?欢迎点赞?收藏⭐留言?如有错误敬请指正!?未来很长,值得我们全力奔赴更美好的生活✨------------------❤️分割线❤️-------------------------
嗨~大家好,这里是可莉!今天给大家带来的是7个C语言的经典基础代码~那一起往下看下去把【程序一】打印100到200之间的素数#includeintmain(){ inti; for(i=100;i 【程序二】输出乘法口诀表#includeintmain(){inti;for(i=1;i 【程序三】判断1000年---2000年之间的闰年#includeintmain(){intyear;for(year=1000;year 【程序四】给定两个整形变量的值,将两个值的内容进行交换。这里提供两种方法来进行交换,第一种为创建临时变量来进行交换,第二种是不创建临时变量而直接进行交换。1.创建临时变量来
最近在学习CAN,记录一下,也供大家参考交流。推荐几个我觉得很好的CAN学习,本文也是在看了他们的好文之后做的笔记首先是瑞萨的CAN入门,真的通透;秀!靠这篇我竟然2天理解了CAN协议!实战STM32F4CAN!原文链接:https://blog.csdn.net/XiaoXiaoPengBo/article/details/116206252CAN详解(小白教程)原文链接:https://blog.csdn.net/xwwwj/article/details/105372234一篇易懂的CAN通讯协议指南1一篇易懂的CAN通讯协议指南1-知乎(zhihu.com)视频推荐CAN总线个人知识总
Transformers开始在视频识别领域的“猪突猛进”,各种改进和魔改层出不穷。由此作者将开启VideoTransformer系列的讲解,本篇主要介绍了FBAI团队的TimeSformer,这也是第一篇使用纯Transformer结构在视频识别上的文章。如果觉得有用,就请点赞、收藏、关注!paper:https://arxiv.org/abs/2102.05095code(offical):https://github.com/facebookresearch/TimeSformeraccept:ICML2021author:FacebookAI一、前言Transformers(VIT)在图
我正在尝试学习Ruby词法分析器和解析器(whitequarkparser)以了解更多有关从Ruby脚本进一步生成机器代码的过程。在解析以下Ruby代码字符串时。defadd(a,b)returna+bendputsadd1,2它导致以下S表达式符号。s(:begin,s(:def,:add,s(:args,s(:arg,:a),s(:arg,:b)),s(:return,s(:send,s(:lvar,:a),:+,s(:lvar,:b)))),s(:send,nil,:puts,s(:send,nil,:add,s(:int,1),s(:int,3))))任何人都可以向我解释生成的
关闭。这个问题不符合StackOverflowguidelines.它目前不接受答案。我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。关闭3年前。Improvethisquestion我正处于学习Ruby的阶段,我想查看一些小型库的源代码以了解它们是如何构建的。我不知道什么是小型图书馆,但希望SO能推荐一些易于理解的图书馆来学习。因此,如果有人知道一两个非常小的库,这是新手Rubyists学习的好例子,请推荐!我想使用Manveru'sInnatelib,因为它试图保持在2000LOC以下,但我还不熟悉其中经常使用的Ruby速记。也许大约100-5
由于匿名block和散列block看起来大致相同。我正在玩它。我做了一些严肃的观察,如下所示:{}.class#=>Hash好的,这很酷。空block被视为Hash。print{}.class#=>NilClassputs{}.class#=>NilClass为什么上面的代码和NilClass一样,下面的代码又显示了Hash?puts({}.class)#Hash#=>nilprint({}.class)#Hash=>nil谁能帮我理解上面发生了什么?我完全不同意@Lindydancer的观点你如何解释下面几行:print{}.class#NilClassprint[].class#A
下面的代码工作正常:person={:a=>:A,:b=>:B,:c=>:C}berson={:a=>:A1,:b=>:B1,:c=>:C1}kerson=person.merge(berson)do|key,oldv,newv|ifkey==:aoldvelsifkey==:bnewvelsekeyendendputskerson.inspect但是如果我在“ifblock”中添加return,我会得到一个错误:person={:a=>:A,:b=>:B,:c=>:C}berson={:a=>:A1,:b=>:B1,:c=>:C1}kerson=person.merge(berson