https://www.luogu.org/problem/show?pid=2831

BFS

看到N这么小就可以想到搜索,求最少步数显然应该用BFS。

在这题中过两猪可以唯一确定一条抛物线,每一步可以发射两只猪确定的一条抛物线(打下这条抛物线上的所有猪),也可以发射一条只经过一只猪的抛物线(只打下这只猪)。

这时可以想到状压存储每个状态,并且读入所有猪时预处理一下每两只猪确定的抛物线。

两点确定一条过原点的抛物线y=ax2+bx的方法:点(x1, y1) (x2, y2)过抛物线,得y1=ax12+bx1,y2=ax22+bx2,两式分别变形得y1/x1-ax1=y2/x2-ax2=b,再整理得a=(y2/x2-y1/x1)/(x2-x1)。求出a后往回代可求出b。

记得记录每个状态是否已经搜索过,避免重复的状态入队。由于共有2n个状态,故时间复杂度是O(2n)的。如果不剪枝就是O(n!)。

题目给的m应该是用来xjb剪枝用的,但是m=2怎么用我也不懂……

注意事项:

  • 判断两个浮点数相等要考虑精度误差。
  • stl的queue不开优化会很慢,可以考虑手写队列。
#include <algorithm>
#include <cstring>
#include <iostream>
#include <queue>
#include <vector>
#include <cmath>
using namespace std;
bool equal(double x, double y, double eps = 1e-)
{
return fabs(x - y) <= eps;
}
int n;
bool m[];
pair<double, double> pigs[];
unsigned target; // 搜索的最终目标,即第1~n位均为1
unsigned pwxs[][]; // pwxs[i][j] => i、j两只猪所在抛物线上的所有猪
bool visited[( << )];
int bfs()
{
typedef pair<unsigned, int> state; // <状态, 步数>
queue<state> q;
q.push(make_pair(, ));
while (!q.empty())
{
state x = q.front();
q.pop(); // 用一只鸟打一只猪的情况
for (int i = ; i <= n; i++)
{
if (!(x.first & ( << i))) // 如果i猪还没被打下
{
state y = x;
y.first |= ( << i);
y.second++;
if (y.first == target)
return y.second;
if (!visited[y.first] && !(m[] && y.second > (int)(n * 1.0 / + )))
{
q.push(y);
visited[y.first] = true;
}
}
} // 用一只鸟打两只猪的情况
for (int i = ; i <= n; i++)
{
for (int j = i + ; j <= n; j++)
{
if (!(x.first & ( << i)) && !(x.first & ( << j))) // 如果i、j猪都还没被打下
{
state y = x;
y.first |= pwxs[i][j];
y.second++;
if (y.first == target)
return y.second;
if (!visited[y.first] && !(m[] && y.second > (int)(n * 1.0 / + )))
{
q.push(y);
visited[y.first] = true;
}
}
}
}
}
return -;
}
int main()
{
ios::sync_with_stdio(false);
int t;
cin >> t;
while (t--)
{
memset(visited, false, << ); int c;
cin >> n >> c;
m[c] = true; for (int i = ; i <= n; i++)
for (int j = ; j <= n; j++)
pwxs[i][j] = ; target = ;
for (int i = ; i <= n; i++)
target |= ( << i); double a, b;
for (int i = ; i <= n; i++)
{
cin >> a >> b;
pigs[i] = make_pair(a, b);
} for (int i = ; i <= n; i++)
{
for (int j = i + ; j <= n; j++)
{
/* ∵ y1=ax1^2+bx1, y2=ax2^2+bx2
∴ b=y1/x1-ax1=y2/x2-ax2
∴ a=(y2/x2-y1/x1)/(x2-x1) */
double &x1 = pigs[i].first, &y1 = pigs[i].second;
double &x2 = pigs[j].first, &y2 = pigs[j].second;
a = (y2 / x2 - y1 / x1) / (x2 - x1);
b = y1 / x1 - a * x1;
if (a < )
{
for (int k = ; k <= n; k++)
{
double &x = pigs[k].first, &y = pigs[k].second;
if (equal(a * x * x + b * x, y))
pwxs[i][j] |= ( << k);
}
}
}
}
cout << bfs() << endl;
}
return ;
}

看到N这么小就可以想到搜索,求最少步数显然应该用BFS。

在这题中过两猪可以唯一确定一条抛物线,每一步可以发射两只猪确定的一条抛物线(打下这条抛物线上的所有猪),也可以发射一条只经过一只猪的抛物线(只打下这只猪)。

这时可以想到状压存储每个状态,并且读入所有猪时预处理一下每两只猪确定的抛物线。

记得记录每个状态是否已经搜索过,避免重复的状态入队。

【NOIP2016提高组】愤怒的小鸟的更多相关文章

  1. [NOIp2016提高组]愤怒的小鸟

    题目大意: 平面直角坐标系的第一象限有n(n<=18)个点,你可以每次给出一个形如y=ax^2+bx的函数把经过这条函数的点消掉,问消掉所有的点至少要多少函数? 思路: 枚举每两个点对,可以唯一 ...

  2. 【题解】NOIP2016提高组 复赛

    [题解]NOIP2016提高组 复赛 传送门: 玩具谜题 \(\text{[P1563]}\) 天天爱跑步 \(\text{[P1600]}\) 换教室 \(\text{[P1850]}\) 组合数问 ...

  3. 【题解】NOIP2016 提高组 简要题解

    [题解]NOIP2016 提高组 简要题解 玩具迷题(送分) 用异或实现 //@winlere #include<iostream> #include<cstdio> #inc ...

  4. NOIP2016提高组解题报告

    NOIP2016提高组解题报告 更正:NOIP day1 T2天天爱跑步 解题思路见代码. NOIP2016代码整合

  5. [日记&做题记录]-Noip2016提高组复赛 倒数十天

    写这篇博客的时候有点激动 为了让自己不颓 还是写写日记 存存模板 Nov.8 2016 今天早上买了两个蛋挞 吃了一个 然后就做数论(前天晚上还是想放弃数论 但是昨天被数论虐了 woc noip模拟赛 ...

  6. 【NOIP2016提高组】 Day2 T3 愤怒的小鸟

    题目传送门:https://www.luogu.org/problemnew/show/P2831 说个题外话:NOIP2014也有一道题叫做愤怒的小鸟. 这题自测时算错了eps,导致被卡了精度,从1 ...

  7. NOIP2016提高组复赛C 愤怒的小鸟

    题目链接:http://uoj.ac/problem/265 题目大意: 太长了不想概括... 分析: 状压DP的模板题,把所有可能的抛物线用二进制表示,然后暴力枚举所有组合,详情见代码内注释 代码如 ...

  8. 【NOIP2016提高组day2】愤怒的小鸟

    分析 Kiana最近沉迷于一款神奇的游戏无法自拔. 简单来说,这款游戏是在一个平面上进行的. 有一架弹弓位于 (0, 0) 处,每次Kiana可以用它向第一象限发射一只红色的小鸟, 小鸟们的飞行轨迹均 ...

  9. 【noip2016提高组day2T3】【愤怒的小鸟】状压dp转移时的集合包含

    (上不了p站我要死了,图来自百度,侵权度娘背锅) 调死我了... 标题就说明了,死在了集合包含上.因为这道题与其他的状压题不同,其他的题基本上都是要求集合不重合,而这道题完全是可以的. 废话不多说,先 ...

随机推荐

  1. HDU 5182

    #include <iostream> #include <algorithm> #include <cstring> using namespace std; / ...

  2. ajax跨域请求解决方案

    大家好,今天我们学习了js的跨域请求的解决方案,由于JS中存在同源策略,当请求不同协议名,不同端口号.不同主机名下面的文件时,将会违背同源策略,无法请求成功!需要进行跨域处理! 方案一.后台PHP进行 ...

  3. PHP(一)OOP基础

    [面向过程&面向对象] 1.面向过程:专注于解决一件事情的过程.最大的特点,是有一个个函数来实现功能需求 2.面向对象:专注于有哪一个对象来实现这个功能,最大的特点,时产生一个个具有属性和方法 ...

  4. kettle介绍

    Kettle也叫PDI,在2006年Kettle加入了开源的BI组织Pentaho,正式命名为PDI,英文全称为Pentaho Data Integeration.Kettle是"Kettl ...

  5. vue初级学习--控制台创建vue项目

    一.导语 困困的我,好想睡啊,但还是敲下这边有的没的的笔记吧 二.正文 1.输入 vue list 可查看此环境下,vue能创建的项目模板 没将webpack.vue-route啃精了的话,老实用we ...

  6. Apple 公司开发者账号添加团队成员

    html,body,div,span,applet,object,iframe,h1,h2,h3,h4,h5,h6,p,blockquote,pre,a,abbr,acronym,address,bi ...

  7. [J2EE] 有关 PreparedStatement

    今天同事遇到一个问题,简言之,就是PreparedStatement的预编译究竟是怎么发挥作用的... 嘿嘿,说来惭愧,我以前就只知道PreparedStatement比Statement要好,要防S ...

  8. 约瑟夫环C#解决方法

    /*约瑟夫环 (问题描述) 约瑟夫问题的一种描述是:编号为1,2,......n,的n个人按顺时针方向围坐一圈,每个人持有一个密码(正整数).一开始任意选 一个正整数作为报数的上限值m,从第一个人开始 ...

  9. Python字符编码详解(转)

    http://www.cnblogs.com/huxi/archive/2010/12/05/1897271.html 在没有reload(sys)之前调用sys.setdefaultencoding ...

  10. [转]动态管理视图和函数 (Transact-SQL)

    动态管理视图和函数返回可用于监视服务器实例的运行状况.诊断故障以及优化性能的服务器状态信息. 重要提示 动态管理视图和函数返回特定于实现的内部状态数据. 在未来的 SQL Server 版本中,它们的 ...