csu1812
csu1812
题意
求三角形和矩形交的面积。
分析
半平面交。把三角形的三条边当作直线去切割矩形,最后求切割后的多边形面积即可。
code
#include<algorithm>
#include<cstdio>
#include<cmath>
#include<cstring>
using namespace std;
const double eps = 1e-8;
int sgn(double x) { // 误差
if(fabs(x) < eps) return 0;
return x < 0 ? -1 : 1;
}
struct P {
double x, y;
P() {}
P(double x, double y) : x(x), y(y) {}
P operator + (const P p) const {
return P(x + p.x, y + p.y);
}
P operator - (const P p) const {
return P(x - p.x, y - p.y);
}
P operator * (const double tt) const {
return P(x * tt, y * tt);
}
P operator / (const double tt) const {
return P(x / tt, y / tt);
}
bool operator < (const P &p) const { // 坐标排序规则
return x < p.x || (x == p.x && y < p.y);
}
double dot(const P &p) const { // 点积
return x * p.x + y * p.y;
}
double det(const P &p) const { // 叉积
return x * p.y - y * p.x;
}
};
P tri[4], rec[5];
P b[5], c[5];
double A, B, C;
int m;
double cross(P o, P p, P q) { // 向量 op 和 oq 的叉积
return (p.x - o.x) * (q.y - o.y) - (q.x - o.x) * (p.y - o.y);
}
void init() { // 顺时针排序
if(sgn(cross(tri[0], tri[1], tri[2])) < 0)
reverse(tri, tri + 3);
}
// 得到直线 pq : A * x + B * y + C = 0
// f(x,y) = A * x + B * y + C
// f(x,y) < 0 表示点(x,y)在直线pq的左边(此时可把pq当做向量)
void getLine(P p, P q) {
A = q.y - p.y;
B = p.x - q.x;
C = q.det(p);
}
// 直线 A * x + B * y + C = 0 与 直线 pq 的交点
P intersect(P p, P q) {
double u = fabs(A * p.x + B * p.y + C);
double v = fabs(A * q.x + B * q.y + C);
return P((p.x * v + q.x * u) / (u + v), (p.y * v + q.y * u) / (u + v));
}
double cal(P p) {
return sgn(A * p.x + B * p.y + C);
}
void cut() {
int tmpm = 0;
for (int i = 0; i < m; i++) {
if (cal(b[i]) <= 0) { // 判断是否在多边形内,这里指点在直线A * x + B * y + C = 0的左边或直线上
c[tmpm++] = b[i];
} else { // 虽然不在多边形内,但是可能和多边形内的点组成的直线与多边形产生新的交点
if (cal(b[(i - 1 + m) % m]) < 0) {
c[tmpm++] = intersect(b[(i - 1 + m) % m], b[i]);
}
if (cal(b[i + 1]) < 0) {
c[tmpm++] = intersect(b[i + 1], b[i]);
}
}
}
for (int i = 0; i < tmpm; i++)
b[i] = c[i];
b[tmpm] = c[0];
m = tmpm;
}
void solve() {
tri[3] = tri[0];
rec[4] = rec[0];
memcpy(b, rec, sizeof rec);
m = 4;
for(int i = 0; i < 3; i++) {
getLine(tri[i], tri[i + 1]);
cut();
}
}
// 求多边形面积 (a[]为多边形的点 n为点的个数)
double polygon_area(P *a, int n) {
a[n] = a[0];
double area = 0;
for(int i = 0; i < n; i++) {
area += a[i].det(a[i + 1]);
}
return fabs(area) / 2;
}
int main() {
double x1, y1, x2, y2, x3, y3, x4, y4;
while(~scanf("%lf%lf%lf%lf%lf%lf%lf%lf", &x1, &y1, &x2, &y2, &x3, &y3, &x4, &y4)) {
tri[0] = P(x1, y1);
tri[1] = P(x2, y1);
tri[2] = P(x1, y2);
init();
rec[0] = P(x3, y3);
rec[1] = P(x4, y3);
rec[2] = P(x4, y4);
rec[3] = P(x3, y4);
solve();
printf("%.8f\n", polygon_area(b, m));
}
return 0;
}
csu1812的更多相关文章
- 【CSU1812】三角形和矩形 【半平面交】
检验半平面交的板子. #include <stdio.h> #include <bits/stdc++.h> using namespace std; #define gg p ...
- 2017年暑假ACM集训日志
20170710: hdu1074,hdu1087,hdu1114,hdu1159,hdu1160,hdu1171,hdu1176,hdu1010,hdu1203 20170711: hdu1231, ...
随机推荐
- 打包成apk,生成apk文件,上传到网站服务器提供链接下载
Android开发把项目打包成apk: 做完一个Android项目之后,如何才能把项目发布到Internet上供别人使用呢?我们需要将自己的程序打包成Android安装包文件--APK(Android ...
- apt-get阿里源
备份原有配置文件 mv /etc/apt/sources.list /etc/apt/sources.list.bak 新建一个文件 vi /etc/apt/sources.list 复制以下内容到新 ...
- Python全栈工程师 (类变量、方法、继承、覆盖)
ParisGabriel 每天坚持手写 一天一篇 决定坚持几年 为了梦想为了信仰 开局一张图 Python人工智能从入门到精通 补充: 实例方法都是操作实例对象 ...
- 【志银】Dev-Cpp配置OpenGL图形库(成功版本:Dev-Cpp 5.7.1 MinGW 4.8.1)
★配置前须知:Dev-Cpp自带OpenGL的使用和OpenGL简介 (附Dev-Cpp下载地址:http://sourceforge.net/projects/orwelldevcpp/?sourc ...
- NodeJs04
REST API的设计 前言 客户端通过请求URL,传递参数,去获取指定的数据,这就是API(ApplicationProgramInterface). API是前端和客户端操作后端数据的一种方式,一 ...
- FTP数字代码的意义
110 重新启动标记应答.120 服务在多久时间内ready.125 数据链路端口开启,准备传送.150 文件状态正常,开启数据连接端口.200 命令执行成功.202 命令执行失败.211 系统状态或 ...
- ZOJ 3606 Lazy Salesgirl ( 线段树 + 思路 )
卖切糕的小女孩 http://www.cnblogs.com/wuyiqi/archive/2012/04/28/2474672.html #include <cstdio> #inclu ...
- axis2实践(二)Restful入门示例
1. 实例说明 本示例直接参照了RESTful Web Services with Apache Axis2,本示例基本就是沿用的原示例,就是一个对学生信息(包括姓名,年龄,课程)的管理的例子,提供如 ...
- jQuery仿3D旋转木马效果插件(带索引按钮)
项目中需要用到旋转木马效果,但是我在网上找的插件,基本都是不带按钮或者只是带前后按钮的,而项目要求的是带索引按钮,也就是说有3张图片轮播,对应的要有3个小按钮,点击按钮,对应的图片位于中间位置.于是就 ...
- Codeforces 433 Div.2(A、B、C、D)
A. Fraction 暴力遍历1-1000,取组成的真分数比值最大且分子分母gcd为1时更新答案 代码: #include <stdio.h> #include <algorith ...