【模板】【洛谷 P4724】三维凸包(增量法)

传送门

先随机扰动一下变成每个面都恰好只有三个点
考虑增量法
每次新加一个点,把所有从这个点能看到的面删去
加入新的面
大概就是这样

(图是洛谷题解区搬来的)
在这里插入图片描述
在这里插入图片描述

判断 p p p是否能看到一个面 ( v 1 , v 2 , v 3 ) (v1,v2,v3) (v1,v2,v3)就是
( ( v 2 − v 1 ) ∗ ( v 3 − v 2 ) ) • ( p − v 1 ) ((v2-v1)*(v3-v2))•(p-v1) ((v2v1)(v3v2))(pv1)
就是垂直面的向量和 p p p到面的向量的关系

直接做就是了
边数最大约是 3 n 3n 3n,面数为 2 n 2n 2n
洛谷数据比较水。。。面数连 n n n都卡不到

#include<bits/stdc++.h>
using namespace std;
#define cs const
#define re register
#define pb push_back
#define pii pair<int,int>
#define ll long long
#define fi first
#define se second
#define bg begin
cs int RLEN=1<<20|1;
inline char gc(){
    static char ibuf[RLEN],*ib,*ob;
    (ib==ob)&&(ob=(ib=ibuf)+fread(ibuf,1,RLEN,stdin));
    return (ib==ob)?EOF:*ib++;
}
inline int read(){
    char ch=gc();
    int res=0;bool f=1;
    while(!isdigit(ch))f^=ch=='-',ch=gc();
    while(isdigit(ch))res=(res+(res<<2)<<1)+(ch^48),ch=gc();
    return f?res:-res;
}
inline ll readll(){
    char ch=gc();
    ll res=0;bool f=1;
    while(!isdigit(ch))f^=ch=='-',ch=gc();
    while(isdigit(ch))res=(res+(res<<2)<<1)+(ch^48),ch=gc();
    return f?res:-res;
}
inline int readstring(char *s){
	int top=0;char ch=gc();
	while(isspace(ch))ch=gc();
	while(!isspace(ch)&&ch!=EOF)s[++top]=ch,ch=gc();
	return top;
}
template<typename tp>inline void chemx(tp &a,tp b){a<b?a=b:0;}
template<typename tp>inline void chemn(tp &a,tp b){a>b?a=b:0;}
cs double eps=1e-9;
inline double reps(){
	return ((1.0*rand()/RAND_MAX)-0.5)*eps;
}
cs int N=4005;
struct pt{
	double x,y,z;
	void shake(){x+=reps(),y+=reps(),z+=reps();}
	pt(double _x=0,double _y=0,double _z=0):x(_x),y(_y),z(_z){}
	friend inline pt operator +(cs pt&a,cs pt&b){
		return pt(a.x+b.x,a.y+b.y,a.z+b.z);
	}
	friend inline pt operator -(cs pt&a,cs pt&b){
		return pt(a.x-b.x,a.y-b.y,a.z-b.z);
	}
	friend inline pt operator *(cs pt &a,cs pt &b){
		return pt(a.y*b.z-a.z*b.y,a.z*b.x-a.x*b.z,a.x*b.y-a.y*b.x);
	}
	friend inline double operator ^(pt a,pt b){
		return a.x*b.x+a.y*b.y+a.z*b.z;
	}
	inline double len(){return sqrt(x*x+y*y+z*z);}
}p[N];
struct flat{
	int v[3];
	flat(){}
	flat(int a,int b,int c){v[0]=a,v[1]=b,v[2]=c;}
	inline pt calc()cs{return (p[v[1]]-p[v[0]])*(p[v[2]]-p[v[0]]);}
	inline double S()cs{return calc().len()*0.5;}
}f[N],tp[N];
int cnt,n,vis[N][N];
inline bool See(cs flat &a,cs pt &b){
	return ((b-p[a.v[0]])^(a.calc()))>0;
}
int main(){
	#ifdef Stargazer
	freopen("lx.in","r",stdin);
	#endif
	scanf("%d",&n);
	for(int i=1;i<=n;i++)scanf("%lf%lf%lf",&p[i].x,&p[i].y,&p[i].z),p[i].shake();
	f[1].v[0]=1,f[1].v[1]=2,f[1].v[2]=3;
	f[2].v[0]=3,f[2].v[1]=2,f[2].v[2]=1;
	cnt=2;
	for(int i=4;i<=n;i++){
		int c=0;
		for(int j=1;j<=cnt;j++){
			int t=See(f[j],p[i]);
			if(!t)tp[++c]=f[j];
			for(int k=0;k<3;k++)vis[f[j].v[k]][f[j].v[(k+1)%3]]=t;
		}
		for(int j=1;j<=cnt;j++){
			for(int k=0;k<3;k++){
				int x=f[j].v[k],y=f[j].v[(k+1)%3];
				if(vis[x][y]&&!vis[y][x])tp[++c]=flat(x,y,i); 
			}
		}
		for(int j=1;j<=c;j++)f[j]=tp[j];
		cnt=c;
	}double ans=0;
	for(int i=1;i<=cnt;i++)ans+=f[i].S();
	printf("%.3lf",ans);return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值