【原+转】STL之优先队列的cmp函数和sort以及qsort函数的cmp

sort以及qsort函数的cmp 转自 http://blog.csdn.net/lionel_d/article/details/41746135 写的很好,直接复制粘贴过来了,感谢


一、sort 以及 qsort

首先,我们来谈谈大名鼎鼎的void qsort(void *base,int nelem,int width,int (*fcmp)(const void *,const void *));

它属于C语言标准库函数,应该是运用最多的了,今天我不是来教你们怎么用qsort的,只是来交流一下排序函数的最后一个参数cmp()(它不仅可以叫cmp,你还可以给他取名叫什么pig啊dog的只要是英文单词都可以,cmp只是人们对compare的一种常用缩写)比较函数的写法。

下面是cmp的写法:

[cpp]  view plain  copy
 print ?
  1. int cmp(const void *a ,const void *b)  
  2. {  
  3.     return *(int *)a - *(int *)b ;          //从小到大排序,把a,b位置反过来就是从大到小   
  4. }  


注意:qsort的cmp()函数千万别写成下面这样

[cpp]  view plain  copy
 print ?
  1. //错错错错错错错 错错错错错错错 错错错错错错错 错错错错错错错   
  2. int cmp(const void *a ,const void *b)  
  3. {  
  4.     return *(int *)a > *(int *)b ;       // > 与 < 都不行 !      
  5. }  
  6. //XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX  


有一次就是写了下面的cmp(),结果排序死活不对!


下面是完整的测试代码:

[cpp]  view plain  copy
 print ?
  1. #include   
  2. #include   
  3.   
  4. int cmp(const void *a ,const void *b)  
  5. {  
  6.     return *(int *)a - *(int *)b ;      //从小到大排序,把a,b位置反过来就是从大到小   
  7. }  
  8. int main()  
  9. {  
  10.     int a[10]={-1,9,5,7,-11,2,6,8,9,6};  
  11.     qsort(a,10,sizeof(int),cmp);  
  12.     for(int i = 0 ; i < 10 ; ++i)  
  13.     {  
  14.         printf("%d ",a[i]);  
  15.     }  
  16.     printf("\n") ;  
  17.     return 0 ;  
  18. }  

测试结果:

【原+转】STL之优先队列的cmp函数和sort以及qsort函数的cmp_第1张图片





第二个cmp(),就是void sort( iterator start, iterator end, StrictWeakOrdering cmp );下面是标准声明:

  template 
  void sort (RandomAccessIterator first, RandomAccessIterator last, Compare comp);
他的头文件是,这个是标准C++头文件里的。

[cpp]  view plain  copy
 print ?
  1. bool cmp(int a ,int b)  
  2. {  
  3.     return a < b ;       //从小到大排序,把 < 换成 > 就是从大到小   
  4. }  

好的,问题又来了,这个cmp与qsort的cmp不一样了,正好相反,他不能写成下面这样:

[cpp]  view plain  copy
 print ?
  1. //错错错错错错错 错错错错错错错 错错错错错错错 错错错错错错错   
  2. bool cmp(int a ,int b)  
  3. {  
  4.     return a - b ;            
  5. }  
  6. //XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX  


切记,切记!

下面是sort的测试代码:

[cpp]  view plain  copy
 print ?
  1. #include   
  2. #include   
  3.   
  4. using namespace std ;  
  5.   
  6. bool cmp(int a ,int b)  
  7. {  
  8.     return a < b ;           //从小到大排序,把 < 换成 > 就是从大到小   
  9. }  
  10.   
  11. int main()  
  12. {  
  13.     int a[10]={-1,9,5,7,-11,2,6,8,9,6};  
  14.     sort(a,a+10,cmp);  
  15.     for(int i = 0 ; i < 10 ; ++i)  
  16.     {  
  17.         cout<" " ;   
  18.     }  
  19.     cout<
  20.     return 0 ;  
  21. }  

测试结果:

【原+转】STL之优先队列的cmp函数和sort以及qsort函数的cmp_第2张图片




在C++中,我们经常需要用到set,map等容器,他们的cmp基本写法都与sort的相同,当然set,map的cmp可不仅仅是函数了,而是函数对象:

[cpp]  view plain  copy
 print ?
  1. struct cmp{  
  2.     bool operator ()(const int a , const int b)  
  3.     {  
  4.         return a < b ;           // 从小到大,反过来就是从大到小   
  5.     }  
  6. };  

下面仅仅对set做代码测试:

[cpp]  view plain  copy
 print ?
  1. #include   
  2. #include   
  3. #include   
  4.   
  5. using namespace std ;  
  6.   
  7. struct Person{  
  8.     int age;  
  9.     char name[20];  
  10.     Person(int Age , const char Name[]):age(Age){strcpy(name,Name);}  
  11. };  
  12. struct cmp{  
  13.     bool operator ()(const Person a , const Person b)  
  14.     {  
  15.         return a.age < b.age ;           //  从小到大 ;   
  16.     }  
  17. };  
  18.   
  19. int main()  
  20. {  
  21.     set s ;  
  22.     Person n1(46,"ggg");  
  23.     Person n2(-16,"fff");           //年龄无负数,只是为了测试代码,下同   
  24.     Person n3(45,"eee");  
  25.     Person n4(-25,"ddd");  
  26.     Person n5(34,"ccc");  
  27.     Person n6(22,"bbb");  
  28.     Person n7(2,"aaa");  
  29.     s.insert(n1);  
  30.     s.insert(n2);  
  31.     s.insert(n3);  
  32.     s.insert(n4);  
  33.     s.insert(n5);  
  34.     s.insert(n6);  
  35.     s.insert(n7);  
  36.     set::iterator begin = s.begin();  
  37.     set::iterator end = s.end();  
  38.     for(set::iterator i = begin ; i != end ; ++i)  
  39.     {  
  40.         cout<age<<" "<name<
  41.     }  
  42.     return 0 ;  
  43. }  

测试结果:

【原+转】STL之优先队列的cmp函数和sort以及qsort函数的cmp_第3张图片



好的,废话也说了这么多了,那我们就来个总结吧:

在调用C的库函数qsort时,因为C语言没有明确的定义bool类型,只是笼统的说,零为假,任何非零都是真,而qsort的cmp函数是返回int的,通过<和>比较两个数据只能返回非零值(真)或零(假),具体返回多少,得看编译器,据我猜测qsort内部是根据返回的正或负或0来判断两个数之间的大小或等于关系的,这时用<或>就不能正常排序了。

而在C++中,已经定义了bool类型,而且sort的cmp函数返回的是bool类型的,说明sort的判断方式与qsort不同的,需要返回一个布尔值,来判断两个数之间的关系的。


二 、优先队列的cmp
1.在优先队列中存储常用类型时,利用自定义结构体cmp来实现自定义的<比较。
2.在优先队列中存储自定义数据类型时,只需要在自定义类型中重载小于号。


#include 
#include 
#include 
#include  ///greater 函数在里面。
using namespace std;

///默认优先级,从大到小,大的先出队
priority_queue  que1;


///使用系统比较函数greater 从小到大的出队
         ///数据类型 容器类型   比较函数
priority_queue ,greater > que2;///注意有> >之间有空格


///自定义优先级
struct cmp1{    ///不允许起名为 cmp ?
    bool operator ()(int &a,int &b){
        return a>b;///从小到大,最小值优先。
    }
};
priority_queue ,cmp1> que3;



///结构体的自定义优先级
struct number{
    int x;
    int y;
    bool operator < (const number &a)const{
        if(x == a.x) return y>a.y;///x相等的情况下按y从小到大,最小值优先
        else
            return x>a.x;///x从小到大,最小值优先
    }
};
priority_queue  que4;
int data1[5] = {88,88,56,62,99};
number data2[5]={{88,7},{88,6},{56,5},{62,3},{99,4}};

int main()
{
    for(int i=0;i<5;++i)
        que1.push(data1[i]);
    while(!que1.empty()){
        printf("%d ",que1.top());
        que1.pop();
    }
    printf("\n-----------------------------------------------------\n");

    for(int i=0;i<5;++i)
        que2.push(data1[i]);
    while(!que2.empty()){
        printf("%d ",que2.top());
        que2.pop();
    }
    printf("\n-----------------------------------------------------\n");

    for(int i=0;i<5;++i)
        que3.push(data1[i]);
    while(!que3.empty()){
        printf("%d ",que3.top());
        que3.pop();
    }
    printf("\n-----------------------------------------------------\n");

    for(int i=0;i<5;++i)
        que4.push(data2[i]);
    printf("x  y\n");
    while(!que4.empty()){
        printf("%d %d\n",que4.top().x,que4.top().y);
        que4.pop();
    }
    return 0;
}

结果:

99 88 88 62 56
-----------------------------------------------------
56 62 88 88 99
-----------------------------------------------------
56 62 88 88 99
-----------------------------------------------------
x  y
56 5
62 3
88 6
88 7
99 4

你可能感兴趣的:(【原+转】STL之优先队列的cmp函数和sort以及qsort函数的cmp)