【计算机理论】CSAPP ch2
信息存储
十六进制表示法
(略)
字数据大小
- 大多数计算机使用8bit的块(字节)作为最小的可寻址的内存单元
- 字长指明了指针数据的标称大小(?)
- 64位系统和32位系统向后兼容
- C语言中有些数据类型的具体大小依赖于程序的编译,C99引入
int32_t
和int64_t
类型,指明数据的长度 - C标准对不同数据类型的数字的范围设置了下界,没有设置上界
寻址和字节顺序
- 存储规则:对象的地址是什么?在内存中如何排列这些字节?
- 地址:多字节对象的存储地址为连续字节序列的最小字节地址
- 排列方式:
- 小端法:低字节存在低地址
- 大端法:低字节存在高地址
- 默认规则:书写时,地址从左往右表示从低到高,字节从左往右表示从高到低
- 关心字节顺序的场合:
- 网络数据传输
- 阅读字节序列
- 规避正常的类型系统,比如强制类型转换或者联合允许一种数据类型引用一个对象
- 代码示例:按字节读取内容
//按照字节打印数据中的内容
#include<bits/stdc++.h>
using namespace std;
typedef unsigned char *byte_pointer;
void show_bytes(byte_pointer start, int len){
int i;
for(i = 0; i < len; ++i){
printf(" %.2x", start[i]);
}
printf("\n");
}
void show_int(int x){
show_bytes((byte_pointer)&x, sizeof(int));
}
int main(){
int num;
cin >> num;
show_int(num);
return 0;
}
字符串的表示
- 用ASCII码表示
- ASCII码只能表示英文字符,更一般的字符采用Unicode
代码表示
- 二进制代码不兼容,即不同系统上编译得到的二进制代码不同
布尔代数
- 运算:与、或、非
- 位向量
- 位向量表示有限集合
C语言中的位运算
- 将十六进制转换为二进制执行二进制运算,然后将结果转换为十六进制
C语言中的逻辑运算
- 0表示false,非0表示true
- 与位运算的不同:
- 位运算的参数是0或1
- 逻辑运算有短路原则
C语言中的移位运算
\([x_{\omega - 1}, x_{\omega - 2}, ..., x_0]\)
- 左移:高位舍弃,低位补0 $ [x_{\omega - 1 - k}, x_{\omega - 2 - k}, ... x_0, \underbrace{0, \cdots, 0}_{k个}]$
- 右移:
- 算数右移:补符号位 \([\underbrace{x_{\omega - 1}, ..., x_{\omega -1}}_{k个}, x_{\omega -1}, x_{\omega - 2}, ..., x_k]\)
- 逻辑右移:补0 \([\underbrace{0, ..., 0}_{k个}, x_{\omega - 1}, ..., x_k]\)
整数的表示
整型数据类型
- 典型实现中,负数比正数多1个
- C语言标准中规定(1)固定了数据大小(2)正数和负数对称
数据编码
假设整型数据类型有\(\omega\)位,将其表示记作位向量\(\vec x = [x_{\omega - 1}, x_{\omega - 2}, \cdots, x_0]\)
无符号数编码
- 定义:
\]
范围:\(Umax = \sum\limits_{i = 0}^{\omega - 1} 2 ^ i = 2 ^\omega -1\)
示例:
- 定理:\(B2U_\omega(\vec x)\)是一个双射
补码表示
定义:
\[B2T_{\omega}(\vec x) = -x_{\omega - 1}2^{\omega - 1} + \sum_{i = 0}^{\omega - 2}x_i 2 ^i【注】补码表示中,最高位被理解为负权
\]【注】补码表示中,最高位被理解为负权
范围:\(Tmin = -2^{\omega - 1}, Tmax = \sum\limits_{i = 0} ^ {\omega - 2}2 ^ i = 2 ^{\omega - 1} - 1\)
示例:
定理:\(B2T_\omega(\vec x)\)是一个双射
【注】
补码的范围中:\(\vert Tmin\vert = \vert Tmax\vert + 1\)
补码和无符号数:\(Umax = 2Tmax + 1\)
所有机器都将有符号数表示为补码,尽管C语言标准没有明确规定
反码表示
- 定义:
\[B2O_{\omega}(\vec x) = -x_{\omega - 1}(2^{\omega - 1} - 1) + \sum_{i = 0}^{\omega - 2}x_i2^i
\]
原码表示
- 定义:
\]
有符号数和无符号数之间的转换
- C语言强制类型转换:不改变位级表示,只改变位的解释方式
补码转无符号数
补码转为无符号数:
\[T2U_\omega(x) = \begin{cases}
x + 2 ^\omega & x < 0,\\
x & x \geq 0
\end{cases}
\]推导:
B2U_\omega (\vec x) - B2T_\omega(\vec x) &= x_{\omega - 1}2^\omega
\Longrightarrow B2U_\omega(\vec x) = x_{\omega - 1}2^\omega + B2T_\omega(\vec x)\\
T2U_\omega(x) &= B2U_\omega(T2B_\omega(x)) \\
&= x_{\omega -1}2^\omega + B2T_{\omega}(T2B_\omega)(x)\\
&= x_{\omega- 1}2^\omega + x
\end{align}
\]
- 理解:将补码中最高位解释为负权变为解释为正权,将补码看做无符号数时,正数不变,负数变为一个更大的数
- 示例:
无符号数转补码
- 转换:
u & u \leq Tmax_\omega \\
u - 2 ^ \omega & u > Tmax_\omega
\end{cases}
\]
- 推导(同补码转无符号数)
总结
C语言中的有符号数和无符号数的转换
- C语言中默认数字都是有符号
- 声明一个无符号常量需要在数字后面缀上‘U’
- 一个运算中同时出现有符号数和无符号数时,会被隐式转换为无符号数
扩展数字的位表示
无符号数的0扩展
- \(u_\omega = [u_{\omega -1}, \cdots, u_0]\longrightarrow u'_{\omega + k} = [\underbrace{0, \cdots, 0}_{k个}, u_{\omega -1}, \cdots, u_0]\)
补码的符号扩展
- 定义
\]
- 推导
\[\begin{align}
B2T_{\omega + 1}(\vec x') &= -x_{\omega -1}2^\omega + \sum_{i = 0}^{\omega - 1}x_i2^i\\
&= -x_{\omega - 1}2^\omega + x_{\omega -1}2^{\omega - 1} + \sum_{i=0}^{\omega - 2}x_i 2^i\\
&= -x_{\omega - 1}2^{\omega -1}+\sum_{i=0}^{\omega - 2}x_i 2^i\\
&= B2T_\omega(\vec x)
\end{align}
\]
截断数字的位表示
无符号数的截断
- 定理
x' = x \mod 2^k
\]
- 推导:截断的部分对应的位权为\(2^{k}, \cdots, 2^\omega\)
补码的截断
- 定理
x' = U2T_k(x\mod 2^k)
\]
- 推导:以无符号数为中介
整数的运算
无符号数加法
- 定义运算\(+_\omega^u\),将无符号数\(x, y\)相加后截断\(\omega\)位,可以视作是一种模运算
- 无符号数加法:
对于满足\(0\leq x, y < 2^\omega\),有:
x + y & x + y < 2 ^ \omega \\
x + y - 2 ^ \omega & x + y \geq 2 ^\omega
\end{cases}
\]
- 溢出判断:对于\(s = x+_\omega^uy\),当且仅当\(s < x\quad or\quad s < y\)时发生溢出
- 无符号数求反:模数加法形成了一个阿贝尔群,定义\(x\)的反为:
x & x = 0\\
2^\omega - x & x > 0
\end{cases}
\]
补码加法
- 定义运算\(+_\omega^t\),将有符号数\(x, y\)相加后截断\(\omega\)位,可以视作是一种模运算
- 补码加法:
x+y-2^\omega & 2 ^{\omega - 1} \leq x + y\\
x+y & -2^{\omega -1 } \leq x + y < 2 ^{\omega -1}\\
x+y+2^\omega &x + y < -2^{\omega -1}
\end{cases}
\]
【注】还是来源于截断,当没有溢出时,不发生截断;溢出的情况只会是“正+正”或者“负+负”;“正+正”溢出时,符号位为0,数字部分的最高位变为1(进位导致增加的数字),截断后原有的符号位会被丢弃,符号位变为1,真实的值为\(x+y=2^{\omega -1} + \sum\limits_{i=0}^{\omega -2}x_i2^i\),截断后的值为\(x+_\omega^t y- = 2^{\omega -1} + \sum\limits_{i=0}^{\omega -2}x_i2^i\),因此\(x+_\omega^t y = x + y - 2^\omega\);负溢出解释相似。
- 推导:
有符号和无符号具有相同的位级表示,因此先将参数转换为无符号数,然后按照无符号数运算,最后将结果转换为补码
x+_\omega^u y &= U2T_\omega(T2U_\omega(x) + T2U_\omega(y))\\
&= U2T_\omega((x_{\omega-1}2^\omega + y_{\omega - 1}2^\omega+x+y)\mod 2^\omega)\\
&= U2T_\omega((x+y)\mod 2^\omega)
\end{align}
\]
记\(z=x+y, z' = z\mod 2^\omega, z'' = U2T_\omega(z')\)
若\(-2^{\omega}\leq z < -2^{\omega-1}\),则\(z'=z+2^\omega\),因此\(0\leq z' < 2^{\omega -1}\),故\(z''=z'\)
其余三种情况以此类推
- 补码加法溢出判断:“正+正=负”或者“负+负=正”
补码的非
- 定义:对于满足\(Tmin\leq x \leq Tmax\)的\(x\)
Tmin &x = Tmin\\
-x & x>Tmin
\end{cases}
\]
- 根据位级表示计算补码的非:
- 各位取反再加1:-5=[1011], [0100] + [0001] = [0101] = 5
- 将符号位(包含符号位)与最后一个1之间的位取反:5=[0101], [1011] = -5
无符号乘法
- 定义:\(x*_\omega^uy=(xy)\mod 2^\omega\)
补码乘法
- 定义:\(x*_\omega^t y=U2T_\omega({(xy)\mod 2^\omega})\)
- 无符号和补码乘法的位级等价性
给定长度为\(\omega\)的位向量\(\vec x, \vec y\),用补码形式定义的整数为\(x, y\),用无符号定义的整数为\(x', y'\),则:
\]
证明
(x'*y')\mod 2^\omega &= ((x_{\omega -1}2^\omega+x) \cdot (y_{\omega -1}2^\omega + y))\mod 2^\omega\\
&= (xy)\mod 2^\omega\\
T2U_\omega(x*_\omega^t y)&=T2U_\omega(U2T_\omega((x\cdot y)\mod 2^\omega))\\
&= x\cdot y \mod 2^\omega\\
T2B_\omega(x*_\omega^t y)&=U2B_\omega(T2U_\omega(x*_\omega^t y))\\
&=U2B_\omega((x\cdot y)\mod 2^\omega)
\end{align}
\]
乘常数
- 乘2的幂:\(x\cdot 2^k = x << k\)
- 乘以常数K:将常数K分解为{0,1}序列,然后通过移位和加法
- 计算\(x\cdot K\)
- 分解K:\(K = [(0...0)(1...1)(0...0)...(1...1)]\)
- 计算:考虑从\(n\)位到\(m\)位(\(n \geq m\))为连续的1,有两种计算方案:
- \((x<<n) + (x <<(n-1))+...+(x<<m)\)
- \((x<<(n+1)) - (x<<m)\)
除以2的幂
对于无符号数,\(x>>k\)产生数值\(\lfloor x/2^k\rfloor\)
对于有符号数
\(x>0\)时同无符号数
\(x<0\)时,\(x>>k\)产生数值\(\lfloor x/2^k\rfloor\),\((x+(1<<k) - 1)>>k\)产生数值\(\lceil x/2^k\rceil\)
偏置的设计:
注意到\(\lceil x/y\rceil = \lfloor (x+y-1)/y\rfloor\),设\(x = qy+r\),因此\(\lfloor (x+y-1)/y\rfloor=q+\lfloor(r+y-1)/y\rfloor\),当\(r\)为0时,后面一项为0,当\(r>0\)时后面为1。取\(y=2^k\),即可得到带偏置的除法公式
【计算机理论】CSAPP ch2的更多相关文章
- 《计算机组成原理/CSAPP》网课总结(一)
现在是2022年4月17日晚10点,本月计划的网课<csapp讲解>视频课看到了第八章"异常"第三讲,视频讲的很好但更新很慢,暂时没有最新的讲解,所以先做一个简单总结. ...
- 深入理解计算机(CSAPP)资源汇总
用于资源记录. 视频: 卡内基梅隆大学 Introduction to Computer Systems CMU 15-213 Fall 作业(labs): Lab Assignments
- 《计算机组成原理/CSAPP》网课总结(二)——编译原理基础
这部分是四月份的安排,拖到五一放假了,主要是对源码编译过程的一次总结,总的来说,大致可分为预编译.编译.汇编和链接四部分.这里简单记录一下: 一 概述 1.预处理 或者说是预编译,指的是在编译前需要做 ...
- 计算机专业-世界大学学术排名,QS排名,U.S.NEWS排名
2015年美国大学计算机专业排名 计算机专业介绍:计算机涉及的领域非常广泛,其分支学科也是非常多.所以在美国将主要的专业方向分为人工智能,程序应用,计算机系统(Systems)以及计算机理论(theo ...
- 计算机与linux操作系统的发展
一.计算机 (一)计算机的概念 1.概念:计算机(computer)俗称电脑,是一种用于高速计算的电子计算机器,可以进行数值计算,又可以进行逻辑计算,还具有存储记忆功能.是能够按照程序运行,自动.高速 ...
- 这个大学在Github开源了计算机课程,看完在家上个 985
微信搜「后端技术学堂」有干货,本文已收录于Github:https://github.com/imcoderlemon/CodeClass 内含原创干货文章,千本计算机电子书,3本LeetCode题解 ...
- 三月总结&四月计划
三月总结 1. 主要任务 <C++设计模式>网课 ① 进展: 看完,对设计模式的整体设计思路和几个重要的设计模式重点学习了一下,目前对设计模式的认识还比较浅 ② 总结: 做完总结 ③ 反思 ...
- 一个人的 ClojureScript 技术栈
作者:题叶链接:https://zhuanlan.zhihu.com/p/24425284来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载请注明出处.今天(昨天)分享完关于 Cloj ...
- [Z]CS权威会议
CS Conference TOP 40 计算机会议TOP40 一.A 类 15 个 ASPLOS: Architecture Support for Programming Languages an ...
随机推荐
- awk 循环过滤EPC脚本
无处理中EPC过滤脚本 #!/bin/bash if [ "$#" -lt 3 -o "$#" -gt 4 ];then #if [ "$#" ...
- 如何把maven项目转为eclipse项目
如何把maven项目转为eclipse项目,按照如下操作便可. 在cmd窗口, 载cmd窗口进入到maven项目所在目录下,输入如下命令: mvn eclipse:eclipse 这样便可.
- JAVA导入(读取)Excel中的数据(支持xls与xlsx文件)
一.导入jar包 poi-3.7.jarpoi-scratchpad-3.7.jarpoi-examples-3.7.jarpoi-ooxml-3.7.jarpoi-ooxml-schemas-3.7 ...
- Guava Retryer实现接口重试
前言 小黑在开发中遇到个问题,我负责的模块需要调用某个三方服务接口查询信息,查询结果直接影响后续业务逻辑的处理: 这个接口偶尔会因网络问题出现超时,导致我的业务逻辑无法继续处理: 这个问题该如何解决呢 ...
- 【LeetCode】46. Permutations 解题报告(Python & C++)
作者: 负雪明烛 id: fuxuemingzhu 个人博客:http://fuxuemingzhu.cn/ 目录 题目描述 解题方法 方法一:库函数 方法二:递归 方法三:回溯法 日期 题目地址:h ...
- 【LeetCode】468. Validate IP Address 解题报告(Python)
[LeetCode]468. Validate IP Address 解题报告(Python) 标签(空格分隔): LeetCode 作者: 负雪明烛 id: fuxuemingzhu 个人博客: h ...
- 编写Java程序,以树形结构显示国家-直辖市/省/州信息
返回本章节 返回作业目录 需求说明: 以树形结构显示国家-直辖市/省/州信息 实现思路: 创建显示树形结构的类Tree,在该类中定义Map类型的全局实例属性countryMap,该Map集合用于存放所 ...
- 每天学一点——while循环(2)、for循环
while循环(2) while+continue 打印数字的话相信朋友们在python中不会一个个的print吧 eg: 或者是打印列表里的元素 eg 这种方法只适用于你知道里面有多少个元素, 不然 ...
- 从零开始学springboot-2.配置项目
### 配置项目 #### 将application.properties改名为application.yml #### 在resources文件夹中(和上面那个配置文件同一路径下)新建一个文件app ...
- dokcer部署Redis哨兵模式
架构图 哨兵的介绍 sentinel , 中文是哨兵. 哨兵是redis 集群架构中非常重要的一个组件,主要功能如下: (1)集群监控:负责监控reidis master 和slave 进程是否正常工 ...