首页 > 其他分享 >洛谷P10136 暨 USACOJan2024S T3 题解

洛谷P10136 暨 USACOJan2024S T3 题解

时间:2024-02-07 10:45:11浏览次数:22  
标签:约数 题解 ll USACOJan2024S P10136 枚举

题意简述

原题已经很简了,没有什么简述的必要了。

思维路径

请注意本题解可以保证正确性但不保证如果有极端的 Hack 数据能够通过。

拿到这道题上来的暴力想必是很容易的,即枚举每个 \(L\) 判断是否合法。

接着我们就考虑优化,减少需要枚举的 \(L\) 的量。题目中要求余数最多有 \(3\) 种,而余数相同对的数相减所得的数的约数才有可能作为 \(L\)。

我们对 \(a\) 数组进行排序,则由于抽屉原理,最小的四个数一定有两个是同余的,枚举他们的差的约束,如果算过就直接跳过,否则就暴力算是否满足。如果存在一个数是满足上述条件的,那么它的约数就一定也满足,可以直接标记。

在目前的数据下这个做法是可以通过的,但是要是这个差很大就 emmmm。

AC 代码

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const ll N=100009;
ll n,a[N],nn,ans,x;
map<ll,ll> ok,vst;

void input(){
	cin>>n;
	for(ll i=1;i<=n;i++) cin>>a[i];
	sort(a+1,a+n+1);
	for(ll i=1,j=1;i<=n;i++){
		if(a[i]==a[i-1]) continue;
		a[j]=a[i]; nn=j; j++;
//		cout<<a[j];
	}
	n=nn;
}

void cal(ll u){
	vst.clear();
	ll cnt=0;
	for(ll i=1;i<=n;i++){
		if(!vst[a[i]%u]) cnt++;
		vst[a[i]%u]=1;
	}
	if(cnt<=3){
		for(ll i=1;i*i<=u;i++){
			if(u%i) continue;
			if(!ok[i]) ans+=i;
			ok[i]=1;
			if(!ok[u/i]) ans+=u/i;
			ok[u/i]=1;
		}
	}
}

void solve(){
	if(n<=3){
		cout<<(1+a[1]/4)*(a[1]/4)/2;
		return;
	}
	ok[1]=ok[2]=ok[3]=1; ans=6;
	x=a[2]-a[1];
	for(ll i=1;i*i<=x;i++){
		if(x%i==0){
			if(x/i<=a[i]/4&&(!ok[x/i])) cal(x/i);
			if(i<=a[i]/4&&(!ok[i])) cal(i);
		}
	}
	x=a[3]-a[1];
	for(ll i=1;i*i<=x;i++){
		if(x%i==0){
			if(x/i<=a[i]/4&&(!ok[x/i])) cal(x/i);
			if(i<=a[i]/4&&(!ok[i])) cal(i);
		}
	}
	x=a[4]-a[1];
	for(ll i=1;i*i<=x;i++){
		if(x%i==0){
			if(x/i<=a[i]/4&&(!ok[x/i])) cal(x/i);
			if(i<=a[i]/4&&(!ok[i])) cal(i);
		}
	}
	x=a[3]-a[2];
	for(ll i=1;i*i<=x;i++){
		if(x%i==0){
			if(x/i<=a[i]/4&&(!ok[x/i])) cal(x/i);
			if(i<=a[i]/4&&(!ok[i])) cal(i);
		}
	}
	x=a[4]-a[2];
	for(ll i=1;i*i<=x;i++){
		if(x%i==0){
			if(x/i<=a[i]/4&&(!ok[x/i])) cal(x/i);
			if(i<=a[i]/4&&(!ok[i])) cal(i);
		}
	}
	x=a[4]-a[3];
	for(ll i=1;i*i<=x;i++){
		if(x%i==0){
			if(x/i<=a[i]/4&&(!ok[x/i])) cal(x/i);
			if(i<=a[i]/4&&(!ok[i])) cal(i);
		}
	}
	cout<<ans;
}

int main(){
	input();
	solve();
	return 0;
}

标签:约数,题解,ll,USACOJan2024S,P10136,枚举
From: https://www.cnblogs.com/lemon-cyy/p/18010713

相关文章

  • CF1408F Two Different 题解
    解题思路先考虑如何将一堆数变为相同的。显然,这里有一个条件\(n=2^k,k\in\mathbbZ\),证明如下:每次操作只能将两个数变为相同的,那么一个数在使得其他数变为相同数的操作中(我们不妨将所有数进行这种操作称为一轮操作),一个数最多被使用一次;按照错位操作,即第一轮\(1\)和\(2\)......
  • CF1428D Bouncing Boomerangs 题解
    解题思路很简单的贪心。观察发现以下性质:当\(a_i=2\)时,这一行一定只有两个目标,且第二个目标一定位于一个\(a_j=1\)的格子内;当\(a_i=3\),那么当前列右边某一列发生转向的地方,\(a_j\not=0\);那么这道题就基本已经做出来了。因为\(a_i=3\)的格子转向格可以在任意非\(0\)......
  • CF1401E Divide Square 题解
    解题思路其实多看几组也能发现块数等于交点的数量加上两个端点都在边上的线段数量再加一。证明如下(图见样例):对于两条只有一个端点位于边上的线段,因为保证有一个端点位于边上,那么这两条线段的交点一定会和已存在的点、边构成一个新的矩形;对于其中有一条为两个端点均位于边上的......
  • CF1401F Reverse and Swap 题解
    解题思路巧妙的数据结构题,非常巧妙的利用的线段树的奇妙性质。操作逐条维护:Replace:直接线段树上单点修改;Sum:线段树查询区间和;Reverse:考虑线段树的形态。线段树第\(i\)层(除最后一层)有\(2^{i-1}\)个节点,那么将所有\(i\ge1\)的区间\([(i-1)\times2^k,i\times2^k]\)......
  • CF1408E Avoid Rainbow Cycles 题解
    解题思路第一眼看过去感觉不是很可做……但是我们可以发现,如果有两个点在不同的集合中出现过,那么一定会存在彩虹环,那么两个点最多出现一次。同时我们考虑将题意转化一下,变成求最大能选取的点,使得不出现彩虹环。根据刚刚的性质,我们可以考虑每个点向它所在的集合连一条边权为\(a_......
  • CF1338C Perfect Triples 题解
    解题思路没什么好说的,就是打表找规律……表在这里不难发现,三元组中第一个数的最后两位按照\(00\to01\to10\to11\)的顺序变化,其他位也一样,同样,第二个数和第三个数中每两位分别按照\(00\to10\to11\to01\)和\(00\to11\to01\to10\)的顺序变化,且与第一个数对应变化......
  • CF1379C Choosing flowers 题解
    解题思路不是那么显然的,当只选一种\(b_i\)或全选\(a_i\)时最优。那么我们可以先对\(a_i\)从大到小排序,枚举每一个\(b_i\),然后二分找到第一个大于等于\(b_i\)的\(a_j\),判断\(a_1\sima_j\)中是否包含\(a_i\),如果包含,当前的答案为\(\displaystyle\left(\sum_{k=1}^{......
  • CF1385F Removing Leaves 题解
    解题思路简单贪心,优先选择叶子节点最多的,这样能够保证一定能把所有能删的都删了。因为要建一个可删除的图,所以我们可以使用set来存边,不然就需要维护一堆东西……那么我们肯定是从有叶子节点的点向父亲更新的,那么我们每次选择叶子节点最多的点,然后删除\(k\)个叶子,判断一下删......
  • CF1415E New Game Plus! 题解
    解题思路简单贪心题,我们可以把整个序列看作\(k+1\)个可重集,首先可以得到一个显然的结论:较大的数一定比较小的数先放入一个集合中。同样,由于每一轮\(ans\getsans+\maxsum_i\),其中\(sum_i\)表示第\(i\)个集合的元素和,那么,我们一定会将当前的元素放入当前和最大的哪个集合......
  • CF1416B Make Them Equal 题解
    解题思路观察可以发现,每次操作后序列元素之和不变,那么我们可以将每一次操作看作是\(a_i\)向\(a_j\)移动了\(i\timesx\)。由此可得,若序列和\(sum\bmodn\not=0\),那么一定无解,否则一定存在一个合法的操作方案。因为每次移动时移动的数应为\(i\)的倍数,所以\(a_1\)可以向......