计算几何+2sat:1020T3

2023-10-20 13:28
文章标签 计算 几何 2sat 1020t3

本文主要是介绍计算几何+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;
}

这篇关于计算几何+2sat:1020T3的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/247386

相关文章

uva 10387 Billiard(简单几何)

题意是一个球从矩形的中点出发,告诉你小球与矩形两条边的碰撞次数与小球回到原点的时间,求小球出发时的角度和小球的速度。 简单的几何问题,小球每与竖边碰撞一次,向右扩展一个相同的矩形;每与横边碰撞一次,向上扩展一个相同的矩形。 可以发现,扩展矩形的路径和在当前矩形中的每一段路径相同,当小球回到出发点时,一条直线的路径刚好经过最后一个扩展矩形的中心点。 最后扩展的路径和横边竖边恰好组成一个直

poj 1113 凸包+简单几何计算

题意: 给N个平面上的点,现在要在离点外L米处建城墙,使得城墙把所有点都包含进去且城墙的长度最短。 解析: 韬哥出的某次训练赛上A出的第一道计算几何,算是大水题吧。 用convexhull算法把凸包求出来,然后加加减减就A了。 计算见下图: 好久没玩画图了啊好开心。 代码: #include <iostream>#include <cstdio>#inclu

uva 1342 欧拉定理(计算几何模板)

题意: 给几个点,把这几个点用直线连起来,求这些直线把平面分成了几个。 解析: 欧拉定理: 顶点数 + 面数 - 边数= 2。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cstring>#include <cmath>#inc

uva 11178 计算集合模板题

题意: 求三角形行三个角三等分点射线交出的内三角形坐标。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cstring>#include <cmath>#include <stack>#include <vector>#include <

XTU 1237 计算几何

题面: Magic Triangle Problem Description: Huangriq is a respectful acmer in ACM team of XTU because he brought the best place in regional contest in history of XTU. Huangriq works in a big compa

poj 3304 几何

题目大意:给出n条线段两个端点的坐标,问所有线段投影到一条直线上,如果这些所有投影至少相交于一点就输出Yes!,否则输出No!。 解题思路:如果存在这样的直线,过投影相交点(或投影相交区域中的点)作直线的垂线,该垂线(也是直线)必定与每条线段相交,问题转化为问是否存在一条直线和所有线段相交。 若存在一条直线与所有线段相交,此时该直线必定经过这些线段的某两个端点,所以枚举任意两个端点即可。

POJ 2318 几何 POJ 2398

给出0 , 1 , 2 ... n 个盒子, 和m个点, 统计每个盒子里面的点的个数。 const double eps = 1e-10 ;double add(double x , double y){if(fabs(x+y) < eps*(fabs(x) + fabs(y))) return 0 ;return x + y ;}struct Point{double x , y

poj 2653 几何

按顺序给一系列的线段,问最终哪些线段处在顶端(俯视图是完整的)。 const double eps = 1e-10 ;double add(double x , double y){if(fabs(x+y) < eps*(fabs(x) + fabs(y))) return 0 ;return x + y ;}struct Point{double x , y ;Point(){}Po

音视频入门基础:WAV专题(10)——FFmpeg源码中计算WAV音频文件每个packet的pts、dts的实现

一、引言 从文章《音视频入门基础:WAV专题(6)——通过FFprobe显示WAV音频文件每个数据包的信息》中我们可以知道,通过FFprobe命令可以打印WAV音频文件每个packet(也称为数据包或多媒体包)的信息,这些信息包含该packet的pts、dts: 打印出来的“pts”实际是AVPacket结构体中的成员变量pts,是以AVStream->time_base为单位的显

计算数组的斜率,偏移,R2

模拟Excel中的R2的计算。         public bool fnCheckRear_R2(List<double[]> lRear, int iMinRear, int iMaxRear, ref double dR2)         {             bool bResult = true;             int n = 0;             dou