4513: [Sdoi2016]储能表

题意:求$$

\sum_{i=0}{n-1}\sum_{j=0}{m-1} max((i\oplus j)-k,0)

\[
***

写出来好开心啊...虽然思路不完全是自己的但代码是按照自己的想法用记忆化搜索写的啊

</br>

小于k的直接不用考虑
考虑二进制上数位DP,从高到低考虑每一位
$n,m,k$变成了三条天际线,小于等于$n,m$并且大于等于$k$
$f[i][s1][s2][s3]$表示第i位三条天际线状态s1s2s3时满足条件的方案数和异或和
每一位枚举i和j这一位是什么转移就行了
最后的答案就是 异或和-方案数*k
然后计算某一位异或和的贡献时需要乘上2的幂和后面的方案数
</br>
然后本题不一样的地方在于**必须把三条天际线记忆化**,之前的题目不用记忆化是因为只有一种转移会到天际线,而本题有多种转移可以到相同的天际线...不然T成暴力分

</br>
```cpp
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
typedef long long ll;
#define pii pair<ll, ll>
#define MP make_pair
#define fir first
#define sec second
const int N=65;
inline ll read(){
char c=getchar();ll x=0,f=1;
while(c<'0'||c>'9'){if(c=='-')f=-1;c=getchar();}
while(c>='0'&&c<='9'){x=x*10+c-'0';c=getchar();}
return x*f;
}

ll n, m, k, p, mi[N]; int tot;
pii f[N][2][2][2], im;
struct meow{
int n, a[N];
int& operator[](int x) {return a[x];}
void read(ll x) {memset(a,0,sizeof(a));n=0; while(x>0) a[++n]=x&1, x>>=1;}
void print() {printf("digit ");for(int i=n; i>=1; i--) printf("%d",a[i]);puts("");}
}a, b, c;

inline void mod(ll &x) {if(x>=p) x-=p;}
pii dfs(int d, int s1, int s2, int s3) {
if(d==0) return MP(1, 0); //printf("dfs %d %d %d %d\n",d,s1,s2,s3);
if(f[d][s1][s2][s3] != im) return f[d][s1][s2][s3];
pii now(0, 0);
int lim1 = s1 ? a[d] : 1, lim2 = s2 ? b[d] : 1, lim3 = s3 ? c[d] : 0;
//printf("lim %d %d %d\n",lim1,lim2,lim3);
for(int i=0; i<=lim1; i++)
for(int j=0; j<=lim2; j++) if((i^j)>=lim3) { //printf("ij %d %d %d %lld\n",i,j, i^j, (i^j)*mi[d-1]);
pii t = dfs(d-1, s1 && i==lim1, s2 && j==lim2, s3 && (i^j)==lim3);
mod(now.fir += t.fir);
mod(now.sec += (t.sec + (i^j) * mi[d-1]%p * t.fir%p)%p);
}
//printf("now %d %d %d %d %lld %lld\n\n",d,s1,s2,s3,now.fir, now.sec);
return f[d][s1][s2][s3]=now;
}
int main() {
//freopen("menci_table.in","r",stdin);
//freopen("menci_table.out","w",stdout);
int T=read();
im=MP(-1, -1);
while(T--) {
n=read(); m=read(); k=read(); p=read();
n--; m--;
mi[0]=1; for(int i=1; i<=60; i++) mi[i] = (mi[i-1]<<1)%p;
a.read(n); b.read(m); c.read(k);
//a.print(); b.print(); c.print();
tot=max(a.n, max(b.n, c.n));
for(int i=0; i<=tot; i++) for(int j=0;j<2;j++) for(int k=0;k<2;k++) f[i][j][k][0]=f[i][j][k][1]=im;
pii ans = dfs(tot, 1, 1, 1);
//printf("ans %lld %lld\n",ans.sec,ans.fir);
printf("%lld\n", (ans.sec - ans.fir*(k%p)%p + p)%p);
}
}
```\]

BZOJ 4513: [Sdoi2016]储能表 [数位DP !]的更多相关文章

  1. BZOJ.4513.[SDOI2016]储能表(数位DP)

    BZOJ 洛谷 切了一道简单的数位DP,终于有些没白做题的感觉了...(然而mjt更强没做过这类的题也切了orz) 看部分分,如果\(k=0\),就是求\(\sum_{i=0}^n\sum_{j=0} ...

  2. 4513: [Sdoi2016]储能表 数位DP

    国际惯例的题面: 听说这题的正解是找什么规律,数位DP是暴力......好的,我就写暴力了QAQ.我们令f[i][la][lb][lc]表示二进制从高到低考虑位数为i(最低位为1),是否顶n上界,是否 ...

  3. 【BZOJ4513】[Sdoi2016]储能表 数位DP

    [BZOJ4513][Sdoi2016]储能表 Description 有一个 n 行 m 列的表格,行从 0 到 n−1 编号,列从 0 到 m−1 编号.每个格子都储存着能量.最初,第 i 行第 ...

  4. bzoj 4513 [Sdoi2016]储能表

    题面 https://www.lydsy.com/JudgeOnline/problem.php?id=4513 题解 要求的式子 用数位dp的方法去做 我们把式子拆开 变成 $\sum_{i=0}^ ...

  5. [SDOI2016]储能表——数位DP

    挺隐蔽的数位DP.少见 其实减到0不减了挺难处理.....然后就懵了. 其实换个思路: xor小于k的哪些都没了, 只要留下(i^j)大于等于k的那些数的和以及个数, 和-个数*k就是答案 数位DP即 ...

  6. BZOJ4513: [Sdoi2016]储能表(数位dp)

    题意 题目链接 Sol 一点思路都没有,只会暴力,没想到标算是数位dp??Orz 首先答案可以分成两部分来统计 设 \[ f_{i,j}= \begin{aligned} i\oplus j & ...

  7. [bzoj4513][SDOI2016]储能表——数位dp

    题目大意 求 \[\sum_{i = 0}^{n-1}\sum_{j=0}^{m-1} max((i\ xor\ j)\ -\ k,\ 0)\ mod\ p\] 题解 首先,开始并没有看出来这是数位d ...

  8. 4513: [Sdoi2016]储能表

    4513: [Sdoi2016]储能表 链接 分析: 数位dp. 横坐标和纵坐标一起数位dp,分别记录当前横纵坐标中这一位是否受n或m的限制,在记录一维表示当前是否已经大于k了. 然后需要两个数组记录 ...

  9. 搜索(四分树):BZOJ 4513 [SDOI2016 Round1] 储能表

    4513: [Sdoi2016]储能表 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 395  Solved: 213[Submit][Status] ...

随机推荐

  1. 【Git】Git基础操作

    repository:版本库又名仓库,可以简单理解成一个目录,这个目录里面的所有文件都可以被Git管理起来,每个文件的修改.删除,Git都能跟踪,以便任何时刻都可以追踪历史,或者在将来某个时刻可以&q ...

  2. TCP 和 UDP

    TCP协议与UDP协议的区别    首先咱们弄清楚,TCP协议和UCP协议与TCP/IP协议的联系,很多人犯糊涂了,一直都是说TCP/IP协议与UDP协议的区别,我觉得这是没有从本质上弄清楚网络通信! ...

  3. PHPStorm+PHPStudy新建第一个PHP项目

    img { max-width: 100% } 熟悉了.net的编程,偶尔也来客串一下PHP.前几天闲来无事随便加了一个PHP的开发群,入群之后傻眼了,群里有大小各位程工1600多人,所以决定学习一下 ...

  4. js数组操作记录

    一 .splice() 方法向/从数组中添加/删除项目,然后返回被删除的项目. arrayObject.splice(index,howmany,item1,.....,itemX) 参数 描述 in ...

  5. Linux/Unix系统SSH远程按Backspace键删除时出现^H的处理方法

    在linux/unix系统中连接SSH远程工作时,输出字符后按Backspace键删除时,会出现^H,这对习惯了按Backspace键删除的用户来说,感觉非常别扭,虽然可以通过Ctrl+Backspa ...

  6. 将js进行到底:node学习笔记1

    废话:自高中以来一直对编程充满激情,磨剑五年,如今要毕业了,我不想用我已经擅长的知识敷衍,而想以一个全新的领域去面向我的毕设--是时候学习一下node.js node.js基础 对于JavaScrip ...

  7. nodejs+websocket实时聊天系统

    介绍下websocket: webSocket协议本质上是一个基于tcp的协议; 建立一个websocket连接,大体的过程: 1.客户端浏览器首先向服务器发起一个http请求,这个请求和平常的请求有 ...

  8. 2017-07-04(sudo wc sort)

    sudo 作用 root把本来只能超级用户执行的命令,赋予普通用户执行. 添加 运行visudo命令,在文件底部添加信息即可! sudo -l  查看用户可以运行的命令 use1  ALL=(ALL) ...

  9. [转]解决error: "net.ipv4.netfilter.ip_conntrack_max" is an unknown key错误

    今天在新买的vps上执行sysctl -p,报下面的错误:net.ipv4.ip_forward = 0net.ipv4.conf.default.rp_filter = 1net.ipv4.conf ...

  10. Switch-case 内定义变量的问题

    Switch-case 内定义变量的问题 这个问题需要分开讨论,C 语言和 C++ 的标准定义是不同的. C++ int Caset(int a) { switch (a) { case 1: int ...