P2831 愤怒的小鸟(状压dp)

P2831 愤怒的小鸟

我们先预处理出每个猪两两之间(设为$u,v$)和原点三点确定的抛物线(当两只猪横坐标相等时显然无解)

处理出$u,v$确定的抛物线一共可以经过多少点,记为$lines[u][v]$

设$f[i]$表示已经被消灭的猪的集合为二进制表示为$i$时,需要的最小抛物线数

显然$f[0]=0$

$f[i|(1<<(u-1))]=min(f[i|(1<<(u-1)],f[i]+1)$(一条抛物线只串一个点)

$f[i|lines[u][v]]=min(f[i|lines[u][v]],f[i]+1)$

然鹅这是$O(Tn^{2}2^{n})$,ccf的老爷机会T

那么我们考虑优化

我们发现加上抛物线时,先串$1,4$与先串$2,3$没有区别,但是我们两种都用不同的顺序枚举了一遍。

那么我们可以处理出集合$i$的$mex$(在集合中没有出现的最小正整数)

枚举时加上限制条件:一定要包含$mex[i]$

于是枚举就从$O(n^{2})$降到了$O(n)$

总复杂度就降到了$O(Tn2^{n})$

end.

#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#define re register
using namespace std;
typedef double db;
int min(int &a,int &b){return a<b?a:b;}
#define N 524589
const db eps=1e-8;
db x[25],y[25];
int t,n,m,lines[25][25],f[N],mex[N];
void calc(db &a,db &b,db x1,db y1,db x2,db y2){
    b=(x1*x1*y2-x2*x2*y1)/(x1*x1*x2-x1*x2*x2);
    a=(y1-x1*b)/(x1*x1);
}//用于解一元二次方程组
int main(){
    for(re int i=0,j;i<262144;++i){//预处理mex
        for(j=0;(i&(1<<j))&&j<18;++j);
        mex[i]=j;
    }scanf("%d",&t);
    while(t--){
        scanf("%d%d",&n,&m); db a,b;
        for(re int i=1;i<=n;++i)
            scanf("%lf%lf",&x[i],&y[i]);
        for(re int i=1;i<=n;++i)
            for(re int j=1;j<=n;++j){
                lines[i][j]=0;
                if(fabs(x[i]-x[j])<eps) continue;//横坐标相等无解
                calc(a,b,x[i],y[i],x[j],y[j]);
                if(a>-eps) continue;
                for(re int u=1;u<=n;++u)
                    if(fabs(a*x[u]*x[u]+b*x[u]-y[u])<eps)
                        lines[i][j]|=(1<<(u-1));//找到这条抛物线能连到的所有点
            }
        memset(f,63,sizeof(f)); f[0]=0;
        for(re int i=0,j=mex[i];i<(1<<n);j=mex[++i]){
            f[i|(1<<j)]=min(f[i|(1<<j)],f[i]+1); //单个点用掉一条的情况
            for(re int u=1;u<=n;++u)//枚举的抛物线必须穿过j
                f[i|lines[j+1][u]]=min(f[i|lines[j+1][u]],f[i]+1);
        }
        printf("%d\n",f[(1<<n)-1]);
    }return 0;
}

原文地址:https://www.cnblogs.com/kafuuchino/p/9823485.html

时间: 2024-07-31 04:28:37

P2831 愤怒的小鸟(状压dp)的相关文章

P2831 愤怒的小鸟——状压

P2831 愤怒的小鸟 抛物线过原点,只要再找两个就能确定抛物线: 处理出两两之间的抛物线能过哪些点,状态压缩: 但是直接枚举每一条抛物线常数太大会T,所以我们需要预处理一个low_bit表示当前状态下第一个没选的,即是二进制下第一个不是1的位置: 因为我们早晚都要把它变成1,所以先处理他就可以达到要求: 注意精度问题: #include<cmath> #include<cstdio> #include<cstring> #include<algorithm>

[NOIP2016]愤怒的小鸟 D2 T3 状压DP

[NOIP2016]愤怒的小鸟 D2 T3 Description Kiana最近沉迷于一款神奇的游戏无法自拔. 简单来说,这款游戏是在一个平面上进行的. 有一架弹弓位于(0,0)处,每次Kiana可以用它向第一象限发射一只红色的小鸟,小鸟们的飞行轨迹均为形如y=ax2+bx的曲线,其中a,b是Kiana指定的参数,且必须满足a<0. 当小鸟落回地面(即x轴)时,它就会瞬间消失. 在游戏的某个关卡里,平面的第一象限中有n只绿色的小猪,其中第i只小猪所在的坐标为(xi,yi). 如果某只小鸟的飞行

ZOJ3305Get Sauce 状压DP,

状压DP的题目留个纪念,首先题意一开始读错了,搞了好久,然后弄好了,觉得DFS可以,最后超时,修改了很久还是超时,没办法看了一下n的范围,然后觉得状压可以,但是没有直接推出来,就记忆化搜索了一下,可是一直错,莫名奇妙,然后没办法看了一下题解,发现了下面这个比较好的方法,然后按照这个方程去推,然后敲,也是WA了好多把,写的太搓了,没人家的清楚明了,唉~也算是给自己留个纪念,状压一直做的都不太好~唉~还好理解了, 参考了  http://blog.csdn.net/nash142857/articl

poj 2411 Mondriaan&#39;s Dream(状压DP)

Mondriaan's Dream Time Limit: 3000MS   Memory Limit: 65536K Total Submissions: 12232   Accepted: 7142 Description Squares and rectangles fascinated the famous Dutch painter Piet Mondriaan. One night, after producing the drawings in his 'toilet series

(状压dp)uva 10817 Headmaster&#39;s Headache

题目地址 1 #include <bits/stdc++.h> 2 typedef long long ll; 3 using namespace std; 4 const int MAX=1e5+5; 5 const int INF=1e9; 6 int s,m,n; 7 int cost[125]; 8 //char sta[MAX]; 9 string sta; 10 int able[125]; 11 int dp[125][1<<8][1<<8]; 12 in

HDU5816 Hearthstone(状压DP)

题目 Source http://acm.hdu.edu.cn/showproblem.php?pid=5816 Description Hearthstone is an online collectible card game from Blizzard Entertainment. Strategies and luck are the most important factors in this game. When you suffer a desperate situation an

HDU 4336 容斥原理 || 状压DP

状压DP :F(S)=Sum*F(S)+p(x1)*F(S^(1<<x1))+p(x2)*F(S^(1<<x2))...+1; F(S)表示取状态为S的牌的期望次数,Sum表示什么都不取得概率,p(x1)表示的是取x1的概率,最后要加一因为有又多拿了一次.整理一下就可以了. 1 #include <cstdio> 2 const int Maxn=23; 3 double F[1<<Maxn],p[Maxn]; 4 int n; 5 int main() 6

Travel(HDU 4284状压dp)

题意:给n个城市m条路的网图,pp在城市1有一定的钱,想游览这n个城市(包括1),到达一个城市要一定的花费,可以在城市工作赚钱,但前提有工作证(得到有一定的花费),没工作证不能在该城市工作,但可以走,一个城市只能工作一次,问pp是否能游览n个城市回到城市1. 分析:这个题想到杀怪(Survival(ZOJ 2297状压dp) 那个题,也是钱如果小于0就挂了,最后求剩余的最大钱数,先求出最短路和 Hie with the Pie(POJ 3311状压dp) 送披萨那个题相似. #include <

BZOJ 1087: [SCOI2005]互不侵犯King( 状压dp )

简单的状压dp... dp( x , h , s ) 表示当前第 x 行 , 用了 h 个 king , 当前行的状态为 s . 考虑转移 : dp( x , h , s ) = ∑ dp( x - 1 , h - cnt_1( s ) , s' ) ( s and s' 两行不冲突 , cnt_1( s ) 表示 s 状态用了多少个 king ) 我有各种预处理所以 code 的方程和这有点不一样 ------------------------------------------------

BZOJ 1072 排列 状压DP

题意:链接 方法:状压DP? 题解:这题其实没啥好写的,不算很难,推一推就能搞出来. 首先看到这个问题,对于被d整除这个条件,很容易就想到是取余数为0,所以想到可能状态中刚开始含有取余数. 先说我的第一个想法,f[i][j]表示选取i个数且此时的mod为j,这样的思想是第一下蹦出来的,当时想的就是在线来搞最终的答案.不过转瞬即发现,这TM不就是暴力吗魂淡!并没有什么卵用,于是开始想这个状态可不可以做什么优化. 显然第二维的j并不需要太大的优化,暂且先将其搁置一边,来考虑第一维的i怎么优化. 把滚