A.Contest for Robots(贪心)

题意:有n道题。事先知道两个机器人(R,B)分别能答对哪几道。现在要分配每题得分使得机器人R一定能赢(至少1分),问怎么分配使得所有题的最高分最低。

题解:贪心。分别计算R对B错和R错B对的数量,然后把R错B对的题全部设置为1分。所以R对B错的题尽可能平均分且超过R错B对的题数即可。

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxn = 2e5 + 5;
const int mod = 1e9 + 7;
const int INF = 0x3f3f3f3f;
const ll INFL = 0x3f3f3f3f3f3f3f3fLL;
int r[105], b[105];
int main()
{
    int n;
    scanf("%d", &n);
    int cnt1 = 0, cnt2 = 0;
    for (int i = 0; i < n; i++)
        scanf("%d", &r[i]);
    for (int i = 0; i < n; i++)
    {
        scanf("%d", &b[i]);
        if (r[i] > b[i])
            cnt1++;
        if (r[i] < b[i])
            cnt2++;
    }
    if (cnt1 == 0)
        puts("-1");
    else
        printf("%d\n", cnt2 / cnt1 + 1);
    return 0;
}

B.Journey Planning

题意:n个地点编号1~n。每个地点有一个美丽指数bi。现在要制定一个浏览顺序使得浏览过的城市的美丽指数之和最大。要求如下(1)浏览顺序必须为严格递增序列(2)相邻浏览的城市必须满足编号的差等于美丽指数的差

题解:bi-bj=i-j,即bi-i=bj-j,将bi转移为bi-i即可,map记录总和,最后扫描一遍取最大值。

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxn = 2e5 + 5;
const int mod = 1e9 + 7;
const int INF = 0x3f3f3f3f;
const ll INFL = 0x3f3f3f3f3f3f3f3fLL;
map<int, ll> mp;
int main()
{
    int n;
    scanf("%d", &n);
    for (int i = 1, x; i <= n; i++)
    {
        scanf("%d", &x);
        mp[x - i] += 1ll * x;
    }
    ll ans = 0;
    for (auto i : mp)
        ans = max(ans, i.second);
    printf("%lld\n", ans);
    return 0;
}

C.Remove Adjacent

题意:给定一个小写字母字符串。具体操作规则如下:若一个字母的相邻字母为它字典序中的前一个字母,则可以将这个字母删除。问最多能操作几次。

题解:注意到字符串长度仅100,所以暴力解决。顺序从z扫描到a。每次删除一个字母后记得重新扫描一遍即可。理论上最坏复杂度10010026

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxn = 2e5 + 5;
const int mod = 1e9 + 7;
const int INF = 0x3f3f3f3f;
const ll INFL = 0x3f3f3f3f3f3f3f3fLL;
map<int, ll> mp;
int main()
{
    int n;
    scanf("%d", &n);
    string s;
    cin >> s;
    bool flag = true;
    while (flag)
    {
        flag = false;
        for (char i = 'z'; i > 'a'; i--)
        {
            for (int j = 0; j < s.length(); j++)
                if (s[j] == i)
                    if ((j > 0 && s[j - 1] == i - 1) || (j + 1 < s.length() && s[j + 1] == i - 1))
                    {
                        s.erase(s.begin() + j);
                        flag = true;
                        break;
                    }
            if (flag)
                break;
        }
    }
    printf("%d\n", n - s.length());
    return 0;
}

D.Navigation System

题意:给定一个单向图和一个规划完的s~t路径。有一个导航仪会实时显示当前点到t的最短路径(但并不会已经规划完的路径产生影响)。如果规划的路径和导航路径不同,那么走到下一个点后,导航仪会重新规划路径。问导航仪最少和最多重新规划几次。

题解:

设p为规划路径。dis[p[i]]为p[i]~t的最短距离。导航仪在如下几种情况下会重新规划:

  (1)dis[p[i]]-1≠dis[p[i+1]]。即一定不是最短路径。

  (2)存在一个不为p[i+1]的点w使得dis[p[i]]-1=dis[w]。即存在一条其他的路径为最短路径。

不难发现如果第一种情况发生,那么第二种情况一定存在。所以满足第一条即为最少的规划次数,满足第二条的即为最多的规划次数。

对于dis数组的获得逆向bfs一下即可。

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxn = 2e5 + 5;
const int mod = 1e9 + 7;
const int INF = 0x3f3f3f3f;
const ll INFL = 0x3f3f3f3f3f3f3f3fLL;
vector<int> G1[maxn], G2[maxn];
int p[maxn], dis[maxn];
int n, m, k;
void bfs(int s)
{
    queue<int> q;
    memset(dis, INF, sizeof(dis));
    q.push(s);
    dis[s] = 0;
    while (!q.empty())
    {
        int u = q.front();
        q.pop();
        for (auto v : G2[u])
        {
            if (dis[v] == INF)
            {
                dis[v] = dis[u] + 1;
                q.push(v);
            }
        }
    }
}
int main()
{
    scanf("%d%d", &n, &m);
    for (int i = 0; i < m; i++)
    {
        int x, y;
        scanf("%d%d", &x, &y);
        G1[x].push_back(y);
        G2[y].push_back(x);
    }
    scanf("%d", &k);
    for (int i = 1; i <= k; i++)
        scanf("%d", &p[i]);
    bfs(p[k]);
    int ans1 = 0, ans2 = 0;
    for (int i = 1; i < k; i++)
    {
        int u = p[i], v = p[i + 1];
        if (dis[u] - 1 != dis[v])
            ans1++;
        for (auto j : G1[u])
        {
            if (j != v && (dis[u] - 1) == dis[j])
            {
                ans2++;
                break;
            }
        }
    }
    printf("%d %d\n", ans1, ans2);
    return 0;
}

E.World of Darkraft: Battle for Azathoth(二维偏序+线段树)

题意:有n把武器,m件防具,p只怪物,每把武器有攻击值ai和价格cai,每件防具有防御值bi和价格cbi,怪物有防御值xi、攻击值yi和奖励zi。勇者可以(必须)选择购买一把武器和一件防具,当勇者的攻击值大于怪物的防御值且勇者防御值大于怪物攻击值,则勇者可以击杀这只怪物并获得其对应的奖励。勇者可以击杀多只怪物,每只怪物只能被击杀一次。求勇者能获得的最大收益(打怪奖励-武器价格-防具价格)。

题解:

二维偏序,可以先按武器的攻击值从小到大排序,怪物的防御值从小到大排序,那么对于每把武器,利用双指针可以求出所有防御值比这把武器攻击值低的怪物
然后考虑防具和怪兽的攻击值也就是第二维,首先先定义lb数组,lb[i]表示所选防具至少为i的最少花费
可以发现如果选到防御值为i的防具,那么所有当前可以打败的怪兽中,只要攻击值小于i都可以累加到答案中那么考虑mx[i]表示维护防御值最大为i的最大获利(包含防具的开支,武器的开支在最后计算)对于一只新增的怪兽只要i大于其防御值,那么mx[i]都可以加上这只怪兽的获利,而这可以用线段树来维护
线段树内存的值为防御值至少为i的最大获利

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxn = 2e5 + 5;
const int MAXN = 1e6 + 5;
const ll INF = 0x3f3f3f3f3f3f3f3fll;
int n, m, p;
ll lb[MAXN], ans, mxb;
struct tool
{
    ll f, cost;
    bool operator<(const tool &C) const
    {
        return f < C.f;
    }
} a[maxn], b[maxn];
struct monsters
{
    ll x, y, z;
    bool operator<(const monsters &C) const
    {
        return x < C.x;
    }
} c[maxn];
struct node
{
    ll sum, lazy;
} tr[MAXN << 2];
void pushup(int rt)
{
    tr[rt].sum = max(tr[rt << 1].sum, tr[rt << 1 | 1].sum);
}
void pushdown(int rt)
{
    if (tr[rt].lazy != 0)
    {
        tr[rt << 1].sum += tr[rt].lazy;
        tr[rt << 1].lazy += tr[rt].lazy;
        tr[rt << 1 | 1].sum += tr[rt].lazy;
        tr[rt << 1 | 1].lazy += tr[rt].lazy;
        tr[rt].lazy = 0;
    }
}
void build(int l, int r, int rt)
{
    if (l == r)
    {
        tr[rt].sum = -lb[l];
        return;
    }
    int mid = l + r >> 1;
    build(l, mid, rt << 1);
    build(mid + 1, r, rt << 1 | 1);
    pushup(rt);
}
void update(int rt, int l, int r, int L, int R, int val)
{
    if (L <= l && r <= R)
    {
        tr[rt].sum += val;
        tr[rt].lazy += val;
        return;
    }
    pushdown(rt);
    int mid = l + r >> 1;
    if (L <= mid)
        update(rt << 1, l, mid, L, R, val);
    if (R > mid)
        update(rt << 1 | 1, mid + 1, r, L, R, val);
    pushup(rt);
}
int main()
{
    scanf("%d%d%d", &n, &m, &p);
    for (int i = 1; i <= n; i++)
        scanf("%lld%lld", &a[i].f, &a[i].cost);
    for (int i = 1; i <= m; i++)
        scanf("%lld%lld", &b[i].f, &b[i].cost);
    for (int i = 1; i <= p; i++)
        scanf("%lld%lld%lld", &c[i].x, &c[i].y, &c[i].z);
    sort(a + 1, a + n + 1);
    sort(c + 1, c + p + 1);
    for (int i = 1; i <= m; i++)
        mxb = max(mxb, b[i].f);
    for (int i = 1; i <= mxb; i++)
        lb[i] = INF;
    for (int i = 1; i <= m; i++)
        lb[b[i].f] = min(lb[b[i].f], b[i].cost);
    for (int i = mxb - 1; i >= 1; i--)
        lb[i] = min(lb[i], lb[i + 1]);
    ans = -INF;
    build(1, mxb, 1);
    for (int i = 1, j = 1; i <= n; i++)
    {
        while (c[j].x < a[i].f && j <= p)
        {
            if (c[j].y < mxb)
                update(1, 1, mxb, c[j].y + 1, mxb, c[j].z);
            j++;
        }
        ans = max(ans, -a[i].cost + tr[1].sum);
    }
    printf("%lld\n", ans);
    return 0;
}