AC修炼计划(AtCoder Beginner Contest 335)A-F

传送门:

AtCoder Beginner Contest 335 (Sponsored by Mynavi) - AtCoder

A,B,C,D还算比较基础,没有什么思路,纯暴力就可以过。

这里来总结一下E和F

E - Non-Decreasing Colorful Path

最开始以为是树形dp,一直在纠结与树形的结构,后来发现我们可以从权值小的向权值大的数遍历,这样的话我们可以保证先提条件就是非递减的,而后在考虑树形的链接关系,由大权值的数字来迭代小权值的数字:

cpp 复制代码
// #pragma GCC optimize(3)  //O2优化开启
#include<bits/stdc++.h>
using namespace std;
#define int long long
typedef long long ll;
typedef pair<int,int> PII;
const int mod=998244353;
const int MX=0x3f3f3f3f3f3f3f3f;
int n,m;
int a[500005];
int pre[500005];
int v[500005];
int find(int x){
    if(pre[x]==x)return x;
    return pre[x]=find(pre[x]);
}
bool cmp(int x,int y){
    return a[x]<a[y];
}
void icealsoheat(){
    cin>>n>>m;
    vector<vector<int>>ve(n+5);
    vector<int>dp(n+5,-0X3f3f3f3f);
    for(int i=1;i<=n;i++){
        cin>>a[i];
        pre[i]=i;
        v[i]=i;
    }
    sort(v+1,v+1+n,cmp);
    // for(int i=1;i<=n;i++)cout<<v[i]<<"+++\n";
    for(int i=1;i<=m;i++){
        int l,r;
        cin>>l>>r;
        if(a[l]==a[r]){
            pre[find(l)]=find(r);
        }
        ve[l].push_back(r);
        ve[r].push_back(l);
    }
    dp[find(1)]=1;
    for(int i=1;i<=n;i++){
        for(auto j:ve[v[i]]){
            if(a[find(v[i])]>a[find(j)]){
                dp[find(v[i])]=max(1+dp[find(j)],dp[find(v[i])]);
            }
        }
    }
    // cout<<dp[find(n)];
    dp[find(n)]=max(0ll,dp[find(n)]);
    cout<<dp[find(n)];


}
signed main(){
    ios::sync_with_stdio(false);
    cin.tie();
    cout.tie();
    int _yq;
    _yq=1;
    // cin>>_yq;
    while(_yq--){
        icealsoheat();
    }
}

F - Hop Sugoroku

这个题比较巧妙,最初我们能够很容易的想到时间复杂度为O()的dp迭代方法,但2e5的数据很显然不能直接这样迭代,所以我们需要想到优化的方法。我们发现如果i+a[i]*x==j,那么可以等价于i%a[i]==j%a[i],我们可以用dp[a[i]][i%a[i]]来存储每一个,而对于大于a[i]>sqrt(n)的数字来说,我们可以直接遍历求解,这样我们能够保证所有情况的时间复杂度为O():

代码如下:

cpp 复制代码
// #pragma GCC optimize(3)  //O2优化开启
#include<bits/stdc++.h>
using namespace std;
#define int long long
typedef long long ll;
typedef pair<int,int> PII;
const int mod=998244353;
const int MX=0x3f3f3f3f3f3f3f3f;
int n,m;
int a[500005];
int cnt[500005];
void icealsoheat(){
    cin>>n;
    for(int i=1;i<=n;i++)cin>>a[i];
    vector<vector<int>>dp(sqrt(n)+5,vector<int>(n+5,0));
    int ans=0;
    cnt[1]=1;
    for(int i=1;i<=n;i++){
        for(int j=1;j<=sqrt(n);j++){
            cnt[i]+=dp[j][i%j];
            cnt[i]%=mod;
        }
        ans+=cnt[i];
        ans%=mod;
        if(a[i]>sqrt(n)){
            int x=a[i];
            while(i+x<=n){
                cnt[i+x]+=cnt[i];
                cnt[i+x]%=mod;
                x+=a[i];
            }
        }
        else{
            dp[a[i]][i%a[i]]+=cnt[i];
            // dp[a[i]][i%a[i]]++;
            dp[a[i]][i%a[i]]%=mod;
        }
    }
    // cout<<cnt[1]<<"+++\n";
    // for(int i=1;i<=n;i++)cout<<cnt[i]<<"+++\n";
    cout<<ans;

    

}
signed main(){
    ios::sync_with_stdio(false);
    cin.tie();
    cout.tie();
    int _yq;
    _yq=1;
    // cin>>_yq;
    while(_yq--){
        icealsoheat();
    }
}
相关推荐
艾莉丝努力练剑14 分钟前
【LeetCode&数据结构】单链表的应用——反转链表问题、链表的中间节点问题详解
c语言·开发语言·数据结构·学习·算法·leetcode·链表
_殊途2 小时前
《Java HashMap底层原理全解析(源码+性能+面试)》
java·数据结构·算法
还债大湿兄2 小时前
《C++内存泄漏8大战场:Qt/MFC实战详解 + 面试高频陷阱破解》
c++·qt·mfc
珊瑚里的鱼5 小时前
LeetCode 692题解 | 前K个高频单词
开发语言·c++·算法·leetcode·职场和发展·学习方法
AI+程序员在路上5 小时前
QTextCodec的功能及其在Qt5及Qt6中的演变
开发语言·c++·qt
Risehuxyc5 小时前
C++卸载了会影响电脑正常使用吗?解析C++运行库的作用与卸载后果
开发语言·c++
秋说6 小时前
【PTA数据结构 | C语言版】顺序队列的3个操作
c语言·数据结构·算法
lifallen7 小时前
Kafka 时间轮深度解析:如何O(1)处理定时任务
java·数据结构·分布式·后端·算法·kafka
liupenglove7 小时前
自动驾驶数据仓库:时间片合并算法。
大数据·数据仓库·算法·elasticsearch·自动驾驶
python_tty8 小时前
排序算法(二):插入排序
算法·排序算法