bzoj 4541: [Hnoi2016]矿区

学习了一下平面图剖分的姿势,orz cbh

每次只要随便选择一条边,然后不停尽量向左转就行

#include <bits/stdc++.h>
#define N 1300000
#define M 5000013
#define LL long long
#define pb push_back
using namespace std;
LL n, m, k;
struct point 
{
    LL x, y;
} S[N];
vector <LL> bi[N];
vector <LL> re[N], vis[N], s1[N], s2[N];
vector <LL> bt[N], ca[N], cb[N];
LL tot_are;
LL siz[N], sum1[N], sum2[N], tot[N];
LL nwc;
LL comp(LL a, LL b)
{
    return atan2(S[a].y - S[nwc].y, S[a].x - S[nwc].x) > atan2(S[b].y - S[nwc].y, S[b].x - S[nwc].x);
}
LL getsiz(point a, point b, point c)
{
    return (b.x - a.x) * (c.y - a.y) - (c.x - a.x) * (b.y - a.y);
}
  
namespace addr
{
    LL hs[M]; LL dt[M];
    LL get(LL a, LL b)
    {
        return 1ll * a * 1000000 + b;
    }
    LL & find(LL a, LL b)
    {
        LL p = get(a, b); LL q = p % M;
        while (hs[q] && hs[q] != p) q = (q + 1) % M;
        hs[q] = p; return dt[q];
    }
}
LL tvis[N], fa[N];
void dfs(LL t)
{
    //cout << t << "\n";
     
    tvis[t] = 1;
    sum1[t] = siz[t] * siz[t];
    sum2[t] = siz[t] * 2;
    for (LL i = 0; i < bt[t].size(); ++ i)
        if (!tvis[bt[t][i]])
        {
            dfs(bt[t][i]); fa[bt[t][i]] = t;
            sum1[t] += sum1[bt[t][i]];
            sum2[t] += sum2[bt[t][i]];
            s1[ca[t][i]][addr :: find(ca[t][i], cb[t][i])] = sum1[bt[t][i]];
            s2[ca[t][i]][addr :: find(ca[t][i], cb[t][i])] = sum2[bt[t][i]];
            s1[cb[t][i]][addr :: find(cb[t][i], ca[t][i])] = -sum1[bt[t][i]];
            s2[cb[t][i]][addr :: find(cb[t][i], ca[t][i])] = -sum2[bt[t][i]];
        }
}
LL read()
{
    LL 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*10+ch-'0';ch=getchar();}
    return x*f;
}
LL c = 0;
int main()
{
    //freopen("mine2.in", "r", stdin);
    n = read(); m = read(); k = read();
    for (LL i = 1; i <= n; ++ i) S[i].x = read(), S[i].y = read();
    for (LL i = 1; i <= m; ++ i)
    {
        LL a, b;
        a = read(); b = read();
        bi[a].push_back(b);
        bi[b].push_back(a);
    }
    for (LL i = 1; i <= n; ++ i) re[i].resize(bi[i].size()), s1[i] = s2[i] = vis[i] = re[i];
    for (LL i = 1; i <= n; ++ i) nwc = i, sort(bi[i].begin(), bi[i].end(), comp);
    for (LL i = 1; i <= n; ++ i)
        for (LL j = 0; j < bi[i].size(); ++ j)
            addr :: find(i, bi[i][j]) = j;
    for (LL i = 1; i <= n; ++ i)
        for (LL j = 0; j < bi[i].size(); ++ j)
            re[i][j] = addr :: find(bi[i][j], i);
          
    for (LL i = 1; i <= n; ++ i)
        for (LL j = 0; j < bi[i].size(); ++ j)
            if (!vis[i][j])
            {
                tot_are ++;
                for (LL k = i, p = j; !vis[k][p]; )
                {
                    siz[tot_are] += getsiz(S[i], S[k], S[bi[k][p]]);
                    vis[k][p] = tot_are;
                    LL np = (re[k][p] + 1) % bi[bi[k][p]].size();
                    k = bi[k][p];
                    p = np;
                }
                if (siz[tot_are] < 0) c = tot_are;
            }
    for (LL i = 1; i <= n; ++ i)
        for (LL j = 0; j < bi[i].size(); ++ j)
            bt[vis[i][j]].push_back(vis[bi[i][j]][re[i][j]]),
            ca[vis[i][j]].push_back(i),
            cb[vis[i][j]].push_back(bi[i][j]);
    dfs(c);
    for (LL i = 1, last = 0; i <= k; ++ i)
    {
        LL d, ns1 = 0, ns2 = 0;
        d = read(); d = (d + last) % n + 1;
        LL fs, ls, nw;
        fs = read(); fs = (fs + last) % n + 1; ls = fs;
        for (LL j = 2; j <= d; ++ j, ls = nw)
        {
            nw = read(); nw = (nw + last) % n + 1;
            ns1 += s1[ls][addr :: find(ls, nw)];
            ns2 += s2[ls][addr :: find(ls, nw)];
        }
        ns1 += s1[ls][addr :: find(ls, fs)];
        ns2 += s2[ls][addr :: find(ls, fs)];
        LL g = __gcd(ns1, ns2);
        cout << ns1 / g << " " << ns2 / g << "\n";
        last = ns1 / g;
    }
}

 

放在class里的东西还会爆栈QAQ,以后不敢用了

posted @ 2016-12-03 22:50  AwD!  阅读(412)  评论(0编辑  收藏  举报