emm

位操作实现技巧:

获得第i位的数据:  if(!(data & (1<< i)))  则data的第 i 位为0,else 为 1

设置第i位为1,data=(data | (1<< i));

设置第i位为0,data=(data & (~(1<< i)))

将第i位取反,data=(data ^ (1<< i)

取出一个数的最后一个1 (lowbit):(data & (-data))

(二进制数从右往左最右为第0位)

Codeforces gym 101343 J

题意:给出一片田野,其中有0有1,1的地方可以选,挨着的两个格不能选,问有多少种选法

思路:把田野上可以选的状态用二进制表示,进行dp,因为每一行的情况只与它上一行有关

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cmath>
#include <cstring>
#include <algorithm>
using namespace std;
const int SZ = ;
const int INF = ;
const int mod = ;
int tmp;
int f[][SZ], st[SZ], a[];
int m, n;
bool check(int x, int y)//判断田野情况为x时,y方案能不能放
{
bool flag = true;
for(int i = ; i < n; i++)
if((x&( << i)) == && (y&( << i)) > ) flag = false;//田野上这个点是0,而方案中这个点要放,不可以呀
return flag;
}
void init()//初始化出所有可能存在的状态
{
int tot = ;
for(int i = ; i < (<<n); i++)
if(!(i & (i<<))) st[++tot] = i;
}
int main()
{
scanf("%d %d", &m, &n);
for(int i = ; i <= m; i++) a[i] = , f[i] = ;
for(int i = ; i <= m; i++)
for(int j = ; j <= n; j++)
{
scanf("%d", &tmp);
if(tmp) a[i] = a[i] | (<<(n-j));//将这一位设为1
}
for(int i = ; i <= tot; i++)
{
if(check(a[], st[i])) f[][i] = ;
}
for(int i = ; i <= m; i++)
for(int j = ; j <= tot; j++)//枚举第i行的状态
if(check(a[i], st[j]))//第i行的j状态合法
for(int k = ; k <= tot; k++)//枚举第i-1行状态
if(check(a[i-], st[k]) && !(st[k] & st[j]))
f[i][j] = (f[i][j] + f[i-][k]) % mod;
int ans = ;
for(int i = ; i <= tot; i++)
ans = (ans + f[m][i]) % mod;
printf("%d\n", ans);
return ;
}

POJ 3254

题意:构造一个序列,包括给出的n个子序列,问这个序列的最短长度

思路:第一反应贪心 然鹅是枚举去做状压dp

f[i][j] 表示状态 i 中最后一个序列是 j 的长度,下次更新的时候在它后面接k

把k接在它后面 f[i|(1<<(k-1))][k] = min{f[i][j] + num[k] - arr[j][k]};

细节:预处理每两个数组可以互相覆盖的大小之前要处理一下完全包含的情况

然后。。1<<(i-1) QAQ窝数组下标是从1开始开的哇。。

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cmath>
#include <cstring>
#include <algorithm>
#include <vector>
using namespace std;
const int SZ = ;
const int INF = ;
const int mod = ;
int num[], a[][], arr[][];
int f[SZ][], vis[];
int main()
{
int n;
scanf("%d", &n);
for(int i = ; i <= n; i++)
{
scanf("%d", &num[i]);
for(int j = ; j <= num[i]; j++)
scanf("%d", &a[i][j]);
}
for(int i = ; i <= n; i++)//除去被a[i]包含的串//1 2 3 4 5
{
if(vis[i]) continue;
for(int j = ; j <= n; j++)//看看a[j]有没有被a[i]包含//2 3 4
{
if(i == j || vis[j] || num[i] < num[j]) continue;
for(int k = ; k <= num[i]; k++)
{
if(k + num[j] - > num[i]) break;
bool same = true;
for(int tj = ; tj <= num[j]; tj++)
if(a[i][k+tj-] != a[j][tj]) same = false;
if(same) {vis[j] = ; break;}
}
}
}
int idx = ;
for(int i = ; i <= n; i++)
if(!vis[i])
{
num[++idx] = num[i];
for(int j = ; j <= num[i]; j++)
a[idx][j] = a[i][j];
}
n = idx;
for(int i = ; i <= n; i++)//把j接在i后面
for(int j = ; j <= n; j++)
{
if(i == j) continue;
for(int len = ; len <= num[i]; len++)
{
bool flag = true;
for(int k = ; k <= len; k++)
if(a[i][num[i]-len+k] != a[j][k]) flag = false;
if(flag) arr[i][j] = len;
}
}
memset(f, , sizeof(f));
for(int i = ; i <= n; i++)
f[<<(i-)][i] = num[i];
for(int i = ; i < (<<n); i++)
for(int j = ; j <= n; j++)
for(int k = ; k <= n; k++)//在j后放k
if(i&(<<(j-)) || !i&(<<(k-)))
f[i|(<<(k-))][k] = min(f[i|(<<(k-))][k], f[i][j] + num[k] - arr[j][k]);
int ans = INF;
for(int i = ; i <= n; i++)
ans = min(ans, f[(<<n)-][i]);
printf("%d\n", ans);
return ;
}

BZOJ 1087

题意:在N×N的棋盘里面放K个国王,使他们互不攻击,共有多少种摆放方案。国王能攻击到它一圈的八个地方(1 <=N <=9, 0 <= K <= N * N)

思路:看到这个数据范围就想到状压,每一行的情况都能用一个9位二进制数表示

f[i][j] 表示第 i 行放 j 情况的方法数,从f[i-1][k]转移而来,转移时判断 j 和 k 是否冲突

因为总共的数量有限制,所以再加一维表示到当前放了多少个

c数组表示这种情况下有多少个1

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cmath>
#include <cstring>
#include <algorithm>
using namespace std;
typedef long long LL;
const int SZ = ;
const int INF = ;
const int mod = ;
LL f[][SZ][], st[SZ], c[SZ];
int m, n;
//f[i][j][k] 第i行,状态为st[j],已经放了k个
bool check(int x, int y)
{
bool flag = true;
if(y&x || y&(x<<) || y &(x>>)) flag = false;
return flag;
} int main()
{
scanf("%d %d", &n, &m);
int tot = ;
for(int i = ; i < (<<n); i++)
if(!(i & (i<<)))
{
st[++tot] = i;
for(int j = ; j < n; j++)
if(i&(<<j)) c[tot]++;
}
for(int i = ; i <= tot; i++)
f[][i][c[i]] = ;
for(int i = ; i <= n; i++)
for(int j = ; j <= tot; j++)//枚举第i行的状态
for(int k = ; k <= tot; k++)//枚举第i-1行状态
{
if(check(st[j], st[k]))
{
for(int t = c[j]; t <= m; t++)
f[i][j][t] += f[i-][k][t-c[j]];
}
}
LL ans = ;
for(int i = ; i <= tot; i++)
ans += f[n][i][m];
printf("%lld\n", ans);
return ;
}

最新文章

  1. LINQ to SQL语句(8)之Concat/Union/Intersect/Except
  2. 在SQL Serve里停用行和页层级锁
  3. 关于vue.js中class与style绑定的学习
  4. js动态生成表格
  5. JavaScript:变量对象(Variable Object)
  6. luke 操作记录
  7. codeforces 671C Ultimate Weirdness of an Array 线段树+构造
  8. Linux第三方源
  9. 一个简单的Garbage Collector的实现
  10. 完美解决ie浏览器location.href不刷新页面的问题,进入页面只刷新一次
  11. 【转自知乎】:localhost、127.0.0.1 和 本机IP 三者的区别?
  12. http_build_query()函数使用方法
  13. eclipse打包java项目
  14. LeetCode算法题-Merge Sorted Array(Java实现)
  15. centos7下kubernetes(7.kubernetesScale Up/Down)
  16. (转)IBM mq基本使用
  17. Configure GenieACS
  18. devexpress 之 ChartControl
  19. [转]Java 变量和常量
  20. oracle创建HR示例数据库脚本hr_main.sql分享

热门文章

  1. vmware实现与windows下的共享文件
  2. apache禁止使用IP访问的实现方法
  3. shell脚本只提供整数算术运算(三种方式)—((表达式))、let &quot;表达式&quot;、value=`expr 表达式右边` (转载)
  4. JAVA基础--函数和数组03
  5. html行内要素与块级要素
  6. 51nod 1095【映射】
  7. poj 3294 Life Forms【SA+二分】
  8. 洛谷P2585 [ZJOI2006]三色二叉树(树形dp)
  9. Luogu P2326 AKN&#39;s PPAP【按位贪心】
  10. Cloudera Manager是啥?主要是干啥的?