C代码优化转载.docx

上传人:b****6 文档编号:6325742 上传时间:2023-01-05 格式:DOCX 页数:17 大小:22.62KB
下载 相关 举报
C代码优化转载.docx_第1页
第1页 / 共17页
C代码优化转载.docx_第2页
第2页 / 共17页
C代码优化转载.docx_第3页
第3页 / 共17页
C代码优化转载.docx_第4页
第4页 / 共17页
C代码优化转载.docx_第5页
第5页 / 共17页
点击查看更多>>
下载资源
资源描述

C代码优化转载.docx

《C代码优化转载.docx》由会员分享,可在线阅读,更多相关《C代码优化转载.docx(17页珍藏版)》请在冰豆网上搜索。

C代码优化转载.docx

C代码优化转载

谈到优化,很多人都会直接想到汇编。

难道优化只能在汇编层次吗?

当然不是,C++层次一样可以作代码优化,其中有些常常是意想不到的。

在C++层次进行优化,比在汇编层次优化具有更好的移植性,应该是优化中的首选做法。

1.确定浮点型变量和表达式是float型

为了让编译器产生更好的代码(比如说产生3DNow!

或SSE指令的代码),必须确定浮点型变量和表达式是float型的。

要特别注意的是,以";F";或";f";为后缀(比如:

3.14f)的浮点常量才是float型,否则默认是double型。

为了避免float型参数自动转化为double,请在函数声明时使用float。

2.使用32位的数据类型

  编译器有很多种,但它们都包含的典型的32位类型是:

int,signed,signedint,unsigned,unsignedint,long,signedlong,longint,signedlongint,unsignedlong,unsignedlongint。

尽量使用32位的数据类型,因为它们比16位的数据甚至8位的数据更有效率。

3.明智使用有符号整型变量

  在很多情况下,你需要考虑整型变量是有符号还是无符号类型的。

比如,保存一个人的体重数据时不可能出现负数,所以不需要使用有符号类型。

但是,如果是要保存温度数据,就必须使用到有符号的变量。

  在许多地方,考虑是否使用有符号的变量是必要的。

在一些情况下,有符号的运算比较快;但在一些情况下却相反。

  比如:

整型到浮点转化时,使用大于16位的有符号整型比较快。

因为x86构架中提供了从有符号整型转化到浮点型的指令,但没有提供从无符号整型转化到浮点的指令。

看看编译器产生的汇编代码:

  不好的代码:

编译前编译后

doublex;mov[foo+4],0

unsignedinti;moveax,i

x=i;mov[foo],eax

flidqwordptr[foo]

fstpqwordptr[x]

  上面的代码比较慢。

不仅因为指令数目比较多,而且由于指令不能配对造成的FLID指令被延迟执行。

最好用以下代码代替:

推荐的代码:

编译前编译后

doublex;filddwordptr

inti;fstpqwordptr[x]

x=i;

  在整数运算中计算商和余数时,使用无符号类型比较快。

以下这段典型的代码是编译器产生的32位整型数除以4的代码:

  不好的代码

编译前编译后

inti;moveax,i

i=i/4;cdq

andedx,3

addeax,edx

sareax,2

movi,eax

推荐的代码

编译前编译后

unsignedinti;shri,2

i=i/4;

 总结:

 无符号类型用于:

除法和余数,循环计数,数组下标

有符号类型用于:

整型到浮点的转化

4.whileVS.for

  在编程中,我们常常需要用到无限循环,常用的两种方法是while

(1)和for(;;)。

这两种方法效果完全一样,但那一种更好呢?

然我们看看它们编译后的代码:

编译前编译后

while

(1);moveax,1

testeax,eax

jefoo+23h

jmpfoo+18h

编译前编译后

for(;;);jmpfoo+23h

  一目了然,for(;;)指令少,不占用寄存器,而且没有判断跳转,比while

(1)好。

5.使用数组型代替指针型

  使用指针会使编译器很难优化它。

因为缺乏有效的指针代码优化的方法,编译器总是假设指针可以访问内存的任意地方,包括分配给其他变量的储存空间。

所以为了编译器产生优化得更好的代码,要避免在不必要的地方使用指针。

一个典型的例子是访问存放在数组中的数据。

C++允许使用操作符[]或指针来访问数组,使用数组型代码会让优化器减少产生不安全代码的可能性。

比如,x[0]和x[2]不可能是同一个内存地址,但*p和*q可能。

强烈建议使用数组型,因为这样可能会有意料之外的性能提升。

不好的代码

typedefstruct

{

  floatx,y,z,w;

}VERTEX;

typedefstruct

{

  floatm[4][4];

}MATRIX;

voidXForm(float*res,constfloat*v,constfloat*m,intnNumVerts)

{

  floatdp;

  inti;

   constVERTEX*vv=(VERTEX*)v;

   for(i=0;i<;nNumVerts;i++)

  {

    dp=vv->;x**m++;

    dp+=vv->;y**m++;

    dp+=vv->;z**m++;

    dp+=vv->;w**m++;

    *res++=dp;      //写入转换了的x

    dp=vv->;x**m++;

    dp+=vv->;y**m++;

    dp+=vv->;z**m++;

    dp+=vv->;w**m++;

    *res++=dp;     //写入转换了的y

    dp=vv->;x**m++;

    dp+=vv->;y**m++;

    dp+=vv->;z**m++;

    dp+=vv->;w**m++;

    *res++=dp;    //写入转换了的z

    dp=vv->;x**m++;

    dp+=vv->;y**m++;

    dp+=vv->;z**m++;

    dp+=vv->;w**m++;

    *res++=dp;    //写入转换了的w

    vv++;       //下一个矢量

    m-=16;

  }

}

推荐的代码

typedefstruct

{

  floatx,y,z,w;

}VERTEX;

typedefstruct

{

  floatm[4][4];

}MATRIX;

voidXForm(float*res,constfloat*v,constfloat*m,intnNumVerts)

{

  inti;

  constVERTEX*vv=(VERTEX*)v;

  constMATRIX*mm=(MATRIX*)m;

  VERTEX*rr=(VERTEX*)res;

  for(i=0;i<;nNumVerts;i++)

  {

    rr->;x=vv->;x*mm->;m[0][0]+vv->;y*mm->;m[0][1]

        +vv->;z*mm->;m[0][2]+vv->;w*mm->;m[0][3];

    rr->;y=vv->;x*mm->;m[1][0]+vv->;y*mm->;m[1][1]

        +vv->;z*mm->;m[1][2]+vv->;w*mm->;m[1][3];

    rr->;z=vv->;x*mm->;m[2][0]+vv->;y*mm->;m[2][1]

        +vv->;z*mm->;m[2][2]+vv->;w*mm->;m[2][3];

    rr->;w=vv->;x*mm->;m[3][0]+vv->;y*mm->;m[3][1]

        +vv->;z*mm->;m[3][2]+vv->;w*mm->;m[3][3];

  }

}

  注意:

源代码的转化是与编译器的代码发生器相结合的。

从源代码层次很难控制产生的机器码。

依靠编译器和特殊的源代码,有可能指针型代码编译成的机器码比同等条件下的数组型代码运行速度更快。

明智的做法是在源代码转化后检查性能是否真正提高了,再选择使用指针型还是数组型。

6.充分分解小的循环

  要充分利用CPU的指令缓存,就要充分分解小的循环。

特别是当循环体本身很小的时候,分解循环可以提高性能。

BTW:

很多编译器并不能自动分解循环。

不好的代码推荐的代码

//3D转化:

把矢量V和4x4矩阵M相乘

for(i=0;i<;4;i++)

{

  r=0;

  for(j=0;j<;4;j++)

  {

    r+=M[j]*V[j];

  }

}

r[0]=M[0][0]*V[0]+M[1][0]*V[1]+M[2][0]*V[2]+M[3][0]*V[3];

r[1]=M[0][1]*V[0]+M[1][1]*V[1]+M[2][1]*V[2]+M[3][1]*V[3];

r[2]=M[0][2]*V[0]+M[1][2]*V[1]+M[2][2]*V[2]+M[3][2]*V[3];

r[3]=M[0][3]*V[0]+M[1][3]*V[1]+M[2][3]*V[2]+M[3][3]*v[3];

7.避免没有必要的读写依赖

  当数据保存到内存时存在读写依赖,即数据必须在正确写入后才能再次读取。

虽然AMDAthlon等CPU有加速读写依赖延迟的硬件,允许在要保存的数据被写入内存前读取出来,但是,如果避免了读写依赖并把数据保存在内部寄存器中,速度会更快。

在一段很长的又互相依赖的代码链中,避免读写依赖显得尤其重要。

如果读写依赖发生在操作数组时,许多编译器不能自动优化代码以避免读写依赖。

所以推荐程序员手动去消除读写依赖,举例来说,引进一个可以保存在寄存器中的临时变量。

这样可以有很大的性能提升。

下面一段代码是一个例子:

不好的代码

floatx[VECLEN],y[VECLEN],z[VECLEN];

......

for(unsignedintk=1;k<;VECLEN;k++)

{

  x[k]=x[k-1]+y[k];

}

for(k=1;k<;VECLEN;k++)

{

  x[k]=z[k]*(y[k]-x[k-1]);

}

  推荐的代码

floatx[VECLEN],y[VECLEN],z[VECLEN];

......

floatt(x[0]);

for(unsignedintk=1;k<;VECLEN;k++)

{

  t=t+y[k];

  x[k]=t;

}

t=x[0];

for(k=1;k<;VECLEN;k++)

{

  t=z[k]*(y[k]-t);

  x[k]=t;

}

8.Switch的用法

  Switch可能转化成多种不同算法的代码。

其中最常见的是跳转表和比较链/树。

推荐对case的值依照发生的可能性进行排序,把最有可能的放在第一个,当switch用比较链的方式转化时,这样可以提高性能。

此外,在case中推荐使用小的连续的整数,因为在这种情况下,所有的编译器都可以把switch转化成跳转表。

不好的代码

intdays_in_month,short_months,normal_months,long_months;

......

switch(days_in_month)

{

  case28:

  case29:

    short_months++;

    break;

  case30:

    normal_months++;

    break;

  case31:

    long_months++;

    break;

  default:

    cout<;<;";monthhasfewerthan28ormorethan31days";<;<;endl;

    break;

}

推荐的代码

intdays_in_month,short_months,normal_months,long_months;

......

switch(days_in_month)

{

  case31:

    long_months++;

    break;

  case30:

    normal_months++;

    break;

  case28:

  case29:

    short_months++;

    break;

  default:

    cout<;<;";monthhasfewerthan28ormorethan31days";<;<;endl;

    break;

}

9.所有函数都应该有原型定义

  一般来说,所有函数都应该有原型定义。

原型定义可以传达给编译器更多的可能用于优化的信息。

  尽可能使用常量(const)。

C++标准规定,如果一个const声明的对象的地址不被获取,允许编译器不对它分配储存空间。

这样可以使代码更有效率,而且可以生成更好的代码。

10.提升循环的性能

  要提升循环的性能,减少多余的常量计算非常有用(比如,不随循环变化的计算)。

  不好的代码(在for()中包含不变的if())推荐的代码

for(i...)

{

  if(CONSTANT0)

  {

    DoWork0(i);//假设这里不改变CONSTANT0的值

  }

  else

  {

    DoWork1(i);//假设这里不改变CONSTANT0的值

  }

}

if(CONSTANT0)

{

  for(i...)

  {

    DoWork0(i);

  }

}

else

{

  for(i...)

  {

    DoWork1(i);

  }

}

  如果已经知道if()的值,这样可以避免重复计算。

虽然不好的代码中的分支可以简单地预测,但是由于推荐的代码在进入循环前分支已经确定,就可以减少对分支预测的依赖。

  把本地函数声明为静态的(static)

  如果一个函数在实现它的文件外未被使用的话,把它声明为静态的(static)以强制使用内部连接。

否则,默认的情况下会把函数定义为外部连接。

这样可能会影响某些编译器的优化——比如,自动内联。

11.考虑动态内存分配

  动态内存分配(C++中的";new";)可能总是为长的基本类型(四字对齐)返回一个已经对齐的指针。

但是如果不能保证对齐,使用以下代码来实现四字对齐。

这段代码假设指针可以映射到long型。

  例子

  double*p=(double*)newBYTE[sizeof(double)*number_of_doubles+7L];

double*np=(double*)((long(p)+7L)&;–8L);

  现在,你可以使用np代替p来访问数据。

注意:

释放储存空间时仍然应该用deletep。

12.使用显式的并行代码

  尽可能把长的有依赖的代码链分解成几个可以在流水线执行单元中并行执行的没有依赖的代码链。

因为浮点操作有很长的潜伏期,所以不管它被映射成x87或3DNow!

指令,这都很重要。

很多高级语言,包括C++,并不对产生的浮点表达式重新排序,因为那是一个相当复杂的过程。

需要注意的是,重排序的代码和原来的代码在代数上一致并不等价于计算结果一致,因为浮点操作缺乏精确度。

在一些情况下,这些优化可能导致意料之外的结果。

幸运的是,在大部分情况下,最后结果可能只有最不重要的位(即最低位)是错误的。

  不好的代码

doublea[100],sum;

inti;

sum=0.0f;

for(i=0;i<;100;i++)

  sum+=a;

推荐的代码

doublea[100],sum1,sum2,sum3,sum4,sum;

inti;

sum1=sum2=sum3=sum4=0.0;

for(i=0;i<;100;i+=4)

{

  sum1+=a;

  sum2+=a[i+1];

  sum3+=a[i+2];

  sum4+=a[i+3];

}

sum=(sum4+sum3)+(sum1+sum2);

  要注意的是:

使用4路分解是因为这样使用了4阶段流水线浮点加法,浮点加法的每一个阶段占用一个时钟周期,保证了最大的资源利用率。

13.提出公共子表达式

  在某些情况下,C++编译器不能从浮点表达式中提出公共的子表达式,因为这意味着相当于对表达式重新排序。

需要特别指出的是,编译器在提取公共子表达式前不能按照代数的等价关系重新安排表达式。

这时,程序员要手动地提出公共的子表达式(在VC.net里有一项“全局优化”选项可以完成此工作,但效果就不得而知了)。

推荐的代码

floata,b,c,d,e,f;

...

e=b*c/d;

f=b/d*a;

floata,b,c,d,e,f;

...

constfloatt(b/d);

e=c*t;

f=a*t;

推荐的代码

floata,b,c,e,f;

...

e=a/c;

f=b/c;

floata,b,c,e,f;

...

constfloatt(1.0f/c);

e=a*t;

f=b*t;

14.结构体成员的布局

  很多编译器有“使结构体字,双字或四字对齐”的选项。

但是,还是需要改善结构体成员的对齐,有些编译器可能分配给结构体成员空间的顺序与他们声明的不同。

但是,有些编译器并不提供这些功能,或者效果不好。

所以,要在付出最少代价的情况下实现最好的结构体和结构体成员对齐,建议采取这些方法:

  A按类型长度排序

  把结构体的成员按照它们的类型长度排序,声明成员时把长的类型放在短的前面。

  把结构体填充成最长类型长度的整倍数

  把结构体填充成最长类型长度的整倍数。

照这样,如果结构体的第一个成员对齐了,所有整个结构体自然也就对齐了。

下面的例子演示了如何对结构体成员进行重新排序:

  不好的代码,普通顺序推荐的代码,新的顺序并手动填充了几个字节

struct

{

  chara[5];

  longk;

  doublex;

}baz;

struct

{

  doublex;

  longk;

  chara[5];

charpad[7];

}baz;

  这个规则同样适用于类的成员的布局。

  B按数据类型的长度排序本地变量

  当编译器分配给本地变量空间时,它们的顺序和它们在源代码中声明的顺序一样,和上一条规则一样,应该把长的变量放在短的变量前面。

如果第一个变量对齐了,其它变量就会连续的存放,而且不用填充字节自然就会对齐。

有些编译器在分配变量时不会自动改变变量顺序,有些编译器不能产生4字节对齐的栈,所以4字节可能不对齐。

下面这个例子演示了本地变量声明的重新排序:

  不好的代码,普通顺序推荐的代码,改进的顺序

shortga,gu,gi;

longfoo,bar;

doublex,y,z[3];

chara,b;

floatbaz;

doublez[3];

doublex,y;

longfoo,bar;

floatbaz;

shortga,gu,gi;

14.避免不必要的整数除法

  整数除法是整数运算中最慢的,所以应该尽可能避免。

一种可能减少整数除法的地方是连除,这里除法可以由乘法代替。

这个替换的副作用是有可能在算乘积时会溢出,所以只能在一定范围的除法中使用。

  不好的代码推荐的代码

inti,j,k,m;

m=i/j/k;

inti,j,k,m;

m=i/(j*k);

15.把频繁使用的指针型参数拷贝到本地变量

  避免在函数中频繁使用指针型参数指向的值。

因为编译器不知道指针之间是否存在冲突,所以指针型参数往往不能被编译器优化。

这样是数据不能被存放在寄存器中,而且明显地占用了内存带宽。

注意,很多编译器有“假设不冲突”优化开关(在VC里必须手动添加编译器命令行/Oa或/Ow),这允许编译器假设两个不同的指针总是有不同的内容,这样就不用把指针型参数保存到本地变量。

否则,请在函数一开始把指针指向的数据保存到本地变量。

如果需要的话,在函数结束前拷贝回去。

  

不好的代码

//假设q!

=r

voidisqrt(unsignedlonga,unsignedlong*q,unsignedlong*r)

{

  *q=a;

  if(a>;0)

  {

    while(*q>;(*r=a/*q))

    {

      *q=(*q+*r)>;>;1;

    }

  }

  *r=a-*q**q;

}

推荐的代码

//假设q!

=r

voidisqrt(unsignedlonga,unsignedlong*q,unsignedlong*r)

{

  unsignedlongqq,rr;

  qq=a;

  if(a>;0)

  {

    while(qq>;(rr=a/qq))

    {

      qq=(qq+rr)>;>;1;

    }

  }

  rr=a-qq*qq;

  *q=qq;

  *r=rr;

}

16.赋值与初始化

先看看以下代码:

classCInt

{

  intm_i;

public:

  CInt(inta=0):

m_i(a){cout<;<;";CInt";<;<;endl;}

  ~CInt(){cout<;<;";~CInt";<;<;endl;}

  CIntoperator+(constCInt&;a){returnCInt(m_i+a.GetInt());}

  voidSetInt(constinti)  {m_i=i;}

  intGetInt()const      {returnm_i;}

};

展开阅读全文
相关资源
猜你喜欢
相关搜索
资源标签

当前位置:首页 > 表格模板 > 合同协议

copyright@ 2008-2022 冰豆网网站版权所有

经营许可证编号:鄂ICP备2022015515号-1