学习一发平面图的姿势……ref

#include <algorithm>
#include <iostream>
#include <cstdio>
#include <vector>
#include <cmath>
using namespace std;
typedef long long ll;
int n, m, k, cnt, uu, vv, nxt[1200005], bel[1200005], belcnt, rot, fa[1200005];
int ask[200005];
bool vis[1200005], isn[1200005];
ll s[1200005], sp[1200005];
struct Point{
int x, y;
Point operator-(const Point &u){
return (Point){x-u.x, y-u.y};
}
ll operator*(const Point &u){
return (ll)x*u.y-(ll)y*u.x;
}
}p[200005];
struct Edge{
int fro, too, idx;
double ang;
Edge(int f=0, int t=0, int i=0){
fro = f; too = t; idx = i;
ang = atan2(p[t].y-p[f].y, p[t].x-p[f].x);
}
bool operator<(const Edge &x)const{
return ang<x.ang;
}
}edge[1200005];
vector<Edge> w[200005], tr[1200005];
void add_edge(int fro, int too){
edge[cnt] = Edge(fro, too, cnt);
w[fro].push_back(edge[cnt]);
cnt++;
}
ll getGcd(ll a, ll b){
return !b?a:getGcd(b, a%b);
}
void rn(int &x){
char ch=getchar();
x = 0;
int f=1;
while(ch<'0' || ch>'9'){
if(ch=='-') f = -1;
ch = getchar();
}
while(ch>='0' && ch<='9'){
x = x * 10 + ch - '0';
ch = getchar();
}
x *= f;
}
int findEdge(int f, const Edge &x){
int l=0, r=w[f].size()-1, mid, re;
while(l<=r){
mid = (l + r) >> 1;
if(w[f][mid]<x) l = mid + 1;
else re = mid, r = mid - 1;
}
return re;
}
void dfs(int x){
vis[x] = true;
sp[x] = s[x] * s[x];
s[x] <<= 1;
for(int i=0; i<tr[x].size(); i++){
int t=tr[x][i].too;
if(vis[t]) continue;
fa[t] = x;
isn[tr[x][i].idx] = isn[tr[x][i].idx^1] = true;
dfs(t);
s[x] += s[t];
sp[x] += sp[t];
}
}
int main(){
rn(n); rn(m); rn(k);
for(int i=1; i<=n; i++){
rn(p[i].x);
rn(p[i].y);
}
for(int i=1; i<=m; i++){
rn(uu); rn(vv);
add_edge(uu, vv);
add_edge(vv, uu);
}
for(int i=1; i<=n; i++)
sort(w[i].begin(), w[i].end());
for(int i=0; i<cnt; i++){
int qwq=findEdge(edge[i].too, edge[i^1])-1;
if(qwq==-1) qwq = w[edge[i].too].size() - 1;
nxt[i] = w[edge[i].too][qwq].idx;
}
for(int i=0; i<cnt; i++)
if(!bel[i]){
bel[i] = bel[nxt[i]] = ++belcnt;
for(int j=nxt[i]; edge[j].too!=edge[i].fro; j=nxt[j]){
s[belcnt] += (p[edge[j].fro]-p[edge[i].fro]) * (p[edge[j].too]-p[edge[i].fro]);
bel[nxt[j]] = belcnt;
}
if(s[belcnt]<=0) rot = belcnt;
}
for(int i=0; i<cnt; i++)
tr[bel[i]].push_back(Edge(bel[i], bel[i^1], i));
dfs(rot);
ll p=0, q=0;
int d;
while(k--){
rn(d); d = (d + p) % n + 1;
for(int i=1; i<=d; i++){
rn(ask[i]);
ask[i] = (ask[i] + p) % n + 1;
}
ask[d+1] = ask[1];
p = q = 0;
for(int i=1; i<=d; i++){
int j=w[ask[i]][findEdge(ask[i], Edge(ask[i],ask[i+1],0))].idx;
if(!isn[j]) continue;
if(fa[bel[j]]==bel[j^1])
p += sp[bel[j]], q += s[bel[j]];
else
p -= sp[bel[j^1]], q -= s[bel[j^1]];
}
ll gcd=getGcd(p, q);
p /= gcd; q /= gcd;
printf("%lld %lld\n", p, q);
}
return 0;
}

loj2052 「HNOI2016」矿区的更多相关文章

  1. LOJ#2052. 「HNOI2016」矿区(平面图转对偶图)

    题面 传送门 题解 总算会平面图转对偶图了-- 首先我们把无向边拆成两条单向边,这样的话每条边都属于一个面.然后把以每一个点为起点的边按极角排序,那么对于一条边\((u,v)\),我们在所有以\(v\ ...

  2. 【LOJ】#2052. 「HNOI2016」矿区

    题解 之前尝试HNOI2016的时候弃坑的一道,然后给补回来 (为啥我一些计算几何就写得好长,不过我写啥都长orz) 我们尝试给这个平面图分域,好把这个平面图转成对偶图 怎么分呢,我今天也是第一次会 ...

  3. 「HNOI2016」矿区

    https://loj.ac/problem/2052 题解 平面图转对偶图.. 首先我们转的话需要给所有的平面标号,然后找到每条边看看他们隔开了哪两个平面. 做法就是对每个点维护它的所有排好序的出边 ...

  4. 「HNOI2016」数据结构大毒瘤

    真是 \(6\) 道数据结构毒瘤... 开始口胡各种做法... 「HNOI2016」网络 整体二分+树状数组. 开始想了一个大常数 \(O(n\log^2 n)\) 做法,然后就被卡掉了... 发现直 ...

  5. 「HNOI2016」树 解题报告

    「HNOI2016」树 事毒瘤题... 我一开始以为每次把大树的子树再接给大树,然后死活不知道咋做,心想怕不是个神仙题哦 然后看题解后才发现是把模板树的子树给大树,虽然思维上难度没啥了,但是还是很难写 ...

  6. 「HNOI2016」序列 解题报告

    「HNOI2016」序列 有一些高妙的做法,懒得看 考虑莫队,考虑莫队咋移动区间 然后你在区间内部找一个最小值的位置,假设现在从右边加 最小值左边区间显然可以\(O(1)\),最小值右边的区间是断掉的 ...

  7. 「HNOI2016」网络 解题报告

    「HNOI2016」网络 我有一个绝妙的可持久化树套树思路,可惜的是,它的空间是\(n\log^2 n\)的... 注意到对一个询问,我们可以二分答案 然后统计经过这个点大于当前答案的路径条数,如果这 ...

  8. 「HNOI2016」最小公倍数 解题报告

    「HNOI2016」最小公倍数 考虑暴力,对每个询问,处理出\(\le a,\le b\)的与询问点在一起的联通块,然后判断是否是一个联通块,且联通块\(a,b\)最大值是否满足要求. 然后很显然需要 ...

  9. loj #2051. 「HNOI2016」序列

    #2051. 「HNOI2016」序列 题目描述 给定长度为 n nn 的序列:a1,a2,⋯,an a_1, a_2, \cdots , a_na​1​​,a​2​​,⋯,a​n​​,记为 a[1: ...

随机推荐

  1. CocoStudio UIButton setPressedActionEnabled(true) 子控件不跟着缩放

    具体情况是这样的:美术给了我 一个按钮的背景图片  一个按钮的文字图片,用背景图片创建一个button,然后把文字图片添加进去(注意关闭文字图片的交互功能) 设置UIButton setPressed ...

  2. 不该被忽视的CoreJava细节(一)

    一.系列文章导言 <不该被忽视的CoreJava细节>系列文章将会持续更新.我希望自己通过这一系列文章的写作,能与读者一起进步,逐步完善对Java体系结构的了解. 二.本期关注点 几乎翻看 ...

  3. python3基础08(exec、bytearray使用等)

    #!/usr/bin/env python# -*- coding:utf-8 -*- str="test"print(ascii(str))a=bytearray("a ...

  4. LeetCode Find Peak Element 找临时最大值

    Status: AcceptedRuntime: 9 ms 题意:给一个数组,用Vector容器装的,要求找到一个临时最高点,可以假设有num[-1]和num[n]两个元素,都是无穷小,那么当只有一个 ...

  5. 实现Hbase的分页

    作者:R星月 出处:http://www.cnblogs.com/rxingyue 欢迎转载,也请保留这段声明.谢谢! 做一个项目中由于数据量比较大,并且需要定时增量分析,做了hbase的分页.项目中 ...

  6. python_71_json序列化1

    #序列化:序列化 (Serialization)将对象的状态信息转换为可以存储或传输的形式的过程. #本例把字典数据类型存成字符串存在硬盘 #文件只能存字符串和二进制码,字典之类的不可以 info={ ...

  7. IBM MQ安装

    一.下载MQ 可以去官方网站下载,我这次下了一个下载器从官方,然后通过下载器进行MQ的下载. 地址:https://www.ibm.com/developerworks/cn/downloads/ws ...

  8. Oracle 配置文件目录

    Oracle 配置文件目录 ① 在oracle安装目录下,找D:\oracle\product\10.2.0\client_1\NETWORK\ADMIN中的tnsnames.ora文件,找到之后,配 ...

  9. java基础面试题:Math.round(11.5)等於多少? Math.round(-11.5)等於多少?

    package com.swift; public class Math_Round { public static void main(String[] args) { /* * Math roun ...

  10. nodejs 用户登录密码md5加密

    jade文件 div.login ul.inp-content  li span= '用户名:' input.ui-input1#input1(placeholder='请输入手机号')  li sp ...