计算几何+2sat:1020T3

http://cplusoj.com/d/senior/p/SS231019C

我们进行这样的转化

在这里插入图片描述

则0/1必选一个,2/3必选一个

那么就变成一个2sat问题

两三角形有交,则一个选,一个不能选

对角三角形一个选,一个不选。一个不选,一个选

三角形不合法,则选向不选连边,代表必须不选

// 5.3k
#include<bits/stdc++.h>
using namespace std;
#define int long long
inline int read(){int x=0,f=1;char ch=getchar(); while(ch<'0'||
ch>'9'){if(ch=='-')f=-1;ch=getchar();}while(ch>='0'&&ch<='9'){
x=(x<<1)+(x<<3)+(ch^48);ch=getchar();}return x*f;}
#define Z(x) (x)*(x)
#define pb push_back
//mt19937 rand(time(0));
//mt19937_64 rand(time(0));
//srand(time(0));
#define N 200010
//#define M
//#define mo
#define eps 1e-4
double H, W, X[N], Y[N]; 
struct Point {
	double x, y; 
	Point operator - (const Point &A) const {
		Point B; B.x=x-A.x; B.y=y-A.y; 
		return B; 
	}
	double operator ^ (const Point &A) const {
		return x * A.y - y * A.x; 
	}
	bool check() {
//		printf("Checking %lf %lf\n", x, y); 
		if(x < 0 || x > W) return false; 
		if(y < 0 || y > H) return false; 
		return true; 
	}
};
namespace Cross {
	bool Line_cross(Point p1, Point p2, Point p3, Point p4) {
		double a1 = (p1 - p2) ^ (p3 - p2); 
		double a2 = (p1 - p2) ^ (p4 - p2); 
		if(a1 * a2 >= 0) return false; 
		return true; 
	}
	bool cross(Point p1, Point p2, Point p3, Point p4) {
//		printf("(%lf %lf) (%lf %lf) || (%lf %lf) (%lf %lf)\n", p1.x, p1.y, p2.x, p2.y, p3.x, p3.y, p4.x, p4.y); 
		bool t1 = Line_cross(p1, p2, p3, p4); 
		bool t2 = Line_cross(p3, p4, p1, p2); 
		return t1 && t2; 
	}
}
struct node {
	Point a[3]; 
	void make_node(int i, int j, double k) {
		a[0] = {X[i], Y[i]}; 
		if(j == 0 || j == 3) a[1] = {X[i], Y[i] + k}; 
		if(j == 3 || j == 1) a[2] = {X[i] + k, Y[i]}; 
		if(j == 1 || j == 2) a[1] = {X[i], Y[i] - k}; 
		if(j == 2 || j == 0) a[2] = {X[i] - k, Y[i]}; 
	}
	bool check() {
		if(a[0].check() == false) return false; 
		if(a[1].check() == false) return false; 
		if(a[2].check() == false) return false; 
		return true; 
	}
	void print() {
		printf("%lf %lf\n", a[0].x, a[0].y); 
		printf("%lf %lf\n", a[1].x, a[1].y); 
		printf("%lf %lf\n", a[2].x, a[2].y); 
		printf(">> %lld\n", check()); 
	}
	bool operator ^(const node &A) const {
//		; A.print(); 
		int f0, f1, g0, g1; 
		for(f0 = 0; f0 <= 2; ++f0) 
			for(f1 = f0+1; f1 <= 2; ++f1)
				for(g0 = 0; g0 <= 2; ++g0)
					for(g1 = g0+1; g1<=2; ++g1) {
						if(Cross::cross(a[f0], a[f1], A.a[g0], A.a[g1])) 
							return true; 
					}
		return false; 
	}
	
}p[N]; 
int n, m, i, j, k, T;
namespace Graph {
	int make_node(int i, int j) { 
		return j * n + i; 
	}
	struct two_sat {
		int dfn[N], low[N], vis[N], col[N], tot, i; 
		vector<int>G[N]; 
		stack<int>z; 
		void clear() {
			tot = 0; 
			while(!z.empty()) z.pop(); 
			memset(dfn, 0, sizeof(dfn)); 
			memset(low, 0, sizeof(low)); 
			memset(vis, 0, sizeof(vis)); 
			memset(col, 0, sizeof(col)); 
			for(i=0; i<N; ++i) G[i].clear(); 
		}
		void add_edge(int u,  int v) {
//			printf("%lld  %lld\n", u, v); 
			G[u].pb(v); 
		}
		void dfs(int x) {
			dfn[x] = low[x] = ++tot; 
			z.push(x); vis[x] = 1; 
			for(auto y : G[x]) {
				if(vis[y] == -1) continue; 
				if(!vis[y]) dfs(y), low[x] = min(low[x], low[y]); 
				else low[x] = min(low[x], low[y]); 
			}
//			printf("\t %lld : %lld %lld\n", x, dfn[x], low[x]); 
			if(dfn[x] == low[x]) {
				while(z.top() != x) col[z.top()] = x, vis[z.top()] = -1, z.pop(); 
				col[x] = x; z.pop(); vis[x] = -1; 
			}
//			vis[x] = -1; 
		}
		void tarjan() {
			for(i=1; i<=8*n; ++i) 
				if(!vis[i]) dfs(i); 
//			for(i=1; i<=8*n; ++i) printf("%3lld", i); 
//			printf("\n"); 
//			for(i=1; i<=8*n; ++i) printf("%3lld", col[i]); 
//			printf("\n"); 
		}
		int pan(int u, int v) {
			return col[u] == col[v]; 
		}
	}Sat;
}
using namespace Graph; 
double l, r, mid, ans; 

bool check(double k) {
//	printf("# Now is %.3lf\n", k); 
	int i, j, x, y, u, v; 
	Sat.clear(); 
	for(i=1; i<=n; ++i) {
		for(j=0; j<=3; ++j) {
			u = make_node(i, j); 
			v = make_node(i, j^1); 
			Sat.add_edge(u, v + 4*n); 
			Sat.add_edge(v + 4*n, u ); 
			p[u].make_node(i, j, k); 
//			p[u].print(); 
			if(p[u].check() == false) 
				Sat.add_edge(u, u + 4*n);
//				 Sat.add_edge(u + 4*n, u); 
		}
//		printf("-----------\n"); 
	}
		
	for(i=1; i<=n; ++i) 
		for(j=i+1; j<=n; ++j) 
			for(x=0; x<=3; ++x) 
				for(y=0; y<=3; ++y) {
//					if(i == j) continue; 
					u = make_node(i, x); 
					v = make_node(j, y); 
					if(p[u] ^ p[v]) {
//						printf("%lld %lld || %lld %lld\n", i, j, x, y); 
						Sat.add_edge(u, v + 4*n); 
						Sat.add_edge(v, u + 4*n); 
					}
				}
	Sat.tarjan(); 
	for(i=1; i<=n; ++i) 
		for(j=0; j<=3; ++j) {
			u = make_node(i, j); 
			v = make_node(i, j^1); 
//			printf("%lld | %lld %lld %lld\n", i, j, u, u+4*n) ; 
//				u+4*n, v+4*n); 
			if(Sat.pan(u, u + 4*n)) return false; 
//			if(Sat.pan(u + 4*n, v + 4*n)) return false; 
		}
//	printf("\t ok\n"); 
	return true; 
	
}

signed main()
{
//	freopen("in.txt", "r", stdin);
//	freopen("out.txt", "w", stdout);
		freopen("city.in", "r", stdin);
	freopen("city.out", "w", stdout);	
//	T=read();
//	while(T--) {
//
//	}
	scanf("%lf%lf%lld", &W, &H, &n); 
	for(i=1; i<=n; ++i) {
		scanf("%lf%lf", &X[i], &Y[i]); 
	}
	l = 0; r = max(H, W); 
	while(r - l > eps){
		mid = (l + r) / 2; 
		if(check(mid)) l = mid, ans = mid; 
		else r = mid; 
	}
	printf("%.2lf", ans*2); 
	return 0;
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值