我的编程空间,编程开发者的网络收藏夹
学习永远不晚

你知道如何自定义sort函数中的比较函数

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

你知道如何自定义sort函数中的比较函数

如何自定义sort函数中的比较函数

在做剑指offer时,有一道题:

题目描述

输入一个正整数数组,把数组里所有数字拼接起来排成一个数,打印能拼接出的所有数字中最小的一个。例如输入数组{3,32,321},则打印出这三个数字能排成的最小数字为321323。

思路

自定义比较器,若a+b>b+a则a>b,即”3”+”23”>”23”+”3”则3>23,并且我们希望在排序的时候将23排在3的前面,也就是升序排列。


class Solution {
public:
    static bool compare(const string& s1, const string& s2)
    {
        string ab = s1 + s2;
        string ba = s2 + s1;
        return ab < ba; //升序排列。如改为ab > ba, 则为降序排列
    }
    string PrintMinNumber(vector<int> numbers) 
    {
        string result;
        if(numbers.size() <= 0) return result;
        vector<string> num2str;
        for(int i = 0; i < numbers.size(); i++)
        {
            stringstream ss;
            ss << numbers[i];
            string s = ss.str();
            num2str.push_back(s);
        }
        sort(num2str.begin(), num2str.end(), compare);
        for(int i = 0; i < num2str.size(); i++)
        {
            result.append(num2str[i]);
        }
        return result;
    }
};

这道题的解法中用到了自定义比较器。也就是自定义了campare函数。

但是为什么当ab<ba的时候就是升序排列了呢?十分不解,在网上搜了好多资料,最后看到了c++的技术文档,醍醐灌顶!!!强烈推荐直接看技术文档,比在网上查来查去明白的更快!

http://www.cplusplus.com/reference/algorithm/sort/

这里写图片描述


// sort algorithm example
#include <iostream>     // std::cout
#include <algorithm>    // std::sort
#include <vector>       // std::vector
bool myfunction (int i,int j) { return (i<j); }
struct myclass {
  bool operator() (int i,int j) { return (i<j);}
} myobject;
int main () {
  int myints[] = {32,71,12,45,26,80,53,33};
  std::vector<int> myvector (myints, myints+8);               // 32 71 12 45 26 80 53 33
  // using default comparison (operator <):
  std::sort (myvector.begin(), myvector.begin()+4);           //(12 32 45 71)26 80 53 33
  // using function as comp
  std::sort (myvector.begin()+4, myvector.end(), myfunction); // 12 32 45 71(26 33 53 80)
  // using object as comp
  std::sort (myvector.begin(), myvector.end(), myobject);     //(12 26 32 33 45 53 71 80)
  // print out content:
  std::cout << "myvector contains:";
  for (std::vector<int>::iterator it=myvector.begin(); it!=myvector.end(); ++it)
    std::cout << ' ' << *it;
  std::cout << '\n';
  return 0;
}

可以看到comp的定义:comp函数返回一个bool类型的值,这个值表示了在严格弱排序中(可以理解为升序排序)第一参数是否位于第二个参数之前。

也就是说如果comp返回true,则第一个参数小于第二个参数,sort根据compare的返回值将第一个参数排在第二个参数之前。

如果comp返回false,则第一个参数大于第二个参数,sort根据compare的返回值将第一个参数排在第二个参数之后。

  • 传入A,B,定义bool myfunction (int i,int j) { return (i<j); },作为comp函数,则排列AB。
  • 传入BA,则排列AB。

可以看出是升序排列的。(sort函数默认的comp函数也是默认升序的)

而如果我们定义bool myfunction (int i,int j) { return (i>j); },作为comp函数,

  • 传入AB,返回false,则排列为BA,
  • 传入BA,返回true,排列为BA。

是降序排列的。

回到最初的问题中


    static bool compare(const string& s1, const string& s2)
    {
        string ab = s1 + s2;
        string ba = s2 + s1;
        return ab < ba; //升序排列。如改为ab > ba, 则为降序排列
    }

我们可以看出 如果s1=”3”, s2=”23”

ab = “323”;

ba = “233”;

事实上ab>ba,所以comp会返回false,sort根据compare返回的false将s2排列在s1之前,也就是排列成”23”,”3”这也就是我们想要的结果。

总结起来就是

sort函数根据comp函数的返回值,对comp函数的两个参数排序。

如果comp返回true,排序为“参数1”“参数2”,否则排序为“参数2”“参数1”。

  • 想要升序排列,则return parameter1<parameter2
  • 想要降序排列,则return parameter1>parameter2

sort()基本用法

做ACM题的时候,排序是一种经常要用到的操作。如果每次都自己写个冒泡之类的O(n^2)排序,不但程序容易超时,而且浪费宝贵的比赛时间,还很有可能写错。STL里面有个sort函数,可以直接对数组排序,复杂度为n*log2(n)。使用这个函数,需要包含头文件。

这个函数可以传两个参数或三个参数。第一个参数是要排序的区间首地址,第二个参数是区间尾地址的下一地址。也就是说,排序的区间是[a,b)。简单来说,有一个数组int a[100],要对从a[0]到a[99]的元素进行排序,只要写sort(a,a+100)就行了,默认的排序方式是升序。

拿我出的“AC的策略”这题来说,需要对数组t的第0到len-1的元素排序,就写sort(t,t+len);

对向量v排序也差不多,sort(v.begin(),v.end());

排序的数据类型不局限于整数,只要是定义了小于运算的类型都可以,比如字符串类string。

如果是没有定义小于运算的数据类型,或者想改变排序的顺序,就要用到第三参数——比较函数。比较函数是一个自己定义的函数,返回值是bool型,它规定了什么样的关系才是“小于”。想把刚才的整数数组按降序排列,可以先定义一个比较函数cmp


bool cmp(int a,int b)
{
return a>b;
}

排序的时候就写sort(a,a+100,cmp);

假设自己定义了一个结构体node


struct node{
int a;
int b;
double c;
}

有一个node类型的数组node arr[100],想对它进行排序:先按a值升序排列,如果a值相同,再按b值降序排列,如果b还相同,就按c降序排列。就可以写这样一个比较函数:

以下是代码片段:


bool cmp(node x,node y)
{
if(x.a!=y.a) return x.a
if(x.b!=y.b) return x.b>y.b;
return return x.c>y.c;
} 

排序时写sort(arr,a+100,cmp);


qsort(s[0],n,sizeof(s[0]),cmp);
int cmp(const void *a,const void *b)
{
return *(int *)a-*(int *)b;
}

对int类型数组排序


int num[100];
Sample:
int cmp ( const void *a , const void *b )
{
return *(int *)a - *(int *)b;
}
qsort(num,100,sizeof(num[0]),cmp);

对char类型数组排序(同int类型)


char word[100];
Sample:
int cmp( const void *a , const void *b )
{
return *(char *)a - *(int *)b;
}
qsort(word,100,sizeof(word[0]),cmp);

对double类型数组排序(特别要注意)


double in[100];
int cmp( const void *a , const void *b )
{
return *(double *)a > *(double *)b ? 1 : -1;
}
qsort(in,100,sizeof(in[0]),cmp);

对结构体一级排序


struct In
{
double data;
int other;
}s[100]
//按照data的值从小到大将结构体排序,关于结构体内的排序关键数据data的类型可以很多种,参考上面的例子写
int cmp( const void *a ,const void *b)
{
return ((In *)a)->data - ((In *)b)->data ;
}
qsort(s,100,sizeof(s[0]),cmp);

对结构体


struct In
{
int x;
int y;
}s[100];
//按照x从小到大排序,当x相等时按照y从大到小排序
int cmp( const void *a , const void *b )
{
struct In *c = (In *)a;
struct In *d = (In *)b;
if(c->x != d->x) return c->x - d->x;
else return d->y - c->y;
}
qsort(s,100,sizeof(s[0]),cmp);

对字符串进行排序


struct In
{
int data;
char str[100];
}s[100];
//按照结构体中字符串str的字典顺序排序
int cmp ( const void *a , const void *b )
{
return strcmp( ((In *)a)->str , ((In *)b)->str );
}
qsort(s,100,sizeof(s[0]),cmp);

计算几何中求凸包的cmp


int cmp(const void *a,const void *b) //重点cmp函数,把除了1点外的所有点,旋转角度排序
{
struct point *c=(point *)a;
struct point *d=(point *)b;
if( calc(*c,*d,p[1]) < 0) return 1;
else if( !calc(*c,*d,p[1]) && dis(c->x,c->y,p[1].x,p[1].y) < dis(d->x,d->y,p[1].x,p[1].y)) //如果在一条直线上,则把远的放在前面
return 1;
else return -1;
}

以上为个人经验,希望能给大家一个参考,也希望大家多多支持编程网。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

你知道如何自定义sort函数中的比较函数

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Android自定义view 你所需要知道的基本函数总结

Android自定义view 你所需要知道的基本函数 首先 往Canvas上面draw需要一个Paint。 画笔常用的函数有哪些呢。由于木有调试环境,函数基本上默写,有错请评论提出,蟹蟹!Paint p = new Paint(); //
2022-06-06

Python中如何自定义函数

这篇文章主要介绍了Python中如何自定义函数,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
2023-01-04

如何在shell中自定义函数

这篇文章给大家介绍如何在shell中自定义函数,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。一,调用函数必须在定义函数的后,不然会报错的funfun (){ echo "aaaaaa"}fun返回结果如下:[root@
2023-06-09

linux中shell如何自定义函数

小编给大家分享一下linux中shell如何自定义函数,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!一、定义shell函数(define function)语法:
2023-06-09

如何在 PHP 中自定义函数参数

php 中自定义函数参数包括:参数类型提示:指定函数参数的预期类型,防止意外数据类型和运行时错误。默认值:为参数指定默认值,在未提供实际参数时使用。可选参数:可以使用方括号括起来定义,可以传递或不传递给函数,并且可以有默认值。如何在 PHP
如何在 PHP 中自定义函数参数
2024-04-26

Java如何使用用户自定义的比较函数对数组中的键名进行排序

Java中可使用Arrays.sort()方法对数组键名进行自定义排序。首先创建自定义比较函数,实现Comparator接口的compare()方法,指定键名大小比较逻辑。然后在Arrays.sort()中传入自定义比较函数作为参数。该方法将按指定逻辑对键名进行排序。示例代码展示了按忽略大小写顺序排序键名的过程。通过自定义比较函数,可根据特定需求对键名进行灵活排序。
Java如何使用用户自定义的比较函数对数组中的键名进行排序
2024-04-02

PHP如何使用用户自定义的比较函数对数组中的键名进行排序

PHP提供了ksort()和krsort()函数用于按键名排序,但默认按ASCII值排序。要按自定义逻辑排序,需使用回调函数作为比较函数,接受两个键名参数并返回整数。PHP提供uksort()和ukrsort()函数,使用自定义比较函数对键名排序。示例演示了如何按键名的长度排序:自定义比较函数返回键名长度差值。uksort()使用该函数对数组键名排序,输出按长度递增的数组。自定义比较函数应返回整数,并能接收其他参数,但需在调用uksort()或ukrsort()时指定。
PHP如何使用用户自定义的比较函数对数组中的键名进行排序
2024-04-02

python中如何调用自定义函数

要调用自定义函数,首先需要定义该函数,然后在需要调用该函数的地方使用函数名加上括号来调用它。例如:def my_function():print("Hello, world!")my_function() # 调用自定义函数在上面的例子
python中如何调用自定义函数
2024-03-14

PHP 中如何使用自定义比较函数对数组进行排序,并保留键名?

php 中使用自定义比较函数可以对数组进行排序,并保留键名。为了做到这一点,可以使用 usort() 函数,它需要一个数组和一个回调函数作为参数。回调函数接收两个数组元素,返回一个整数 (-1、0 或 1) 来指示排序顺序。PHP 中使用自
PHP 中如何使用自定义比较函数对数组进行排序,并保留键名?
2024-05-04

如何在golang中自定义实现函数?

在 go 中实现自定义函数,需要使用 func 关键字后跟函数名、参数列表和返回类型(可选)。通过调用函数名和提供适当参数即可调用自定义函数。自定义函数可用于各种任务,例如处理数据、格式化输出或创建可重用代码块。如何在 Go 中自定义实现函
如何在golang中自定义实现函数?
2024-04-28

如何在SQLite中使用自定义函数

在SQLite中使用自定义函数可以通过以下步骤实现:创建一个自定义函数:CREATE FUNCTION my_function(param1 TEXT, param2 TEXT) RETURNS TEXT ASBEGIN-- 在这里编写自
如何在SQLite中使用自定义函数
2024-03-14

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录