2182.构造限制重复的字符串(模拟 贪心 优先队列 C++ Go)

简介: 【2月更文挑战第19天】2182.构造限制重复的字符串(模拟 贪心 优先队列 C++ Go)

题目链接

题意

给你一个字符串 s 和一个整数 repeatLimit ,用 s 中的字符构造一个新字符串 repeatLimitedString ,使任何字母 连续 出现的次数都不超过 repeatLimit 次。你不必使用 s 中的全部字符。

返回 字典序最大的 repeatLimitedString 。

如果在字符串 a 和 b 不同的第一个位置,字符串 a 中的字母在字母表中出现时间比字符串 b 对应的字母晚,则认为字符串 a 比字符串 b 字典序更大 。如果字符串中前 min(a.length, b.length) 个字符都相同,那么较长的字符串字典序更大。
提示:

$1 <= repeatLimit <= s.length <= 10^5$
s 由小写英文字母组成

思路

贪心的构造

  • 每次将当前剩余的字典序最大的字符添加到答案里,如果该字符已经连续出现了repeatLimit次,则先将当前剩余的字典序次大的字符添加到答案里,再继续将当前剩余的字典序最大的字符添加到答案里,直到该最大的字符用完或是没有次大的字符可以插入

有两种写法

  1. 多层for循环,不断尝试填入新字符
    • 结合上述的思路可以得知,每个字符i最多被添加min(repeatLimit,mp[i])次,其中mp[i]为字符i的出现次数
    • 可以先统计字符串s里每个字符的出现次数
    • 倒序进行遍历,这时候i里维护的就是当前剩余的字典序最大的字符
    • 尝试将字符i添加到答案字符串里
    • 如果i已经填完了的话,跳出循环,找下一个当前剩余的字典序最大的字符
    • 否则的话找第一个存在且字典序次大的字符,插入到答案字符串里,这样后面就可以继续填字母i
  2. 使用优先队列维护字典序最大字符和次大字符
    • 思路1的本质是通过for循环找当前剩余的字典序最大/次大的字符,可以通过优先队列来维护

代码

func repeatLimitedString(s string, repeatLimit int) string {
   
    mp := make(map[rune]int, 26)
    for _, ch := range s {
   
        mp[ch]++
    }
    ans := make([]rune,0,len(s))
    for i := 'z'; i >= 'a'; i-- {
   //倒序填入字母
        las := i - 1//记录次小的字母值
        for {
   
            for j := 0; j < repeatLimit && mp[i] > 0; j++ {
   //最多填入min(repeatLimit,mp[i])个字母i
                mp[i]--
                ans = append(ans, i)
            }
            if mp[i] == 0 {
   //i填完了 找下一个字典序最大的字母
                break
            }
            for las >= 0 && mp[las] == 0 {
   //找到第一个存在的字典序次大的字母
                las--
            }
            if las < 0 {
   //找不到 跳出
                break
            }
            //先填入次大字母 后面可以继续填最大字母i
            mp[las]--
            ans = append(ans, las)
        }
    }
    return string(ans)
}
class Solution {
   
    public:
        string repeatLimitedString(string s, int repeatLimit) {
   
            int mp[26];
            for(int i=0; i<26; i++) {
   
                mp[i]=0;
            }
            for(char ch:s) {
   
                mp[ch-'a']++;
            }
            string ans;
            for(int i=25; i>=0; i--) {
   
                int las = i-1;
                while(1) {
   
                    for(int j=0; j<repeatLimit&&mp[i]>0; j++) {
   
                        mp[i]--;
                        ans.push_back(i+'a');
                        //ans = ans + char(i+'a');
                    }
                    if(mp[i]==0) {
   
                        break;
                    }
                    while(las>=0&&mp[las]==0) {
   
                        las--;
                    }
                    if(las < 0) {
   
                        break;
                    }
                    mp[las]--;
                    ans.push_back(las+'a');
                //    ans = ans + char(las+'a');
                }
            }
            return ans;
        }
};
目录
相关文章
|
4天前
|
设计模式 算法 调度
【C++】开始使用优先队列
优先队列的使用非常灵活,它适合于任何需要动态调整元素优先级和快速访问最高(或最低)优先级元素的场景。在使用时,需要注意其插入和删除操作的时间复杂度,以及如何根据实际需求选择合适的仿函数。
20 4
|
6天前
|
Go
|
7天前
|
存储 设计模式 算法
【C++/STL】stack和queue(容器适配器、优先队列、双端队列)
【C++/STL】stack和queue(容器适配器、优先队列、双端队列)
13 1
|
14天前
|
编解码 JavaScript 前端开发
【专栏】介绍了字符串Base64编解码的基本原理和在Java、Python、C++、JavaScript及Go等编程语言中的实现示例
【4月更文挑战第29天】本文介绍了字符串Base64编解码的基本原理和在Java、Python、C++、JavaScript及Go等编程语言中的实现示例。Base64编码将24位二进制数据转换为32位可打印字符,用“=”作填充。文中展示了各语言的编码解码代码,帮助开发者理解并应用于实际项目。
|
18天前
|
存储 编译器 C语言
C++字符串大小写之for语句
C++字符串大小写之for语句
18 0
|
20天前
|
存储 Go 开发者
Golang深入浅出之-Go语言字符串操作:常见函数与面试示例
【4月更文挑战第20天】Go语言字符串是不可变的字节序列,采用UTF-8编码。本文介绍了字符串基础,如拼接(`+`或`fmt.Sprintf()`)、长度与索引、切片、查找与替换(`strings`包)以及转换与修剪。常见问题包括字符串不可变性、UTF-8编码处理、切片与容量以及查找与替换的边界条件。通过理解和实践这些函数及注意事项,能提升Go语言编程能力。
27 0
|
20天前
|
C++
【代码片段】【C++】获取当前时间戳并生成固定格式字符串
【代码片段】【C++】获取当前时间戳并生成固定格式字符串
15 0
|
1天前
|
存储 编译器 Go
Go语言学习12-数据的使用
【5月更文挑战第5天】本篇 Huazie 向大家介绍 Go 语言数据的使用,包含赋值语句、常量与变量、可比性与有序性
37 6
Go语言学习12-数据的使用
|
3天前
|
Java Go
一文带你速通go语言指针
Go语言指针入门指南:简述指针用于提升效率,通过地址操作变量。文章作者sharkChili是Java/CSDN专家,维护Java Guide项目。文中介绍指针声明、取值,展示如何通过指针修改变量值及在函数中的应用。通过实例解析如何使用指针优化函数,以实现对原变量的直接修改。作者还邀请读者加入交流群深入探讨,并鼓励关注其公众号“写代码的SharkChili”。
9 0
|
3天前
|
存储 缓存 Java
来聊聊go语言的hashMap
本文介绍了Go语言中的`map`与Java的不同设计思想。作者`sharkChili`是一名Java和Go开发者,同时也是CSDN博客专家及JavaGuide项目的维护者。文章探讨了Go语言`map`的数据结构,包括`count`、`buckets指针`和`bmap`,解释了键值对的存储方式,如何利用内存对齐优化空间使用,并展示了`map`的初始化、插入键值对以及查找数据的源码过程。此外,作者还分享了如何通过汇编查看`map`操作,并鼓励读者深入研究Go的哈希冲突解决和源码。最后,作者提供了一个交流群,供读者讨论相关话题。
14 0