COGS 862. 二进制数01串【dp+经典二分+字符串】
862. 二进制数01串
★ 输入文件:kimbits.in
输出文件:kimbits.out
简单对比
时间限制:1 s 内存限制:128 MB
USACO/kimbits(译 by !Starliu )
描述
考虑排好序的N(N<=31)位二进制数。
你会发现,这很有趣。因为他们是排列好的,而且包含所有可能的长度为N且含有1的个数小于等于L(L<=N)的数。
你的任务是输出第I(1<=I<=长度为N的二进制数的个数)大的,长度为N,且含有1的个数小于等于L的那个二进制数。
注意:这里“长度为N”包括长度小于N的数(我们认为高位用0补齐)
格式
PROGRAM NAME: kimbits
INPUT FORMAT:(file kimbits.in)
共一行,用空格分开的三个整数N,L,I。
OUTPUT FORMAT:(file kimbits.out)
共一行,输出满足条件的第I大的二进制数。
SAMPLE INPUT
(file kimbits.in)
5 3 19
SAMPLE OUTPUT
(file kimbits.out)
10011
题目链接:http://cogs.cf/cogs/problem/problem.php?pid=862
分析:这道题很难,构思巧妙,先用dp求出所要求dp[i,j]前i位1的个数不大于j的方案数,然后便是printf了。
先讲一下如何计算dp[i,j]。
dp[i,j]表示前i位,1的个数不大于j的方案数。
显然:dp[i,j]=dp[i-1,j]+dp[i-1,j-1];
dp[i-1,j]表示当前第i位以0开头所得到的方案数,dp[i-1,j-1]表示当前第i位以1开头得到的方案数。
如何根据得到的dp[i,j]来printf呢?如果当前我要确定第i位,那么肯定要看dp[i-1]集合中的值判断,例如我当前确定第5位,前4位不超过3个1的方案数为15,而我现在要求第19位,则第5为1,因为19>15,为什么呢?因为第5位可能为0,1,而为0的占了15个,为1的开头也是占15个,显然19属于为1开头的数,所以输出1,输出1的同时还要更新数据即可。输出0就什么也不用更新。
更新数据是指:当前第19位要减去15,3个1要变为2个1,以更新完的数据去得到第i+1个数才能是正确的。二分对每一位判断当改位是0的时候的有多少个符合条件的数就行了
貌似代码跑的飞起来,COGS上第一道排名NO.1的题,纪念一下!
下面给出AC代码:
1 #include <bits/stdc++.h>
2 using namespace std;
3 const int maxn=32+5;
4 typedef long long ll;
5 ll dp[maxn][maxn],n,o=0;
6 ll C(ll a,ll b)
7 {
8 if(a==0)
9 return 1;
10 if(dp[a][b]!=-1)
11 return dp[a][b];
12 return a==1?b:dp[a][b]=(C(a-1,b)*(b-a+1))/a;
13 }
14 ll total(ll num,ll len,ll maxo)
15 {
16 ll cnt=0;
17 len=n-len-1;
18 for(ll i=0;(i+o)<=maxo;i++)
19 {
20 cnt+=C(i,len);
21 }
22 return cnt;
23 }
24 int main()
25 {
26 ll l,i,len;
27 char ans[maxn];
28 memset(dp,-1,sizeof(dp));
29 memset(ans,0,sizeof(ans));
30 freopen("kimbits.in","r",stdin);
31 freopen("kimbits.out","w",stdout);
32 cin>>n>>l>>i;
33 for(len=0;len<n;len++)
34 {
35 ll temp=total(0,len,l);
36 if(temp>=i)
37 ans[len]=0+'0';
38 else
39 {
40 ans[len]=1+'0';
41 i=i-temp;
42 o++;
43 }
44 }
45 cout<<ans<<endl;
46 return 0;
47 }
- nGrinder 简易使用教程
- UI设计高效学习网站&工具,来自学长的收藏夹哦
- 安装git出现templates not found的问题
- 时间戳 时间
- jenkins 设置 gitlab web hooks
- 测试流程?项目管理流程?
- 学web前端开发写给新手的建议,超实用!
- 价值22万的5字母域名sanwa.com被启用
- Django admin 一些有用的设置
- mysql @value := 用法
- css样式—字体垂直、水平居中
- maven可选依赖(Optional Dependencies)和依赖排除(Dependency Exclusions)
- gitignore 使用
- 区块链学堂——区块链词汇手册
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- 新特性解读 | 数组范围遍历功能
- 技术分享 | MySQL 内存管理初探
- 新特性解读 | 窗口函数的适用场景
- Android自定义View 仿QQ侧滑菜单的实现代码
- Android view随触碰滑动效果
- TextView使用SpannableString设置复合文本 SpannableString实现TextView的链接效果
- FragmentTabHost使用方法详解
- Android编程实现仿优酷圆盘旋转菜单效果的方法详解【附demo源码下载】
- Android绘制圆形百分比加载圈效果
- Android自定义view实现动态柱状图
- Kubernetes集群高可用&备份还原概述 | 知识分享月第三期直播回顾
- SpringCloud2020 学习笔记(一)springboot和springcloud技术选型以及版本选择
- SpringCloud2020 学习笔记(二)父工程搭建
- SpringCloud2020 学习笔记(三) cloud-api-commons通用模块搭建
- SpringCloud2020 学习笔记(四) cloud-provider-payment8001支付模块