AtcoderABC253场

A - Median?A - Median?

在这里插入图片描述在这里插入图片描述

题目大意

给定三个整数a、b和c,判断b是否是这些整数的中位数。

思路分析

判断升序降序两种情况

时间复杂度分析

O(1)

代码

#include<iostream>
using namespace std;
int main()
{
int a,b,c;
cin>>a>>b>>c;
if((b>=a&&c>=b)||(a>=b&&b>=c))
cout<<"Yes"<<endl;
else cout<<"No"<<endl;
}

B - Distance Between TokensB - Distance Between Tokens

在这里插入图片描述在这里插入图片描述
在这里插入图片描述

题目大意

给定一个网格,其中有两个不同的方块上面有棋子。方块的状态由H行W列的字符串S1, …, SH表示。如果Si,j = ‘o’,表示第i行第j列的方块上有一个棋子;如果Si,j= ‘-’,表示该方块上没有棋子。允许将其中一个棋子移动到其四周相邻的方块中,但不能移动到网格外部。求使得两个棋子相遇所需的最小移动次数。

思路分析

先找出两个棋子的位置,然后计算它们之间的水平和垂直距离,将两者相加即可得到最小移动次数。

时间复杂度分析

O(H * W)
遍历网格需要O(H * W)的时间复杂度,计算最小移动次数只需常数时间。因此,整体的时间复杂度为O(H * W)。

代码

#include<bits/stdc++.h>
using namespace std;
const int N = 105;

int main() {
    int h, w;
    cin >> h >> w;
    char a[N][N];
    vector<pair<int, int>> b;
    
    for(int i = 0; i < h; i++) {
        for(int j = 0; j < w; j++) {
            cin >> a[i][j];
            if(a[i][j] == 'o') {
                b.push_back({i, j});
            }
        }
    }
    
    cout << abs(b[0].first - b[1].first) + abs(b[0].second - b[1].second) << endl;

    return 0;
}

C - Max - Min QueryC - Max - Min Query

在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述

题目大意

题目要求处理一系列的查询操作。

  • 查询1,将一个整数插入到数列S中;
  • 查询2,计算数列中移除指定整数的最多c个元素(不超过数列有指定整数的个数);
  • 查询3,计算multiset中最大值和最小值之间的差值。

思路分析

首先,我们声明一个空的multiset容器 multiset st; ,用于存储整数元素。

然后,使用循环处理所有的查询。在每个查询中,根据不同的操作类型执行相应的操作:

  • 对于查询1,读取整数x,并使用 st.insert(x) 将x插入到multiset中。
  • 对于查询2,读取整数x和次数c。通过循环 while (c-- and st.find(x) != st.end()) ,删除multiset中最多c个等于x的元素。
  • 对于查询3,在multiset中找到最大值和最小值,计算并输出它们的差值 *st.rbegin() - *st.begin() 。

知识点积累

std::multiset<int> st;

1.erase() 当集合st包含{3, 3, 4}时,执行下面的代码会发生什么?

st.erase(3);

我们期望只删除一个3,但实际上st的内容变成了{4}。如果将一个值作为erase()的参数,它将删除所有具有该值的元素,因此需要小心处理。

如果要删除单独的x,应该这样编写代码:

st.erase(st.find(x));

2.count()的复杂度 使用st.count(x)的计算时间为O(k + log N),其中N是st中元素的数量,k是st中包含x的数量。

因此,如果在处理查询2时使用st.count(x),最坏情况下总共需要O(Q^2)的时间。相反,你应该使用O(log N)的操作,它的时间复杂度为O(log N)。由于st.find(x)最多被调用O(Q)次,这样可以适应时间限制。

时间复杂度分析

O(Q^2 log N^)

代码

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

int main() {
    int q;
    cin >> q;
    multiset<int> st;
    while (q--) {
        int t;
        cin >> t;
        if (t == 1) {
            int x;
            cin >> x;
            st.insert(x);
        } else if (t == 2) {
            int x, c;
            cin >> x >> c;
            while (c-- and st.find(x) != st.end()) {
                st.erase(st.find(x));
            }//c-- 表示每次循环时将c的值减1,并检查结果是否为非零。当c的值降至0时,循环将停止。
            //st.find(x) 用于查找multiset st 中第一个等于x的元素的迭代器。
        } 
        else {
            cout << *st.rbegin() - *st.begin() << endl;//st.begin() 返回multiset st 中第一个(最小的)元素的迭代器。
            //st.rbegin() 返回multiset st 中最后一个(最大的)元素的反向迭代器。
        }
    }
}
  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值