zoukankan      html  css  js  c++  java
  • AtCoder Beginner Contest 158

    题目链接

    D:
    完全模拟一定超时,设2个数组来存加在前面的/加在后面的,如果出现reverse操作,交换这两个数组即可

    #include<bits/stdc++.h>
    using namespace std;
    #define ms(x,y) memset(x, y, sizeof(x))
    #define lowbit(x) ((x)&(-x))
    #define sqr(x) ((x)*(x))
    typedef long long LL;
    typedef pair<int,int> pii;
    typedef pair<LL,LL> pll;
    
    
    void run_case() {
        string s;
        int q;
        cin >> s >> q;
        vector<char> store[2];
        int sta = 0;
        while(q--) {
            int type;
            cin >> type;
            if(type == 1) sta = !sta;
            else {
                static int pos;
                static char ch;
                cin >> pos >> ch;
                pos--;
                store[pos^sta].push_back(ch);
            }
        }
        string ans;
        for(auto i = store[0].rbegin(); i != store[0].rend(); i++) ans += *i;
        ans += s;
        for(auto ch: store[1]) ans += ch;
        if(sta) reverse(ans.begin(), ans.end());
        cout << ans;
    }
    
    
    int main() {
        ios::sync_with_stdio(false), cin.tie(0);
        cout.flags(ios::fixed);cout.precision(9);
        //int t; cin >> t;
        //while(t--)
        run_case();
        cout.flush();
        return 0;
    }
    

    E:
    (O(n^2))算法很容易想到怎么做,优化这种问题也是比较常见的形式,这里我们可以使用前缀和的形式。
    (num_i)为第(i)位到末尾的十进制数的大小,那么问题可转换为(frac{num_i-num_j}{10^{i-j+1}} (i < j))是否能被(p)整除,也就是(frac{num_i-num_j}{10^{i-j+1}}\%p=0 (i < j))
    转换一下,得到同余式:(num_i equiv num_j (mod p)),那我们可以借用偏序的思想,从尾向头扫,统计每一个(num_i)出现的数量,若重复出现,则一定满足同余式,更新答案即可,注意要特判(p=2/5)的情况

    #include<bits/stdc++.h>
    using namespace std;
    #define ms(x,y) memset(x, y, sizeof(x))
    #define lowbit(x) ((x)&(-x))
    #define sqr(x) ((x)*(x))
    typedef long long LL;
    typedef pair<int,int> pii;
    typedef pair<LL,LL> pll;
    
    
    void run_case() {
        int n, p;
        string s;
        cin >> n >> p >> s;
        LL ans = 0;
        if(p == 2) {
            for(int i = 0; i < n; ++i) if((s[i]-'0') % 2 == 0) ans += i+1;
        } else if(p == 5) {
            for(int i = 0; i < n; ++i) if((s[i]-'0') % 5 == 0) ans += i+1;
        } else {
            map<int, int> mp;
            mp[0]++;
            int now = 0, idx = 1;
            for(int i = n-1; i >= 0; --i) {
                now = (now + (s[i]-'0')*idx) % p;
                ans += mp[now]++;
                idx = idx*10%p;
            }
        }
        cout << ans;
    }
    
    
    int main() {
        ios::sync_with_stdio(false), cin.tie(0);
        cout.flags(ios::fixed);cout.precision(9);
        //int t; cin >> t;
        //while(t--)
        run_case();
        cout.flush();
        return 0;
    }
    

    F:
    本题要考虑到前者会对后者的影响,那我们也可以像E题一样,从后往前考虑,我们可以先进行排序使之有规律
    假设现在有两个点(i,j(X_i<X_j)), (dp_x为移动第x个后剩下的不同集合的数量), 总集合为(s), 第(j)个移动后的集合为(s_1),若(X_i+D_i < X_j), 那么(i)移动后(j)不移动,则剩下的集合为(s-{i}, dp_i=dp_j+1),若(X_i+D_i geq X_j), 则(j)也会跟着移动,剩下的集合就为(s_1-{1}), 假设(k)为满足(X_j+D_j < X_k), 则(dp_i=dp_k+1),若不激活第(i)个机器人,则(dp_i=dp_j)
    将激活与不激活两种情况相加即可

    #include<bits/stdc++.h>
    using namespace std;
    #define ms(x,y) memset(x, y, sizeof(x))
    #define lowbit(x) ((x)&(-x))
    #define sqr(x) ((x)*(x))
    typedef long long LL;
    typedef pair<int,int> pii;
    typedef pair<LL,LL> pll;
    
    const LL MOD = 998244353;
    
    void run_case() {
        int n; cin >> n;
        vector<pll> a(n), sta;
        vector<LL> dp(n+1);
        for(auto &x: a) cin >> x.first >> x.second;
        sort(a.begin(), a.end());
        sta.emplace_back(n, 3e9);
        dp[n] = 1;
        for(int i = n-1; i >= 0; --i) {
            while(sta.back().second < a[i].first + a[i].second) sta.pop_back();
            dp[i] += dp[sta.back().first];
            sta.emplace_back(i, a[i].first);
            dp[i] = (dp[i] + dp[i+1]) % MOD;
        }
        cout << dp[0];
    }
    
    
    int main() {
        ios::sync_with_stdio(false), cin.tie(0);
        cout.flags(ios::fixed);cout.precision(9);
        //int t; cin >> t;
        //while(t--)
        run_case();
        cout.flush();
        return 0;
    }
    
  • 相关阅读:
    第10节--单行函数之数学函数
    第九节--单行函数之日期函数
    【译】什么是游戏开发实体系统框架(终)
    【译】什么是游戏开发实体系统框架(一)
    【译】游戏《实体系统框架》译文——序
    IDEA 快捷键
    Java学习笔记1
    Java面试题
    Java学习笔记——集合
    Java学习笔记——String类常用方法
  • 原文地址:https://www.cnblogs.com/GRedComeT/p/13026161.html
Copyright © 2011-2022 走看看