题解 P2831 【愤怒的小鸟】

题目

我的天,这题是真的卡精度......

主要是精度很不好处理,经本蒟蒻测验,精度在\(10^{-6}\)会比较好优雅


【分析】

对于这种某个变量特别小\((\leq 31)\)的题目,本蒟蒻第一反应就是状压

对于某个抛物线,一定要打到起码一个小猪(不然不如不要这一条抛物线)

有人觉得最少会打掉两只小猪的,可以仔细想一下,万一\(a \geq 0\)呢......

好的,我们继续

那么,我们可以这么考虑,枚举每一只小猪的坐标

首先,有一条抛物线是只过它的

其次,再枚举其他小猪,算出抛物线方程(见下方),若\(a<0\)就记录这条抛物线,否则可以直接跳出

记录完这条抛物线后,我们枚举其他的小猪,查看是否在线上

这里要注意精度问题,如果计算出来的\(y\)与题目所给的\(y\)偏差值不超过\(10^{-6}\),就直接视为同一个值(毕竟计算是有一定的精度问题)

那么对于接下来,我已经预处理过了所有的小鸟的轨迹(抛物线),只需要在状压方程中直接算就可以了

在打\(0\)只小猪的时候,需要用\(0\)只小鸟,于是有:

\(dp[0]=0\)

假设当前状态为\(i\),抛物线为第\(j\)条,抛物线打掉的小猪状态为\(para[j]\),那么有:

\(dp[i|para[j]]=min(dp[i|para[j]],dp[i]+1)\)



接下来我们说一下怎么求\(a\)和\(b\):

假设枚举到两个小猪,坐标分别为\((x_1,y_1)\)和\((x_2,y_2)\),那么就对应地会有:

\(\begin{cases} y_1=ax_1^2+bx_1\\ \\y_2=ax_2^2+bx_2\end{cases}\)

这里有一个很明显的矩阵关系:

\(\left[\begin{matrix}\ y_1\ \\ \\y_2\end{matrix}\right]=\left[\begin{matrix}\ x_1^2&x_1\ \\ \\x_2^2&x_2\end{matrix}\right]\times\left[\begin{matrix}\ a\ \\ \\b\end{matrix}\right]\)

于是有:

\(\left[\begin{matrix}\ a\ \\ \\b\end{matrix}\right]=\left[\begin{matrix}\ x_1^2&x_1\ \\ \\x_2^2&x_2\end{matrix}\right]^{-1}\times\left[\begin{matrix}\ y_1\ \\ \\y_2\end{matrix}\right]\)

又因为:

\(\left|\begin{matrix}\ x_1^2&x_1\\ \\x_2^2&x_2\end{matrix}\right|=x_1^2x_2-x_1x_2^2=(x_1-x_2)x_1x_2\)

所以有:

\(\left[\begin{matrix}\ x_1^2&x_1\ \\ \\ x_2^2&x_2\end{matrix}\right]^{-1}=\frac{1}{(x_1-x_2)x_1x_2}\left[\begin{matrix}\ x_2&-x_1\ \\ \\-x_2^2&x_1^2\end{matrix}\right]\)

(逆矩阵的求法)

所以有:

\(\left[\begin{matrix}\ a\ \\ \\b\end{matrix}\right]={1\over(x_1-x_2)x_1x_2}\left[\begin{matrix}\ x_2&-x_1\ \\ \\-x_2^2&x_1^2\end{matrix}\right]\times\left[\begin{matrix}\ y_1\ \\ \\y_2\end{matrix}\right]\)

即:

\(\begin{cases}a={1\over(x_1-x_2)x_1x_2}\times(x_2y_1-x_1y_2)\\ \\b={1\over(x_1-x_2)x_1x_2}\times(x_1^2y_2-x_2^2y_1)\end{cases}\)



还有一件事

我们对于抛物线,在后面枚举状态的时候是不需要知道除了能打的小猪以外的所有信息

所以直接维护这个信息就可以了,其他都没有必要维护

这个众位大犇可以直接一下本蒟蒻的代码



【代码】

那本蒟蒻就放代码了:

#include<cstdio>
#include<cstring>
using namespace std;
#define min(a,b) ((a<b)?a:b)
inline void built(double &a,double &b,double x1,double y1,double x2,double y2){
    a=(x2*y1-x1*y2)/(x1*x2*(x1-x2));
    b=(x1*x1*y2-x2*x2*y1)/(x1*x2*(x1-x2));
}//计算a,b
inline bool inc(double a,double b,double x,double y){
    double abs=a*x*x+b*x-y;
    if(abs<0) abs=-abs;
    return abs<=0.000001;
}//判定某个小猪是否在抛物线上
inline int read(){
    int ans=0;char c=getchar();bool neg=0;
    while((c<'0')|(c>'9')) { neg^=!(c^'-'); c=getchar(); }
    while((c>='0')&(c<='9')) { ans=(ans<<3)+(ans<<1)+c-'0'; c=getchar(); }
    return neg?-ans:ans;
}//无聊的读入优化
int n,para[200],dp[1<<18],countpara;
inline void pre(){
    memset(dp,0x3f,sizeof(dp));
    countpara=0;
    double x[18]={0},y[18]={0};
    n=read();read();
    for(int i=0;i<n;i++) scanf("%lf %lf",&x[i],&y[i]);
    //dp定义为无限大,抛物线的条数清空,读入
    for(int i=0;i<n;i++){
        para[countpara++]=(1<<i);
        //只打一只小猪的
        for(int j=i+1,vis=0;j<n;j++)//定义vis表示打到过的小猪,避免重复枚举
            if((vis>>j)&1) continue;
            else{
                double a,b;
                built(a,b,x[i],y[i],x[j],y[j]);
                if(a>=0) continue;
                para[countpara]=(1<<i);
                for(int k=j;k<n;k++)//枚举小猪,查看是否在线上
                    if(inc(a,b,x[k],y[k])){
                        vis|=(1<<k);
                        para[countpara]|=(1<<k);
                    }
                countpara++;
            }
    }
}
inline int ans(){//状压
    dp[0]=0;
    for(int i=0;i<(1<<n);i++)
        for(int j=0;j<countpara;j++)
            dp[i|para[j]]=min(dp[i|para[j]],dp[i]+1);
    return dp[(1<<n)-1];
}
int main(){
    int t=read();
    while(t--){
        pre();//先皮一下
        printf("%d\n",ans());
    }
    return 0;
}

最后安利一下 本蒟蒻的博客

原文地址:https://www.cnblogs.com/JustinRochester/p/12243152.html

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

题解 P2831 【愤怒的小鸟】的相关文章

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

P2831 愤怒的小鸟——状压

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

【luogu P2831 愤怒的小鸟】 题解

题目链接:https://www.luogu.org/problemnew/show/P2831 写点做题总结:dp,搜索,重在设计状态,状态设的好,转移起来也方便. 对于一条抛物线,三点确定.(0,0)是固定的,所以我们一条抛物线要用两只猪确定.再多的猪就只能用来判断是不是在这条抛物线上了. 于是我们把猪分成两种:在已有方程里的猪,单独的猪还没有确定方程. 那么对于一只猪,就会有被以前方程覆盖/和前面单独的猪构成新抛物线/自己单独. #include <cmath> #include <

P2831 愤怒的小鸟

传送门 看到数据范围就知道是搜索或状压DP 算了一波复杂度搜索好像过不了极限数据 搞状压 设 f [ i ] 表示所有猪的状态为 i (二进制下1表示死了,0表示没死)时需要的最少发射次数 设 p [ i ] [ j ] 存经过第 i 只猪和第 j 只猪的抛物线经过的猪的状态(可以$n^2$预处理出来,解方程都会吧..) 找到第一个没死的猪 i ,然后枚举所有其他没死的猪 j ,进行转移: f [ i|p [ i ] [ j ] ] = min ( f [ i|p [ i ] [ j ] ],f

【2019.10.18】luogu TG5动态规划进阶

树形dp P1352 没有上司的舞会 P2607 骑士(review) 对于每一个"联通快" 只有根节点有机会形成环 强制不选\(rt\)和\(rt\)的父亲 各跑一遍 P1131 时态同步(review) 贪心 显然增加深度约小的边越优 从下到上来调整 先将同一个点的儿子们延伸到一样 再往上进行一样的操作 //apio 烟火 树上背包? 一棵\(n\)个点的树,有点权.选择一个大小不超过\(K\)的联通块,使得点权和最大.\(n ≤ 2000\) \(f(x, i)\)表示\(x\)

【题解】NOIP2016愤怒的小鸟

一眼n<=18状压dp--方程什么的都很显然,枚举两只小鸟,再将这条抛物线上的小鸟抓出来就好啦.只是这样O(n^3)的dp必然是要TLE的,我一开始这样交上去显然跑得巨慢无比,后来转念一想:面对一个崭新的情况的时候,只有搭配的优劣之分,没有先后的区别,所以最外面的一层可以直接去掉,变成O(n^2)的dp.这样就跑的很快啦~ PS:print()函数只是调试输出,作用是输出now 的二进制形式+dp[now]: #include <bits/stdc++.h> using namespac

[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). 如果某只小鸟的飞行

noip做题记录+挑战一句话题解?

2017 小凯的疑惑 傻逼数论题,打表猜结论就好 比较有追求的可以搞下之前专题分享里面叶佬港数论的时候提到的一个,什么来着忘了QAQ 数论证明待落实 时间复杂度 真 傻逼模拟题 耐心做下就成,注意细节,好像没辽 这道题是真滴恶心,除了逼着本压行选手硬生生打出了156行代码以外没有任何意义:) 逛公园 还没有落实,题目还没看呢QAQ 奶酪 开始看到的时候并没有思路啊感觉,看到了一个并查集的标签但是没有想到?就上去硬艹了个,算是贪心的玩意儿,,,就AC了,,,是数据太水了??? 不管怎么说并查集的解

洛谷 P1079 Vigen&#232;re 密码 题解

此文为博主原创题解,转载时请通知博主,并把原文链接放在正文醒目位置. 题目链接:https://www.luogu.org/problem/show?pid=1079 题目描述 16 世纪法国外交家 Blaise de Vigenère 设计了一种多表密码加密算法――Vigenère 密 码.Vigenère 密码的加密解密算法简单易用,且破译难度比较高,曾在美国南北战争中为 南军所广泛使用. 在密码学中,我们称需要加密的信息为明文,用 M 表示:称加密后的信息为密文,用 C 表示:而密钥是一种