首页 > 其他分享 >231101校内赛

231101校内赛

时间:2023-11-05 10:44:07浏览次数:37  
标签:pre 校内 limits int sum times 231101 mod

T1 茵蒂克丝

模拟题,用一个栈模拟就完了,挺简单的有人没看见是非负数

#include<bits/stdc++.h>
#define pii pair<int,int>
#define fi first
#define se second
#define N 1000010
using namespace std;
int n,k,a[N],ans[N],cnt,top,tot;
pii stk[N],b[N];
bool cmp(pii x,pii y){
	return x.se==y.se?x.fi<y.fi:x.se<y.se;
}
void del(){
	while(top>1&&stk[top-1].fi==stk[top].fi){
		stk[top-1] = {stk[top-1].fi+1,stk[top].se};
		top--;
	}
}
int main(){
	freopen("index.in","r",stdin);
	freopen("index.out","w",stdout);
	ios::sync_with_stdio(0);
	cin.tie(0);cout.tie(0);
	cin>>n>>k;
	for(int i = 1;i<=n;i++){
		cin>>a[i];
		while(top>0&&stk[top].fi<a[i]){
			b[++tot] = stk[top];
			stk[top].fi++;
			del();
		}
		stk[++top] = {a[i],i};
		del();
	}
	while(1){
		if(top==1&&stk[top].fi==30) break;
		b[++tot] = stk[top];
		stk[top].fi++;
		del();
		if(top==1&&stk[top].fi==30) break;
	}
	/*cerr<<tot<<"\n";
	for(int i = 1;i<=tot;i++)
		cerr<<b[i].fi<<" "<<b[i].se<<"\n";*/
	int tmp = 1;
	while(tot<k){
		while(b[tmp].fi==0) tmp++;
		b[tmp].fi--;
		b[++tot] = b[tmp];
	}
	sort(b+1,b+tot+1,cmp);
	/*cerr<<tot<<"\n";
	for(int i = 1;i<=tot;i++)
		cerr<<b[i].fi<<" "<<b[i].se<<"\n";*/
	tot = 1;
	for(int i = 1;i<=n;i++){
		ans[++cnt] = a[i];
		while(b[tot].se==i){
			ans[++cnt] = b[tot].fi;
			tot++;
		}
	}
	for(int i = 1;i<=cnt;i++)
		cout<<ans[i]<<" ";
	return 0;
}

T2 初春饰利

\(\mathcal O(nq\log n)\) 飞过去了?

就是每一行预处理当前行排列之后的答案

具体做法就是先把当前行踢出矩阵找每一列最大值,再用当前行更新找到最佳答案

每一次 \(n\) 行分别二分查找答案,输出最大的那个

优化掉一个 \(n\) 也是十分简单

预处理多处理一个每个答案的最大要求是多少

#include<bits/stdc++.h>
#define N 1010
using namespace std;
int n,q,b[N][N],a[N][N];
struct stlist{
	int st[N][20];
	inline int query(int l,int r){
		if(l>r) return 0;
		int k = log2(r-l+1);
		return max(st[l][k],st[r-(1<<k)+1][k]);
	}
	inline void pre(){
		for(int i = 1;(1<<i)<=n;i++)
			for(int j = 0;j+(1<<(i-1))<=n;j++)
				st[j][i] = max(st[j][i-1],st[j+(1<<(i-1))][i-1]);
	}
}T[N];
int main(){
	freopen("uiharu.in","r",stdin);
	freopen("uiharu.out","w",stdout);
	ios::sync_with_stdio(0);
	cin.tie(0);cout.tie(0);
	cin>>n>>q;
	for(int i = 1;i<=n;i++)
		for(int j = 1;j<=n;j++){
			cin>>T[j].st[i][0];
			a[i][j] = T[j].st[i][0];
		}
	for(int i = 1;i<=n;i++)
		T[i].pre();
	for(int i = 1;i<=n;i++){
		for(int j = 1;j<=n;j++)
			b[i][j] = max(T[j].query(1,i-1),T[j].query(i+1,n));
		sort(a[i]+1,a[i]+n+1);
		sort(b[i]+1,b[i]+n+1);
		int cnt = n;
		for(int j = 1;j<=n;j++)
			if(b[i][j]<a[i][cnt]){
				b[i][j] = a[i][cnt];
				cnt--;
			}
		sort(b[i]+1,b[i]+n+1);
	}
	while(q--){
		int x,ans = 0;cin>>x;
		for(int i = 1;i<=n;i++){
			int tmp = lower_bound(b[i]+1,b[i]+n+1,x)-b[i];
			ans = max(ans,n-tmp+1);
			if(ans==n) break;
		}
		cout<<ans<<"\n";
	}
	return 0;
}

T3 御坂美琴

麻烦的很,看懂了题解但是自己说不出来系列

首先定义 \(s_i\) 表示 \(\sum \limits_{j=1}^{i} j\) ,再定义 \(sq_i\) 表示 \(\sum \limits_{j = 1}^{i} j^2\)

为了方便假设 \(a_i\) 互不相同,显然每一个数都是答案独立的

设 \(F(P_2,S_2)\) 为 \(i \in P_2,a_i \in S_2\) 的方案总数,\(f_2(i)\) 表示 \(i\in P_2\) 的方案数,\(g_2(i)\) 表示 \(i\in S_2\) 的方案数

因为对于 \(i,a_i\) 的限制是独立的,所以可以写为 \(f_2(i) \times g_2(i)\)

但是又因为对于 \(i,a_i\) 的限制是独立的,所以 \(f_2=g_2\)

根据定义 \(f_2(i)\) 表示 \(i\in P_2 \subseteq P_1 \subseteq [1,n]\) ,不妨反过来考虑,将其定义为在位置 \(i\) 两侧嵌套两层括号的总方案数

对于左右括号来说也是相互独立的,先考虑左括号:第一层左括号在 \(j\) 的两侧时,第二层只能在套在 \([1,j]\) ,即 \(s_j\) 种,同理右括号方案数为 \(s_{n-j+1}\)

所以 \(f_2(i) = s_i\times s_{n-i+1}\) 由此推出 \(a_i\) 对 \(F(P_2,S_2)\) 的贡献为 \(f_2(i)\times g_2(i) = s_i\times s_{n-i+1} \times s_{a_i} \times s_{n-a_i+1}\)

再求 \(a_i\) 对于 \(F(P_1,S_1)\) 的贡献,即所有 \(i\in P_1,a_i \in S_1\) 的方案数

注意此时每一个区间 \(P_1(S_1)\) 并不是只计算一次,还要考虑子区间 \(P_2(S_2)\) 的个数

我们可以如法炮制设 \(f_1(i),g_1(i),F(P_1,S_1) = f_1(i) \times g_1(i)\) ,同理可得 \(f_1 = g_1\)

枚举 \(P_1\) 左右端点 \(l,r(l\le i,i\le r)\) 再乘上子区间个数,可以得:

\[f(i) = \sum\limits_{l = 1}^i\sum\limits_{r = i}^n\binom {r-l+2} 2 \]

注意此处 \(r-l+2\) 是因为区间长度 \(P_1\) 为 \(r-l+1\) ,而长度为 \(len\) 的区间有 \(\binom {len+1} 2\) 个子区间

上式可化为

\[\frac 1 2\sum\limits_{l = 1}^i\sum\limits_{r = i}^n (r-l+2)(r-l+1) \]

拆开可以得到

\[ \frac 1 2\sum\limits_{l = 1}^i\sum\limits_{r = i}^n r^2+l^2-2lr+3r-3l+2 \]

上面可以 \(\mathcal O(n)\) 预处理 \(\mathcal O(1)\) 求

化简一下其实并不简

\[2f(i) = i(sq_n-sq_{i-1})+sq_i(n-i+1)-2s_i(s_n-s_{i-1})\\ +3i(s_n-s_{i-1}-3s_i(n-i+1)+2i(n-i+1)) \]

好像已经解决了这道题,但是还有一点没有解决

我们一开始假设都不相同

但是实际上只有对位置上限制的情况需要重新计算,对数值上的并不需要重算

根据上述假设,显然有 \(a_i = a_j(i<j)\) ,对于 \(a_j\) 来说,所有左端点在 \(i\) 左边的 \(P_1(P_2)\) 都没有贡献

因此假设 \(f'_1(pre,i)\) 表示 \(P_1\) 左端点在 \(pre\) 右侧的总情况数,表达式如下

\[f'_1(pre,i) = \sum\limits_{l = pre+1}^i \sum\limits_{r = i}^n \binom {r-l+2} 2 \]

对比 \(f_1(i)\) ,可以发现 \(f'_1\) 相比较于 \(f_1\) 只有左端点 \(l\) 下界改变,即 \(f_1(i) = f'_1(1,i)\)

因此,\(a_i\) 对 \(F(P_1,S_1)\) 的贡献等于 \(f'_1(pre,i)\times f_1(a_i)\) ,其中 \(pre_i\) 定义为在 \(a_i\) 之前与 \(a_i\) 相等的最右边的数的位置,没有则为 \(0\)

同理,可以推出 \(f'_2(pre,i) = \sum\limits_{j = pre+1}^i j\) 因为嵌套的第一层左括号只能在 \(pre\) 右边,\(i\) 左边

对于 \(f'_2,f_2\) 来说,同样只有左端点 \(l\) 的下界改变,即 \(f_2(i) = f'_2(1,i)\)

那么 \(a_i\) 对答案的贡献即为 \(f'_1(pre,i)\times f_1(a_i) - f'_2(pre,i)\times f_2(a_i)\)

距离解出这道题还差一个总方案数,因为 \(P_1,P_2\) 与 \(S_1,S_2\) 等价,所以可由一边的数量平方得到

不妨枚举 \(P_1\) 区间长度 \(len\) ,那么就有 \(n-len+1\) 种情况,此时其子区间 \(P_2\) 有 \(\binom {len+1} 2\) 种情况

不难发现 \(s_i = \binom {i+1} 2\) 所以:

\[cnt = \sum\limits_{i = 1}^n (n-i+1)s_i \]

至此,本题就解决了,时间复杂度 \(\mathcal O(n)\)抄袭题解讲解完毕

#include<bits/stdc++.h>
#define N 500010
#define mod ((int)1e9+7)
#define int long long
using namespace std;
int n,ans,cnt,s[N],sq[N],t[N];
int ksm(int x,int y){
	int res = 1;
	while(y){
		if(y&1) res = 1ll*res*x%mod;
		x = 1ll*x*x%mod;
		y>>=1;
	}
	return res;
}
int inv(int x){
	return ksm(x,mod-2);
}
int calcs(int l,int r){
	return (s[r]-s[l-1]+mod)%mod;
}
int calcsq(int l,int r){
	return (sq[r]-sq[l-1]+mod)%mod;
}
int f1(int l,int r){
	return (calcsq(l,r)*(n-r+1)%mod+calcsq(r,n)*(r-l+1)%mod-
		2*calcs(l,r)*calcs(r,n)%mod-3*calcs(l,r)*(n-r+1)%mod+
		3*calcs(r,n)*(r-l+1)%mod+2*(n-r+1)*(r-l+1)%mod+mod)*(mod+1)/2%mod;
}
int f2(int l,int r){
	return calcs(l,r)*s[n-r+1]%mod;
}
signed main(){
	freopen("misaka.in","r",stdin);
	freopen("misaka.out","w",stdout);
	ios::sync_with_stdio(0);
	cin.tie(0);cout.tie(0);
	cin>>n;
	for(int i = 1;i<=n;i++){
		s[i] = (s[i-1]+i)%mod;
		sq[i] = (sq[i-1]+i*i)%mod;
	}
	int pre;
	for(int i = 1;i<=n;i++){
		int x;cin>>x;
		pre = t[x]+1;t[x] = i;
		ans = (ans+f1(pre,i)*f1(1,x)-f2(pre,i)*f2(1,x)%mod+mod)%mod;
		cnt = (cnt+(n-i+1)*s[i])%mod;
	}
	cout<<ans*inv(cnt*cnt%mod)%mod;
	return 0;
}

标签:pre,校内,limits,int,sum,times,231101,mod
From: https://www.cnblogs.com/cztq/p/17810302.html

相关文章

  • 每日总结20231101
    代码时间(包括上课)6h代码量(行):100行博客数量(篇):1篇相关事项:1、今天是周三,上的是软件构造,软件构造讲的是对于csv文件的读写操作。2、今天下午开会然后上班,把erp的作业也完成了,需要加速看软考了。3、今天还打算看看软件设计师相关的题目,我要过!......
  • 20231101构造题记录
    20231101构造题记录A.人生的经验可以对于每个长度为\(l-1\)的串建一个点,每个点有\(c\)个后继状态,也有\(c\)个入边,所以一定可以找到一个欧拉路因此答案为\(c^l+l-1\)即所有可能的串首尾相接拼起来的长度考虑用一个圈套圈求欧拉路,即每次拓展一个点,用栈维护,如果不能继......
  • 20231101
    T1考虑先跑m遍KMP,记录下每个可以造成贡献的起点,再直接\(O(n^2)\)DP就可以了。思路比较好想,据说可以AC自动机做得分:没交上去......T2观察前50%的数据,发现O(nk)可以直接过。再考虑第四个子任务。所有颜色相同,那么其他的K-1种颜色都是连通图,通过边数判断一下是否是树即可。正解考虑......
  • 231030校内赛
    T1新的阶乘有三种做法,第一种也就是我写的这种容易被评测机波动坑,复杂度玄学考虑处理出每个数的质因数,然后就暴力除每个数的质因数的种类次非常的简单,也容易被卡第二种是与第一种差距不大,就是在线性筛中处理出质因数之后,再对每个数除以线筛中处理的质因数,将它的答案加到除数和......
  • 231023校内赛
    T1区间题解很容易想到的一点是如果\(k\)足够大,那么把区间单独放到一个组里总比多个区间在一个组优对于多个区间来说,区间之间如果两两不包含的话这道题会是比较好做的就可以注意到如果一个大区间包含了一个小区间,那么大区间要么单独一组,要么和小区间同一组,这样会是比较优的......
  • 231019校内赛
    T1机器人题解傻逼题,但是有人\(90\)分一开始十分想直接暴力\(2^n\)判断每一步选不选求出所有可能性但是会发现它有制约关系有些步走了之后,有些就必须走了所以需要用一个数组记录当前位置走没走过,或者是不是障碍注意走没走过不能直接赋值\(1,0\)因为回溯时会直接将前......
  • 231019校内赛
    T1购买饮料题解简单且傻逼的题目有人更傻逼没做出来很容易就会想去拿最后能喝多少瓶去做未知量来求然后就有一个严重的问题,它会赊账非常明显这样算是不得行的那么考虑换个思路以能喝多少套饮料为未知量,先除去第一套,免得一套都买不起时赊账买了饮料然后将剩余的钱除以\(......
  • [校内]此方(konata)
    2023-10-14题目LittleBrother题目描述难度&重要性(1~10):7题目来源CQYC题目算法几何,二分解题思路Sol我们知道,对于两个圆,我们无非就只有三种情况:相离,相切,相交。而这道题目是不允许其他圆相交,而两个圆不相交只有两种情况:包含,不包含。根据垂径定理得知,过线段两端的圆的......
  • 231011校内赛
    T1树上的数题解比上一次好一些的第一题不过我还是没做出来一眼树形\(dp\)不过状态设计和转移不是很好列容易想到对于子树枚举,记录\(f_{i,j}\)表示\(i\)的子树空出了\(j\)个点时的方案数对于每一个节点的初始状态都是\(f_{i,0}=n-dep_i\\\f_{i,1}=1\)为......
  • 231009校内赛
    T1里群题解阴间第一题题目中有一个很明显的建图方法就是对于第\(i\)天入群的人在第\(j\)天退群那么就在\(i,j\)之间连一条边首先有一个结论,管理员个数不大于\(3\)对于这个结论,证明如下:首先第一次删除出现后就一定需要两个管理员了如果某次删除只删掉了某一个管理......