MIT 6.172 L2 Bentley Rules for Optimizing Work
课程主页:
课程视频:
https://www.bilibili.com/video/BV1wA411h7N7
这次回顾第二讲,这一讲介绍了减少工作量(work)的一些方法。
Lecture 2 Bentley优化工作规则
工作量(work)
程序的工作量(在给定的输入上)是程序执行的所有操作的总和。
后续将介绍优化工作的”Bentley”规则,整体的规划如下:
数据结构
打包和编码
打包的思想是在一个机器字中存储多个数据值。编码的相关思想是将数据值转换成较少比特的表示。
例子:考虑日期数据“September 11, 2018”,该日期需要18比特表示。
编码日期:
如果我们只存储公元前4096年到公元4096年的日期,如果用数字表示日期,那么需要的位数为:
打包日期:
typedef struct {
int year: 13;
int month: 4;
int day: 5;
} date_t;
该打包表示需要22位。
增强
数据结构扩充的思想是向数据结构中添加信息,使常用操作减少工作量。
例子:单向链表拼接操作,可以通过增加尾指针减少运算时间:
预计算
预计算的思想是预先进行计算,以避免在“任务关键”时进行计算。
例子:二项式系数的计算。
方法1:
int choose(int n, int k) {
if (n < k) return 0;
if (n == 0) return 1;
if (k == 0) return 1;
return choose(n-1, k-1) + choose(n-1, k);
}
该方法会有很多重复计算,可以通过预计算的方式避免这点。
方法2:
#define CHOOSE_SIZE 100
int choose[CHOOSE_SIZE][CHOOSE_SIZE];
void init_choose() {
for (int n = 0; n < CHOOSE_SIZE; ++n) {
choose[n][0] = 1;
choose[n][n] = 1;
}
for (int n = 1; n < CHOOSE_SIZE; ++n){
choose[0][n] = 0;
for (int k = 1; k < n; ++k){
choose[n][k] = choose[n-1][k-1] + choose[n-1][k];
choose[k][n] = 0;
}
}
}
编译时初始化
编译时初始化的思想是在编译期间存储常量的值,从而节省执行时的工作。
如果表很大怎么办?可以通过程序来编写该表:
int main(int argc, const char *argv[]){
init_choose();
printf("int choose[10][10] = {\n");
for (int a = 0; a < 10; ++a) {
printf(" {");
for (int b = 0; b < 10; ++b) {
printf("%3d, ", choose[a][b]);
}
printf("}, \n" );
}
printf("}; \n");
}
缓存
缓存的思想是存储最近访问过的结果,这样程序就不需要再计算它们了。
例子:
double cached_A = 0.0;
double cached_B = 0.0;
double cached_h = 0.0;
inline double hypotenuse(double A, double B){
if (A == cached_A && B == cached_B){
return cached_h;
}
cached_A = A;
cached_B = B;
cached_h = sqrt(A*A + B*B);
return cached_h;
}
稀疏性
利用稀疏性的思想是为了避免在零上存储和计算。因为最快的计算方法是根本不计算。
例子:矩阵乘法。
该矩阵只有14项非零,只需考虑这些项即可:
压缩稀疏行(CSR):
rows表示每行开始部分的索引,cols表示对应列,vals为对应值,代码实现如下:
typedef struct {
int n, nnz;
int *rows; //length n
int *cols; //length nnz
double *vals; //length nnz
}sparse_matrix_t;
void spmv(sparse_matrix_t *A, double *x, double *y) {
for (int i = 0; i < A->n; i++){
y[i] = 0;
for (int k = A->rows[i]; k < A->rows[i+1]; k++){
int j = A->cols[k];
y[i] += A->vals[k] * x[j];
}
}
}
存储静态稀疏图(和之前类似):
逻辑
常量折叠和传播
常量折叠和传播的思想是在编译过程中计算常量表达式并将结果替换为其他表达式:
#include <math.h>
void orrery() {
const double radius = 6371000.0;
const double diameter = 2 * radius;
const double circumference = M_PI * diameter;
const double cross_area = M_PI * radius * radius;
const double surface_area = circumference * diameter;
const double volume = 4 * M_PI * radius * radius * radius / 3;
// ...
}
通过足够高的优化级别,可以在编译时计算所有表达式。
共同子表达式消除
共同子表达式消除的思想是通过对表达式进行一次求值并存储结果以备以后使用,避免多次计算同一表达式。
代数恒等式
利用代数恒等式的思想是用需要较少工作的代数等价式代替昂贵的代数表达式。
短路
在执行一系列测试时,短路的想法是在知道答案后立即停止评估。
排序测试
考虑执行一系列逻辑测试的代码。排序测试的思想是在很少成功的测试之前执行那些通常“成功”的测试——测试会选择一个特定的替代方案。同样,廉价的测试应该先于昂贵的测试。
创建快速路径
组合测试
组合测试的思想是用一个测试或switch替换一系列测试。
循环
Hoisting
hoisting的目标(也称为循环不变代码移动)是避免每次通过循环主体重新计算循环不变代码。
哨兵(Sentinel)
哨兵是放置在数据结构中的特殊虚拟值,以简化边界条件的逻辑,尤其是简化循环退出测试的处理。
循环展开
循环展开尝试通过将循环的多次连续循环合并为单个循环来节省工作,减少了循环的总数,从而减少了控制循环必须执行的指令的运行次数。
全循环展开:展开所有循环。
部分循环展开:展开了几个(但不是全部)循环。
循环融合
循环融合(也称为jamming)的想法是将同一索引范围内的多个循环合并为一个循环主体,从而节省了循环控制的开销。
消除浪费的循环
消除浪费的循环的想法是修改循环边界,以避免在本质上为空的循环主体上执行循环:
函数
内联
内联的想法是通过用函数本身的主体替换对函数的调用来避免函数调用的开销。内联函数可以像宏一样高效,但是它们的结构更好。
消除尾递归
消除尾递归的想法是用分支替换在函数的最后一步发生的递归调用,从而节省了函数调用的开销。
粗化(Coarsening)递归
粗化递归的想法是增加基本情形的大小,并使用更有效的代码来处理它,从而避免函数调用的开销。
总结
- 避免过早优化。首先获取正确的工作代码,然后进行优化,通过回归测试保留正确性。
- 减少程序的工作量并不一定会减少其运行时间,但这是一种很好的启发式方法。
- 编译器可自动执行许多底层优化。
- 要了解编译器是否实际上正在执行特定的优化,请查看汇编代码。