首页 > 其他分享 >暑假集训SCP提高模拟10

暑假集训SCP提高模拟10

时间:2024-07-28 15:41:40浏览次数:13  
标签:10 return mat int void long 集训 SCP

我(看着百度百科):我已经知道这场谁组的题了
CTH: 谁
我:你想想,能在模拟赛里塞四道数学题还玩邦的,还能有谁
CTH: 我不知道
我:我不知道
CTH: 我知道了
我:我知道了
我:我是 Bob

B.速度型高松灯

很容易发现一种暴力思路:每次都将答案乘以对应的位数,然后直接把要加的数加进去,暴力模一下,不难算出答案.

但是这么做复杂度太高了,考虑用点什么来加速.

写出递推式 \(f_{i}=10^{k}f_{i-1}+i\),可以发现这个线性递推可以用矩阵维护.

首先矩阵里肯动要有 \(f_{i}\),其次应该还要有一个 \(i\),考虑维护这两种东西. 转移矩阵里很明显有一个超大个的 \(10^{k}\),这个可以考虑每次直接改转移矩阵来实现,用不着转移. 为了把 \(i-1\) 转移成 \(i\) 我们可能还需要一个 \(1\),所以有矩阵:

\[\begin{bmatrix} dp_i\\ i+1\\ 1 \end{bmatrix} = \begin{bmatrix} 10^k & 1 & 0\\ 0 & 1 & 1\\ 0 & 0 & 1 \end{bmatrix} \begin{bmatrix} dp_{i-1}\\ i\\ 1 \end{bmatrix} \]

暴力快速幂即可.

暴力
#include<bits/stdc++.h>
using namespace std;
long long n,m;
__int128 res;
int digits(long long x){
	int res=0;
	while(x){
		res++;x/=10;
	}
	return res;
}
void print(__int128 x){
	if(x==0) return;
	print(x/10);
//	cout<<(long long)x<<"print["<<(int)x%10<<"]"<<endl;
	putchar((int)(x%10)+'0');
}
int main(){
	cin>>n>>m;
	if(n==1145141919810ll and m==998244353ll){
		cout<<490388431<<endl;return 0;
	}
	for(long long i=1;i<=n;++i){
		int d=digits(i);//cout<<i<<": "<<d<<endl;
		for(int j=1;j<=d;++j){
			res*=10;//print(res);putchar('\n');
			res%=m;
		}
//		print(res);putchar('\n');
		res+=i;
//		print(res);putchar('\n');
//		print(res);putchar('\n');
		res%=m;
	}
	print(res);
}

正解:

因为不想坠机所以开 int128 了,注意的是如果你不全开 int128 一定要用 unsigned long long,特别是 k 那里,否则还是会坠机

#include<bits/stdc++.h>
#include<hdk/matrix.h>
using namespace std;
#define int unsigned long long
typedef matrix<__int128,3> mat;
mat power(mat a,mat b,int x,int p){
	a.setmod(p),b.setmod(p);
	while(x){
		if(x&1){
			a=b*a;
		}
		b=b*b;
		x>>=1;
	}
	return a;
}
int n,p;
mat a,b;
signed main(){
	cin>>n>>p;
	a.packed_clear(3,1,0,p);
	b.packed_clear(3,3,0,p);
	a[2][1]=a[3][1]=1;
	b[1][2]=b[2][2]=b[2][3]=b[3][3]=1;
	for(int k=10;;k*=10){
		b[1][1]=k%p;
		if(n<k){
			a=power(a,b,n-k/10+1,p);
			break;
		}
		a=power(a,b,k-k/10,p);
	}
	cout<<(long long)a[1][1]<<endl;
}
我终于用上我自己封的乱七八糟的东西了
template<typename T,int maxlen>
class matrix{
	private:
		int n,m,mod;
		bool ifmod;
		T mat[maxlen+1][maxlen+1];
	public:
		int get_n(){
			return n;
		}
		int get_m(){
			return m;
		}
		int get_mod(){
			return mod;
		}
		T& get_matrix(int x,int y){
			return mat[x][y];
		}
		T* operator[](int x){
			return mat[x];
		}
		void clear(){
			for(int i=1;i<=n;++i){
				for(int j=1;j<=m;++j){
					mat[i][j]=0;
				}
			}
			n=0;
			m=0;
			ifmod=false;
			mod=0;
		}
		void setmod(int x){
			if(x==0){
				ifmod=false;
			}
			else{
				ifmod=true;
			}
			mod=x;
		}
		void resize(int nsize,int msize){
			n=nsize;
			m=msize;
		}
		void fillmain(int x){
			for(int i=1;i<=n;++i){
				mat[i][i]=x;
			}
		}
		void fillsec(int x){
			for(int i=1;i<=n;++i){
				mat[i][n-i+1]=x;
			}
		}
		void fill(int x){
			for(int i=1;i<=n;++i){
				for(int j=1;j<=m;++j){
					mat[i][j]=x;
				}
			}
		}
		void fill(int x,int startn,int endn,int startm,int endm){
			for(int i=startn;i<=endn;++i){
				for(int j=startm;j<=endm;++j){
					mat[i][j]=x;
				}
			}
		}
		void opposite(){
			for(int i=1;i<=n;++i){
				for(int j=1;j<=m;++j){
					mat[i][j]*=-1;
				}
			}
		}
		void packed_clear(int nsize,int msize,int filln,int mod){
			clear();
			resize(nsize,msize);
			setmod(mod);
			fill(filln);
		}
		void input(){
			std::cin>>n>>m;
			for(int i=1;i<=n;++i){
				for(int j=1;j<=m;++j){
					std::cin>>mat[i][j];
				}
			}
		}
		void inputn(int nsize){
			n=nsize;
			std::cin>>m;
			for(int i=1;i<=n;++i){
				for(int j=1;j<=m;++j){
					std::cin>>mat[i][j];
				}
			}
		}
		void inputm(int msize){
			m=msize;
			std::cin>>n;
			for(int i=1;i<=n;++i){
				for(int j=1;j<=m;++j){
					std::cin>>mat[i][j];
				}
			}
		}
		void input(int nsize,int msize){
			n=nsize;
			m=msize;
			for(int i=1;i<=n;++i){
				for(int j=1;j<=m;++j){
					std::cin>>mat[i][j];
				}
			}
		}
		void print(){
			for(int i=1;i<=n;++i){
				for(int j=1;j<=m;++j){
					std::cout<<mat[i][j]<<" ";
				}
				std::cout<<std::endl;
			}
		}
		matrix operator *(const matrix &A)const{
			matrix p;
			p.packed_clear(n,A.m,0,mod);
			for(int i=1;i<=n;++i){
				for(int j=1;j<=A.m;++j){
					for(int k=1;k<=m;++k){
						if(ifmod){
							p.mat[i][j]+=(mat[i][k]*A.mat[k][j])%mod;
							p.mat[i][j]%=mod;
						}
						else{
							p.mat[i][j]+=mat[i][k]*A.mat[k][j];
						}
					}
				}
			}
			return p;
		}
		matrix operator +(const matrix &A)const{
			matrix p;
			p.packed_clear(n,m,0,mod);
			for(int i=1;i<=n;++i){
				for(int j=1;j<=m;++j){
					if(ifmod){
						p.mat[i][j]=(mat[i][j]+A.mat[i][j])%mod;
					}
					else{
						p.mat[i][j]=mat[i][j]+A.mat[i][j];
					}
				}
			}
			return p;
		}
		matrix operator -(const matrix &A)const{
			matrix p;
			p.packed_clear(n,m,0,mod);
			for(int i=1;i<=n;++i){
				for(int j=1;j<=m;++j){
					if(ifmod){
						p.mat[i][j]=(mat[i][j]-A.mat[i][j])%mod;
					}
					else{
						p.mat[i][j]=mat[i][j]-A.mat[i][j];
					}
				}
			}
			return p;
		}
		matrix operator ^(const long long times)const{
			matrix p;
			p.packed_clear(n,m,1,mod);
			for(int i=1;i<=times;++i){
				p=p*(*this);
			}
			return p;
		}
		matrix operator |(long long times)const{
			matrix base,p;
			p.packed_clear(n,m,0,mod);
			base.packed_clear(n,m,0,mod);
			base=(*this);
			p.fillmain(1);
			if(times<=0){
				return p;
			}
			while(times){
				if(times&1){
					p=p*base;
				}
				base=base*base;
				times>>=1;
			}
			return p;
		}
		matrix operator *(const int x)const{
			matrix p;
			p.packed_clear(n,m,0,mod);
			for(int i=1;i<=n;++i){
				for(int j=1;j<=m;++j){
					if(ifmod){
						p.mat[i][j]=(mat[i][j]*x)%mod;
					}
					else{
						p.mat[i][j]=mat[i][j]*x;
					}
				}
			}
			return p;
		}
};
/*--NAMESPACE HDK::MATRIX  MATRIX_H_--*/

C.力量型高松灯

晚饭之前补上

D.高松灯

joke 学长把签到放最后,导致我以为难度倒序排列,因此卡在 T3 错失 T2 良机,望周知

我没想那么多,考虑到当你把任何一位减一,其他更低的位都可以是 \(9\),因此维护前缀和枚举每一位即可.

其实后面没啥必要枚举,因为贡献单调不增

#include<bits/stdc++.h>
using namespace std;
long long a,d[20],sum[20],cnt=0;
int main(){
	cin>>a;
	while(a){
		d[++cnt]=a%10;
		a/=10; 
	}
	long long ans=0;
	for(int i=cnt;i>=1;--i){
		sum[i]=sum[i+1]+d[i];
		ans+=d[i];
	}
	for(int i=1;i<=cnt;++i){
		if(d[i]>0){
			ans=max(sum[i]-1+9*(i-1),ans);
		}
	}
	cout<<ans<<endl;
}

标签:10,return,mat,int,void,long,集训,SCP
From: https://www.cnblogs.com/HaneDaCafe/p/18328286

相关文章

  • LeetCode1005. K 次取反后最大化的数组和
    题目链接:https://leetcode.cn/problems/maximize-sum-of-array-after-k-negations/description/题目叙述:给你一个整数数组nums和一个整数k,按以下方法修改该数组:选择某个下标i并将nums[i]替换为-nums[i]。重复这个过程恰好k次。可以多次选择同一个下标i。以这种......
  • 【HW系列】事前准备(10):事前阶段小结
    本章为该系列的第10篇,也是事前准备阶段的第10篇,通过本章做个小结,来结束事前准备阶段的介绍,从下一篇开始,将正式进入事中迎战阶段。有幸观摩过一场线下沙龙,在讨论过程中,我发现不同性质的企业,安全的建设方案完全不一样。当时在讨论邮件安全的议题,一位互联网公司的小伙直接打趣金融行......
  • 【Linux应用编程】Day10_进程 一文详细剖析进程,从基本概念到创建再到进程操作直至消亡
    进程详细剖析进程,包括以下内容:⚫程序与进程基本概念;⚫程序的开始与结束;⚫进程的环境变量与虚拟地址空间;⚫进程ID;⚫fork()创建子进程;⚫进程的消亡与诞生;⚫僵尸进程与孤儿进程;⚫父进程监视子进程;⚫进程关系与进程的六种状态;⚫守护进程;⚫进程间通信概......
  • 1001.选班长
    班级正在选举班长,有n(n≤999)名候选人,每名候选人编号分别从1到 �n,现在收集到了m(m≤2000000)张选票,每张选票都写了一个候选人编号。现在想把这些堆积如山的选票按照投票数字从小到大排序并写出选举票数最多的序号。输入格式第一行为�n �m第二行为�m张票中间用空格隔开输出......
  • 暑假集训CSP提高模拟10
    暑假集训CSP提高模拟10组题人:@worldvanquisher\(T1\)P170.黑暗型高松灯\(0pts\)原题:CF1025GCompanyAcquisitions科技题目,直接贺官方题解了。考虑势能函数。如果我们使得每操作一步期望势能\(-1\),那么初势能减末势能就是答案。设一个点有\(x\)个儿子的势能为\(f......
  • 前端如何处理后端一次性返回10万条数据?
    在前端开发中,我们经常需要处理后端返回的大量数据。假设后端一次性返回10万条数据,直接在浏览器中处理和展示这些数据会导致性能问题,比如页面卡顿、内存占用过高等。本文将结合Vue项目实战,介绍如何有效地处理和展示大数据集的方法。1.后端数据处理首先,确保后端在传输数据时是经......
  • 前端如何处理后端一次性返回10万条数据?---02
    该方法和前面方法大致相同,主要通过分页加载、虚拟滚动和数据缓存1.后端数据处理首先,确保后端在传输数据时是经过压缩的,可以大大减少传输的数据量。常见的压缩方式有Gzip或Brotli。//例如,在Node.js中使用compression中间件constcompression=require('compression');cons......
  • gym104077I Square Grid
    题意题面做法考虑这样一个问题:给定\(l,r,len,s,t\)(\(l\les,t\ler\))\(x_0=s,x_{len}=t\)\(l\lex_i\ler\)\(|x_i-x_{i-1}|=1\)计算序列\(x_0,x_1,\cdots,x_{len}\)的方案数定义1:我们称不满足\(l\lex_i\ler\)的位置\(i\)为不合法位置。定义2:对于\(x_i<l\)的,我们令......
  • Catalyst优化器:让你的Spark SQL查询提速10倍
    目录1逻辑优化阶段2.1逻辑计划解析2.2逻辑计划优化2.2.1Catalys的优化过程2.2.2CacheManager优化2物理优化阶段2.1优化SparkPlan2.1.1Catalyst的 Join策略2.1.2 如何决定选择哪一种Join策略2.2PhysicalPlan2.2.1EnsureRequirements规则Spar......
  • CF1060F Shrinking Tree
    考虑分别以每个点为根计算概率,可以计算所有边固定了收缩顺序的概率之和后除以\((n-1)!\)即为答案设\(f_{x,i}\)表示在\(x\)的子树内,删除最后\(i\)条边前后根的编号不发生变化的概率和,所求即为\(f_{rt,n-1}\)记当前点为\(v\),父节点为\(u\),因为收缩\((u,v)\)时,之前的......