day5-李佳衡

T1 CF611G New Year and Cake

  • 给定一个 n 个顶点的严格凸多边形。
  • 要求 \(\frac{n(n-3)}2\) 个*由对角线将多边形割成两个部分的面积差 2 之和。
  • \(n \le 5 \times 10^5\),答案对 10^9+7 取模。

首先,知道一点,求一个多边形的面积,可以将它所有点按顺时针排序,然后\(\displaystyle \sum p[i \%n+1] * p[i]\)就可以得到这个多边形的面积。

然后,就可以维护 \(p[i] * p[i + 1]\) 的前缀和的前缀和,然后再维护 \(p[i]\)的前缀和。再用类似双指针的方式进行统计答案即可。具体的统计方法看代码吧,我的语文能力不支持我将它表示出来。

#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdlib>
#include<string>
#include<cstdio>
#include<vector>
#include<queue>
#include<cmath>
#include<set>
#include<map>
using namespace std;
typedef long long LL;
const int N = 1e6+10 , mod = 1e9+7;
#define int long long
inline int read()
{
	register int x = 0 , f = 0; register char c = getchar();
	while(c < '0' || c > '9') f |= c == '-' , c = getchar();
	while(c >= '0' && c <= '9') x = (x << 3) + (x << 1) + c - '0' , c = getchar();
	return f ? -x : x;
}
int n;
struct node
{
	int x , y;
	node(int x = 0 , int y = 0) : x(x) , y(y) {}
	node operator - (const node &A) const { return node(x - A.x , y - A.y); }
	node operator + (const node &A) const { return node((x + A.x) % mod , (y + A.y) % mod); }
	LL operator * (const node &A) const {return (LL)x * A.y - (LL)y * A.x; }
}p[N] , sum[N];
LL s[N] , ss[N];
signed main()
{
	n = read();
	for(int i = 1 ; i <= n ; ++i) p[i].x = read() , p[i].y = read();
	unsigned long long all = 0;
	for(int i = 1 ; i <= n ; ++i) all += p[i % n + 1] * p[i]; // cout << all << '\n';
	for(int i = n + 1 ; i <= 2 * n ; ++i) p[i] = p[i - n];
	for(int i = 1 ; i <= n * 2 ; ++i) s[i] = p[i] * p[i-1] % mod;
	for(int i = 1 ; i <= n * 2 ; ++i) s[i] = (s[i-1] + s[i]) % mod;
	for(int i = 1 ; i <= n * 2 ; ++i) ss[i] = (ss[i-1] + s[i]) % mod;
	for(int i = 1 ; i <= n * 2 ; ++i) sum[i] = sum[i-1] + p[i];
	int R = 1; unsigned long long S = 0; LL ans = 0;
	for(int i = 1 ; i <= n ; ++i)
	{
		while(S + p[R] * p[i] + p[R + 1] * p[R] + p[i] * p[R + 1] <= all / 2) S += p[R] * p[i] + p[R + 1] * p[R] + p[i] * p[R + 1] , R++; // +
		if(i != R)
		{
			LL tp = ((ss[R] - ss[i] - (R - i) * s[i] % mod + p[i] * (sum[R] - sum[i]) % mod) % mod + mod) % mod;
            ans = (ans + (all % mod * (R - i - 1) % mod - 2 * tp % mod + mod) % mod) % mod;
            S += p[i] * p[i+1] + p[i+1] * p[R] + p[R] * p[i]; // -
		}
	}
	cout << ans << '\n';
	return 0;
}


T2 AT4994 [AGC034D] Manhattan Max Matching

在一个二维坐标系内,点$ (RX_i,RY_i) $上有 \(RC_i\) 个红球,点$ (BX_i,BY_i)$ 上有$ BC_i$个蓝球,且保证 \(\sum_{i=1}^{n}RC_i=\sum_{i=1}^{n}BC_i\)

现在要你将这些红球蓝球一一配对,配对的价值为两球所在点之间的曼哈顿距离,请你求出配对完它们的最大价值和。n <= 1000

这个题一开始,我的想法是暴力建边网络流,然后发现边的数量到了n ^ 2 , 网络流肯定是不能跑出来的。

这就要用到曼哈顿距离的一个性质,枚举两个符号是+还是-,可以得到四个,然后最大的就是正确的那个。

这样边数就成了n的级别。 跑费用流即可。

#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdlib>
#include<string>
#include<cstdio>
#include<vector>
#include<queue>
#include<cmath>
#include<set>
#include<map>
using namespace std;
typedef long long LL;
#define int long long
const int N = 1010 , inf = 1e16;
inline int read()
{
	register int x = 0 , f = 0; register char c = getchar();
	while(c < '0' || c > '9') f |= c == '-' , c = getchar();
	while(c >= '0' && c <= '9') x = (x << 3) + (x << 1) + c - '0' , c = getchar();
	return f ? -x : x;
}

int n , S , T , cnt = 1;
int head[N << 1] , f[N << 1] , pre[N << 1] , vis[N << 1];
LL dis[N << 1];
struct node{ int x , y , c; }p1[N] , p2[N];
struct edge{ int v , nex , c; LL val; }e[(N * (N + 2)) << 1];
inline void add(int u , int v , int c , int val)
{
	e[++cnt].v = v; e[cnt].nex = head[u]; e[cnt].c = c; e[cnt].val = val; head[u] = cnt;
	e[++cnt].v = u; e[cnt].nex = head[v]; e[cnt].c = 0; e[cnt].val = -val; head[v] = cnt;
}

queue<int> q;
bool spfa()
{
	for(int i = 1 ; i <= T ; ++i) dis[i] = -inf , pre[i] = 0 , f[i] = 0; dis[S] = 0; f[S] = inf; q.push(S);
	while(q.size())
	{
		int x = q.front(); q.pop(); vis[x] = 0;
		for(int i = head[x] , v; i ; i = e[i].nex)
		{
			v = e[i].v;
			if(e[i].c && dis[v] < dis[x] + e[i].val)
			{
				dis[v] = dis[x] + e[i].val; pre[v] = i; f[v] = min(f[x] , e[i].c);
				if(!vis[v]) vis[v] = 1 , q.push(v);
			}
		}
	}
	return dis[T] != -inf;
}

LL calc()
{
	LL cost = 0;
	while(spfa())
	{
		cost += (LL)f[T] * dis[T]; int t = T , i;
		do i = pre[t] , e[i].c -= f[T] , e[i^1].c += f[T] , t = e[i^1].v; while(t != S);
	}
	return cost;
}

inline int Dis(node a , node b) { return abs(a.x - b.x) + abs(a.y - b.y); }

signed main()
{
	n = read(); int t1 = n + n + 1 , t2 = t1 + 1 , t3 = t2 + 1 , t4 = t3 + 1; S = t4 + 1; T = S + 1;
	for(int i = 1 ; i <= n ; ++i) p1[i].x = read() , p1[i].y = read() , p1[i].c = read();
	for(int i = 1 ; i <= n ; ++i) p2[i].x = read() , p2[i].y = read() , p2[i].c = read();
	for(int i = 1 ; i <= n ; ++i)
	{
		add(S , i , p1[i].c , 0);
		add(i , t1 , inf , p1[i].x + p1[i].y);
		add(i , t2 , inf , p1[i].x - p1[i].y);
		add(i , t3 , inf , -p1[i].x + p1[i].y);
		add(i , t4 , inf , -p1[i].x - p1[i].y);
	}
	
	for(int i = 1 ; i <= n ; ++i)
	{
		add(i + n , T , p2[i].c , 0);
		add(t1 , i + n , inf , -p2[i].x - p2[i].y);
		add(t2 , i + n , inf , -p2[i].x + p2[i].y);
		add(t3 , i + n , inf , p2[i].x - p2[i].y);
		add(t4 , i + n , inf , p2[i].x + p2[i].y);
	}
	
	cout << calc() << '\n'; return 0;
}

posted @ 2020-06-05 17:25  沙野博士  阅读(204)  评论(0)    收藏  举报