蓝桥杯2021年第十二届真题第一场-杨辉三角形

题目

题目链接

题解

思维。


请添加图片描述

杨辉三角对称,所以我们只看左半边(含中间列)。

将全部的1忽略掉,中间列具有特点, C 2 1 C_2^1 C21 C 4 2 C_4^2 C42 C 6 3 C_6^3 C63、……、 C 2 x x C_{2x}^x C2xx

每个中间列的元素都可以视为一条斜线的一端,如上图的黄线,写线上的元素也具有特点,自右上角至左下角 C 2 x x C_{2x}^x C2xx C 2 x + 1 x C_{2x+1}^x C2x+1x、……、 C 2 x + i x C_{2x+i}^x C2x+ix,且这些数是满足单调递增的。


如果一个数字能出现在靠下的黄线中,那么它出现的一定是最早的,也就是说如果一个数出现在多条黄线中,那么最靠下的一条黄线中的该数对应的位置一定是最小的。(证明不会,但是直观上去理解还是可以明白的)


x x x大的斜行开始枚举,对于每个斜行,我们去判断输入的数是否在该斜线中,如果在,我们需要知道该数在这条斜线的位置,即自右上往左下数第几个,知道位置就可以,也知道当前枚举到的是第几条斜线,就可以直接算出其前面的数的个数了(本质上我们知道的并不是“自右上往左下数第几个”,而是其他信息,但是都可以求出最终答案,之所以这样说,是先让你有个整体思路而已)


如果再一个个去枚举每条斜线上的元素,时间复杂度就太大了,因为每条斜线都具有单调递增的特性( C 2 x + i x < C 2 x + i + 1 x C_{2x+i}^x < C_{2x+i+1}^x C2x+ix<C2x+i+1x),所以我们二分确定 2 x + i 2x+i 2x+i,即l~r=2x~n,其中x为枚举到的斜线号(从上到下为1、2、……)

代码

#include<bits/stdc++.h>
using namespace std;
typedef long long LL;

LL n;

LL C (LL b, LL a) {
	LL res = 1;
	for (int i = 0;i < a;i ++) {
		res *= b - i;
		res /= i + 1;
		if (res > n) return res; // 必须有这句话,要不AC不了,应该是存在溢出吧 
	}		
	return res;
}

bool check (LL x) {
	LL l = 2*x, r = n, mid; // 这里令右边界为n,因为C(n,x)一定大于n 
	while (l < r) {
		mid = l + r >> 1;
		if (C (mid, x) >= n) r = mid;
		else l = mid + 1;
	}
	
	if (C (l, x) == n) {
		cout << (l + 1) * l / 2 + x + 1 << endl;
		return true;
	}
	return false;
}

int main()
{
	cin >> n;
	if (n == 1)
		cout << 1;
	else 
		for (LL i = 16;i >= 1;i --) 
			if (check (i)) break;

	
	return 0;
}	
  • 1
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

不牌不改

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值