首页 > 其他分享 >ABC324题解

ABC324题解

时间:2023-10-15 10:58:27浏览次数:46  
标签:return int 题解 tr mid long ABC324 size

A/B

赛时没打。

C

暴力判断是相等 s[i] == t 还是替换了一个字符,或者是添加/删除了一个字符。

最后两个判断只需要交换一下 \(s\) 和 \(t\) 的顺序就可以共用一个函数了。

D

注意到 \(N\le 13\),所以平方数不会超过 \(v=10^{13}\),很容易想到暴力枚举 \(\sqrt v\) 以内的数,判断是否能够被表示出来。

时间复杂度 \(O(N\sqrt v)\),或者说 \(O(N10^{\frac{N}{2}})\)

E

注意到如果两个字符串 \(S1,S2\) 拼起来可以包含 \(T\) 那么一定可以被前后分割成 \(T=T1+T2\) 使得 \(T1\subset S1, T2\subset S2\)。

预处理出每个 \(S_i\) 包含 \(T\) 的最长前后缀的长度,使用双指针解决。

复杂度 \(O(\sum_i |S_i|)\)

查询时计算前缀后缀和大于 \(|T|\) 的 \((i,j)\) 对数。

最后对计算出的前后缀长度做一个后缀和来优化查询。

时间复杂度 \(O(N+\sum_i |S_i|)\)。

F

分数规划裸题。

二分答案,然后在 DAG 上跑个最长路 dp 就完事了。

> 什么是分数规划?<

G

将排列上的数放在二维平面上,每个数可以表示为 \((i,a_i)\) 的形式,对于每个序列可以写作 \((l,r,u,d)\),表示包含了 \(l\le i\le r, u\le a_i\le d\) 的所有数。注意到这相当于包含了一个矩形内的所有点,可以使用主席树来维护。

具体的,把点按照第二维排序,依次加入主席树中。

对于两个操作,相当于在矩形 \(s\) 里划一条平行于坐标轴的分界线(自己模拟一下)。

怎么找这条分界线呢?

对于每个 1 操作,在主席树上找到 \(l_s-1\) 的排名 \(rk\),然后主席树上二分出 \(rk+x\) 的值 \(v\) 就得到了分界线。

将 \((l,r,u,d)\) 变成了 \((l,v,u,d)(v+1,r,u,d)\)

对于每个 2 操作,分界线就是 \(x\),直接分割就行了。

将 \((l,r,u,d)\) 变成了 \((l,r,u,v)(l,r,v+1,d)\)

查询是容易的。

附 C,D,E,F,G 代码

C

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;

const int N = 5e5 + 5;
string s[N], t;
int n;

bool chk1(string s1, string s2)
{
    int cnt = 0;
    for(int i = 0; i < s1.size(); i ++)
        cnt += s1[i] != s2[i];
    return cnt <= 1;
}

bool chk2(string s1, string s2)
{
    int dt = 0;
    for(int i = 0; i < s1.size(); i ++)
    {
        while(!dt && s1[i] != s2[i + dt]) dt ++;
        if(s1[i] != s2[i + dt]) return false;
    }
    return true;
}

signed main()
{
    ios::sync_with_stdio(0);cin.tie(0);
    cin >> n >> t;
    vector<int> ans;
    for(int i = 1; i <= n; i ++)
    {
        cin >> s[i];
        if(s[i].size() < t.size() - 1 || s[i].size() > t.size() + 1) continue;
        if(s[i] == t)
        {
            ans.push_back(i);
            continue;
        }
        if(s[i].size() == t.size() && chk1(s[i], t))
        {
            ans.push_back(i);
            continue;
        }
        if(s[i].size() == t.size() - 1 && chk2(s[i], t))
        {
            ans.push_back(i);
            continue;
        }
        if(s[i].size() == t.size() + 1 && chk2(t, s[i]))
        {
            ans.push_back(i);
            continue;
        }
    }
    cout << ans.size() << "\n";
    for(int i : ans) cout << i << " ";

    return 0;
}

D

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;

#define int ll

const int N = 15;
int a[N], n, p[N];
string s;

inline bool chk(int x)
{
    int b[N];
    memcpy(b, a, sizeof a);
    while(x)
    {
        b[x % 10] --;
        if(b[x % 10] < 0) return false;
        x /= 10;
    }
    for(int i = 1; i < 10; i ++)
        if(b[i] > 0) return false;
    return true;
}

signed main()
{
    ios::sync_with_stdio(0);cin.tie(0);
    cin >> n >> s;
    p[0] = 1;
    for(int i = 1; i <= n; i ++) p[i] = p[i - 1] * 10ll;
    for(int i = 0; i < n; i ++) a[s[i] - '0'] ++;
    int ans = 0;
    for(int i = 0; i * i < p[n]; i ++)
        ans += chk(i * i);
    cout << ans;

    return 0;
}

E

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;

#define int ll

const int N = 5e5 + 5;
string s[N], t;
int n, l[N], r[N], sl[N], sr[N];

signed main()
{
    ios::sync_with_stdio(0);cin.tie(0);
    cin >> n >> t;
    for(int i = 1; i <= n; i ++)
    {
        cin >> s[i];
        int k = 0;
        for(int j = 0; k < t.size() && j < s[i].size(); j ++)
            if(s[i][j] == t[k]) k ++;
        l[i] = k;
        k = t.size() - 1;
        for(int j = s[i].size() - 1; k >= 0 && j >= 0; j --)
            if(s[i][j] == t[k]) k --;
        r[i] = t.size() - 1 - k;
        sl[l[i]] ++;
        sr[r[i]] ++;
    }
    for(int i = N - 5; i >= 0; i --) sl[i] += sl[i + 1];
    for(int i = N - 5; i >= 0; i --) sr[i] += sr[i + 1];
    int ans = 0;
    for(int i = 1; i <= n; i ++)
    {
        ans += sr[t.size() - l[i]] + sl[t.size() - r[i]];
    }
    cout << ans / 2;
    

    return 0;
}

F

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef long double D;

#define int ll

const int N = 2e5 + 5;

struct edge{int v, a, b;};
vector<edge> e[N];
int deg[N], n, m;
D f[N];

bool check(D x)
{
    queue<int> q;
    int d[N];
    memcpy(d, deg, sizeof deg);
    memset(f, -0x10, sizeof f);
    f[1] = 0;
    for(int i = 1; i <= n; i ++)
        if(!d[i]) q.push(i);
    while(!q.empty())
    {
        int t = q.front();q.pop();
        for(auto i : e[t])
        {
            if(--d[i.v] == 0) q.push(i.v);
            f[i.v] = max(f[i.v], f[t] + (i.a - x * i.b));
        }
    }
    return f[n] >= 0.L;
}

signed main()
{
    ios::sync_with_stdio(0);cin.tie(0);
    cin >> n >> m;
    for(int i = 1; i <= m; i ++)
    {
        int x, y, a, b;
        cin >> x >> y >> a >> b;
        e[x].push_back({y, a, b});
        deg[y] ++;
    }

    D l = 0, r = 1e4;
    while(l + (D)1e-12 < r)
    {
        D mid = (l + r) / 2.L;
        if(check(mid)) l = mid;
        else r = mid;
    }
    printf("%.17Lf", r);

    return 0;
}

G

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N = 2e5 + 5;
struct node{int l, r, sz;};
int n, m, a[N];
struct sgtv
{
    node tr[N << 5];
    int root[N], idx;
    int insert(int lstt, int l, int r, int x)
    {
        int nowt = ++idx;
        tr[nowt] = tr[lstt];
        if(l == r)
        {
            tr[nowt].sz++;
            return nowt;
        }
        int mid = (l + r) >> 1;

        if(mid >= x) tr[nowt].l = insert(tr[lstt].l, l, mid, x);
        else tr[nowt].r = insert(tr[lstt].r, mid + 1, r, x);
        tr[nowt].sz = tr[tr[nowt].l].sz + tr[tr[nowt].r].sz;
        return nowt;
    }
    int query(int lt, int rt, int l, int r, int k)
    {
        if(l > k || r < k) return 0;
        if(l == r) return tr[rt].sz - tr[lt].sz;
        int mid = (l + r) >> 1;
        if(mid >= k) return query(tr[lt].l, tr[rt].l, l, mid, k);
        else return tr[tr[rt].l].sz - tr[tr[lt].l].sz + query(tr[lt].r, tr[rt].r, mid + 1, r, k);
    }
    int find(int lt, int rt, int l, int r, int k)
    {
        if(l == r) return r;
        int mid = (l + r) >> 1;
        int cnt = tr[tr[rt].l].sz - tr[tr[lt].l].sz;
        if(cnt >= k) return find(tr[lt].l, tr[rt].l, l, mid, k);
        else return find(tr[lt].r, tr[rt].r, mid + 1, r, k - cnt);
    }
} tr;

struct M{int l, r, u, d;}b[N];
int h = 0;

int main()
{
    ios::sync_with_stdio(0);cin.tie(0);
    cin >> n;
    vector<pair<int, int>> v;
    for (int i = 1; i <= n; i++)
    {
        cin >> a[i];
        v.push_back({a[i], i});
    }

    sort(v.begin(), v.end());

    for(int i = 1; i <= n; i ++)
        tr.root[i] = tr.insert(tr.root[i - 1], 0, n + 1, v[i - 1].second);

    cin >> m;
    b[0] = {1, n, 1, n};
    while(m --)
    {
        int t, s, x;
        cin >> t >> s >> x;
        if(t == 1)
        {
            int L = tr.query(tr.root[b[s].u - 1], tr.root[b[s].d], 0, n + 1, b[s].l - 1);
            int r = tr.find(tr.root[b[s].u - 1], tr.root[b[s].d], 0, n + 1, L + x);
            r = min(r, b[s].r);
            r = max(r, b[s].l - 1);
            b[++h] = {r + 1, b[s].r, b[s].u, b[s].d};
            b[s].r = r;
        }
        else
        {
            x = min(x, b[s].d);
            x = max(x, b[s].u - 1);
            b[++h] = {b[s].l, b[s].r, x + 1, b[s].d};
            b[s].d = x;   
        }
        cout << tr.query(tr.root[b[h].u - 1], tr.root[b[h].d], 0, n + 1, b[h].r) - 
        tr.query(tr.root[b[h].u - 1], tr.root[b[h].d], 0, n + 1, b[h].l - 1) << "\n";
    }

    return 0;
}

标签:return,int,题解,tr,mid,long,ABC324,size
From: https://www.cnblogs.com/adam01/p/17765361.html

相关文章

  • P9517 drink 题解
    P9517drink题解Part1提示题目传送门欢迎大家指出错误并私信这个蒟蒻欢迎大家在下方评论区写出自己的疑问(记得@这个蒟蒻)Part2更新日志2023-08-1218:06文章完成2023-08-1415:53文章通过审核Part3解析这道题考场上用的查找做的。先用一个结构体分别表示......
  • P9686 Judg. 题解
    P9686Judg.题解Part1提示题目传送门欢迎大家指出错误并私信这个蒟蒻欢迎大家在下方评论区写出自己的疑问(记得@这个蒟蒻)Part2更新日志2023-10-0215:50文章完成2023-10-0412:37文章通过审核Part3解析一道简单模拟。这道题最简单的方法就是直接在for循......
  • P8741 [蓝桥杯 2021 省 B] 填空问题 题解
    P8741[蓝桥杯2021省B]填空问题题解题目传送门欢迎大家指出错误并联系这个蒟蒻更新日志2023-05-0923:19文章完成2023-05-0923:20通过审核2023-06-2021:03优化了文章代码格式试题A:空间【解析】本题考察计算机存储的基础知识,只要掌握空间存储的换算方法,就能......
  • P8679 [蓝桥杯 2019 省 B] 填空问题 题解
    P8679[蓝桥杯2019省B]填空问题题解题目传送门欢迎大家指出错误并联系这个蒟蒻更新日志2023-05-2521:02文章完成2023-05-2711:34文章通过审核2023-06-2021:03优化了文章代码格式试题A:组队【解析】本题是一道经典的DFS搜索题,每次对各号位的选手进行DFS,......
  • P8684 [蓝桥杯 2019 省 B] 灵能传输 题解
    P8684[蓝桥杯2019省B]灵能传输题解Part1提示题目传送门欢迎大家指出错误并私信这个蒟蒻欢迎大家在下方评论区写出自己的疑问(记得@这个蒟蒻)Part2更新日志2023-06-2021:46文章完成2023-07-0308:57文章通过审核2023-08-2118:14更改了文章格式,使文章看起......
  • ABC324总结
    ABC324solved:CDErating:UnratedC:7min(+1)D:7minE:9minCconstintN=2e5+5;\(\rightarrow\)constintN=5e5+5;。。。F01分数规划,没学过。后:就是二分答案加上推式子啊。G一眼主席树上二分。但是没调出来。码力\(\downarrow\)\(\downarrow\)\(\do......
  • ABC324F Beautiful Path
    给出一张DAG,每条边有两种边权\(b\)与\(c\),求一条从\(1\)到\(n\)的路径,问路径经过的边的\(\dfrac{\sumb}{\sumc}\)的最大值是多少。\(n,m\le2\times10^5\)。这不是经典01分数规划吗?将题目中的要求写成如下形式:\[\begin{aligned}&\text{Maximize}\dfrac......
  • 算法题解——买卖股票的最佳时机
    解题思路先考虑最简单的「暴力遍历」,即枚举出所有情况,并从中选择最大利润。设数组prices的长度为n,由于只能先买入后卖出,因此第1天买可在未来n−1天卖出,第2天买可在未来n-2天卖出……以此类推,共有\[(n-1)+(n-2)+\cdots+0=\frac{n(n-1)}{2}\]种情况,时间复......
  • Atcoder Beginner Contest 324 F Beautiful Path 题解-分数规划
    为了更好的阅读体验,请点击这里分数规划小技巧:尽可能将式子写成存在某种取值,使得不等式成立的形式。不然可能需要绕几个弯才能想出来。题目链接题目大意:给出一个DAG,每条边有一个\(b_i,c_i\),保证从编号小的边向编号大的边连边,且\(1\)到\(n\)必有路径,求\(1\)到\(n\)......
  • CF1204D2 Kirk and a Binary String (hard version) 题解
    CF1204D2KirkandaBinaryString(hardversion)题解分析先来分析\(01\)串的最长不下降子序列。全是\(0\)显然是不下降的,如果中间出现一个\(1\),为了维护不下降的性质,后面就只能全是\(1\)。一句话概括一下,\(0\)后面能跟\(0,1\),\(1\)后面只能跟\(1\)。现在来分析这......