首页 > 其他分享 >[44] (多校联训) A层冲刺NOIP2024模拟赛04

[44] (多校联训) A层冲刺NOIP2024模拟赛04

时间:2024-10-09 19:12:24浏览次数:8  
标签:std num string 04 int 44 long 联训 include

A.02 表示法

这题放在 ABC D 题我可能不会奇怪,但是它放模拟赛 T1 了

赛时代码
#include<bits/stdc++.h>
using namespace std;
class num_string{
	private:
		string x;
		inline void devided_2(){
			string ans="";int now=0;
			for(int i=0;i<=(int)x.length()-1;++i){
				now=now*10+x[i]-'0';
				if(now<2){
					if(!ans.empty()) ans.push_back('0');
					continue;
				}
				ans.push_back(now/2+'0');
				now%=2;
			}
			if(ans.empty()) ans.push_back('0');
			x=ans;
		}
	public:
		num_string(string A){x=A;}
		inline num_string operator >>(int _x){
			num_string a(x); 
			while(_x--){a.devided_2();}
			return a;
		}
		inline num_string operator >>=(int _x){
			*this=*this>>_x;
			return *this;
		}
		friend ostream& operator <<(ostream& out,num_string&A){
			out<<A.x;
			return out;
		}
		friend istream& operator >>(istream& in,num_string&A){
			in>>A.x;
			return in;
		}
		string& operator()(){return x;}
		num_string to_bit(){
			num_string an(x);
			string ans="";
			while(an().size()!=1 or an()[0]!='0'){
				ans.push_back((an().back()-'0')%2+'0');
				an>>=1;
			}
			std::reverse(ans.begin(),ans.end());
			return num_string(ans);
		}
};
num_string a("");
template<typename T>
std::string to_string(T x){
	std::string res;bool f=false;
	if(x<0){
		f=true;
		x*=-1;
	}
	while(x){
		res.push_back((int)(x%10)+'0');
		x/=10;
	}
	reverse(res.begin(),res.end());
	if(f) res.push_back('-');
	if(res.empty()) res.push_back('0');
	return res;
}
void solve(num_string x){
	x=x.to_bit();
	bool flag=false;
	for(int i=0;i<=(int)x().size()-1;++i){
		if(x()[i]=='1'){
			if(flag) putchar('+');
			int w=(int)x().size()-1-i;
			if(w==1){
				cout<<"2";
			}
			else if(w==0){
				cout<<"2(0)";
			}
			else{
				cout<<"2(";
				solve(to_string(w));
				cout<<")";
			}
			flag=true;
		}
	}
}
int main(){
	freopen("pow.in","r",stdin);
	freopen("pow.out","w",stdout);
//	freopen("code/by/t1.in","r",stdin);
//	freopen("out.out","w",stdout);
	cin>>a;
	solve(a);
}

hdk::bitset 一解

#include<bits/stdc++.h>
#include<bitset.h>
#include<tool.h>
#include<hdk_string.h>
using namespace std;
hdk::bitset::bitset<10000>b; 
void solve(hdk::string x){
	b.operator=<2>(x);x=b;
	bool flag=false;
	for(int i=0;__array_of(x)){
		if(x[i]=='1'){
			if(flag) putchar('+');
			int w=x.size()-1-i;
			if(w==1) cout<<"2";
			else if(w==0) cout<<"2(0)";
			else{
				cout<<"2(";
				solve(hdk::tool::to_string(w));
				cout<<")";
			}
			flag=true;
		}
	}
}
int main(){
	cin>>a;solve(a);
}

B.子串的子串

\(n^2q\log\)

哈希,暴力枚举子区间判断即可

和 \(n^2q\) 一个分

\(n^2+q\)

考虑对一段区间 DP 求差分

设差分数组为 \(f_{l,r}\),先考虑我们怎么求,我们可以根据一个区间的子区间求前缀和来得到:

\[f_{l,r}=f_{l+1,r}+f{l,r-1}-f_{l+1,r-1}+1 \]

这是经典的二位前缀和方法,后面加的那个 \(1\) 是它自己这个子串(显然,没有字符串能和它相等)

然后我们会发现 \([l,r]\) 会被子区间覆盖,所以要长度从大往小求前缀和,防止被覆盖

然后是差分怎么求的问题

假设右边那段红色的区间为 \([l,r]\),左边找到了一个和它相同的字符串,那么我们就将前面那个字符串的贡献减掉 \(1\)(在差分数组上做这个相当于给整个黑色大区间的贡献减掉 \(1\),是对的)

那么为什么我们要减掉的是前面的那个而不是后面的那个呢

假设我们减掉后面那个,然后用如下的方式来计算前缀和

for(int i=n;i>=1;--i){
	for(int j=i;j<=n;++j){
		f[i][j]+=f[i+1][j]+f[i][j-1]-f[i+1][j-1]+1;
	}
}

\(1:1+1-1+1-2=0\)

\(2:1+1-1+1-1=1\)

\(3:0+1-(-1)+1=3\)

可以发现小区间答案错了,先遍历的字符串反而没有贡献

然而如果我们放在前面

那么就有

\(1:1+1-1+1-1=1\)

\(2:1+1-1+1-2=0\)

\(3:1+0-(-1)+1=3\)

不知道出题人咋想到的

#include<bits/stdc++.h>
using namespace std;
int n,q;
string a;
unsigned long long num=233;
unsigned long long __hash[3001];
unsigned long long base[3001];
void prework(){
	base[0]=1;
	for(int i=1;i<=n;++i){
		base[i]=base[i-1]*num;
		__hash[i]=__hash[i-1]*num+a[i];
	}
}
inline unsigned long long _hash(int l,int r){
	return __hash[r]-__hash[l-1]*base[r-l+1];
}
int f[3001][3001];
unordered_map<unsigned long long,int>mp;
int main(){
	scanf("%d %d",&n,&q);
	cin>>a;a=" "+a;
	prework();
	for(int len=1;len<=n;++len){
		mp.clear();
		for(int i=1;i+len-1<=n;++i){
			int j=i+len-1;
			f[mp[_hash(i,j)]][j]--;
			mp[_hash(i,j)]=i;
		}
	}
	for(int i=n;i>=1;--i){
		for(int j=i;j<=n;++j){
			f[i][j]+=f[i+1][j]+f[i][j-1]-f[i+1][j-1]+1;
		}
	}
	while(q--){
		int l,r;
		cin>>l>>r;
		cout<<f[l][r]<<"\n";
	}
}

C.魔法咒语

正序插入一颗字典树,如果字典树上某一个节点不存在某个字母 \(\alpha\),则可以将所有字符串的所有后缀中以 \(\alpha\) 结尾的字符串拼到后面,这样可以保证不会重复

证明:假设最后拼成的子串 \(A=B+C\),为了防止重复,钦定使 \(B\) 尽可能长,则就是上述 “字典树上某一个节点不存在某个字母 \(\alpha\)” 的限制条件,不妨设 \(S'=\alpha+C\),实际上我们开这个 cnt 找的是 \(S'\),为了恰好能拼上,因此需要找倒序状态下以 \(\alpha\) 结尾的字符串

然而当一个词的长度为 \(1\),那么它完全没办法通过这种办法被拼出来,所以需要特判

知道思路以后就水了

#include<bits/stdc++.h>
using namespace std;
#define int long long
struct trie1{
	int tot=0;
	int to[400001][26];
	void insert(string x){
		int pos=0;
		for(int i=0;i<=(int)x.length()-1;++i){
			if(!to[pos][x[i]-'a']){
				to[pos][x[i]-'a']=++tot;
			}
			pos=to[pos][x[i]-'a'];
		}
	} 
}tr1;
struct trie2{
	int tot=0;
	int to[400001][26];
	int cnt[26];
	void insert(string x){
		reverse(x.begin(),x.end());
		int pos=0;
		for(int i=0;i<=(int)x.length()-1;++i){
			if(!to[pos][x[i]-'a']){
				to[pos][x[i]-'a']=++tot;
				cnt[x[i]-'a']++;
			}
			pos=to[pos][x[i]-'a'];
		}
	} 
}tr2;
int n,ans;
string s[400001];
bool vis[26],flag[26];
signed main(){
	freopen("magic.in","r",stdin);
	freopen("magic.out","w",stdout);
	cin>>n;
	for(int i=1;i<=n;++i){
		cin>>s[i];
		tr1.insert(s[i]);
		tr2.insert(s[i]);
		vis[s[i].back()-'a']=true;
		if(s[i].length()==1 and flag[*s[i].begin()-'a']==false){
			ans++;
			flag[*s[i].begin()-'a']=true;
		}
	}
	for(int i=1;i<=tr1.tot;++i){
		for(int j=0;j<=25;++j){
			if(tr1.to[i][j]==0){
				ans+=tr2.cnt[j];
			}
			else ans+=vis[j];
		}
	}
	cout<<ans<<'\n';
}

D.表达式

部分分分块就能算

类似的思想(即若 \(f=g(h)\),则 \(f(x)=g(h(x))\)),尝试把操作放到线段树上

当 \(P\) 很小的时候,直接对 \(x\mod p\) 维护值就行了

否则,观察到题中的 \(P\) 都可以分解成很小的质数乘积,所以拆开做然后 CRT 就行

注意 CRT 要求模数互质,\(100\) 不能拆完,否则就有重复的数了,拆成 \(25\times 4\) 比较好

在补了

标签:std,num,string,04,int,44,long,联训,include
From: https://www.cnblogs.com/HaneDaCafe/p/18454113

相关文章

  • 多校A层冲刺NOIP2024模拟赛04
    多校A层冲刺NOIP2024模拟赛04学校OJ赛时rank28,T10pts,T2100pts,T320pts,T425ptsaccodersrank15,T1100pts,T2100pts,T320pts,T425pts不是,也没人告诉我两个OJ文件名不一样啊02表示法递归+高精除低精。点此查看代码#include<bits/stdc++.h>#include<bits/extc++.h>//......
  • 『模拟赛』多校A层冲刺NOIP2024模拟赛04
    Rank赤石场。A.02表示法签。若干天前在洛谷随到过,不过当时只看了眼讨论区就走了www还好本来不是很难。发现大体上是一个拆分二的幂的问题,从大到小枚举2的幂,判断有没有这个幂只用比较大小关系,然后再对指数做一个同样的操作,递归至不大于2为止,注意\(2^1\)不用输出(1......
  • 多校A层冲刺NOIP2024模拟赛04
    多校A层冲刺NOIP2024模拟赛04\(T1\)A.02表示法\(0pts/100pts\)弱化版:luoguP1010[NOIP1998普及组]幂次方递归模拟即可,二进制分解时需要写高精除低精。点击查看代码intr[810],t[810];chars[810],id[810][10];stringa;intchu(chars[]){ intn=strlen(s......
  • IEC104规约的秘密之七----配置参数t1,t2,t3
    104通讯前需要配置通讯参数,一般有如下参数:IP地址,端口号,k,w,t1,t2,t3,公共地址,遥控超时参数,104主规约还有一个t0参数。本次只讲解t1,t2,t3这两个参数。这三个都是超时时间,t1用于两个地方,一个是发送的I帧没有得到及时的确认,在规约文本中有如下图:B站发送I(0,0)帧后,开始计时,A站回......
  • QOJ #9448. Product Matrix
    题面传送门这居然是能快速插值的?首先令\(A_{i,x,y}=a_{x,y}2^i+b_{x,y}\),则\(\prod\limits_{i=k}^{k+m-1}A_i\)的\((1,1)\)位置相当于将\(x=2^k\)代入答案多项式得到的点值。可以用矩阵乘法在\(O(mn^3)\)时间复杂度内求出这\(m+1\)个点值。然后我们需要插值出原来......
  • C++ day04(友元 friend、运算符重载、String字符串)
    目录【1】友元friend1》概念2》友元函数 3》友元类 4》友元成员函数 【2】运算符重载1》概念2》友元函数运算符重载 ​编辑 3》成员函数运算符重载4》赋值运算符与类型转换运算符重载 5》注意事项【3】String字符串类【1】友元friend1》概念定义:......
  • 代码随想录算法训练营第九天|344.反转字符串, 541. 反转字符串II,卡码网:54.替换数字
    344.反转字符串反转字符串比较简单,除了用reverse,可以用for循环,两头向中间夹,进行swapclassSolution{public:voidreverseString(vector<char>&s){inthalf=s.size()/2;intlength=s.size();for(inti=0,j=length-1;i<half;i++,j--){......
  • XYD1004CSPS
    T1序列[贪心,模拟]Description给定一个序列,每次可以将一个数减一,求让所有数字互不相同的最小操作数,\(0\)除外,也就是\(0\)可以相同。Solution贪心地,从大到小考虑每个数,都只需要减到第一个没有数出现的数。开个桶从大到小累计答案即可。T2XYD[构造]Description给定\(......
  • 【堆】【优先队列】[NOIP2004]合并果子
    https://ac.nowcoder.com/acm/contest/22669/I堆的用法Type:队列中存储元素的类型。例如int,double,pair<int,int>等。Container:底层存储数据的容器类型,默认为vector,但可以换成deque或其他容器。Compare:比较函数,用于决定优先级顺序。默认是less,表示最大堆。如果使用......
  • [智能网联汽车/数据标准/法规政策] 标准解读:GB/T 44464-2024《汽车数据通用要求》
    0引言随着智能技术的不断发展,智能网联汽车作为新时代移动智能终端的代表,正引领着汽车产业向智能化、网联化深刻转型与升级。智能网联汽车与云端服务器、移动端、车端等设备存在大量的数据交互,包括车辆运行数据、用户个人信息等。缺乏对这些数据实施的有效监管与控制,将潜藏重大......