先吐槽一下题面锅了这回事,再吐槽下出题人英语水平…
A
给定三个数a,b,na,b,na,b,n,问[a,b][a,b][a,b]里面有没有nnn的倍数
如果a,ba,ba,b里面有一个是显然就是
剩下的看⌊an⌋<⌊bn⌋\lfloor\frac{a}{n}\rfloor<\lfloor\frac{b}{n}\rfloor⌊na⌋<⌊nb⌋就可以
int n,a,b;
int main()
{
read(n),read(a),read(b);
if(a%n==0||b%n==0)return puts("OK"),0;
int t1=a/n,t2=b/n;
if(t1<t2)puts("OK");
else puts("NG");
return 0;
}
B
最开始你有100100100块钱,每年获得1%1\%1%的利息,可以利滚利,问多少年后的钱≥X,X≤1018\geq X,X\leq10^{18}≥X,X≤1018
数据范围看上去非常大,但是我们观察一下第二个样例,在X=1018X=10^{18}X=1018的时候,只需要376037603760次,所以直接暴力就可以
为什么呢?我们发现随着时间的推移,他每年获得的钱就越多,所以就瞎搞搞就可以
# define int long long
int n;
int x=100,t;
signed main()
{
read(n);
while(x<n){
x+=x/100;
t++;
}
printf("%lld\n",t);
return 0;
}
C
让你构造长度为nnn,每个数不超过mmm的一个递增数列,让∑di(Abi−Aai=Ci)\sum d_i(A_{b_i}-A_{a_i}=C_i)∑di(Abi−Aai=Ci)最大
好像还是很奇怪
但是我们看他是一个递增数列
也就是说根据插板法原则,最差的爆搜的复杂度是Cn+mnqC_{n+m}^nqCn+mnq左右,显然可以通过这道题
所以这道题又是一个爆搜
int n,m,q;
int a[N],b[N],c[N],d[N];
int val[N];
int ans;
void dfs(int u,int last){
if(u==n+1){
int res=0;
Rep(i,1,q)if(val[b[i]]-val[a[i]]==c[i])res+=d[i];
ans=max(ans,res);
return;
}
Rep(i,last,m)val[u]=i,dfs(u+1,i);
}
int main()
{
read(n),read(m),read(q);
Rep(i,1,q)read(a[i]),read(b[i]),read(c[i]),read(d[i]);
dfs(1,1);
printf("%d\n",ans);
return 0;
}
D
对于给定a,b,na,b,na,b,n,试求0≤x≤n0\leq x\leq n0≤x≤n,使得⌊axb⌋−a⌊xb⌋\lfloor\frac{ax}{b}\rfloor-a\lfloor\frac{x}{b}\rfloor⌊bax⌋−a⌊bx⌋最大,输出这个最大值
显然,这个值是bbb个一循环的,也就是说我们只需要考虑所有[0,b)[0,b)[0,b)范围内的就可以了,那么显然取x=b−1x=b-1x=b−1的时候最优,但是因为有一个≤n\leq n≤n的限制,所以我们取x=min{b−1,n}x=\min\{b-1,n\}x=min{b−1,n},带入求值即可
# define int long long
int a,b,n;
signed main()
{
read(a),read(b),read(n);
int t=min(b-1,n);
printf("%lld\n",a*t/b-t/b*a);
return 0;
}
E
没看懂
F
树上最长上升子序列板子题
这种树上的问题就是考虑做完子树之后把之前对答案的影响撤销掉嘛
我们用fif_ifi表示以iii为结尾的最长上升子序列的长度,那么可以列出一个转移方程
fi=max{fj}+1,i∈subtree(j),aj<aif_i=\max\{f_j\}+1,i\in subtree(j),a_j<a_ifi=max{fj}+1,i∈subtree(j),aj<ai
显然n2n^2n2做法是不行的,那么考虑log\loglog的做法,也就是树状数组或者贪心的做法
树状数组应该是不行的,因为我们要把之前的给撤掉,但是权值线段树大概好像可以(单点赋值,区间取max),但是线段树太长了啊,所以我们写二分的做法
我们用gig_igi表示长度iii的最长上升子序列结尾的数最小是多少,那么显然ggg是单调不降的,可以二分
然后利用aia_iai去更新相应的ggg上面的位置就可以了
我们还需要记录一个原来的最小值,当访问完全部子树之后,再拿原来的最小值去撤掉这个点的贡献
具体看代码,记得离散化
# define int long long
int n;
int a[N],b[N],sz;
int head[N],cnt;
int f[N];
int ans;
int d[N];
struct Edge{
int to,next;
}e[N<<1];
void add(int x,int y){
e[++cnt]=(Edge){y,head[x]},head[x]=cnt;
}
void dfs(int u,int fa)
{
int tmp=lower_bound(d,d+n,a[u])-d;
f[u]=tmp+1;
f[u]=max(f[u],f[fa]);
int t=d[tmp];
d[tmp]=a[u];
RepG(i,u){
int v=e[i].to;
if(v==fa)continue;
dfs(v,u);
}
d[tmp]=t;
}
signed main()
{
memset(d,0x3f,sizeof(d));
memset(head,-1,sizeof(head));
read(n);
Rep(i,1,n)read(a[i]),b[i]=a[i];
sort(b+1,b+n+1);
sz=unique(b+1,b+n+1)-b-1;
Rep(i,1,n)a[i]=lower_bound(b+1,b+sz+1,a[i])-b;
Rep(i,1,n-1){
int x,y;
read(x),read(y);
add(x,y),add(y,x);
}
dfs(1,0);
Rep(i,1,n)printf("%lld\n",f[i]);
return 0;
}

本文解析了五道算法竞赛题目,包括寻找特定倍数、复利计算、构造递增数列、求最大差值及树上最长上升子序列等问题,提供了详细的解题思路与代码实现。
1114

被折叠的 条评论
为什么被折叠?



