多校联测11 THUSC

题目大意

有 n n n个人参加 T H U S C THUSC THUSC,第 i i i个人算法场和工程场的成绩分别为 a i a_i ai和 b i b_i bi,保证不存在两个人两项成绩都相同。

现在招办想给他们排个名。一个合理的排名方案是分别给算法场和工程场一个正的权重 x , y x,y x,y,然后按照 a i x + b i y a_ix+b_iy aix+biy降序排名,相同的可以按照他们的心情排。

对于一个成绩分布,求有多少种可能的排名。

输出答案对 998244353 998244353 998244353取模后的值。

1 ≤ n ≤ 1000 , 1 ≤ a i , b i ≤ 1 0 9 1\leq n\leq 1000,1\leq a_i,b_i\leq 10^9 1≤n≤1000,1≤ai,bi≤109

题解

因为当 x 1 ≠ x 2 , y 1 ≠ y 2 x_1\neq x_2,y_1\neq y_2 x1=x2,y1=y2且 x 1 y 1 = x 2 y 2 \dfrac{x_1}{y_1}=\dfrac{x_2}{y_2} y1x1=y2x2时,权重为 x 1 , y 1 x_1,y_1 x1,y1和 x 2 , y 2 x_2,y_2 x2,y2所得的排名是相同的。所以,我们只需考虑比值 x y \dfrac xy yx,令比值为 w = x y w=\dfrac xy w=yx。

首先,我们把所有满足题意的 w w w分成两个类型:

  • 能使有至少两个人分数一样
  • 能使任意两个人的分数不一样

因为存在分数一样的时候分数相同的任意排,所以所有第二种类型的 w w w所得的排名都一定会在第一种类型的 w w w所得的排名中出现。(可以自己举几个例子试一下)。

那么,我们只需要求能使至少两个人分数一样的 w w w即可,而能使至少两个人分数一样的 w w w不超过 n 2 n^2 n2个(对于任意两对 a i , b i a_i,b_i ai,bi,只有一个 w w w满足两者排名相同)。

枚举两个人 i , j i,j i,j,求出能使 i , j i,j i,j两个人分数相等的 w w w。然后,我们只需考虑在每个不同的 w w w作为比值的情况下有多少种排名即可。

这怎么处理呢?我们在求出能使 i , j i,j i,j两个人分数相等的 w w w的时候,在图上将 i , j i,j i,j两点连一条权值为 w w w的双向边,再从虚拟节点 n + 1 n+1 n+1向 i i i连一条权值为 w w w的单向边。等到全部边都连完了之后,对于每一个不同的 w w w,看虚拟节点有多少个权值为 w w w的边连出去。对于每个从虚拟节点指向的点 i i i,看从 i i i经过权值为 w w w的边能够到达多少个点(包括自己),这些点当前的分数是与 i i i的分数相同的。若能够到达 t t t个点,则贡献为 t ! t! t!。换句话说,设 v w v_w vw表示比值为 w w w时不同排名的数量,则当得知 i i i能到达 t t t个点时, v w = v w × t ! v_w=v_w\times t! vw=vw×t!。

在处理边的时候,我们可以将每个点连出的边按边权从小到大排序,那么走过的边就不用再走了,能保证每条边只会被遍历一次。

因为相邻的两个能使两人分数相等的 w w w之间的那种排名会被计算两次,所以要减去,那么答案为 1 + ∑ ( v w − 1 ) 1+\sum (v_w-1) 1+∑(vw−1)。

枚举 i , j i,j i,j的时间复杂度为 O ( n 2 ) O(n^2) O(n2),排序的时间复杂度为 O ( n 2 log ⁡ n ) O(n^2\log n) O(n2logn)。总共有不超过 3 n 2 3n^2 3n2条边,每条边最多遍历一次,所以图上部分的时间复杂度为 O ( n 2 ) O(n^2) O(n2)。所以,总时间复杂度为 O ( n 2 log ⁡ n ) O(n^2\log n) O(n2logn)。

code

cpp 复制代码
#include<bits/stdc++.h>
using namespace std;
const long long mod=998244353;
const int N=3000000;
int n,w1=0,now,z[N+5],to[1005];
long long ans=0,nd,a[1005],b[1005],jc[1005];
struct node{
	int a,b;
	long long x,y;
	bool operator==(const node ax)const{
		return x==ax.x&&y==ax.y;
	}
	bool operator!=(const node ax)const{
		return x!=ax.x||y!=ax.y;
	}
}w[1000005];
struct pr{
	int x,v;
};
vector<pr>g[1005];
bool cmp(node ax,node bx){
	if(ax.x!=bx.x) return ax.x<bx.x;
	return ax.y<bx.y;
}
long long gcd(long long i,long long j){
	while(j){
		i%=j;swap(i,j);
	}
	return i;
}
void dfs(int u,int t){
	if(z[u]==t) return;
	z[u]=t;
	++now;
	for(int i=to[u];i<g[u].size();++i,++to[u]){
		if(w[g[u][i].v]!=w[t]) break;
		dfs(g[u][i].x,t);
	}
}
int main()
{
	scanf("%d",&n);
	for(int i=1;i<=n;i++){
		scanf("%lld%lld",&a[i],&b[i]);
	}
	jc[0]=1;
	for(int i=1;i<=1000;i++) jc[i]=jc[i-1]*i%mod;
	for(int i=1;i<=n;i++){
		for(int j=i+1;j<=n;j++){
			long long x=b[i]-b[j],y=a[j]-a[i],d;
			if(x*y<0) continue;
			if(x==0||y==0) continue;
			if(x<0){
				x=-x;y=-y;
			}
			d=gcd(x,y);
			x/=d;y/=d;
			w[++w1]=(node){i,j,x,y};
		}
	}
	sort(w+1,w+w1+1,cmp);
	for(int i=1;i<=w1;i++){
		g[w[i].a].push_back((pr){w[i].b,i});
		g[w[i].b].push_back((pr){w[i].a,i});
		g[n+1].push_back((pr){w[i].a,i});
	}
	for(int i=1;i<=w1;i++){
		if(i>1&&w[i]==w[i-1]) continue;
		nd=1;
		for(int j=to[n+1];j<g[n+1].size();++j,++to[n+1]){
			if(w[g[n+1][j].v]!=w[i]) break;
			now=0;
			dfs(g[n+1][j].x,i);
			nd=nd*jc[now]%mod;
		}
		ans=(ans+nd)%mod;
		ans=(ans-1+mod)%mod;
	}
	ans=(ans+1)%mod;
	printf("%lld",ans);
	return 0;
}
相关推荐
StrokeAce1 小时前
linux桌面软件(wps)内嵌到主窗口后的关闭问题
linux·c++·qt·wps·窗口内嵌
家有狸花4 小时前
VSCODE驯服日记(三):配置C++环境
c++·ide·vscode
dengqingrui1235 小时前
【树形DP】AT_dp_p Independent Set 题解
c++·学习·算法·深度优先·图论·dp
C++忠实粉丝5 小时前
前缀和(8)_矩阵区域和
数据结构·c++·线性代数·算法·矩阵
ZZZ_O^O5 小时前
二分查找算法——寻找旋转排序数组中的最小值&点名
数据结构·c++·学习·算法·二叉树
小飞猪Jay8 小时前
C++面试速通宝典——13
jvm·c++·面试
rjszcb9 小时前
一文说完c++全部基础知识,IO流(二)
c++
小字节,大梦想9 小时前
【C++】二叉搜索树
数据结构·c++
吾名招财10 小时前
yolov5-7.0模型DNN加载函数及参数详解(重要)
c++·人工智能·yolo·dnn
我是哈哈hh10 小时前
专题十_穷举vs暴搜vs深搜vs回溯vs剪枝_二叉树的深度优先搜索_算法专题详细总结
服务器·数据结构·c++·算法·机器学习·深度优先·剪枝