C++中cmp()用法

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

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

下面是cmp的写法:

int cmp(const void *a ,const void *b)  
{  
    return *(int *)a - *(int *)b ;          //从小到大排序,把a,b位置反过来就是从大到小   
}  

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

//错错错错错错错 错错错错错错错 错错错错错错错 错错错错错错错   
int cmp(const void *a ,const void *b)  
{  
    return *(int *)a > *(int *)b ;       // > 与 < 都不行 !      
}  

下面是完整的测试代码:

#include <stdio.h>  
#include <stdlib.h>  
  
int cmp(const void *a ,const void *b)  
{  
    return *(int *)a - *(int *)b ;      //从小到大排序,把a,b位置反过来就是从大到小   
}  
int main()  
{  
    int a[10]={-1,9,5,7,-11,2,6,8,9,6};  
    qsort(a,10,sizeof(int),cmp);  
    for(int i = 0 ; i < 10 ; ++i)  
    {  
        printf("%d ",a[i]);  
    }  
    printf("\n") ;  
    return 0 ;  
}  

 

 

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

  template <class RandomAccessIterator, class Compare>
  void sort (RandomAccessIterator first, RandomAccessIterator last, Compare comp);
他的头文件是<algorithm>,这个是标准C++头文件里的。
bool cmp(int a ,int b)  
{  
    return a < b ;       //从小到大排序,把 < 换成 > 就是从大到小   
}  

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


//错错错错错错错 错错错错错错错 错错错错错错错 错错错错错错错   
bool cmp(int a ,int b)  
{  
    return a - b ;            
}  

切记,切记!

下面是sort的测试代码:


#include <iostream>  
#include <algorithm>  
  
using namespace std ;  
  
bool cmp(int a ,int b)  
{  
    return a < b ;           //从小到大排序,把 < 换成 > 就是从大到小   
}  
  
int main()  
{  
    int a[10]={-1,9,5,7,-11,2,6,8,9,6};  
    sort(a,a+10,cmp);  
    for(int i = 0 ; i < 10 ; ++i)  
    {  
        cout<<a[i]<<" " ;   
    }  
    cout<<endl ;   
    return 0 ;  
}  

 

 

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

struct cmp{  
    bool operator ()(const int a , const int b)  
    {  
        return a < b ;           // 从小到大,反过来就是从大到小   
    }  
};  

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

#include <iostream>  
#include <cstring>  
#include <set>  
  
using namespace std ;  
  
struct Person{  
    int age;  
    char name[20];  
    Person(int Age , const char Name[]):age(Age){strcpy(name,Name);}  
};  
struct cmp{  
    bool operator ()(const Person a , const Person b)  
    {  
        return a.age < b.age ;           //  从小到大 ;   
    }  
};  
  
int main()  
{  
    set<Person,cmp> s ;  
    Person n1(46,"ggg");  
    Person n2(-16,"fff");           //年龄无负数,只是为了测试代码,下同   
    Person n3(45,"eee");  
    Person n4(-25,"ddd");  
    Person n5(34,"ccc");  
    Person n6(22,"bbb");  
    Person n7(2,"aaa");  
    s.insert(n1);  
    s.insert(n2);  
    s.insert(n3);  
    s.insert(n4);  
    s.insert(n5);  
    s.insert(n6);  
    s.insert(n7);  
    set<Person,cmp>::iterator begin = s.begin();  
    set<Person,cmp>::iterator end = s.end();  
    for(set<Person,cmp>::iterator i = begin ; i != end ; ++i)  
    {  
        cout<<i->age<<" "<<i->name<<endl ;  
    }  
    return 0 ;  
}  

 

 

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

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

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

所以在C++中应该使用sort中t的cmp函数写法;

这篇文章纯属交流,有错请望指出!

<think>好的,我需要帮助用户了解Python中的cmp函数及其替代方案。首先,我得先回忆一下Python 2和3之间的区别,因为用户提到的问题主要出现在这里。 用户提供的引用资料显示,Python 3中移除了cmp函数,这导致了一些兼容性问题。比如在排序时,使用cmp参数会报错,因为Python 3的sort()和sorted()不再支持这个参数。用户需要知道如何在Python 3中实现类似的功能。 根据引用[2],解决方案是使用functools.cmp_to_key函数,将比较函数转换成键函数。同时,用户可能需要自己编写比较函数,比如例子中的cmp_for_py3。这个函数返回a和b比较后的布尔值之差,模拟了原来的cmp行为。 引用[4]提到可以用operator模块中的函数来替代部分比较操作,比如eq、lt等。这可能适用于简单的比较,但对于复杂的比较逻辑,还是需要自定义函数。 接下来,我需要整理这些信息,分步骤说明Python 2中的cmp用法,以及如何在Python 3中替代。可能需要给出具体的代码示例,比如使用cmp_to_key的示例,以及如何修改旧代码。 同时,用户可能对为什么Python 3要移除cmp函数感兴趣,可能涉及到性能问题,因为键函数比比较函数更高效。这也是一个需要解释的点。 另外,引用[3]提到cmp在字典比较中的应用,但Python 3中可能已经完全改变了这一机制,需要指出现在字典比较的方式是否有变化,或者如何替代。 最后,用户可能需要知道替代方案的具体应用场景,比如在排序复杂对象时如何使用键函数或自定义比较器。 需要确保回答结构清晰,先说明cmp的作用,然后指出Python 3的变化,接着给出替代方法,并提供代码示例。同时,注意使用正确的LaTeX格式,比如行内公式用$...$,独立公式用$$...$$,但在这个问题中可能数学公式较少,但如果有的话需要正确应用。 另外,用户要求生成相关问题,需要根据内容提出相关问题,比如Python 3中排序函数的优化、性能差异等。 检查引用是否正确,引用[1]似乎和cmp函数无关,可能是用户提供的错误引用,但根据用户给出的引用内容,引用[1]提到的是Python3中cmp函数的替代方法,所以可能需要调整引用标记的位置,确保准确对应。 需要确保回答中提到的每个替代方案都有对应的引用来源,例如使用operator模块对应引用[4],使用cmp_to_key对应引用[2]。 总结步骤: 1. 解释Python 2中的cmp函数及其用途。 2. 说明Python 3中移除了cmp函数及其影响。 3. 提供替代方案一:使用functools.cmp_to_key。 4. 提供替代方案二:自定义比较函数。 5. 提到operator模块在简单比较中的使用。 6. 强调键函数在性能上的优势。 7. 给出实际代码示例。 8. 生成相关问题。</think>### Python cmp函数使用说明及替代方案 #### 1. Python 2中的`cmp`函数 `cmp(x, y)`函数用于比较两个对象的大小,返回: - `1`(若`x > y`) - `0`(若`x == y`) - `-1`(若`x < y`) **常见应用场景**: - 排序时自定义比较逻辑:例如对列表排序时通过`sort(cmp=my_cmp)`指定规则[^3]。 - 字典、元组等复杂结构的比较。 #### 2. Python 3中`cmp`的移除 Python 3移除了`cmp`函数及`sort()`的`cmp`参数,原因: - **性能优化**:`key`参数生成排序键值效率更高(时间复杂度$O(n \log n)$),而`cmp`需要多次比较(复杂度$O(n^2)$)[^2]。 - 简化语言设计:提倡更明确的比较逻辑。 #### 3. Python 3中的替代方案 ##### 方案一:使用`functools.cmp_to_key` 将旧版`cmp`函数转换为`key`函数: ```python from functools import cmp_to_key def custom_cmp(a, b): return (a > b) - (a < b) # 模拟cmp的返回值 data = [3, 1, 4, 1, 5] sorted_data = sorted(data, key=cmp_to_key(custom_cmp)) print(sorted_data) # 输出:[1, 1, 3, 4, 5] ``` ##### 方案二:自定义比较函数 通过布尔运算模拟`cmp`逻辑: ```python def cmp_for_py3(a, b): return bool(a > b) - bool(a < b) # 返回1/0/-1 ``` ##### 方案三:利用`operator`模块 对简单比较(如相等、大小),可直接使用`operator`模块: ```python import operator operator.eq(x, y) # 等价于 x == y operator.lt(x, y) # 等价于 x < y ``` #### 4. 性能优化建议 - **优先使用`key`参数**:例如对复杂对象按属性排序: ```python sorted(students, key=lambda x: x.age) ``` - 避免复杂`cmp`逻辑:多次比较会增加时间复杂度。 #### 示例:排序自定义对象 ```python from functools import cmp_to_key class Person: def __init__(self, name, age): self.name = name self.age = age # 按年龄升序,若年龄相同则按姓名降序 def person_cmp(a, b): if a.age != b.age: return a.age - b.age else: return -1 if a.name > b.name else 1 people = [Person("Bob", 30), Person("Alice", 25), Person("Charlie", 25)] sorted_people = sorted(people, key=cmp_to_key(person_cmp)) ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值