MIT 6.828 JOS学习笔记8. Exercise 1.4
Lab 1 Exercise 4
阅读关于C语言的指针部分的知识。最好的参考书自然是"The C Programming Language"。
阅读5.1到5.5节。然后下载pointers.c的代码,并且编译运行它,确保你理解在屏幕上打印出来的所有的值是怎么来的。尤其要重点理解第1行,第6行的指针地址是如何得到的,以及在第2行到第4行的值是如何得到的,还有为什么在第5行打印出来的值看起来像程序崩溃了。
答:
首先编译运行文件pointer.c,得到如下结果:
首先程序声明了3个重要的数组,指针
int型数组--int a[4]; int类型指针--int *b = malloc(16); int类型指针--int *c;
打印的第一句:
printf("1: a = %p, b = %p, c = %p\n", a, b, c);
其中:
a输出的是数组a的首地址,0xbfb4bf84。
b输出的是指针b所指向的操作系统分配给它的空间的起始地址,0x8886008
c输出的是未定义的指针变量c的值,0xb75d8255
打印第二句之前,完成的操作:
c = a; 让c指针和a指向同一个内存地址,0xbfb4bf84;
for(i = 0; i<4; i++) a[i] = 100+i; 这个操作会让数组a的四个单元的值变为,100,101,102,103。
c[0] = 200; 由于c和a指向同一个地方,c[0] = a[0] = 200;
printf("2: a[0] = %d, a[1] = %d, a[2] = %d, a[3] = %d,", a[0], a[1], a[2], a[3]); 所以打印第二句,第一个元素的值会是200.
打印第三句之前,完成操作:
c[1] = 300;
*(c+2) = 301;
3[c] = 302;
分别代表访问数组中的值的三种不同的方法。由于c和a相同,所以它们也是在修改数组a的值。把a[1], a[2],a[3]全都被改变。printf("3: a[0] = %d, a[1] = %d, a[2] = %d, a[3] = %d,", a[0], a[1], a[2], a[3]);
打印第四句前,完成操作:c = c+1;
将指针c指向数组中下一个单元,即a[1]
*c = 400
修该a[1]值为400
printf("4: a[0] = %d, a[1] = %d, a[2] = %d, a[3] = %d,", a[0], a[1], a[2], a[3]);
打印第五句前,完成操作
c = (int *)((char *)c + 1);
先把c强制转换为char类型指针,然后指针加1,此时c的数值应该也加1,由于之前c指向a数组中1号元素的起始地址,1号元素起始地址为0xbfb4bf88,之后加1,变为0xbfb4bf89。然后再把c强制转换回int类型指针,此时c所操作的地址为0xbfb4bf89~0xbfb4bf8c。
*c = 500
把地址地址为0xbfb4bf89~0xbfb4bf8c的值换为500,此时会影响原来数组a的1,2号元素。原来一号元素在内存中存放在地址0xbfb4bf88~0xbfb4bf8b处,存放的值为400,十六进制为0x00000190
0xbfb4bf88 0xbfb4bf89 0xbfb4bf8a 0xbfb4bf8b
90 01 00 00
原来二号元素在内存中存放在地址0xbfb4bf8c~0xbfb4bf8f处,存放的值为301,十六进制为0x0000012D。
0xbfb4bf8c 0xbfb4bf8d 0xbfb4bf8e 0xbfb4bf8f
2D 01 00 00
而现在c操作的地址为0xbfb4bf89~0xbfb4bf8c,并赋值500,十六进制为0x000001F4,所以内存单元变为
0xbfb4bf88 0xbfb4bf89 0xbfb4bf8a 0xbfb4bf8b
90 F4 01 00
0xbfb4bf8c 0xbfb4bf8d 0xbfb4bf8e 0xbfb4bf8f
00 01 00 00
所以此时a[1]的值变为0x0001F490 = 128144, a[2]的值变为0x00000100 = 256.
打印第六句前,完成操作:
b = (int *)a + 1;
这步操作会把b的值加1,由于b现在是int类型指针,所以数值上b的值增加4,所以b的值变为 0xbfb4bf84 + 0x4 = 0xbfb4bf88
c = (int *)((char *)a+1);
这步还是先把a转换为char型指针,然后加1,此时加1会让数值上只增加1,所以c的值变为 0xbfb4bf84 + 0x1 = 0xbfb4bf85
当然运行在你的机器上,这些地址的值一定和我的不一样,但是原理是相同的。
MIT 6.828 JOS学习笔记8. Exercise 1.4的更多相关文章
- MIT 6.828 JOS学习笔记11 Exercise 1.8
Exercise 1.8 我们丢弃了一小部分代码---即当我们在printf中指定输出"%o"格式的字符串,即八进制格式的代码.尝试去完成这部分程序. 解答: 在这个练 ...
- MIT 6.828 JOS学习笔记12 Exercise 1.9
Lab 1中Exercise 9的解答报告 Exercise 1.9: 判断一下操作系统内核是从哪条指令开始初始化它的堆栈空间的,以及这个堆栈坐落在内存的哪个地方?内核是如何给它的堆栈保留一块内存空间 ...
- MIT 6.828 JOS学习笔记13 Exercise 1.10
Lab 1 Exercise 10 为了能够更好的了解在x86上的C程序调用过程的细节,我们首先找到在obj/kern/kern.asm中test_backtrace子程序的地址, 设置断点,并且探讨 ...
- MIT 6.828 JOS学习笔记9. Exercise 1.5
Lab 1 Exercise 5 再一次追踪一下boot loader的一开始的几句指令,找到第一条满足如下条件的指令处: 当我修改了boot loader的链接地址,这个指令就会出现错误. 找到这样 ...
- MIT 6.828 JOS学习笔记5. Exercise 1.3
Lab 1 Exercise 3 设置一个断点在地址0x7c00处,这是boot sector被加载的位置.然后让程序继续运行直到这个断点.跟踪/boot/boot.S文件的每一条指令,同时使用boo ...
- MIT 6.828 JOS学习笔记3. Exercise 1.2
这篇博文是对Lab 1中的Exercise 2的解答~ Lab 1 Exercise 2: 使用GDB的'si'命令,去追踪ROM BIOS几条指令,并且试图去猜测,它是在做什么.但是不需要把每个细节 ...
- MIT 6.828 JOS学习笔记2. Lab 1 Part 1.2: PC bootstrap
Lab 1 Part 1: PC bootstrap 我们继续~ PC机的物理地址空间 这一节我们将深入的探究到底PC是如何启动的.首先我们看一下通常一个PC的物理地址空间是如何布局的: ...
- MIT 6.828 JOS学习笔记0. 写在前面的话
0. 简介 操作系统是计算机科学中十分重要的一门基础学科,是一名计算机专业毕业生必须要具备的基础知识.但是在学习这门课时,如果仅仅把目光停留在课本上一些关于操作系统概念上的叙述,并不能对操作系统有着深 ...
- MIT 6.828 JOS学习笔记7. Lab 1 Part 2.2: The Boot Loader
Lab 1 Part 2 The Boot Loader Loading the Kernel 我们现在可以进一步的讨论一下boot loader中的C语言的部分,即boot/main.c.但是在我们 ...
随机推荐
- bzoj1835[ZJOI2010]base基站选址
据说正解是什么线段树优化DP,但是作为脑子有坑选手,我们需要5k的做法: 主席树+决策单调性..... F[m][i]表示已经放置了m个基站,第m个基站放置在第i个村庄,第i个村庄及之前的村庄的总最少 ...
- Codeforces Round #342 (Div. 2) D. Finals in arithmetic(想法题/构造题)
传送门 Description Vitya is studying in the third grade. During the last math lesson all the pupils wro ...
- [HAOI2009]求回文串
神奇到爆炸的贪心,策略很简单.但是实现上好像比较恶心.换了一种思路.先保存所有点应该转移到的位置,BIT搞个逆序对就好了. 如何找到每个点应该转移到的位置?这个处理方式也是比较玄学.看代码吧. //O ...
- php 实用例子:购物车 涉及session与ajax
login: <div>用户名:<input type="text" id="uid" /></div><div> ...
- 零基础如何系统学习Java Web
零基础如何系统学习Java Web? 我来给你说一说 你要下决心,我要转行做开发,这样你才能学成. 你要会打字,我公司原来有一个程序员,打字都是两个手一指禅,身为程序员你一指禅怎么写出的代码,半个 ...
- mate标签
<meta charset='utf-8'> <!-- 优先使用 IE 最新版本和 Chrome --> <meta http-equiv="X-UA-C ...
- Node.js入门笔记(3):全局对象(2)
buffer 用于更好操作二进制数据,他是一个全局变量.类似数组. var a=new Buffer(); buffer类的三种实现 第一种创建方式 new Buffer(size);size[Num ...
- 忍不住记录下小型的CMDB系统
- LINQ驱动数据的查询功能
一.LINQ概念 LINQ是微软在.NetFramework3.5中新加入的语言功能,在语言中以程序代码方式处理集合的能力. 1.1 LINQ VS 循环处理 在我刚工作时候,对于集合对象的处理一般是 ...
- [Scala] 快学Scala A3L3
Actor 通过尽可能避免锁和共享状态,actor使得我们能够容易地设计出正确.没有死锁或争用状况的程序. Scala类库提供了一个actor模型的简单实现.AKKA是更高级的actor类库. 19. ...