P1781 宇宙总统
题目地址: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 宇宙总统的更多相关文章
- 洛谷 P1781 宇宙总统
P1781 宇宙总统 题目背景 宇宙总统竞选 题目描述 地球历公元6036年,全宇宙准备竞选一个最贤能的人当总统,共有n个非凡拔尖的人竞选总统,现在票数已经统计完毕,请你算出谁能够当上总统. 输入输出 ...
- (大数 string easy。。。)P1781 宇宙总统 洛谷
题目背景 宇宙总统竞选 题目描述 地球历公元6036年,全宇宙准备竞选一个最贤能的人当总统,共有n个非凡拔尖的人竞选总统,现在票数已经统计完毕,请你算出谁能够当上总统. 输入输出格式 输入格式: pr ...
- 洛谷——P1781 宇宙总统
https://www.luogu.org/problem/show?pid=1781 题目背景 宇宙总统竞选 题目描述 地球历公元6036年,全宇宙准备竞选一个最贤能的人当总统,共有n个非凡拔尖的人 ...
- (Java实现) 洛谷 P1781 宇宙总统
题目背景 宇宙总统竞选 题目描述 地球历公元6036年,全宇宙准备竞选一个最贤能的人当总统,共有n个非凡拔尖的人竞选总统,现在票数已经统计完毕,请你算出谁能够当上总统. 输入输出格式 输入格式: pr ...
- 洛谷 P1781 宇宙总统:sort(string)
题目描述 地球历公元6036年,全宇宙准备竞选一个最贤能的人当总统,共有n个非凡拔尖的人竞选总统,现在票数已经统计完毕,请你算出谁能够当上总统. 输入输出格式 输入格式: 第一行为一个整数n,代表竞选 ...
- 洛谷P1781 宇宙总统【排序+字符串】
地球历公元6036年,全宇宙准备竞选一个最贤能的人当总统,共有n个非凡拔尖的人竞选总统,现在票数已经统计完毕,请你算出谁能够当上总统. 输入输出格式 输入格式: president.in 第一行为一个 ...
- 洛谷P1781——宇宙总统(高精度排序)
题目描述 地球历公元6036年,全宇宙准备竞选一个最贤能的人当总统,共有n个非凡拔尖的人竟选总统,现在票数已经统计完毕,请你算出谁能够当上总统. 输入输出格式 输入格式: 第一行为一个整数n,代表竞选 ...
- 洛谷 P1059明明的随机数 & P1068分数线划定 & P1781宇宙总统
题目:https://www.luogu.org/problemnew/show/P1059 思路:STL中的set使用. //#include<bits/stdc++.h> #inclu ...
- 洛谷P1781宇宙总统题解
题目 此题原本是一个简单的排序,但因为数据范围的限制,所以变成了一个需采用字符串排序的题目,接下来我将给大家讲一下如何字符串排序. 首先先判断为位数上是否相同,如果不同再比较那一位的数就可以了 #in ...
随机推荐
- CSS实现背景图片透明和文字不透明效果
1.毛玻璃效果:背景图 + 伪类 + flite:blur(3px) width: 500px; height: 300px; line-height: 50px; text-align: cente ...
- 17 ~ express ~ 分类的显示 ,修改 和 删除
一,前台显示 /views/admin/category.html {% extends 'layout.html' %} {% block main %} <ol class="br ...
- 19 01 17 Django 模板 返回一个页面
模板 问题 如何向请求者返回一个漂亮的页面呢? 肯定需要用到html.css,如果想要更炫的效果还要加入js,问题来了,这么一堆字段串全都写到视图中,作为HttpResponse()的参数吗?这样定义 ...
- win10安装centOS 失去win启动项
联网:执行以下三条命令 yum -y install epel-release yum -y install ntfs-3g grub2-mkconfig -o /boot/grub2/grub.cf ...
- HZNU-ACM寒假集训Day5小结 线段树 树状数组
线段树 什么时候用线段树 1.统计量可合并 2.修改量可合并 3.通过统计量可直接修改统计量 一句话:满足区间加法即可使用线段树维护信息 理解Lazy Tage 蓝色是要把信息及时维护的节点,红色是本 ...
- git修改已经push的commit message
git中修改上一次提交的commit的message git commit --amend -m "你的新的注释" git push -f 多个commit https://www ...
- AD软件将PCB中的元器件旋转45度
- MySql 的操作指令(window)
1.登录: mysql -uroot -p 2.查看所有数据库: show databases 3.切换数据库 : use databasename(数据库名称) 4.查看数据库的所有表格 ...
- html—表单控件
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...
- java课程之团队开发冲刺阶段1.10
一.总结昨天任务: 1.已完成系统规划总结 二.遇到的问题: 1,整合的过程中并没有的很好的处理其中的关系,应当将常用的方法总结成一个工具类便于使用.这样在二次使用的时候会便捷不少. 三.今天的任务 ...