zoukankan      html  css  js  c++  java
  • HDU 3622 Bomb Game(2-sat)

    HDU 3622 Bomb Game

    题目链接

    题意:求一个最大半径,使得每一个二元组的点任选一个,能够得到全部圆两两不相交

    思路:显然的二分半径,然后2-sat去判定就可以

    代码:

    #include <cstdio>
    #include <cstring>
    #include <cstdlib>
    #include <vector>
    #include <cmath>
    #include <algorithm>
    using namespace std;
    
    const int MAXNODE = 105;
    
    struct TwoSet {
    	int n;
    	vector<int> g[MAXNODE * 2];
    	bool mark[MAXNODE * 2];
    	int S[MAXNODE * 2], sn;
    
    	void init(int tot) {
    		n = tot * 2;
    		for (int i = 0; i < n; i += 2) {
    			g[i].clear();
    			g[i^1].clear();
    		}
    		memset(mark, false, sizeof(mark));
    	}
    
    	void add_Edge(int u, int uval, int v, int vval) {
    		u = u * 2 + uval;
    		v = v * 2 + vval;
    		g[u^1].push_back(v);
    		g[v^1].push_back(u);
    	}
    
    	void delete_Edge(int u, int uval, int v, int vval) {
    		u = u * 2 + uval;
    		v = v * 2 + vval;
    		g[u^1].pop_back();
    		g[v^1].pop_back();
    	}
    
    	bool dfs(int u) {
    		if (mark[u^1]) return false;
    		if (mark[u]) return true;
    		mark[u] = true;
    		S[sn++] = u;
    		for (int i = 0; i < g[u].size(); i++) {
    			int v = g[u][i];
    			if (!dfs(v)) return false;
    		}
    		return true;
    	}
    
    	bool solve() {
    		for (int i = 0; i < n; i += 2) {
    			if (!mark[i] && !mark[i + 1]) {
    				sn = 0;
    				if (!dfs(i)){
    					for (int j = 0; j < sn; j++)
    						mark[S[j]] = false;
    					sn = 0;
    					if (!dfs(i + 1)) return false;
    				}
    			}
    		}
    		return true;
    	}
    } gao;
    
    const int N = 105;
    int n;
    
    struct Point {
    	double x, y;
    	void read() {
    		scanf("%lf%lf", &x, &y);
    	}
    } p[N][2];
    
    inline double dis(Point a, Point b) {
    	double dx = a.x - b.x;
    	double dy = a.y - b.y;
    	return sqrt(dx * dx + dy * dy);
    }
    
    inline bool xj(Point a, Point b, double r) {
    	if (dis(a, b) > 2 * r) return false;
    	return true;
    }
    
    bool judge(double r) {
    	gao.init(n);
    	for (int i = 0; i < n; i++) {
    		for (int j = i + 1; j < n; j++) {
    			for (int x = 0; x < 2; x++) {
    				for (int y = 0; y < 2; y++) {
    					if (xj(p[i][x], p[j][y], r))
    						gao.add_Edge(i, x, j , y);
    				}
    			}
    		}
    	}
    	return gao.solve();
    }
    
    int main() {
    	while (~scanf("%d", &n)) {
    		for (int i = 0; i < n; i++)
    			for (int j = 0; j < 2; j++)
    				p[i][j].read();
    		double l = 0, r = 1e5;
    		for (int i = 0; i < 30; i++) {
    			double mid = (l + r) / 2;
    			if (judge(mid)) l = mid;
    			else r = mid;
    		}
    		printf("%.2lf
    ", l);
    	}
    	return 0;
    }


  • 相关阅读:
    <JavaScript> 组合继承
    <JavaScript> 稳妥构造函数模式与工厂模式的区别
    <JavaScript> call()、apply()、bind() 的用法
    <JavaScript>可枚举属性与不可枚举属性
    <JavaScript>闭包(closure)
    在MongoDB中实现聚合函数
    (转)如何入门 Python 爬虫
    Python爬虫实战四之抓取淘宝MM照片
    转载:十年驾车经验总结:活着,才是硬道理
    设计模式之单例模式的七种写法
  • 原文地址:https://www.cnblogs.com/bhlsheji/p/5321189.html
Copyright © 2011-2022 走看看