2021 Jiangsu Collegiate Programming Contest——K. Longest Continuous 1

题目地址https://codeforces.com/gym/103495/problem/K

题目解析

题目

在这里插入图片描述
Example:

4
1
2
3
4

output:

0
1
2
2

思路

如下图,我们打表可以发现在长度 1 e 9 1e9 1e9,最大不过一个二进制数字长度为 29 29 29
在这里插入图片描述
那我们用高中数学的数列求和公式可以求出
L ( n ) = 2 n × ( n − 1 ) + 2 L(n) = 2^{n}\times(n-1)+2 L(n)=2n×(n1)+2
表示以二进制数字长度为 n n n的最后一个数字结尾可以得到的 p k p_k pk的长度。
然后这题是个找规律的题,
如下面一段字符串,我们可以知道是以 8 8 8的二进制结尾。

0110111001011101111000

假设我们输入的 k k k 18 18 18,结合打表得到的数据可以知道,我们得到的答案为 3 3 3,但是如果 k k k 19 19 19,答案就为 4 4 4,因此我们可以发现,连续为 1 1 1的子字符串的最大长度的变化在二进制数字长度变化的交界处。
下面看代码更便于理解

代码

#include<bits/stdc++.h>
#include<algorithm>
#include<functional>
#define inf 1e9
#define ll long long
#define pii pair<int,int>
using namespace std;
const int N = 1e4+10;
int t, n, m, k;
ll len[N], sum[N];
int cnt;
//2^n * (n-1)+2
void init(){
	ll cnt = 2;
	for(int i = 1;i <= 29;i++){
		len[i] = cnt*(i-1)+2;
		cnt*=2;
	}
	// for(int i = 1;i <= 29;i++){
	// 	cout<<i<<" "<<len[i]<<"\n";
	// }
}
void solve(){
	cin>>k;
	if(k==1)cout<<"0\n";
	else{
		int pos = lower_bound(len+1,len+30,k)-len;
		// cout<<pos<<"___\n";
		if(len[pos]==k){
			cout<<pos<<"\n";
		}
		else{
			cout<<pos<<"\n";
		}
	}
}
int main(){
	init();
	cin>>t;
	// t = 1;
	while(t--)
    	solve();
    return 0;
}
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值