ACM___________________________

______________白白の屋
posts - 182, comments - 102, trackbacks - 0, articles - 0

fill memset for 小测试

Posted on 2010-09-02 10:47 MiYu 阅读(1873) 评论(9)  编辑 收藏 引用 所属分类: C/C++ACM_资料

MiYu原创, 转帖请注明 : 转载自 ______________白白の屋    

 

做ACM题目的时候 , 经常使用到  fillmemset  for 操作对 数据进行初始化操作, 在测试数据不大,而且数组范围也不大的情况下,

这几种操作的时间差距不是很明显.  但是!!!!   因为测试数据的数量有时候非常大!!因此对数据初始化 操作的 选择也变得非常重要. 

于是就对3种操作进行了一个小测试.......................... 测试如下:

 

 测试系统 及 配置:        

 测试方案 :     开了3亿的数组, 对每个函数调用三次

 

 测试结果 :

    fill  :                                  G++                                                                 C++

 

  

  memset  :                           G++                                                                C++

 

 

     for :                            G++                                                      C++

  

 

 从上面的 数据对比可以看到 ,  除去误差外, 2种编译器对数据的处理时间 基本是一致的, 对于第一次处理 , 都额外花费了500MS 左右的时间, 因为

 尽管一开始申请了3亿的静态数组空间, 但是系统并不会全部把它给你, 只是代表你有权使用它,  这就要感谢操作系统的 内存管理机制了.   所以第一次

处理都花费了额外的时间来分配内存.  这也是为什么ACM 中很多题, 明明开了1000000 或更大一点的数组  但是内存开销却没超过 1000KB 的原因.

现在我们看后面的数据, 可以看到 memset 的时间优势非常明显,  是 fill 和 for 的2.5倍左右 !!!.   但是 . 我不得不说明其局限性, 因为 memset 是

每字节赋值的, 所以一般情况下, 仅能用与对 0 或 -1 的赋值,   (  memset 在每个字节的末尾填充 设定值 ). 对于每个变量的其他确定性的赋值就显得

力不从心了.  这时候就要用到 fill 或 for 循环赋值了, 原来一直觉得 fill 会很慢, 所以就一直没用,  现在测试了一下才知道, 原来速度是基本一样的, 现在做题可以

偷下懒了,   

 

 

 

 忘记附代码了 ....

代码
/*
MiYu原创, 转帖请注明 : 转载自 ______________白白の屋
          
http://www.cnblog.com/MiYu
Author By : MiYu
Test      : 1
Program   : fill
*/

#include 
<iostream>
#include 
<algorithm>
#include 
<ctime>
using namespace std;  
const int M = 300000000;  
int a[M];
int main ()
{
    time_t beg 
= clock();
    fill ( a,a
+M,0 ); fill ( a,a+M,0xff ); fill ( a,a+M,0 );
    time_t end 
= clock();
    cout 
<< "fill operator Cost 1:";
    cout 
<< end - beg << " MS" << endl;
    beg 
= clock();
    fill ( a,a
+M,0 );fill ( a,a+M,0xff );fill ( a,a+M,0 );
    end 
= clock();
    cout 
<< "fill operator Cost 2:";
    cout 
<< end - beg << " MS" << endl;
    beg 
= clock();
    fill ( a,a
+M,0 );fill ( a,a+M,0xff );fill ( a,a+M,0 );
    end 
= clock();
    cout 
<< "fill operator Cost 3:";
    cout 
<< end - beg << " MS" << endl;
    beg 
= clock();
    fill ( a,a
+M,0 );fill ( a,a+M,0xff );fill ( a,a+M,0 );
    end 
= clock();
    cout 
<< "fill operator Cost 4:";
    cout 
<< end - beg << " MS" << endl;
    beg 
= clock();
    fill ( a,a
+M,0 );fill ( a,a+M,0xff );fill ( a,a+M,0 );
    end 
= clock();
    cout 
<< "fill operator Cost 5:";
    cout 
<< end - beg << " MS" << endl;
    beg 
= clock();
    fill ( a,a
+M,0 );fill ( a,a+M,0xff );fill ( a,a+M,0 );
    end 
= clock();
    cout 
<< "fill operator Cost 6:";
    cout 
<< end - beg << " MS" << endl;
    beg 
= clock();
    fill ( a,a
+M,0 );fill ( a,a+M,0xff );fill ( a,a+M,0 );
    end 
= clock();
    cout 
<< "fill operator Cost 7:";
    cout 
<< end - beg << " MS" << endl;
    system ( 
"pause" ); 
    
return 0;
}

 代码

/*
MiYu原创, 转帖请注明 : 转载自 ______________白白の屋
          
http://www.cnblog.com/MiYu
Author By : MiYu
Test      : 1
Program   : memset
*/

#include 
<iostream>
#include 
<algorithm>
#include 
<ctime>
using namespace std;  
const int M = 300000000;  
int a[M];
int main ()
{
    time_t beg,end;
    beg 
= clock();
    memset ( a, 
0sizeof (a) );memset ( a, 0sizeof (a) );memset ( a, 0sizeof (a) );
    end 
= clock();
    cout 
<< "memset operator Cost 1: ";
    cout 
<< end - beg << " MS" << endl;
    beg 
= clock();
    memset ( a, 
0sizeof (a) );memset ( a, 0sizeof (a) );memset ( a, 0sizeof (a) );
    end 
= clock();
    cout 
<< "memset operator Cost 2: ";
    cout 
<< end - beg << " MS" << endl;
    beg 
= clock();
    memset ( a, 
0sizeof (a) );memset ( a, 0sizeof (a) );memset ( a, 0sizeof (a) );
    end 
= clock();
    cout 
<< "memset operator Cost 3: ";
    cout 
<< end - beg << " MS" << endl;
    beg 
= clock();
    memset ( a, 
0sizeof (a) );memset ( a, 0sizeof (a) );memset ( a, 0sizeof (a) );
    end 
= clock();
    cout 
<< "memset operator Cost 4: ";
    cout 
<< end - beg << " MS" << endl;
    beg 
= clock();
    memset ( a, 
0sizeof (a) );memset ( a, 0sizeof (a) );memset ( a, 0sizeof (a) );
    end 
= clock();
    cout 
<< "memset operator Cost 5: ";
    cout 
<< end - beg << " MS" << endl;
    beg 
= clock();
    memset ( a, 
0sizeof (a) );memset ( a, 0sizeof (a) );memset ( a, 0sizeof (a) );
    end 
= clock();
    cout 
<< "memset operator Cost 6: ";
    cout 
<< end - beg << " MS" << endl;
    beg 
= clock();
    memset ( a, 
0sizeof (a) );memset ( a, 0sizeof (a) );memset ( a, 0sizeof (a) );
    end 
= clock();
    cout 
<< "memset operator Cost 7: ";
    cout 
<< end - beg << " MS" << endl;

    system ( 
"pause" ); 
    
return 0;
}

 

代码
/*
MiYu原创, 转帖请注明 : 转载自 ______________白白の屋
          
http://www.cnblog.com/MiYu
Author By : MiYu
Test      : 1
Program   : for
*/

#include 
<iostream>
#include 
<algorithm>
#include 
<ctime>
using namespace std;  
const int M = 300000000;  
int a[M];
int main ()
{
    time_t beg,end; 
int i;
    beg 
= clock();
    
for ( i = 0; i != M; ++ i ) a[i] = 0;
    
for ( i = 0; i != M; ++ i ) a[i] = 0;
    
for ( i = 0; i != M; ++ i ) a[i] = 0;
    end 
= clock();
    cout 
<< "for operator Cost 1: ";
    cout 
<< end - beg << " MS" << endl;
    beg 
= clock();
    
for ( i = 0; i != M; ++ i ) a[i] = 0;
    
for ( i = 0; i != M; ++ i ) a[i] = 0;
    
for ( i = 0; i != M; ++ i ) a[i] = 0;
    end 
= clock();
    cout 
<< "for operator Cost 2: ";
    cout 
<< end - beg << " MS" << endl;
    beg 
= clock();
    
for ( i = 0; i != M; ++ i ) a[i] = 0;
    
for ( i = 0; i != M; ++ i ) a[i] = 0;
    
for ( i = 0; i != M; ++ i ) a[i] = 0;
    end 
= clock();
    cout 
<< "for operator Cost 3: ";
    cout 
<< end - beg << " MS" << endl;
    beg 
= clock();
    
for ( i = 0; i != M; ++ i ) a[i] = 0;
    
for ( i = 0; i != M; ++ i ) a[i] = 0;
    
for ( i = 0; i != M; ++ i ) a[i] = 0;
    end 
= clock();
    cout 
<< "for operator Cost 4: ";
    cout 
<< end - beg << " MS" << endl;
    beg 
= clock();
    
for ( i = 0; i != M; ++ i ) a[i] = 0;
    
for ( i = 0; i != M; ++ i ) a[i] = 0;
    
for ( i = 0; i != M; ++ i ) a[i] = 0;
    end 
= clock();
    cout 
<< "for operator Cost 5: ";
    cout 
<< end - beg << " MS" << endl;
    beg 
= clock();
    
for ( i = 0; i != M; ++ i ) a[i] = 0;
    
for ( i = 0; i != M; ++ i ) a[i] = 0;
    
for ( i = 0; i != M; ++ i ) a[i] = 0;
    end 
= clock();
    cout 
<< "for operator Cost 6: ";
    cout 
<< end - beg << " MS" << endl;
    beg 
= clock();
    
for ( i = 0; i != M; ++ i ) a[i] = 0;
    
for ( i = 0; i != M; ++ i ) a[i] = 0;
    
for ( i = 0; i != M; ++ i ) a[i] = 0;
    end 
= clock();
    cout 
<< "for operator Cost 7: ";
    cout 
<< end - beg << " MS" << endl;
    system ( 
"pause" ); 
    
return 0;
}

 

Feedback

# re: fill memset for 小测试  回复  更多评论   

2010-09-02 10:48 by 大渊献
还插表情,够辛苦的。

# re: fill memset for 小测试  回复  更多评论   

2010-09-02 17:21 by 糨糊
我认为单纯的从代码层次看或者比较还不是很清楚的说明什么。
也许编译器在处理代码的时候都自己做了很多优化工作,最终殊途同归了?
但不保证每个编译器都是这样的。

反编译到汇编层次分析,才更说服力。

# re: fill memset for 小测试  回复  更多评论   

2010-09-02 23:19 by chaogu
好像clock理解错了吧。
end - begin 应该说的的clock tick而不是MS,要MS
应该还要除以CLK_PER_SEC(还是CLOCK_PER_SEC,忘了自己查查)。

# re: fill memset for 小测试  回复  更多评论   

2010-09-03 10:18 by MiYu
@大渊献
呃, 没啥, 做题累了 ........

# re: fill memset for 小测试  回复  更多评论   

2010-09-03 10:19 by MiYu
@糨糊
表示汇编还没学过.... T.T 路还好长啊....

# re: fill memset for 小测试  回复  更多评论   

2010-09-03 10:23 by MiYu
@chaogu
因为 用的编译器 的CLK_TCK宏, 本身就已经是1000, 所以没有除了, 因为除的话得到的就是 S 了

# re: fill memset for 小测试  回复  更多评论   

2010-09-03 23:48 by flyinghearts

可以直接用 MSVC的 __stosd
VC CRT的memset实现就是使用指令: rep stosd
在一些CPU上该指令要比直接用 mov + 循环控制 要快(对现在主流的CPU这也许是是最快的内存清0方法,如果不使用SSE指令的话)。


另外,for循环好像也可以优化,使用 rep stosd指令。

# re: fill memset for 小测试  回复  更多评论   

2010-09-10 22:30 by MiYu
受教了 ``

只有注册用户登录后才能发表评论。
网站导航: 博客园   IT新闻   BlogJava   博问   Chat2DB   管理