Math teacher's homework
Title:【Math teacher's homework】#
Description##
题目大意:给你n个数m1,m2...mn,求满足X1 xor X2 xor ... xor Xn=k,0<=Xi<=mi (1<=i<=m)的方案数
Input Format##
多组数据(至多100组)每组数据第一行包括两个整数n,k,第二行包括n个整数m1,m2...mn(1<=n<=50,0<=k,mi<=\(2^{31}-1\)(1<=i<=n))
输入以0 0结尾
Output Format##
对于每组数据输出一行表示题目中描述的方案数,方案数可能很大,你只需输出方案数mod 1000000003的结果
Sample Input##
11 2047
1024 512 256 128 64 32 16 8 4 2 1
10 2047
1024 512 256 128 64 32 16 8 4 2
0 0
Sample Output##
1
0
Solution##
还是太菜了,这道题看了很很很很很久
废话不说了
首先,我们把mi+1,那么我们Xi的限制条件自然改变了,变成了1<=Xi<mi(1<=i<=n)
然后我们来分析一下如何进行数位DP
对于Xi,从高位开始比较,必然有若干个高位与mi相等,我们把相等的这些位涂成深蓝色,第一个与mi不同的位涂成绿色(由于Xi一定小于mi,mi的该位必然为1,Xi的该位必然为0)
如果该题没有限制,那么每一位异或为1或0的方案数应该都为\(2^{n-1}\),因为只要n-1个数随便取,用最后一个数改变异或值即可
也就是说,如果Xi已经确定了一位涂上了绿色,那么我们后面的格子就和没有限制是一样的了,那么我们只要固定每列的一个格子,其他格子便可以随便取了
由此我们规定每列最早出现的格子涂成橙色,其余格子涂成浅蓝色,那么,每一列的方案数就是\(2^{浅蓝色格子数-1}\)

有了这些基础我们就可以设计状态了,f[i][j][l]表示处理到第i个数最左边的绿色格子在第j位第j列的异或值为l的方案数
枚举对于f[i][j][0/1]我们枚举上一行的状态f[i-1][k][l],由上一行的状态来推出该行状态
我们用num[i][j]来存储mi的第j位数字(二进制下),用col[i][j]来记录第j位的数值从第一个数异或到mi的异或值,用b[i]表示\(2^i\)的值
接下来,分类讨论~~~
(1)当j>k时 f[i][j][col[i-1][j]]+=f[i-1][k][l]*b[k-1],由于第j格在该行为绿色格子,即其值为0,后面的蓝色格子可以任意取,因此该方程很容易理解
(2)当j=k时 f[i][j][l]+=f[i-1][k][l]*b[j-1],由于第j格在该行为绿色格子,即其值为0,且j,k位于同一列,后面的蓝色格子可以任意取,因此该方程很容易理解
(3)当j<k时 f[i][k][l^num[i][k]]+=f[i-1][k][l]*b[j-1],由于第j格在该行为绿色格子,即其值为0,后面的蓝色格子可以任意取,因此该方程很容易理解
但是由于绿色格子只存在于原数该位值为1的位上,所以方程符合题意当且仅当num[i][j]=1
还有一件事,就是如果最左边的绿色格子如果在第j位,则比j高位的col[n]k都必须与mi的第k位相等
emmmmm,讲到这里,只要记得mod 1000000003应该就没问题了吧(细节错了不要怪我瞎讲)
附代码咯
#include<cstdio>
#define mo 1000000003
#include<cstring>
#define me(a) memset(a,0,sizeof a)
int k,num[55][33],n,m,b[32],col[55][33],f[55][33][2],ans,md[33];
void in(int i){scanf("%d",&k);k++;int t=0;for(;k;)num[i][++t]=k&1,k>>=1;}
int main(){
b[0]=1;
for (int i=1;i<=31;i++) b[i]=b[i-1]*2;
while (~scanf("%d%d",&n,&m)){
if (!n&&!m) return 0;me(col),me(f),me(num),ans=0;f[0][1][0]=1;
for(int i=1;i<=n;i++)in(i);
for(int i=1;i<=n;i++)for(int j=1;j<=32;j++)col[i][j]=col[i-1][j]^num[i][j];
for(int i=1;i<=n;i++)
for(int j=1;j<=32;j++)if(num[i][j])
for(int k=1;k<=32;k++)
for(int l=0;l<=1;l++)if(f[i-1][k][l]){
if(j>k)f[i][j][col[i-1][j]]=(f[i-1][k][l]*1ll*b[k-1]+f[i][j][col[i-1][j]])%mo;
else if(j==k) f[i][j][l]=(f[i-1][k][l]*1ll*b[k-1]+f[i][j][l])%mo;
else f[i][k][l^num[i][k]]=(f[i-1][k][l]*1ll*b[j-1]+f[i][k][l^num[i][k]])%mo;
}
for(int i=1;i<=32;i++)md[i]=m&1,m>>=1;
for(int i=32;md[i+1]==col[n][i+1]&&i;i--)ans=(1ll*ans+f[n][i][md[i]])%mo;
printf("%d\n",ans);
}
}
Math teacher's homework的更多相关文章
- HDU3693 Math Teacher's Homework ---- 数位DP
HDU3693 Math Teacher's Homework 一句话题意 给定$n, k以及m_1, m_2, m_3, ..., m_n$求$x_1 \oplus x_2 \oplus x_3 \ ...
- POJ 3986 Math teacher's homework
题目 给出\(n,m_1,m_2,...,m_n\),求\(x_1 xor x_2 xor ... xor x_n=k (0 \leq x_i \leq m_i)\)的解的数量.二进制位数小于\(32 ...
- 题解 Math teacher's homework
题目传送门 题目大意 给出 \(n,k\) 以及 \(a_{1,2,...,n}\) ,求有多少个 \(m_{1,2,...,n}\) 满足 \(\forall i,m_i\le a_i\) 且 \( ...
- HDU 5068 Harry And Math Teacher
主题链接~~> 做题情绪:的非常高深,有种高大上的感觉. 解题思路: 两层之间的联通能够看成是一个矩阵 代表上下两层都能够联通,,代表下层第1个门与上层第一个门不联通,以此类推联通就能够用矩阵 ...
- HDU 5068 Harry And Math Teacher 线段树+矩阵乘法
题意: 一栋楼有n层,每一层有2个门,每层的两个门和下一层之间的两个门之间各有一条路(共4条). 有两种操作: 0 x y : 输出第x层到第y层的路径数量. 1 x y z : 改变第x层 的 y门 ...
- 【转载】ACM总结——dp专辑
感谢博主—— http://blog.csdn.net/cc_again?viewmode=list ---------- Accagain 2014年5月15日 动态规划一 ...
- 【DP专辑】ACM动态规划总结
转载请注明出处,谢谢. http://blog.csdn.net/cc_again?viewmode=list ---------- Accagain 2014年5月15日 ...
- dp专题训练
****************************************************************************************** 动态规划 专题训练 ...
- 【DP专辑】ACM动态规划总结(转)
http://blog.csdn.net/cc_again/article/details/25866971 动态规划一直是ACM竞赛中的重点,同时又是难点,因为该算法时间效率高,代码量少,多元性强, ...
随机推荐
- Spring boot 搭配 JPA 生成表注释 和 字段注释
原文地址:https://blog.csdn.net/qq_39996837/article/details/84717748 由于在数据库表反向生成过程中呢,需要通过jpa自动生成表,并且这个表必须 ...
- Linux下查看硬盘UUID和修改硬盘UUID(转)
查看硬盘UUID: 1. ls -l /dev/disk/by-uuid 2. blkid /dev/sda5 修改硬盘UUID: 1.新建和改变分区的UUID sudo uuidgen | xarg ...
- 图片在 canvas 中的 选中/平移/缩放/旋转,包含了所有canvas的2D变化,让你认识到数学的重要性
1.介绍 canvas 已经出来好久了,相信大家多少都有接触. 如果你是前端页面开发/移动开发,那么你肯定会有做过图片上传处理,图片优化,以及图片合成,这些都是可以用 canvas 实现的. 如果你是 ...
- SaltStack学习系列之Nginx部署
目录结构 |-- nginx | |-- files #放包文件的 | | |-- admin_22.conf | | |-- fastcgi_params | | |-- jim_fix_param ...
- Logstash学习系列之基础介绍
Logstash功能特性 能集中处理各种类型的数据 能标准化不同模式和格式的数据 能快速的扩展自定义日志的格式 它具有收集,分析和转发数据流的功能 Logstash运行参数 -f 指定配置文件 -e ...
- 015 WAN
Router#config t Enter configuration commands, one per line. End with CNTL/Z. Router(config)#int s0/ ...
- 集成环信时遇到的问题file not found: libEaseMobClientSDK.a
集成环信时遇到的问题 build setting环信SDK集成libEaseMobClientSDKL file not found: libEaseMobClientSDK.a clang: er ...
- iOS 多线程,ARC
iOS自己创建的线程需要自己定时的创建autorelease pools,否则对象不能及时自动释放. 方法1是不对的,while中的对象会无法及时释放. 1:-(void)Thread{ @autor ...
- ldd LD_TRACE_LOADED_OBJECTS
1 该环境变量设置为1的话,只会打印所执行的程序的依赖,即所依赖的动态链接库
- linux之安装软件出现Could not open lock file /var/lib/dpkg/lock - open (13: Permission denied)解决总结
sudo rm -rf /var/lib/dpkg/lock sudo rm -rf /var/cache/apt/archives/lock