计算几何+2sat:1020T3

news2024/12/26 12:41:57

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;
}


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1113461.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

体感互动游戏研发虚拟场景3D漫游

体感互动游戏是一种结合虚拟现实&#xff08;VR&#xff09;或增强现实&#xff08;AR&#xff09;技术的游戏&#xff0c;允许玩家以身体动作和姿势来与游戏互动。这种类型的游戏通常需要特殊的硬件设备&#xff0c;例如体感控制器、摄像头或传感器&#xff0c;以捕捉玩家的动…

spring cloud alibaba 集成seata

1.启动服务端 1.下载 seata-server-1.4.2 2.创建数据库 DROP DATABASE IF EXISTS ry-seata;CREATE DATABASE ry-seata DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci;SET NAMES utf8mb4; SET FOREIGN_KEY_CHECKS 0;USE ry-seata;-- ---------------------------…

Java操作Python数据交互最佳实践

Java操作Python最佳实践 1、Java与Python的互操作性2、Java调用Python脚本及数据交互2.1、准备工作2.2、执行一段Python代码2.3、执行Python文件脚本2.4、执行Python文件中的指定方法2.5、执行含有第三方库的Python文件3、附录1、Java与Python的互操作性 在当今的软件开发领域,…

Linux服务器下装anaconda | 配置深度学习环境 | Pycharm连接远程服务器-经验总结

0 前言 推荐2个工具 WinSCP 一个 Windows 环境下使用的 SSH 的开源图形化 SFTP 客户端。同时支持 SCP 协议。它的主要功能是在本地与远程计算机间安全地复制文件&#xff0c;并且可以直接编辑文件。 WindTerm 一个多平台开源免费的终端软件&#xff0c;用于连接服务器 一…

Redis 主从复制,哨兵,集群——(3)集群篇

目录 1. 前篇回顾 2. Redis 集群是什么&#xff1f; 3. Redis 集群的优点 4. Redis 集群的槽位概念 5. 什么是分片&#xff1f; 6. 如何找到给定key的分片&#xff1f; 7. 分片槽位的设计有什么好处&#xff1f; 8. key映射到节点的三种解决方案 8.1 哈希取余分区 8.…

SortedSet 和 List 异同点

SortedSet 在 Java 的整个集合体系中&#xff0c;集合可以分成两个体系&#xff0c;一个是 Collection 存储单个对象的集合&#xff0c;另一个是 k-v 结构的 Map 集合。 SortedSet 是 Collection 体系下 Set 接口下的派生类&#xff0c;而 Set 集合的特征是不包含重 复的元素的…

【UE】纯蓝图实现:在游戏运行时设置关键点,然后让actor沿着关键点移动

前言 在上一篇博客(【UE】两步实现“从UI中拖出Actor放置到场景中”)中我们已经实现了如何从UI拖拽生成Actor ,本篇博客在此基础上要实现的是:从UI中拖出车,再从UI中拖出关键点,点击“开始移动”按钮后,车会沿着关键点移动,具体效果如下所示。 效果 步骤 1. 首先创建…

WMS透明仓库:实现仓储的全方位可视化与优化

一、WMS透明仓库的定义与特点 1. WMS透明仓库的定义&#xff1a;WMS透明仓库是一种基于信息技术的仓库管理系统&#xff0c;通过实时数据采集、分析和可视化&#xff0c;将仓库内外的物流流程、库存状态、人员活动等信息以透明的方式展示给相关利益方。 2. 实时数据采集&…

RCD负载箱的安全性能和认证标准有哪些?

RCD负载箱的安全性能和认证标准主要包括以下几个方面&#xff1a;RCD负载箱应符合国家标准或国际标准的防护等级要求&#xff0c;通常情况下&#xff0c;RCD负载箱应具备防护等级为IP54或更高级别&#xff0c;以确保在恶劣的环境条件下仍能正常工作。 RCD负载箱的绝缘电阻应满足…

Python+unittest接口自动化测试

首先配置好开发环境&#xff0c;下载安装Python并下载安装pycharm&#xff0c;在pycharm中创建项目功能目录。以下是项目的目录结构&#xff1a; common&#xff1a; 1 2 3 4 5 6 7 8 9 ——configDb.py&#xff1a;这个文件主要编写数据库连接池的相关内容&#xff0c;本项目…

suricata匹配从入门到精通(五)----二次开发保护规则库

0x00 背景 开源的suricata资源包是没有做加密处理,如果想要保护资源包,需要二次开发修改suricata源码。 本文基于suricata6.0.1 版本https://github.com/OISF/suricata/archive/refs/tags/suricata-6.0.1.zip二开。 0x01 实践 通过debug,跟规则处理相关需要修改2个地方。…

iPhone垃圾清理器:AnyMP4 iOS Cleaner for mac

AnyMP4 iOS Cleaner 是一款功能强大的iOS设备清理工具&#xff0c;旨在帮助用户轻松管理和优化iOS设备上的存储空间。该工具支持iPhone、iPad、iPod等设备&#xff0c;可以有效地清理无用的缓存、日志、文稿等&#xff0c;以释放存储空间。它还具有一键清理功能&#xff0c;能够…

二维码智慧门牌管理系统开发:高效解决道路数据难题

文章目录 前言一、二维码智慧门牌管理系统的创新性二、道路数据采集与处理三、二维码智慧门牌管理系统的优势 前言 在当今这个信息化社会&#xff0c;数据的准确性和及时性对于各种项目的成功实施至关重要。尤其是在智慧城市建设中&#xff0c;道路数据是不可或缺的基础性数据…

Spring Boot学习笔记

SpringBoot特征 特征 创建独立的 Spring 应用程序 直接嵌入 Tomcat、Jetty 或 Undertow&#xff08;无需部署 WAR 文件&#xff09; 提供“入门”依赖项以简化构建配置 尽可能自动配置 Spring 和 第三方库 提供生产就绪功能&#xff0c;例如指标、健康检查和外部化配置 完…

Squeeze-and-Attention Networks for Semantic Segmentation

0.摘要 最近&#xff0c;将注意力机制整合到分割网络中可以通过更重视提供更多信息的特征来提高它们的表征能力。然而&#xff0c;这些注意力机制忽视了语义分割的一个隐含子任务&#xff0c;并受到卷积核的网格结构的限制。在本文中&#xff0c;我们提出了一种新颖的squeeze-a…

如何自己的医疗图像分割数据集 使用NNunet进行训练

NNUNet使用自定义医疗图像分割数据集进行分割训练 主要讲解怎么把自己的数据放到nnUnet进行训练&#xff0c;不涉及nnUnet的原理和推导讲解。 1、转换的思路。 从NNUNet的开源代码中可以看到&#xff0c;NNUnetV2已经支持了很多的数据格式。但是因为其底层的逻辑主要是解决医…

pdf怎么压缩?一分钟搞定pdf压缩

在日常生活和工作中&#xff0c;我们经常需要处理大量的PDF文件&#xff0c;这些文件可能包含重要的信息&#xff0c;如合同、报告、说明书等&#xff0c;需要进行有效的管理和传输。然而&#xff0c;PDF文件往往体积较大&#xff0c;给存储和传输带来了不便&#xff0c;那么&a…

电子半导体行业电能质量监测与治理系统解决方案

摘要&#xff1a;本土半导体材料厂商不断提升半导体产品技术水平和研发能力&#xff0c;逐渐打破了国外半导体厂商的垄断格局&#xff0c;推进中国半导体材料国产化进程&#xff0c;促进中国半导体行业的发展。半导体产品的制造使用到的设备如单晶炉、多晶炉等都是恶性的谐波源…

Elasticsearch向量检索的演进与变革:从基础到应用

Elasticsearch向量检索的演进与变革&#xff1a;从基础到应用 1.引言 向量检索已经成为现代搜索和推荐系统的核心组件。 通过将复杂的对象&#xff08;例如文本、图像或声音&#xff09;转换为数值向量&#xff0c;并在多维空间中进行相似性搜索&#xff0c;它能够实现高效的…

内衣专用洗衣机怎么样?选购内衣裤洗衣机的方法

有的小伙伴在问内衣洗衣机有没有必要入手&#xff0c;答案是有必要的&#xff0c;贴身衣物一定要和普通衣服分开来洗&#xff0c;而且用手来清洗衣物真的很耗时间而且还清洗不干净&#xff0c;有了内衣洗衣机&#xff0c;我们不仅可以解放双手&#xff0c;在清洗过程中还能更加…