这次终于是早上考试了

早上考试手感不错,这次刷新了以前的最高排名~

%%%cyh巨佬 \(rk1\)

%%%CT巨佬 \(t2\) 90

纵观前几,似乎我 \(t3\) 是最低的……

总计挂分10分,\(t2\) 写的 \(exgcd\) 因为变量打错没用上


A. 最长不下降子序列

第一眼看上去没思路……

看见 \(n\) 的范围太大了,估计得从数列生成上做文章

一开始研究了半天二次函数之类的东西,试图寻找单调性之类的东东

后来索性不会还是打个表找规律吧

咦?居然循环了?

(这个样例给的刁钻呀)其实换个什么别的模数或者初始项很快就会出现循环节

冷静分析一下,因为模数很小,前一项经过函数作用后结果是固定的,那么最多出现模数个数后就会出现循环

那循环就简单了,有几个循环就有几个相等数,循环节内部也会有,把第一个拿出来和前面的跑个LIS拼一下完事儿

然后写个对拍,发现Wa了!!!

开始手模,发现有一种很神奇的情况,比如:

14523 14523 14523

乍一看前面是3加上后面两个是5,但其实是6

冷静分析一下发现这种情况只有把前 \(len\) 个循环节都揪出去跑才能解决

赶紧改一下然后就过了

代码实现
#include<bits/stdc++.h>
using namespace std;
#define int long long
const int maxn=1e6+5;
int n,t[maxn],a,b,c,d,last[maxn],ed,num,len,st,st1,ans,f[maxn],cc[maxn];
void add(int x,int w){
x++;
for(;x<=151;x+=x&-x){
cc[x]=max(cc[x],w);
}
return ;
}
int ask(int x){
x++;
int ans=0;
for(;x;x-=x&-x)ans=max(ans,cc[x]);
return ans;
}
signed main(){
// freopen("lis0.in","r",stdin);
// freopen("my.out","w",stdout);
cin>>n;
cin>>t[1]>>a>>b>>c>>d;
if(n<=100000){
for(int i=2;i<=n;i++){
t[i]=(a*t[i-1]*t[i-1]+b*t[i-1]+c)%d;
}
for(int i=1;i<=n;i++){
f[i]=1;
f[i]=max(f[i],ask(t[i])+1);
add(t[i],f[i]);
ans=max(ans,f[i]);
}
cout<<ans;
return 0;
}
last[t[1]]=1;
for(int i=2;i<=n;i++){
t[i]=(a*t[i-1]*t[i-1]+b*t[i-1]+c)%d;
if(last[t[i]]){
st=last[t[i]];
len=i-last[t[i]];
break;
}
last[t[i]]=i;
}
for(int i=st+len;i<=st+len*(len+10);i++){
t[i]=(a*t[i-1]*t[i-1]+b*t[i-1]+c)%d;
}
num=(n-st+1)/len-len;
ed=n-num*len;
for(int i=1;i<=ed;i++){
f[i]=1;
f[i]=max(f[i],ask(t[i])+1);
add(t[i],f[i]);
}
st1=ed+1;
for(int i=st1;i<=st1+len-1;i++){
for(int j=1;j<=ed;j++){
if(t[j]<=t[i])ans=max(ans,f[j]);
}
}
cout<<ans+num;
return 0;
}

考完看题解,还有一种不同的方法

对于循环节的 LIS 采用 \(dp\) 的方式

设 \(f[n][i][j]\) 表示以第一个循环节 \(i\) 开头,以后面第 \(n\) 个循环节的 \(j\) 结尾的 LIS 长度,转移:

\[f[n][i][j]=\max_{1\le k \le T,a[i]\le a[k]\le a[j]}(f[n-1][i][k]+f[1][k][j])
\]

然后用了一个很神奇的方法优化一下——广义矩阵乘法

正常的矩乘是先乘后加,发现形式和这个很像,这不过这个是先加后 \(max\),只要把矩乘定义该一下就好了

把 \(F[1]\) 看成 \(base\) 矩阵,那么 \(F[n]=F[n-1]*base\),这样可以矩阵快速幂了

然后 \(F[1]\) 暴力算一下即可


B. 完全背包问题

考场上想过同余最短路,但是看见还有总和不超过 \(C\) 的限制条件就直接跑路了,然后转数学,开始裴蜀定理乱搞,搞不出来

正解就是同余最短路,但是得加个分层来满足限制条件

先来根据同余最短路的套路来设个方程:

\(f[j][k]\) 表示选了的物品模 \(val[0]\) 等于 \(j\),且选了 \(k\) 个物品的最小总价值(之所以是最小,因为最后要加许多个 \(val[0]\),最小的可以表示出全部状态)

转移:

\[f[j][k]+val[i]->f[(j+val[i])\%val[0]][k] (val[i]<L)
\]
\[f[j][k]+val[i]->f[(j+val[i])\%val[0]][k+1] (val[i]\ge L)
\]

发现如果点开成一维的不好转移,那么再加一维代表层数,转移即可

代码实现
#include<bits/stdc++.h>
using namespace std;
#define int long long
#define fi first
#define se second
const int maxn=55,maxm=10005;
int n,m,limit,sum,val[maxn],w,dis[maxm][maxn];
bool vis[maxm][maxn];
int read(){
int x=0,f=1;
char ch=getchar();
while(!isdigit(ch)){
if(ch=='-')f=-1;
ch=getchar();
}
while(isdigit(ch)){
x=x*10+ch-48;
ch=getchar();
}
return x*f;
}
struct Node{
pair<int,int>id;
int dis;
Node(){}
Node(pair<int,int>x,int y):id(x),dis(y){}
bool operator < (const Node & x)const{
return dis>x.dis;
}
};
priority_queue<Node>q;
void dij(){
memset(dis,0x3f,sizeof dis);
dis[0][0]=0;
q.push(Node(make_pair(0,0),0));
while(!q.empty()){
pair<int,int>x=q.top().id;
q.pop();
if(vis[x.fi][x.se])continue;
vis[x.fi][x.se]=true;
for(int i=2;i<=n;i++){
if(val[i]>=limit&&x.se==sum)break;
if(val[i]<limit){
if(dis[(x.fi+val[i])%val[1]][x.se]>dis[x.fi][x.se]+val[i]){
dis[(x.fi+val[i])%val[1]][x.se]=dis[x.fi][x.se]+val[i];
q.push(Node(make_pair((x.fi+val[i])%val[1],x.se),dis[(x.fi+val[i])%val[1]][x.se]));
}
}
else{
if(dis[(x.fi+val[i])%val[1]][x.se+1]>dis[x.fi][x.se]+val[i]){
dis[(x.fi+val[i])%val[1]][x.se+1]=dis[x.fi][x.se]+val[i];
q.push(Node(make_pair((x.fi+val[i])%val[1],x.se+1),dis[(x.fi+val[i])%val[1]][x.se+1]));
}
}
}
}
return ;
}
signed main(){
n=read();
m=read();
for(int i=1;i<=n;i++)val[i]=read();
sort(val+1,val+n+1);
limit=read();
sum=read();
dij();
for(int i=1;i<=m;i++){
w=read();
bool flag=false;
for(int j=0;j<=sum;j++){
if(w>=dis[w%val[1]][j]){
flag=true;
break;
}
}
if(flag)puts("Yes");
else puts("No");
}
return 0;
}

题解上介绍了复杂度更优的方法:


C. 最近公共祖先

每次修改一个点的时候,暴力跳 \(father\),这是祖先的权值对其他子树是有贡献的,在 \(dfs\) 上修改即可

由于每个节点只有第一次走到父亲是有用的,所以最多更新 \(n\) 次,复杂度正确

noip模拟29的更多相关文章

  1. noip模拟29[简单的板子题](虽然我不会)

    \(noip模拟29\;solutions\) 这次考试给我最大的伤害,让我意识到了差距 这场考试可以说是非常的简单,就是简单到,看两眼,打个表就有结果了 但是呢?我考得非常的完蛋,只有30pts 据 ...

  2. NOIP模拟 29

    T1第一眼觉得是网络流 看见4e6条边200次增广我犹豫了 O(n)都过不去的赶脚.. 可是除了网络流板子我还会什么呢 于是交了个智障的EK 还是用dijkstra跑的 居然有50分!$(RP--)$ ...

  3. NOIP 模拟29 B 侥幸

    这次考得好纯属是侥幸,我T3打表试数试了两个小时,没有想打T2的正解(其实是打不出来)所以这个T3A掉纯属是侥幸,以后还是要打正解 (以下博客最好按全选观看,鬼知道为啥这个样子!) 在这里也口胡一下我 ...

  4. Noip模拟29(瞎眼忌) 2021.8.3

    T1 最长不下降子序列 在此记录自己的瞎眼... 考场上像一个傻$der$,自己为了防范上升序列和不下降序列的不同特意的造了一组$hack$数据来卡自己:(第一行是序列长度,第二行是序列) 6 1 5 ...

  5. 2021.8.3考试总结[NOIP模拟29]

    T1 最长不下降子序列 数据范围$1e18$很不妙,但模数$d$只有$150$,考虑从这里突破. 计算的式子是个二次函数,结果只与上一个值有关,而模$d$情况下值最多只有$150$个,就证明序列会出现 ...

  6. NOIP 模拟 $29\; \rm 最近公共祖先$

    题解 \(by\;zj\varphi\) 首先考虑,如果将一个点修改成了黑点,那么它能够造成多少贡献. 它先会对自己的子树中的答案造成 \(w_x\) 的贡献. 考虑祖先时,它会对不包括自己的子树造成 ...

  7. NOIP 模拟 $29\; \rm 完全背包问题$

    题解 \(by\;zj\varphi\) 一道 \(\rm dp\) 题. 现将所有种类从小到大排序,然后判断,若最小的已经大于了 \(\rm l\),那么直接就是一个裸的完全背包,因为选的总数量有限 ...

  8. NOIP 模拟 $29\; \rm 最长不下降子序列$

    题解 \(by\;zj\varphi\) 观察这个序列,发现模数很小,所以它的循环节很小. 那么可以直接在循环节上做最长上升子序列,但是循环节中的逆序对会对拼接后的答案造成影响. 没有必要找逆序对个数 ...

  9. 「题解」NOIP模拟测试题解乱写I(29-31)

    NOIP模拟29(B) T1爬山 简单题,赛时找到了$O(1)$查询的规律于是切了. 从倍增LCA那里借鉴了一点东西:先将a.b抬到同一高度,然后再一起往上爬.所用的步数$×2$就是了. 抬升到同一高 ...

随机推荐

  1. ajax()返回Array

    后台查询的数据为数组$arr,需要将数组 echo json_encode($arr);前台ajax拿到数据 然后用 eval("(+data+)"); 来将json转为json对 ...

  2. ;~ 小部分AutoHotkey脚本源代码测试模板样板.ahk

    ; ;~ 小部分AutoHotkey脚本源代码测试模板样板.ahk ;~ 请把一行或几行少量代码放到此文件中实际测试一下,;~ 看看测试结果如何,等到能够实现代码功能时再复制到自己的脚本代码文件中;~ ...

  3. 在阿里云上单机部署k8s

    系统:CentOS Linux release 8.1.1911 配置主机名 [root@iZwz9e3t4tj14jzewdtvj8Z ~]# hostnamectl set-hostname la ...

  4. IPSec组播概要

    IPSec作为主流IP安全协议之一,在单播环境下,特别是在VPN场景中应用广泛.但是在组播环境貌似看到的不多,通过RFC4301了解到IPSec首先是支持组播的,即通过手动配置的方式可以实现组播包加密 ...

  5. Spring Cloud Aliaba - Ribbon

    Ribbon(有关介绍见RestTemplate末尾) Ribbon负载均衡实现策略 Ribbon负载均衡实现策略通过接口IRule进行实现,默认使用ZoneAvoidanceRule规则进行负载均衡 ...

  6. 【笔记】Ada Boosting和Gradient Boosting

    Ada Boosting和Gradient Boosting Ada Boosting 除了先前的集成学习的思路以外,还有一种集成学习的思路boosting,这种思路,也是集成多个模型,但是和bagg ...

  7. Pikachu-暴力破解模块

    一.概述 "暴力破解"是一攻击具手段,在web攻击中,一般会使用这种手段对应用系统的认证信息进行获取. 其过程就是使用大量的认证信息在认证接口进行尝试登录,直到得到正确的结果. 为 ...

  8. Linux 并发服务器编程(多进程)

    文章目录 说明 注意事项 server.c client.c 运行截图 说明 在Linux中通过流式套接字编程(TCP),实现一个并发服务器的访问回显,适合刚学完Linux套接字编程的朋友进行巩固训练 ...

  9. 带你梳理Jetty自定义ProxyServlet实现反向代理服务

    摘要:最近要做一个将K8s中的某组件UI通过反向代理映射到自定义规则的链接地址上,提供给用户访问的需求.所以顺便研究了一下Jetty的ProxyServlet. 本文分享自华为云社区<Jetty ...

  10. noip35

    T1 考场乱搞出锅了... 正解: 把原序列按k往左和往右看成两个序列,求个前缀和,找下一个更新的位置,直接暴跳. Code #include<cstdio> #include<cs ...