湘大 XTU OJ 1308 比赛 题解:循环结束的临界点+朴素模拟

一、链接

比赛

二、题目

题目描述

有n个人要进行比赛,比赛规则如下:

  1. 假设每轮比赛的人是m,取最大的k,k=2^t且k≤m。这k个人每2人举行一场比赛,胜利者进入一下轮,失败者被淘汰。
  2. 余下的m-k个人,不进行比赛,直接进入下一轮
  3. 直到决出冠军,比赛结束。

比如有5个人参加比赛,第一轮举办2场,剩余3人进入第二轮,第二轮1场,剩余2人进入第三轮,第三轮举办1场决出冠军,所以一共要办4场比赛。 请问一共要举行几轮多少场比赛?

输入

第一行是一个整数K,表示样例的个数。 以后每行一个样例,为n(1≤n≤1000000000)

输出

每行输出两个整数,轮数和比赛场数,中间用一个空格隔开。

样例输入

复制代码
2
1
5

样例输出

复制代码
0 0
3 4

三、题意

有n个人进行比赛,最后只剩下一个人,输出比赛轮数和比赛场数,需要找到最大的小于总人数n的2的指数函数的值k

四、代码

c++

cpp 复制代码
#include<iostream>

using namespace std;

int main()
{
	int t;//表示样例数
	scanf("%d",&t);
	
	while(t--)
	{
		int n,k=1,a=0,b=0;//n表示总人数,k表示最接近n的2的指数函数的值
		//a表示轮数,b表示场数
		scanf("%d",&n);
		
		if(n<2)	printf("0 0\n");//只剩下一个人,就不需要比较,特判
		else
		{
			while(n>1)//只要不是剩下一个人,就一直循环
			{
				while(k<n)	k*=2;//寻找最接近n的2的指数函数的值
				
				if(k!=n)	k/=2;//跳出上面循环会多乘一次,所以除掉一个2
				//当然,k==n不算多乘了一次,条件判断if(k>n)也是可以的
				
				a++;//每一次算一轮
				b+=k/2;//比赛进行k/2场
				n-=k/2;//淘汰k/2个人
			}
			
			printf("%d %d\n",a,b);
		}
	}
	
	return 0;
}

c语言

cpp 复制代码
#include<stdio.h>

int main()
{
	int t;
	scanf("%d",&t);
	
	while(t--)
	{
		int n,k=1,a=0,b=0;
		scanf("%d",&n);
		
		if(n<2)	printf("0 0\n");
		else
		{
			while(n>1)
			{
				while(k<n)	k*=2;
				if(k>n)	k/=2;
				
				a++;
				b+=k/2;
				n-=k/2;
			}
			
			printf("%d %d\n",a,b);
		}
	}
	
	return 0;
}

五、总结

1.怎么找到最大的小于总人数n的2的指数函数数值k:使用一个循环,让k从1开始循环,每一次循环把k乘以2,一直到k大于n的时候跳出循环

cpp 复制代码
while(k<n)	k*=2;

这个时候需要注意一个特判:如果两个数字相等怎么办(如果总人数n是偶数)?好吧,其实是因为我审题的时候没有仔细,题目说了是k<=m,意思也就是说等于也是可以的,当然我们模拟发现也是可以的......(多此一举了)

cpp 复制代码
#include<iostream>

using namespace std;

int main()
{
	int t;//表示样例数
	scanf("%d",&t);
	
	while(t--)
	{
		int n,k=1,a=0,b=0;//n表示总人数,k表示最接近n的2的指数函数的值
		//a表示轮数,b表示场数
		scanf("%d",&n);
		
		if(n<2)	printf("0 0\n");//只剩下一个人,就不需要比较,特判
		else
		{
			while(n>1)//只要不是剩下一个人,就一直循环
			{
				while(k<=n)	k*=2;//寻找最接近n的2的指数函数的值
				
				k/=2;//跳出上面循环会多乘一次,所以除掉一个2
				//当然,k==n不算多乘了一次,条件判断if(k>n)也是可以的
				
				a++;//每一次算一轮
				b+=k/2;//比赛进行k/2场
				n-=k/2;//淘汰k/2个人
			}
			
			printf("%d %d\n",a,b);
		}
	}
	
	return 0;
}

改成这样甚至不用再多加一个条件判断,可以肯定多乘了一次2,所以直接除以一次2即可,这个就是跳出循环的临界条件,只有k>n才会跳出循环,但是这个时候k是不满足条件的,所以需要回退一次,也就是除以2

2.轮数,场数,总人数之间的关系是什么?

每一轮需要算出一个小于等于总人数n的一个最大的2的指数函数数值k

每一轮需要进行k/2场比赛

每一轮需要淘汰k/2个人

不断地更新轮数,场数,总人数,不大于总人数的最大的2的指数函数数值k即可

3.比赛最后只要留下一个冠军,也就是说总人数等于1是结束的标识,跳出循环的临界条件

六、精美图片

相关推荐
未来可期LJ27 分钟前
【C++ 设计模式】单例模式的两种懒汉式和饿汉式
c++·单例模式·设计模式
Trouvaille ~1 小时前
【C++篇】C++类与对象深度解析(六):全面剖析拷贝省略、RVO、NRVO优化策略
c++·c++20·编译原理·编译器·类和对象·rvo·nrvo
little redcap1 小时前
第十九次CCF计算机软件能力认证-乔乔和牛牛逛超市
数据结构·c++·算法
机器视觉知识推荐、就业指导2 小时前
Qt/C++事件过滤器与控件响应重写的使用、场景的不同
开发语言·数据库·c++·qt
muyierfly2 小时前
34.贪心算法1
算法·贪心算法
孤寂大仙v2 小时前
【C++】STL----list常见用法
开发语言·c++·list
咩咩大主教3 小时前
C++基于select和epoll的TCP服务器
linux·服务器·c语言·开发语言·c++·tcp/ip·io多路复用
luthane4 小时前
python 实现average mean平均数算法
开发语言·python·算法
静心问道4 小时前
WGAN算法
深度学习·算法·机器学习
Ylucius4 小时前
动态语言? 静态语言? ------区别何在?java,js,c,c++,python分给是静态or动态语言?
java·c语言·javascript·c++·python·学习