HDU2295 Radar (DLX)
下面的代码99%参考了这个网站http://www.cnblogs.com/183zyz/archive/2011/08/07/2130193.html
人生的第一道DLX肯定是需要作一些参考的啦。
题意:给你N个城市,M个雷达,你要在其中选K个,问当半径最小是多少的时候可以覆盖到所有的N个城市。
做法:二分所需要的半径r,然后处理出这时雷达能够覆盖到哪些位置。然后就转化成了一个重复覆盖问题,跑一下DLX即可。
花了两天的时间学习了一下DLX这种神奇的数据结构,它在解决精确覆盖问题上尤其有帮助,因为随着每次选的一行,对应的包含了这个行的列的行也不会再用到,所以整个图会很快的变得非常小,从而可以极大得加快搜索速度。但是重复覆盖就没有缩小得那么快,包含了对应的列的行是不会消除掉的。
下面对模板的一些数组说下理解 L,R,U,D就是链表的四个方向,其中H是行的头指针,当该行不存在东西的时候H[]=-1,而col[size]表示结点编号为size所在的是哪一列,在精确覆盖的时候还要多维护一个row[size]去表示是在哪一行。对应的link函数是用来加条件的,然后dance就是递归,remove和resume则是对应的删除和恢复,在重复覆盖和精确覆盖问题上这两个操作要做相应的修改。
题目了用了个A*,多亏我有数据结构大作业的基础,A*就是在搜的时候多了个估价函数,但是这个估价函数必须保证不能高估了可能的解,譬如解是6的话你不能估成7,这样的估价函数是不行的。下面给的h函数是一定估小的,虽然我觉得可能优化的不是特别的大。
细节等别的地方后面再慢慢学习,先照着别人的拍一题加深一下认识。。
#pragma warning(disable:4996)
#include<iostream>
#include<cstring>
#include<string>
#include<vector>
#include<cstdio>
#include<cmath>
#include<algorithm>
#include<queue>
using namespace std; #define maxnode 4000
#define maxn 80
#define eps 1e-8
int L[maxnode], R[maxnode], U[maxnode], D[maxnode];
int col[maxnode];
int S[maxn], H[maxn];
int size;
int n, m, K; int ak; // currently best dep void link(int r, int c)
{
S[c]++; col[size] = c;
U[size] = U[c]; D[U[c]] = size;
D[size] = c; U[c] = size; if (H[r] == -1) H[r] = L[size] = R[size] = size;
else{
L[size] = L[H[r]]; R[L[H[r]]] = size;
R[size] = H[r]; L[H[r]] = size;
}
size++;
} void remove(int c)
{
for (int i = D[c]; i != c; i = D[i]){
L[R[i]] = L[i]; R[L[i]] = R[i];
}
} void resume(int c)
{
for (int i = U[c]; i != c; i = U[i]){
L[R[i]] = R[L[i]] = i;
}
}
// evaluating function
int hfunc()
{
bool vis[maxn];
memset(vis, 0, sizeof(vis));
int cnt = 0;
for (int i = R[0]; i; i = R[i]){
if (vis[i]) continue;
cnt++; vis[i] = true;
for (int j = D[i]; j != i; j = D[j]){
for (int k = R[j]; k != j; k = R[k]){
vis[col[k]] = true;
}
}
}
return cnt;
} void dance(int dep)
{
int ans = hfunc();
if (ans + dep > K || ans + dep >= ak) return;
if (R[0] == 0){
if (dep < ak) ak = dep; return;
}
int minv = maxn; int c;
for (int i = R[0]; i; i = R[i]){
if (S[i] < minv) minv = S[i], c = i;
}
//枚举删除掉的每一行
for (int i = D[c]; i != c; i = D[i]){
remove(i);
// 将该行上的对应的列也删掉
for (int j = R[i]; j != i; j = R[j]){
remove(j);
}
dance(dep + 1);
for (int j = L[i]; j != i; j = L[j]){
resume(j);
}
resume(i);
}
return;
} double x[100], y[100];
double xx[100], yy[100]; int dcmp(double x){
return (x > eps) - (x < -eps);
} double dist(double x1, double y1, double x2, double y2){
return sqrt((x1 - x2)*(x1 - x2) + (y1 - y2)*(y1 - y2));
} int main()
{
int T; cin >> T;
while (T--)
{
scanf("%d%d%d", &n, &m, &K);
for (int i = 1; i <= n; i++){
scanf("%lf%lf", &x[i], &y[i]);
}
for (int i = 1; i <= m; i++){
scanf("%lf%lf", &xx[i], &yy[i]);
}
double l = 0, r = 1500;
while (dcmp(r - l) > 0){
for (int i = 0; i <= n; i++){
S[i] = 0; U[i] = D[i] = i;
L[i + 1] = i; R[i] = i + 1;
}
R[n] = 0;
memset(H, -1, sizeof(H)); size = n + 1; double mid = (l + r) / 2;
for (int i = 1; i <= m; i++){
for (int j = 1; j <= n; j++){
if (mid>=dist(xx[i], yy[i], x[j], y[j])){
link(i, j);
}
}
}
ak = maxn;
dance(0);
if (ak <= K) r = mid;
else l = mid;
}
printf("%.6lf\n", l);
}
return 0;
}
HDU2295 Radar (DLX)的更多相关文章
- HDU2295 Radar —— Dancing Links 可重复覆盖
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2295 Radar Time Limit: 2000/1000 MS (Java/Others) ...
- HDU 2295 Radar (DLX + 二分)
Radar Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Submi ...
- HDU 2295.Radar (DLX重复覆盖)
2分答案+DLX判断可行 不使用的估计函数的可重复覆盖的搜索树将十分庞大 #include <iostream> #include <cstring> #include < ...
- 【转】DLX 精确覆盖 重复覆盖
问题描述: 给定一个n*m的矩阵,有些位置为1,有些位置为0.如果G[i][j]==1则说明i行可以覆盖j列. Problem: 1)选定最少的行,使得每列有且仅有一个1. 2)选定最少的行,使得每列 ...
- dancing link 精确覆盖 重复覆盖 (DLX)
申明:因为转载的没有给出转载链接,我就把他的链接附上,请尊重原创: http://www.cnblogs.com/-sunshine/p/3358922.html 如果谁知道原创链接 给一下,请尊重原 ...
- Dancing Links 专题总结
算法详细:Dancing Links博客 1.精确覆盖: ZOJ3209 Treasure Map HUST1017 Exact cover POJ3074 Sudoku 2.可重复覆盖: HDU22 ...
- 搜索(DLX重复覆盖模板):HDU 2295 Radar
Radar Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Submi ...
- [ACM] HDU 2295 Radar (二分法+DLX 重复覆盖)
Radar Problem Description N cities of the Java Kingdom need to be covered by radars for being in a s ...
- HDU 2295 Radar 重复覆盖 DLX
题意: N个城市,M个雷达站,K个操作员,问雷达的半径至少为多大,才能覆盖所有城市.M个雷达中最多只能有K个同时工作. 思路: 二分雷达的半径,看每个雷达可以覆盖哪些城市,然后做重复覆盖,判断这个半径 ...
随机推荐
- Python学习教程(learning Python)--3.3.2 Python的关系运算
如果if的condition不用布尔表达式来做条件判断而采用关系表达式,实际上关系表达式运算的结果要么是True要么是False.下面我们先了解一些有关关系运算符的基础知识,如下表所示. 做个小程序测 ...
- 改用二进制启动Moses translation model提示Can't read ~/working/binarised-model/reordering-table
解决方案: 换成 /home/用户名/working 貌似就好使了...但是时间还是估计太长,明早挂机一天试试,顺便把manual 的详细部分看了
- android Init 相关分析
Init.c主要工作 1. 初始化属性(包括建立/dev./proc等目录.初始化属性.log.执行init.rc等初始化文件中的action等). 2. 解析配置文件的命令(主要是init.rc文件 ...
- 九度oj 1184 二叉树遍历
原题链接:http://ac.jobdu.com/problem.php?pid=1184 简单的二叉树重建,遍历. 如下: #include<cstdio> #include<cs ...
- 触摸屏校准tslib的配置文件
./autogen.sh#sleep 10./configure --prefix=/usr/lxl/tslib --host=arm-linux CC=arm-linux-gcc#sleep 100 ...
- FastDFS安装配置
FastDFS FastDFS为互联网量身定制,充分考虑了冗余备份.负载均衡.线性扩容等机制,并注重高可用.高性能等指标,使用FastDFS很容易搭建一套高性能的文件服务器集群提供文件上传.下载等服务 ...
- 怎样把php数组转换成字符串,php implode()
实例代码 一维数组转换成字符串代码! <?php $arr1=array("shu","zhu","1"); $c=implode(& ...
- 四则运算2+psp0级表格
四则运算2 一.题目和要求 题目:写一个能自动生成小学四则运算题目的程序,要求一次输出不少于30道,只能是整数100以内的四则运算(四则运算1升级版) 要求: 1.题目避免重复 2.可定制(数量/打印 ...
- 主元分析PCA理论分析及应用
首先,必须说明的是,这篇文章是完完全全复制百度文库当中的一篇文章.本人之前对PCA比较好奇,在看到这篇文章之后发现其对PCA的描述非常详细,因此迫不及待要跟大家分享一下,希望同样对PCA比较困惑的朋友 ...
- elasticsearch中的mapping映射配置与查询典型案例
elasticsearch中的mapping映射配置与查询典型案例 elasticsearch中的mapping映射配置示例比如要搭建个中文新闻信息的搜索引擎,新闻有"标题".&q ...