大神们的题解我一个都没看懂。。。。。。。。。。。

十分的尴尬

题意:算出闭区间内二进制中0的个数大于等于1的个数的数字有多少个

思路:

组合数学(n小于500的时候都可以出解,只不过高精比较麻烦)。

这道题还算比较仁慈。。。

Discuss里面有一段说得挺好的

看完各家算法,尝试独立分析一下: 以sample为例子

[2,12]区间的RoundNumbers(简称RN)个数:Rn[2,12]=Rn[0,12]-Rn[0,1]

即:Rn[start,finish]=Rn[0,finish]-Rn[0,start-1] 所以关键是给定一个X,求出Rn[0,X]

现在假设X=10100100 这个X的二进制总共是8位,任何一个小于8位的二进制都小于X

第一部分,求出长度为[0,7]区间内的二进制是RoundNumber的个数

对于一个长度为Len的二进制(最高位为1),如何求出他的RoundNumbers呢(假设为用R(len)来表达),分为奇数和偶数两种情况

1、奇数情况:在Len=2k+1的情况下,最高位为1,剩下2k位,至少需要k+1为0

用C(m,n)表示排列组合数:从m个位置选出n个位置的方法

R(len)=C(2k,k+1)+C(2k,k+2)+…+C(2k,2k). 由于

A:C(2k,0)+C(2k,1)+…+C(2k,2k)=2^(2k)

B:C(2k,0)=C(2k,2k), C(2k,1)=C(2k,2k-1) ,,C(2k,i)=C(2k,2k-i) 于是 C(2k,0)+C(2k,1)+…+C(2k,2k)

= C(2k,0)+C(2k,1)+…+C(2k,k)+C(2k,k+1)+C(2k,K+2)+…+C(2k,2k)

= 2*R(len)+C(2k,k)

=2^(2k)

所以R(len)=1/2*{2^(2k)-C(2k,k)};

2. 偶数情况 len=2*k,类似可以推到 R(len)=1/2*(2^(2k-1)); 第二部分,对于上面这个长度为8的例子:即X=10100100,首先如果本身是RoundNumbers,第二部分的结果总数+1

第一部分已经将长度小于8的部分求出。现在要求长度=8的RoundNumber数目 长度为8,所以第一个1不可改变

现在到第二个1,如果Y是前缀如100*****的二进制,这个前缀下,后面取0和1必然小于X,已经有2个0,一个1,剩下的5个数字中至少需要2个0,

所以把第二个1改为0:可以有C(5,2)+C(5,3)+C(5,4)+C(5,5)

现在第三个1,也就是前最为101000**,同样求出,至少需要0个0就可,所以有C(2,0)+C(2,1)+C(2,2)个RoundNumbers

。。。

将所有除了第一个1以外的1全部变为0,如上算出有多少个RoundNumbers,结果相加(由于前缀不一样,所以后面不管怎么组合都是唯一的)

将第一部分和第二部分的结果相加,就是最后的结果了。 精度要求方面,用int就可以了:two

billion=20亿<2*1024*1024*1024=2^31,需用31位来表示数组,由于第一位总是1,所以求组合数的时候最多求30,C(30,k),k取值区间是[0,30],因为C(k,i)<2^k,所以结果用int表示就可以

(也有人用数位DP、记忆化搜索什么的。。。。 看个人喜好吧)

G++ AC C++WA 鬼知道为什么。。。

// by SiriusRen
#include <bitset>
#include <cstdio>
#include <algorithm>
using namespace std;
bitset<64>n,m;
int C[35][35];
int N,M,maxn=0,maxm=0,cnt1=1,cnt0=0;
int ans=0;
int main()
{
scanf("%d%d",&N,&M);M++;
for(int i=30;i>=0;i--){
if(M&(1<<i))m[i]=1,maxm=max(maxm,i);
if(N&(1<<i))n[i]=1,maxn=max(maxn,i);
}
for(int i=0;i<=31;i++)C[i][0]=1;
for(int i=1;i<=31;i++)
for(int j=1;j<=i;j++)
C[i][j]=C[i-1][j-1]+C[i-1][j];
for(int i=maxm;i>=0;i--)
for(int j=i-1;2*j>=i;j--)ans+=C[i-1][j];
for(int i=maxm-1;i>=0;i--)
if(m[i]){
for(int j=maxm-cnt0-cnt1;2*j+2*cnt0+1>=maxm;j--)
ans+=C[maxm-cnt0-cnt1][j];
cnt1++;
}
else cnt0++;
cnt1=1;cnt0=0;
for(int i=maxn;i>=0;i--)
for(int j=i-1;2*j>=i;j--)ans-=C[i-1][j];
for(int i=maxn-1;i>=0;i--)
if(n[i]){
for(int j=maxn-cnt0-cnt1;2*j+2*cnt0+1>=maxn;j--)
ans-=C[maxn-cnt0-cnt1][j];
cnt1++;
}
else cnt0++;
printf("%d\n",ans);
}

看到另一份题解里有这样一句话:

组合数学,一跪一天,爽!

表示赞同

POJ 3252 组合数学?的更多相关文章

  1. POJ 3252 (数位DP)

    ###POJ 3252 题目链接 ### 题目大意:给你一段区间 [Start,Finish] ,在这段区间中有多少个数的二进制表示下,0 的个数 大于等于 1 的个数. 分析: 1.很显然是数位DP ...

  2. POJ 3252:Round Numbers

    POJ 3252:Round Numbers Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 10099 Accepted: 36 ...

  3. POJ 3252 Round Numbers 组合数学

    Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 13381   Accepted: 5208 Description The ...

  4. POJ 3252 Round Numbers(组合数学)

    Round Numbers Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 10223   Accepted: 3726 De ...

  5. POJ 3252 Round Numbers(组合)

    题目链接:http://poj.org/problem?id=3252 题意: 一个数的二进制表示中0的个数大于等于1的个数则称作Round Numbers.求区间[L,R]内的 Round Numb ...

  6. poj 3252

    http://poj.org/problem?id=3252//自己搞了很长时间...现在刚刚有点明白.. 1 #include <iostream> using namespace st ...

  7. [poj 3252]数位dp前导0的处理

    通过这个题对于数位dp中前导0的处理有了新的认识. 题目链接:http://poj.org/problem?id=3252 //http://poj.org/problem?id=3252 #incl ...

  8. POJ 3252 Round Numbers

     组合数学...(每做一题都是这么艰难) Round Numbers Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 7607 A ...

  9. poj 3252 Round Numbers 【推导·排列组合】

    以sample为例子 [2,12]区间的RoundNumbers(简称RN)个数:Rn[2,12]=Rn[0,12]-Rn[0,1] 即:Rn[start,finish]=Rn[0,finish]-R ...

随机推荐

  1. jQuery基本选择器模块

    选择器模块 1.获取元素的基本操作 案例:给页面中的div和p设置边框样式 1.1 传统方式 -获取元素并设置样式 实现思路 1 通过 标签名 获取元素 2 遍历循环 设置样式 var dvs = d ...

  2. 5) 十分钟学会android--ActionBar知识串烧

    建立ActionBar Action bar 最基本的形式,就是为 Activity 显示标题,并且在标题左边显示一个 app icon.即使在这样简单的形式下,action bar对于所有的 act ...

  3. 【Oracle】权限

    1. 授予权限: GRANT privilege[, privilege...] TO user [, user| role, PUBLIC...]; ①DBA授予用户系统权限 GRANT creat ...

  4. 04--深入探讨C++中的引用

    深入探讨C++中的引用           引用是C++引入的新语言特性,是C++常用的一个重要内容之一,正确.灵活地使用引用,可以使程序简洁.高效.我在工作中发现,许多人使用它仅仅是想当然,在某些微 ...

  5. 优动漫PAINT动画创作功能介绍

    优动漫PAINT也就是我们常说的clip studio paint(CSP)的中文版本,它是一款功能强大的漫画.插画绘制软件,它搭载了绘制漫画和插画所需的所有功能,包括丰富的笔工具.超强的笔压感应和手 ...

  6. 如何构建和设计以确保 API 的安全性

    如何构建和设计以确保 API 的安全性 面对常见的OWASP十大威胁.未经授权的访问.拒绝服务攻击.以及窃取机密数据等类型的攻击,企业需要使用通用的安全框架,来保护其REST API,并保证良好的用户 ...

  7. Centos 搭建activemq

    Centos 搭建activemq 1,官方下载  http://activemq.apache.org/activemq-5122-release.html apache-activemq-5.15 ...

  8. Asp 6种页面转向方法

    asp.net 页面转向方法其实就是两种 服务器端转向和客户端转向 客户端转向实质上是指由浏览器直接向服务器端重新发送一个请求. 而服务器端转向是指服务器内部进行页面的跳转. 服务器端转向和客户端转向 ...

  9. IOS与h5交互记录

    博主之前做过移动端app嵌入网页,与Android和IOS有交互,一直没有时间分享过程.这里不多说Android交互啦-很简单,详细了解IOS与h5的交互吧. IOS不同语法和h5的交互所建立的JSB ...

  10. vue 上传图片到阿里云(前端直传:不推荐)

    为何要这样做:减轻后端数据库压力(个人觉得于前端没啥用,谁返回来都行) 代码部分: <template> <div class="upLoad"> < ...