LUOGU P3048 [USACO12FEB]牛的IDCow IDs(组合数)
2024-08-30 00:27:28
解题思路
组合数学。首先肯定是要先枚举位数,假如枚举到第\(i\)位。我们可以把第一位固定,然后那么后面的随意放\(1\),个数就为\(C_{i-1}^{k-1}\)。然后每次枚举时如果方案\(>n\)就说明位数为\(i\),否则就让\(n-C_{i-1}^{k-1}\),然后继续枚举下去。这样的话我们就确定了第一位,后面的位其实和数位\(dp\)里试填法的思路差不多,就是看\(n\)是否大于当前位为\(0\)时后面的方案数,如果大就把这一位设为\(1\),然后减掉方案。算组合数有一个小\(trick\)就是先用分子约一个分母里比较大的,因为这道题\(k<=10\),所以约完后暴力乘是不会炸\(long long\)的。
代码
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<cstdlib>
using namespace std;
const int MAXN = 1005;
typedef long long LL;
int n,k,ans[1000005];
inline LL C(int x,int y){
if(x<y) return 0;
if(!y) return 0;if(x==y) return 1;
int lim=max(y,x-y),c=min(y,x-y);
LL ret=1;
for(register int i=x;i>=lim+1;i--)
ret=ret*i;
for(register int i=2;i<=c;i++)
ret=ret/i;
return ret;
}
int main(){
scanf("%d%d",&n,&k);
if(n==1) {while(k--) putchar('1');putchar('\n');return 0;}
if(k==1) {putchar('1');n--;while(n--) putchar('0');putchar('\n');return 0;}
n--;int l=k+1;
while(C(l-1,k-1)<=n){
n-=C(l-1,k-1);l++;
}
ans[1]=1;int res=k-1;putchar('1');
for(register int i=2;i<=l;i++){
LL te=C(l-i,res);
if(n>te && res) {ans[i]=1;n-=te;res--;}
putchar(ans[i]+'0');
}putchar('\n');
return 0;
}
最新文章
- [转载]:STM32为什么必须先配置时钟再配置GPIO
- ls目录内容
- 从网页上抓取Windows补丁信息然后整型输出(Python)
- ubuntu 更改时区
- ensure LANG and/or LC_* environment variables are set correctly
- Apache和Nginx配置支持苹果ATS方法
- qt QMessageBox QInputDialog
- CentOS 6破解Mysql5.5的办法
- JSOI2009 游戏
- tomcat6~7~8用户设置及一个独立服务器上跑多个tomcat配置JVM设置优化亲测
- 动态类(Dynamic)应用
- 确认oracle数据库错误日志文件位置
- Oracle 12c 单实例安装
- [原创] debian 9.3 搭建Jira+Confluence+Bitbucket+crowd+seafile (零) 修改端口的问题
- GO slim
- java.util.concurrent ThreadPoolExecutor源码分析
- [arm学习]makefile学习总结
- Java-Java程序设计的基本概念
- PyQt5教程——组件 Ⅱ(八)
- Telnet远程重启路由器TP-LINK