csapp2e-chapter2-homework
一 前言
看了csapp2e第二章,感觉讲的很透彻,理解了一些以前学组成原理没有学懂的东西。这章最让我感觉深刻的还是计算机是怎么实现c语言中的基本数据类型的表示和操作的,这对程序员理解程序无疑是帮助巨大的。也正如这本书的题目--以编程人的视角来理解计算机系统(自己这么翻译的,呵呵!)。第二章的homework很多,很多题目都是和位打交道,并且有些题目还只能使用位操作符,不允许使用循环,比较运算符,这让平时很少使用位操作的我来说确实很捉急啊!不过这也正是锻炼自己位操作的大好机会嘛!下面选了几道我认为还是挺不错的题目拿来分享下,由于homework是没有答案的,所以下面解答只是个人的见解。
二 homework
2.65
这道题目要求写下面这样一个函数:
int even_one(unsigned x);
当x展开的比特位中包含偶数个1时返回1,否则返回0。假设x的位数是32位。只能使用位操作,逻辑操作,+、-操作,并且使用这些操作的总次数不能超过12次。
最直观的的方法当然是循环32次统计x位中1的个数,不过题目要求不能使用循环,所以显然是不可以的。既然不能使用循环,想想什么位操作能统计1的个数呢?很显然是没有的,不过题目也没让统计1的个数,换个角度,它只要统计1的个数为偶数还是奇数个就行了。很直观会想到用异或操作,具体如下:
int even_ones(unsigned x)
{
x ^= (x >> );
x ^= (x >> );
x ^= (x >> );
x ^= (x >> );
x ^= (x >> );
return !(x & );
}
2.66
这道题目要求写下面这样一个函数:
int leftmost_one(unsigned x);
实现x展开的比特位中只保留最高位1,并将结果作为函数值返回,如果x为0,返回0。例如x=0xff00,函数返回0x8000。函数中使用的操作要求和题目2.65一样。
这道题目其实是确定最高位的1在x中是第几位,不过不能使用循环来确定。那就只有把最高位的1右移来填充比这个最高位低的所有位,这样其实就已经确定了最高位1的位置了,具体如下:
int leftmost_one(unsigned x)
{
x |= x >> ;
x |= x >> ;
x |= x >> ;
x |= x >> ;
x |= x >> ;
x &= ~(x >> );
return x;
}
2.73
这道题目要求写下面这样一个函数:
int saturating_add(int x, int y);
如果x + y向上溢出,则函数返回TMax,如果x + y向下溢出,则函数返回TMin,其它都返回x + y。函数中使用的操作要求和题目2.65一样。
先判断x + y是否会溢出,如果会溢出,则根据x的符号位判断是上溢出还是下溢出就可以了。
int saturating_add(int x, int y)
{
int sum = x + y;
int w = (sizeof(int) << ) - ;
int xs = x >> w;
int ys = y >> w;
int sums = sum >> w;
(xs == ys) && (xs != sums) && (!xs && sum = TMax || xs && sum = TMin);
return sum;
}
2.81
对于给定的随机数x和y,它们的类型都为int型,判断c表达式(x > y) == (-x < -y)是否总是产生1或0?
直观上这个等式应该总是1,但由于c中的int类型在计算机中使用补码表示的。而对于补码,TMin是个很特别的数,它的反依旧是TMin,即-TMin == TMin。所以当x = TMin,y = 0时,上面的等式为0。
2.88
题目先定义了六个变量,并假设这些变量都是在IA32的机器上定义的,分别如下:
int x = random();
int y = random();
int z = random();
double dx = (double)x;
double dy = (double)y;
double dz = (double)z;
判断下面的c表达式是否总是为1或0:
A. (double)(float)x == dx
B. dx + dy == (double)(x + y)
C. dx + dy + dz == dz + dy + dx
D. dx / dx == dy / dy
A并非总是1,由于单精度的小数位数只有23位,加隐藏的1位,最高也就24位,所以很显然对于32位的int,有些数它是无法精确表示的,如2 ^ 24 + 1,它是单精度最小的无法精确表示的int数。
B也是并非总是1,由于x + y可能会溢出。
C总是为1的,由于dx,dy,dz是双精度,小数位数有52位,对于只有32的int行来说足够了。当如果dx, dy, dz是float(不考虑dx,dy,dz是由int型转化过来的),就不是总为1了,正如A中所讲的,单精度的小数部位只有24位,如果dx相对dy很小的话,dx会被舍弃掉。如dx = 3.14, dy = 1e10, dz = -1e10, dx + dy + dz == 0,而dz + dy + dx = 3.14。
D并非总是1,如果dx或者dy为0的话,会出现除0情况,这是不允许的。
2.93
这道题目要求写下面这样一个函数:
float_bits float_half(float_bits f);
其中float_bits是32位的无符号整数类型。参数f是其浮点数对应的比特位,函数要求计算0.5 * f,并返回,如果f是NaN(不是一个浮点数),则返回f。并且在电脑上穷举f的2^32种情况来比较电脑中的浮点操作结果和你自己计算的浮点操作的结果。如果需要舍入的话,用round-to-even。
这道题目要求自己实现浮点数的操作,这其实是很不习惯的(反正对于我来是这样,毕竟都没这么搞过啊!)。这道题目需要特殊处理的是浮点数的指数部分(加了偏移量的)为0或1的情况,而其它情况只要对指数减1。如果为0,则必须要将小数向右移动1位,不过这就涉及到舍入的问题了;如果为1,那么需要在小数点中加上隐藏位1,再向右移动一位,并且指数减1,这样就转化为了指数为0的情况。具体实现如下:
#include <stdio.h> typedef unsigned float_bits; float_bits float_half(float_bits f)
{
unsigned sign = f >> ;
unsigned exp = f >> & 0xff;
unsigned frac = f & 0x7fffff; if (exp == 0xff)
return f;
if (exp > )
{
exp--;
if (exp == )
frac += << ;
}
if (exp == )
{
if ((frac & 0x3) == 0x3)
frac = (frac >> ) + ;
else
frac >>= ;
} return ((sign << ) | (exp << ) | frac);
}
int main(void)
{
unsigned i, j;
float f1, f2; for (i = 0x3f800000U; i <= 0xffffffffU; i++)
{
f1 = *((float *)&i);
f1 *= (float)0.5;
j = float_half(i);
f2 = *((float *)&j);
printf("%#x : %f %f\n", i, f1, f2);
}
return ;
}
2.94
这道题目要求写下面这样一个函数:
float_bits float_twice(float_bits f);
题目的要求和2.93一样,只是要求2 * f。
需要特殊处理的是指数部分(加了偏移量的)为0的情况。当指数为0时,左移小数部分1位,如果小数部分的最高位1为第23位(从0开始数),则指数加1,并且小数部分去掉这个最高位的1,否则只要左移小数部分1位就可以了。其它情况只需要将指数加1。具体实现如下:
#include <stdio.h> typedef unsigned float_bits; float_bits float_twice(float_bits f)
{
unsigned sign = f >> ;
unsigned exp = (f >> ) & 0xff;
unsigned frac = f & 0x7fffff; if (exp == 0xff)
return f;
if (exp == )
{
frac <<= ;
if (frac & 0x800000)
{
frac &= 0x7fffff;
exp++;
}
}
else
exp++;
return ((sign << ) | (exp << ) | frac);
}
int main(void)
{
unsigned i, j;
float f1, f2; for (i = 0x37800000U; i <= 0xffffffffU; i++)
{
f1 = *((float *)&i);
f1 *= (float)2.0;
j = float_twice(i);
f2 = *((float *)&j);
printf("%#x : %f %f\n", i, f1, f2);
}
return ;
}
2.95
这道题目要求写下面这样一个函数:
float_bits float_i2f(int i);
函数实现对int类型到float类型的转化。在c中我们可能用简简单单的强制类型转化就搞定了,不过如果是自己来实现,还是有一定的难度的。
这道题主要注意三个方面。一是i为特殊数字的时候,如i=0, i=TMin;二是i为负数的时候,需要求出i的原码来,由于int类型在机器中都是用补码表示的;三是慎重考虑舍入的情况。具体实现如下:
#include <stdio.h> typedef unsigned float_bits;
#define TMin 0x80000000 float_bits float_i2f(int i)
{
unsigned sign, exp, frac, last_bit; sign = i >> ;
if (i == TMin) //处理i为最小负数的时候
{
exp = ;
frac = ;
}
else if (i == ) //处理i为0
{
exp = ;
frac = ;
}
else
{
if (sign) //如果i为负数,需要求其原码
frac = (~i + ) << ;
else
frac = i << ;
exp = ;
while (!(frac & 0x80000000))
{
exp++;
frac <<= ;
}
exp = - exp;
last_bit = ((frac >> ) & );
if (!last_bit) //处理舍入,对于最后1位为0
{
if ((frac & 0xff) == 0x80 || !((frac & 0xff) & 0x80)) //舍弃的位离0更近
frac = ((frac << ) >> );
else
frac = ((frac << ) >> ) + ;
}
else //处理舍入,对于最后1位为1
{
if ((frac & 0xff) == 0x80 || ((frac & 0xff) & 0x80)) //舍弃的位离1更近
{
frac = ((frac << ) >> ) + ;
if (frac & 0x800000)
{
exp++;
frac = ;
}
}
else
frac = ((frac << ) >> );
}
}
return ((sign << ) | (exp << ) | frac);
}
int main(void)
{
unsigned i;
int j, k;
float f1, f2; for (i = 0x80000000U; i <= 0xffffffffU; i++)
{
j = (int)i;
f1 = (float)j;
k = float_i2f(j);
f2 = *((float *)&k);
printf("%#x : %f %f\n", i, f1, f2);
}
return ;
}
2.96
这道题目要求写下面这样一个函数:
int float_f2i(float_bits f);
函数实现对float类型到int类型的转化。如果转化中溢出或者f是NaN,则返回0x80000000。
这道题和2.95差不多,主要是考虑溢出的情况。
#include <stdio.h> typedef unsigned float_bits;
#define TMin 0x80000000 int float_f2i(float_bits f)
{
unsigned sign, exp, frac, last_bit;
int i; sign = f >> ;
exp = (f >> ) & 0xff;
frac = f & 0x7fffff; if (exp == && frac == && sign == ) //处理f = TMin的情况
i = TMin;
else if (exp > || exp == 0xff) //处理f上溢出或f是NaN的情况
i = TMin;
else if (exp < ) //处理f下溢出
i = ;
else
{
exp -= ;
frac |= 0x800000;
if (exp > )
{
exp -= ;
frac <<= exp;
}
else if (exp < )
{
exp = - exp;
frac >>= exp;
}
if (sign == )
i = (~frac) + ;
else
i = frac;
}
return i;
}
int main(void)
{
unsigned i;
int j, k;
float f; for (i = 0x3fbfff70U; i <= 0xffffffffU; i++)
{
f = *(float *)&i;
j = (int)f;
k = float_f2i(i);
printf("%f : %d %d\n", f, j, k);
}
return ;
}
csapp2e-chapter2-homework的更多相关文章
- python核心编程笔记——Chapter2
对于.py文件,任何一个空的式子都不会有什么输出,如下: #!/usr/bin/env python #-*-coding=utf-8-*- #无任何效果,空语句 1 + 2 * 4 对于i++,++ ...
- bzoj 4320: ShangHai2006 Homework
4320: ShangHai2006 Homework Time Limit: 10 Sec Memory Limit: 128 MB Description 1:在人物集合 S 中加入一个新的程序员 ...
- HDU 1789 Doing Homework again(贪心)
Doing Homework again 这只是一道简单的贪心,但想不到的话,真的好难,我就想不到,最后还是看的题解 [题目链接]Doing Homework again [题目类型]贪心 & ...
- hdu-1789-Doing Homework again
/* Doing Homework again Time Limit: 1000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Oth ...
- 《深入PHP与jQuery开发》读书笔记——Chapter2
Pro PHP and jQuery Chapter2 总结 1.理解jQuery脚本的基本行为 jQuery事实上沿用了JavaScript的那一套东西,几乎所有方法都支持链式调用,也就是说方法可以 ...
- HDU 1789 Doing Homework again (贪心)
Doing Homework again http://acm.hdu.edu.cn/showproblem.php?pid=1789 Problem Description Ignatius has ...
- Doing Homework 状态压缩DP
Doing Homework 题目抽象:给出n个task的name,deadline,need. 每个任务的罚时penalty=finish-deadline; task不可以同时做.问按怎样的 ...
- 机器学习 —— 概率图模型(Homework: Exact Inference)
在前三周的作业中,我构造了概率图模型并调用第三方的求解器对器进行了求解,最终获得了每个随机变量的分布(有向图),最大后验分布(双向图).本周作业的主要内容就是自行编写概率图模型的求解器.实际上,从根本 ...
- ###《Effective STL》--Chapter2
点击查看Evernote原文. #@author: gr #@date: 2014-09-15 #@email: forgerui@gmail.com Chapter2 vector和string T ...
- Learning WCF Chapter2 Data Contracts
A data contract describes how CLR types map to XSD schema definitions. Data contracts are the prefer ...
随机推荐
- oracle 字符串分割
); create or replace function strsplit2(p_value varchar2, p_split varchar2 := ',') return str_split ...
- Xcode8中处理打印日志的配置
Xcode8中处理打印日志的配置
- vue-cli 组件的使用
开始项目之前,先了解如何创建项目: http://www.cnblogs.com/pearl07/p/6247389.html 1,项目目录结构(路由是后来建的,将在下一篇使用路由,此处可忽略). 2 ...
- spark 大数据 LR测试
#!/bin/bash size="120Y*10W"date1=`date +%F_%H-%M-%S`config="spark-submit \ --jars /da ...
- Apache服务器在80端口配置多域名虚拟主机的方法
我们在配置一台服务器的时候,如果只运行一个站点,往往过于浪费资源.Nginx和Apache都可以通过配置虚拟主机实现多站点.配置虚拟主机的方式主要有两种,一种是多个不同端口对应的多个虚拟主机站点,一种 ...
- SQL语句大全
经典SQL语句大全(绝对的经典) 一.基础 1.说明:创建数据库CREATE DATABASE database-name 2.说明:删除数据库drop database dbname3.说明:备份s ...
- html 5 实现拖放效果
在html5中要实现拖放操作,相对于以前通过鼠标操作实现,要简单得多,数据安全性也更有保障.只需要以下几步即可. 给被拖拽元素添加draggable属性,如果是文件拖放. 在拖拽元素的dragstar ...
- .NET面试题系列[10] - IEnumerable的派生类
.NET面试题系列目录 IEnumerable分为两个版本:泛型的和非泛型的.IEnumerable只有一个方法GetEnumerator.如果你只需要数据而不打算修改它,不打算为集合插入或删除任何成 ...
- ABP理论学习之设置管理
返回总目录 本篇目录 介绍 定义设置 获取设置值 更改设置 关于缓存 介绍 每个应用程序都需要存储一些设置信息,然后在应用程序中的某个地方使用这些设置.ABP提供了健壮的基础设施来存储或检索服务端和客 ...
- ABP理论学习之发布说明
返回总目录 查看更详细信息以及下载源代码请查看原文档 ABP v0.9.2.0 | [更新日期:2016/6/6 11:21:28 ] 解决方案转换成xproj/project.json格式. 添加了 ...