题目地址:https://www.luogu.com.cn/problem/P1781

题目描述:地球历公元 6036 年,全宇宙准备竞选一个最贤能的人当总统,共有 n 个非凡拔尖的人竞选总统,现在票数已经统计完毕,请你算出谁能够当上总统。

输入格式:第一行为一个整数 n,代表竞选总统的人数。

接下来有 n 行,分别为第一个候选人到第 n 个候选人的票数。第一个人的票数是1,往后依次叠加。

输出格式:共两行,第一行是一个整数 m,为当上总统的人的号数。第二行是当上总统的人的选票。

提示:票数可能会很大,可能会到100位数字。全宇宙有这么多人?宇宙总管啊,你确实要实行计划生育的基本策略了,你看人这么多,搞得我还要用高精做这道题!,1<=n<=20。

这是一道排序和高精同时使用的题目。一般这种需要对多种元素进行排序的题目(比如说本题的m(号数)和n(票数)),可以考虑用结构体做。于是,我创建了一个表示候选人信息的结构体:

struct data
{
char number[];//票数
int rank;//号
void operator=(data& a)//表示data对于等于号的运算法则
{
strcpy(this->number,a.number);//this表示调用此函数的对象本身,是一个指针
this->rank=a.rank;
}
}a[];

其中operator=是最需要讲一讲的。

operator是一种特殊的函数,这种函数的命名是operator后跟一个运算符(中间没有空格)。它只能在一个结构体(或面向对象编程里的“类”)中定义。这种函数是为了明确结构体中运算符的运算法则的。一般的加减乘除等各种运算符,只适用于int等数字类型。如果让一个结构体(如本题的data)使用运算符:

int a,b=;
a=b;//b的值被赋给了a
data c,d;
c=d;//计算机:什么意思?该怎么执行赋值?你告诉我呀!

很明显编译器并不知道怎么执行赋值操作,会报错。

那么,只能让我们告诉编译器了!

上述operator=(data& a)函数表示等号的左右边都是data类型时的操作。还需要介绍一下this指针。它表示使用这个函数的对象本身。举例说明:

struct my_str
{
int num;
void output()
{
cout<<this->num;
}
void compare(my_str other)
{
if(this->num<other.num) cout<<"Smaller";
else if(this->num==other.num) cout<<"Same";
else cout<<"Bigger";
}
};
int main()
{
my_str a,b;
a.num=;
b.num=;
a.output();//a是操纵函数的结构体对象本身,此时output里的this就是a。输出:1
b.compare(a);//此时this是b。由于2>1,所以输出:Bigger。
}

this看起来好像没什么卵用,只是起到让代码意思更清晰而已。但是随着编程的深入,this指针将会很有用。

operator函数可以定义各种类型的运算符,不仅是一般的加减乘除和赋值,还有关系运算符,逻辑运算符,特殊运算符等等(具有极重要意义的运算符除外,比如说作用域解析运算符::和宏符号#)。值得一提的是,由于等号在一般情况下不返回值,所以它被声明为void函数。同理,逻辑运算符因为要返回真或假值以供if或while使用,所以operator!,operator&&和operator||应声明为返回bool。参数也要注意,关系运算符、赋值运算符、&&和||在一般情况下有两个操作数,所以它们的operator函数应该有且只有一个参数表示右操作数(左操作数一般就是调用函数的对象本身,不用再声明)。

回到原题。我的想法是用选择排序,因此需要比较。然而高精度数字本质上是字符串,字符串无法直接用大于小于等于来比较,所以我写了一个比较高精数的函数:

#define LEFT_IS_BIGGER false//为了便于理解
#define RIGHT_IS_BIGGER true
bool WHO_IS_BIGGER(char* a,char* b)
{
int lena=strlen(a);
int lenb=strlen(b);
if(lena!=lenb)//位数不一样,直接确定谁大谁小
{
if(lena>lenb) return LEFT_IS_BIGGER;
else return RIGHT_IS_BIGGER;
}
else
{
for(int i=;i<lena;i++)
{
if(a[i]/*-48*/>b[i]/*-48*/) return LEFT_IS_BIGGER;//从高位向低位,若遇到不同的数字就可以直接判断大小
else if(a[i]<b[i]) return RIGHT_IS_BIGGER;
else continue;
}
}
}

为什么a[i]后面有个注释的-48?他的目的是将字符的0123456789转换为数字的0123456789(0的ASCII码是48)。但其实没必要这样做。可以想到,两个字符的数字比大小和两个转换后的数字比大小返回的结果是一样的。于是我把它注释掉了。

对了,选择排序还需要交换(swap函数),但它只适用于int,所以我写了一个交换data对象的函数:

void my_swap(data& a,data& b)
{
data temp=a;
a=b;
b=temp;
}

和int的swap一样,只是int全部换成了data而已。这里operator=就派上用场了。这个函数里的等号都调用了operator=函数,非常自然,没有一点编译器错误,我们不需要再额外定义一个对于data的赋值函数,用现成的等号就行,简明易懂。这就是operator函数的好处。

最后就是主函数了。道理和选择排序的原理一样,若不理解选择排序,可以查阅相关百科或书籍。

int main()
{
int n;
cin>>n;
for(int i=;i<n;i++)
{
scanf("%s",a[i].number);
a[i].rank=i+;
}
for(int i=;i<n;i++)
{
for(int j=i+;j<n;j++)
{
if(WHO_IS_BIGGER(a[i].number,a[j].number)==RIGHT_IS_BIGGER)
{
my_swap(a[i],a[j]);
}
}
}
cout<<a[].rank<<endl;
printf("%s",a[].number);
}

P1781 宇宙总统的更多相关文章

  1. 洛谷 P1781 宇宙总统

    P1781 宇宙总统 题目背景 宇宙总统竞选 题目描述 地球历公元6036年,全宇宙准备竞选一个最贤能的人当总统,共有n个非凡拔尖的人竞选总统,现在票数已经统计完毕,请你算出谁能够当上总统. 输入输出 ...

  2. (大数 string easy。。。)P1781 宇宙总统 洛谷

    题目背景 宇宙总统竞选 题目描述 地球历公元6036年,全宇宙准备竞选一个最贤能的人当总统,共有n个非凡拔尖的人竞选总统,现在票数已经统计完毕,请你算出谁能够当上总统. 输入输出格式 输入格式: pr ...

  3. 洛谷——P1781 宇宙总统

    https://www.luogu.org/problem/show?pid=1781 题目背景 宇宙总统竞选 题目描述 地球历公元6036年,全宇宙准备竞选一个最贤能的人当总统,共有n个非凡拔尖的人 ...

  4. (Java实现) 洛谷 P1781 宇宙总统

    题目背景 宇宙总统竞选 题目描述 地球历公元6036年,全宇宙准备竞选一个最贤能的人当总统,共有n个非凡拔尖的人竞选总统,现在票数已经统计完毕,请你算出谁能够当上总统. 输入输出格式 输入格式: pr ...

  5. 洛谷 P1781 宇宙总统:sort(string)

    题目描述 地球历公元6036年,全宇宙准备竞选一个最贤能的人当总统,共有n个非凡拔尖的人竞选总统,现在票数已经统计完毕,请你算出谁能够当上总统. 输入输出格式 输入格式: 第一行为一个整数n,代表竞选 ...

  6. 洛谷P1781 宇宙总统【排序+字符串】

    地球历公元6036年,全宇宙准备竞选一个最贤能的人当总统,共有n个非凡拔尖的人竞选总统,现在票数已经统计完毕,请你算出谁能够当上总统. 输入输出格式 输入格式: president.in 第一行为一个 ...

  7. 洛谷P1781——宇宙总统(高精度排序)

    题目描述 地球历公元6036年,全宇宙准备竞选一个最贤能的人当总统,共有n个非凡拔尖的人竟选总统,现在票数已经统计完毕,请你算出谁能够当上总统. 输入输出格式 输入格式: 第一行为一个整数n,代表竞选 ...

  8. 洛谷 P1059明明的随机数 & P1068分数线划定 & P1781宇宙总统

    题目:https://www.luogu.org/problemnew/show/P1059 思路:STL中的set使用. //#include<bits/stdc++.h> #inclu ...

  9. 洛谷P1781宇宙总统题解

    题目 此题原本是一个简单的排序,但因为数据范围的限制,所以变成了一个需采用字符串排序的题目,接下来我将给大家讲一下如何字符串排序. 首先先判断为位数上是否相同,如果不同再比较那一位的数就可以了 #in ...

随机推荐

  1. archlinux下安装mysql

    mysql的安装 这里安装的是mariadb一个mysql的开源版本,实际使用体验没有差别 1. 安装Maria DB sudo pacman -S mariadb 2. 配置目录 sudo mari ...

  2. UVA - 11925 Generating Permutations(生成排列)(构造)

    题意:将序列1,2,3,……,n,用不超过2n^2次操作,通过下列操作变成给定序列.(1<=n<=300) 1.交换前两个元素 2.将第一个元素移到最后 分析:因为将序列变成升序更容易操作 ...

  3. 在MFC做DLL动态链接库时,使用boost,出现断言错误

    建立的MFC DLL工程中有使用boost::thread,就会发生compile正常但是一程式执行或者直接编辑就出現ASSERT错误. 错误位置:dllinit.cpp,Line: 587,ASSE ...

  4. svg教程

    实例 <html> <body> <h1>My first SVG</h1> <svg style="border: 1px solid ...

  5. Hibernate--(二)增删改查

    1.增删改查: public class Test { public static void main(String[] args) { SessionFactory sf = new Configu ...

  6. 下载jQuery

    下载jQuery :https://jquery.com/download/ . 将下载好的文件放到项目中 引入到代码中 <script type="text/javascript&q ...

  7. Vue.js(19)之 封装calendar组件

    效果 需求 1.实现一个日历组件,如图: 2.显示某天的事项: 3.事项是模拟父组件请求接口返回的,数据格式如下: [ { id: '232', date: '2019-06-01', info: ' ...

  8. Docker部署NETCORE应用程序

    Docker部署NETCORE应用程序 创建.netcore项目,本文以ASP.NET Core2.2 版本为例 设置Dockerfile属性始终复制 编辑Dockerfile内容 FROM micr ...

  9. 吴裕雄--天生自然Django框架开发笔记:Django Nginx+uwsgi 安装配置

    Django Nginx+uwsgi 安装配置 使用 python manage.py runserver 来运行服务器.这只适用测试环境中使用. 正式发布的服务,需要一个可以稳定而持续的服务器,比如 ...

  10. SAP_FICO常用事务代码

    1.会计科目维护: FS00:总账科目主记录维护 FSP0:科目表中总账科目主记录维护 FSS0:公司代码中总账科目主记录维护 2.会计凭证创建 FB01:创建凭证 F-02:总账凭证创建(在FB01 ...