【dp】状压dp
二进制的力量
状态压缩DP
愤怒的小鸟
第一次接触状态压缩DP是在NOIP2016的愤怒的小鸟,当时菜得连题目都没看懂,不过现在回过头来看还是挺简单的,那么我们再来看看这道题吧。
考虑预处理出两个点构成的抛物线,因为经过原点,所以对于二次函数
ax2+bx+c
因此已知两个点 (x1,y1),(x2,w2)可得出
a=(y2/x2+y1/x1)/(x2-x1)
b=(y2-ax22)/x2
所以思路也就自然而然的来了,枚举两个点,求出它们所构成的抛物线,再枚举其它的点,看这条抛物线经过另外的哪些点。
然后问题又来了,该如何存储这条抛物线上的信息?
思路1:储存这条抛物线上的个数
这应该是最好想到的一条思路了,但也很快可以否决掉,会发现只存个数的话会无法判断是否打完。
思路2:对每条抛物线开一个数组
这样做的正确性无法否定,可是你的空间呢?
思路3:二进制信息存储
有的时候暴力的思想也是很重要的,这往往预示着正解。
既然我们只要表示某个点的存在或否,我们何不用2进制的一位来表示呢?
这便是状态压缩的核心思想了,对于简单的状态存储,重新开数组太过浪费,可以用一个二进制数来代替数组。
假设我们现在有5个点,那么一条抛物线的初始状态是什么也没有,用一个五位的二进制数表示即 00000。
现在假设我们发现第3个点在这条抛物线上,即要让它变成00100 要怎么办呢。
不难想到我们只要把 1 左移 2 位,再or上去就行了。
因此每当我们发现一个编号为 i 的点,假设表示抛物线的数为f,那么我们可以写出如下运算式:
f=f|2i-1
这样就可以预处理出所有的抛物线了,
然后有了抛物线,状态转移方程也不难写出来。
设dpi表示当前状态最小需要的最小猪,则:dpi|f=min { dpi+1}
其中f为抛物线。
#include<cstdio>
#include<cstring>
#include<cmath>
#define maxm 1000+10
#define maxn 1000+10
#define min(a,b) (a<b?a:b)
using namespace std;
int f[(1<<20)],get[20][20];
double x[20],y[20];
template<class T>
inline bool read(T&n,char ch=getchar(),int sign=1){
    if(ch==EOF)return 0;for(n=0;(ch<'0'||ch>'9')&&ch!='-';ch=getchar());
    if(ch=='-')sign=-1,ch=getchar();for(;ch>='0'&&ch<='9';ch=getchar())
    (n*=10)+=(ch-'0');n*=sign;return 1;
}
inline bool equ(double a,double b){
    return abs(a-b)<10e-8;
}
int main(){
    #ifdef Files
        freopen("f.in","r",stdin);
        freopen("f.out","w",stdout);
    #endif
    #ifdef die
    for(int i=1;i<=19;i++) printf("%d ",un[i]);
    #endif
    int T;read(T);
    for(int i=1;i<=T;i++){
        int n,m,tot=0;read(n),read(m);
        memset(get,0,sizeof(get));
        for(int i=1;i<=n;i++) scanf("%lf%lf",&x[i],&y[i]);
        for(int i=1;i<=n;i++)
            for(int j=i+1;j<=n;j++){
                if(equ(x[i],x[j]))continue;
                double a=(y[j]/x[j]-y[i]/x[i])/(x[j]-x[i]);
                 double b=(y[j]-a*x[j]*x[j])/x[j];
                if(a>-1e-8) continue;
                for(int k=1;k<=n;k++)
                    if(equ(a*x[k]*x[k]+b*x[k],y[k])) get[i][j]|=1<<(k-1);
            }
        memset(f,127,sizeof(f));f[0]=0;
        for(int i=0;i<=(1<<n)-1;i++)
            for(int j=1;j<=n;j++)
                if(!((1<<j-1)&i)){
                    for(int k=j+1;k<=n;k++)
                        f[i|get[j][k]]=min(f[i]+1,f[i|get[j][k]]);
                    f[i|1<<(j-1)]=min(f[i]+1,f[i|1<<(j-1)]);
                }
        printf("%d\n",f[(1<<n)-1]);
    }
}
【dp】状压dp的更多相关文章
- 【BZOJ】1076 [SCOI2008]奖励关 期望DP+状压DP
		[题意]n种宝物,k关游戏,每关游戏给出一种宝物,可捡可不捡.每种宝物有一个价值(有负数).每个宝物有前提宝物列表,必须在前面的关卡取得列表宝物才能捡起这个宝物,求期望收益.k<=100,n&l ... 
- CCF 201312-4	有趣的数 (数位DP, 状压DP, 组合数学+暴力枚举, 推公式, 矩阵快速幂)
		问题描述 我们把一个数称为有趣的,当且仅当: 1. 它的数字只包含0, 1, 2, 3,且这四个数字都出现过至少一次. 2. 所有的0都出现在所有的1之前,而所有的2都出现在所有的3之前. 3. 最高 ... 
- hdu 4352 "XHXJ's LIS"(数位DP+状压DP+LIS)
		传送门 参考博文: [1]:http://www.voidcn.com/article/p-ehojgauy-ot.html 题解: 将数字num字符串化: 求[L,R]区间最长上升子序列长度为 K ... 
- [转]状态压缩dp(状压dp)
		状态压缩动态规划(简称状压dp)是另一类非常典型的动态规划,通常使用在NP问题的小规模求解中,虽然是指数级别的复杂度,但速度比搜索快,其思想非常值得借鉴. 为了更好的理解状压dp,首先介绍位运算相关的 ... 
- 状态压缩dp 状压dp 详解
		说到状压dp,一般和二进制少不了关系(还常和博弈论结合起来考,这个坑我挖了还没填qwq),二进制是个好东西啊,所以二进制的各种运算是前置知识,不了解的话走下面链接进百度百科 https://baike ... 
- 洛谷 P3343 - [ZJOI2015]地震后的幻想乡(朴素状压 DP/状压 DP+微积分)
		题面传送门 鸽子 tzc 竟然来补题解了,奇迹奇迹( 神仙题 %%%%%%%%%%%% 解法 1: 首先一件很明显的事情是这个最小值可以通过类似 Kruskal 求最小生成树的方法求得.我们将所有边按 ... 
- 51nod 1673 树有几多愁(链表维护树形DP+状压DP)
		题意 lyk有一棵树,它想给这棵树重标号. 重标号后,这棵树的所有叶子节点的值为它到根的路径上的编号最小的点的编号. 这棵树的烦恼值为所有叶子节点的值的乘积. lyk想让这棵树的烦恼值最大,你只需输出 ... 
- BZOJ3836  [Poi2014]Tourism  【树形dp +状压dp】
		题目链接 BZOJ3836 题解 显然这是个\(NP\)完全问题,此题的解决全仗任意两点间不存在节点数超过10的简单路径的性质 这意味着什么呢? \(dfs\)树深度不超过\(10\) \(10\)很 ... 
- SCUT - 254 - 欧洲爆破 - 概率dp - 状压dp
		https://scut.online/p/254 思路很清晰,写起来很恶心. #include<bits/stdc++.h> using namespace std; #define l ... 
- bzoj1076: [SCOI2008]奖励关(期望dp+状压dp)
		1076: [SCOI2008]奖励关 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 2989 Solved: 1557[Submit][Statu ... 
随机推荐
- 全套Project版本安装教程及下载地址
			1:Project 2007 安装教程及下载地址 https://mp.weixin.qq.com/s/8iI7x1qjon0yAdo3bStjzw 2:Project 2010 安装教程及下载地址 ... 
- Spring核心结构及核心思想
			Spring核心结构 基本概念 Spring是⼀个分层⾮常清晰并且依赖关系.职责定位⾮常明确的轻量级框架,主要包括⼏个⼤模块:数据处理模块.Web模块.AOP(Aspect Oriented Prog ... 
- 从几道题目带你深入理解Event Loop_宏队列_微队列
			目录 深入探究JavaScript的Event Loop Event Loop的结构 回调队列(callbacks queue)的分类 Event Loop的执行顺序 通过题目来深入 深入探究Java ... 
- SE_WorkX_提问回顾与个人总结
			项目 内容 课程:北航-2020-春-软件工程 博客园班级博客 要求:正所谓"实践是认识的来源.目的.动力以及检验认识真理性的唯一标准",在经历了一个学期的学习和实践后,请大家写一 ... 
- [linux] Git基本概念&操作
			1.基本概念 版本控制系统:一种软体工程技巧,籍以在开发的过程中,确保由不同人所编写的同一项目代码都得到更新.并追踪.记录整个开发过程. 集中式(SVN)/ 分布式(GIT)版本控制系统:SVN的版本 ... 
- Ansible_使用jinja2模板部署自定义文件
			一.jinja2简介 1.jinja2模板 1️⃣:Ansible将jinja2模板系统用于模板文件,Ansible还使用jinja2语法来引用playbook中的变量 2️⃣:变量和逻辑表达式置于标 ... 
- Linux_防火墙与SElinux
			一.防火墙与SElinux 1.防火墙和selinux 防火墙 iptables 默认允许所以 firewalld 默认拒绝所有 ebtables 不认识,不管 se ... 
- SystemVerilog 语言部分(二)
			接口interface: 既可以设计,也可以用来验证. 验证环境:interface使得连接变得简单不容易出错. interface可以定义端口,单双向信号,内控部使用initial always t ... 
- 为什么选择b+树作为存储引擎索引结构
			为什么选择b+树作为存储引擎索引结构 在数据库或者存储的世界里,存储引擎的角色一直处于核心位置.往简单了说,存储引擎主要负责数据如何读写.往复杂了说,怎么快速.高效的完成数据的读写,一直是存储引擎要解 ... 
- 在命令行模式下查看Python帮助文档---dir、help、__doc__
			在命令行模式下查看Python帮助文档---dir.help.__doc__ 1.dir函数式可以查看对象的属性,使用方法很简单,举str类型为例,在Python命令窗口输入 dir(str) 即 ... 
