贪心算法的初涉(双指针 + “过山车思想”)

"过山车"思想

首先我们用一道力扣的题目,来简单了解一下"过山车思想"

3228. 将 1 移动到末尾的最大操作次数 - 力扣(LeetCode)

给你一个

二进制字符串

s

你可以对这个字符串执行 任意次 下述操作:

  • 选择字符串中的任一下标 ii + 1 < s.length ),该下标满足 s[i] == '1's[i + 1] == '0'
  • 将字符 s[i]右移 直到它到达字符串的末端或另一个 '1'。例如,对于 s = "010010",如果我们选择 i = 1,结果字符串将会是 s = "0++001++10"

返回你能执行的最大操作次数。

示例 1:

输入: s = "1001101"

输出: 4

解释:

可以执行以下操作:

  • 选择下标 i = 0。结果字符串为 s = "++001++1101"
  • 选择下标 i = 4。结果字符串为 s = "0011++01++1"
  • 选择下标 i = 3。结果字符串为 s = "001++01++11"
  • 选择下标 i = 2。结果字符串为 s = "00++01++111"

示例 2:

输入: s = "00111"

输出: 0

提示:

  • 1 <= s.length <= 105
  • s[i]'0''1'

思路就是:想象一个过山车,所有的"1"都是小车,如果遇到了就连起来(长度是cnt),所有"0"都是关卡,当小车碰到关卡时:ans += cnt

代码:

cpp 复制代码
class Solution {
public:
    int maxOperations(string s) {
        int n = s.size(),ans = 0,cnt = 0;
        for(int i=0;i<n;i++){
            if(s[i] == '1') cnt ++;
            else if(i && s[i-1] == '1') ans += cnt;
        }
        return ans;
    }
};

双指针

我们来看一个升级版的:双指针+"过山车"

题目来自codeforces,这个平台对于思维的提升非常非常大,如果你需要很强大的思维,这个平台非常适合你

Problem - 1896B - Codeforces

You are given a string ss of length nn consisting of characters AA and BB. You are allowed to do the following operation:

  • Choose an index 1≤i≤n−1 such that si=Aand si+1=B. Then, swap si and si+1.

You are only allowed to do the operation at most once for each index 1≤i≤n−1. However, you can do it in any order you want. Find the maximum number of operations that you can carry out.

Input

Each test contains multiple test cases. The first line contains the number of test cases tt (1≤t≤10001≤t≤1000). Description of the test cases follows.

The first line of each test case contains a single integer nn (2≤n≤2⋅10^5) --- the length of string ss.

The second line of each test case contains the string ss (si=A or si=B).

It is guaranteed that the sum of nn over all test cases does not exceed 2⋅10^5.

Output

For each test case, print a single integer containing the maximum number of operations that you can carry out.

题目的意思大致是:给定一个只包含"AB"的字符串,如果字串是"AB"这种形式那么可以交换i与i+1的位置变成"BA",每个i只能与前面的i+1交换一次

那还是"过山车"思想,所有的"A"是小车,"B"是关卡,但是这里注意的是交换位置。

比如:"AABB",最后应该变成的是:"BABA" 而不是 "BBAA"

这里的话就需要一个双指针来每次更新位置

如果是我们需要找到最左边"A"的位置已经最后边"B"的位置,然后ans += right - left,left = right

为了防止越界每次输入的字符串s后面加一个"C"(这个随便加,加什么都行)然后每次right多走一步

代码:

cpp 复制代码
#include <iostream>
#include <string>
#include <vector>
#include <algorithm>
#include <cstring>
using namespace std;

int main()
{
    int n,t;
    string s;
    cin >> t;
    while(t --){
        cin >> n >> s;
        s += 'C';
        int l = 0,r = 0,ans = 0;
        while(s[l] == 'B') l ++,r ++;
        while(l < n){
            int cnt = 0;
            while(s[r] == 'A') r ++;
            while(s[r] == 'B') r ++,cnt ++;
            // cout << cnt << " " << l << " " << r << endl;
            if(cnt){
                ans += r - l - 1;
                l = r - 1;
            }else break;
        }
        cout << ans << endl;
    }
    return 0;
}

加油

相关推荐
小猿_003 分钟前
C语言程序设计十大排序—插入排序
c语言·算法·排序算法
熊文豪2 小时前
深入解析人工智能中的协同过滤算法及其在推荐系统中的应用与优化
人工智能·算法
siy23335 小时前
[c语言日寄]结构体的使用及其拓展
c语言·开发语言·笔记·学习·算法
吴秋霖5 小时前
最新百应abogus纯算还原流程分析
算法·abogus
灶龙6 小时前
浅谈 PID 控制算法
c++·算法
菜还不练就废了6 小时前
蓝桥杯算法日常|c\c++常用竞赛函数总结备用
c++·算法·蓝桥杯
金色旭光6 小时前
目标检测高频评价指标的计算过程
算法·yolo
he101016 小时前
1/20赛后总结
算法·深度优先·启发式算法·广度优先·宽度优先
Kent_J_Truman6 小时前
【回忆迷宫——处理方法+DFS】
算法
paradoxjun6 小时前
落地级分类模型训练框架搭建(1):resnet18/50和mobilenetv2在CIFAR10上测试结果
人工智能·深度学习·算法·计算机视觉·分类