拼写单词 - LeetCode 1160 - 附C++代码

2023-11-02 06:50

本文主要是介绍拼写单词 - LeetCode 1160 - 附C++代码,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、问题描述

        给你一份『词汇表』(字符串数组) words 和一张『字母表』(字符串) chars。
假如你可以用 chars 中的『字母』(字符)拼写出 words 中的某个『单词』(字符串),那么我们就认为你掌握了这个单词。
        注意:每次拼写(指拼写词汇表中的一个单词)时,chars 中的每个字母都只能用一次。
返回词汇表 words 中你掌握的所有单词的长度之和。

示例 1:
输入:words = [“cat”,“bt”,“hat”,“tree”], chars = “atach”
输出:6
解释: 可以形成字符串 “cat” 和 “hat”,所以答案是两个单词长度之和 3 + 3 = 6。

示例 2:
输入:words = [“hello”,“world”,“leetcode”], chars = “welldonehoneyr”
输出:10
解释:可以形成字符串 “hello” 和 “world”,所以答案是两个单词长度之和 5 + 5 = 10。

原题:LeetCode 1160. 拼写单词

二、思路分析

1、暴力法

        分析题意中的判定条件,“会一个单词word”可以转换为条件:对word中的每一个字母,都有一个chars中的、未使用过的字母与之对应。由此不难想到暴力法的求解思路:每当判定一个word,创建一个与chars的每位对应的标记数组Mark[chars.size()],用于标记chars对应位是否被使用过,然后遍历word中的每一个字母,从chars中查找未用过的相同字母,并相应维护Mark,若word中的每一个字母都能取到,则认为“会”,结果长度增加该单词长度。

2、优化方式1

        分析暴力法的求解,有大量时间花费在了从chars中查找相同字母的操作上,为了优化这步查找,可以对chars建立索引,由于chars中的字母不能重复使用,可以建立<字母,个数>的map<char,int>。查找一个单词时,拷贝一份chars的map,再遍历这个单词中的字母,查找到字母的同时将可用个数减1。倘若未出现个数不够的情况,则认为“会这个单词”。

3、优化方式2

        根据 2 中的思路,已经对chars建立了map,进一步地,我们可以对每一个word也建立相同类型的map,同时无需再次拷贝chars的map,“会这个单词”则转换为条件:word中的每一个字母(即word_map中的每一个key),chars中都有足够的“可用量”(即chars_map[key]足够大)。

三、代码实现

1、暴力法

        略。

2、优化方式1(chars的map重复拷贝)

class Solution1 {
public:int countCharacters(vector<string>& words, string chars) {int res_len = 0;bool miss = false;//Get chars's index map <char,num>.map<char, int> chars_map;for (char ch : chars) {if (chars_map.count(ch)) chars_map[ch]++;else chars_map[ch] = 1;}//Judge each word in words.for (string word : words) {//Copy one map.map<char, int> copy_map(chars_map);miss = false;//Count one word.for (char ch : word) {if (!copy_map.count(ch) || copy_map[ch] == 0) {miss = true;break;}copy_map[ch]--;}if (!miss) res_len += word.size();}return res_len;}
};int main()
{/** 解题思路:本题转换成查找模型,对words中的每个单词,查找chars中是否存在一一对应;* 思路1:为了快速进行查找,对chars建立索引,由于chars中的字母不能重复使用,可以建立<字母,个数>的map<char,int>遍历一遍chars,建立索引map,每找到对应的字母,将对应的int+1;判断一个Word时,拷贝一份map,每当遇到一个字母,map中对应可用个数-1;当遇到剩余可用个数不足时,认为不会,若全部字母个数都够,则认为会。*/cout << "Process start." << endl;Solution1 s;vector<string> words1 = { "cat","bt","hat","tree" };string chars1 = "atach";cout << "Excepted result1 is : 6    Actually result1 is : " << s.countCharacters(words1, chars1) << endl;vector<string> words2 = { "hello","world","leetcode" };string chars2 = "welldonehoneyr";cout << "Excepted result2 is : 10    Actually result2 is : " << s.countCharacters(words2, chars2) << endl;cout<< "Process finish." << endl;return 0;
}

3、优化方式2

#include <iostream>
#include <vector>
#include <string>
#include <map>using namespace std;class Solution {
public:int countCharacters(vector<string>& words, string chars) {int res_len = 0;bool miss = false;//Get chars's index map <char,num>.map<char, int> chars_map;for (char ch : chars)chars_map[ch]++;//Judge each word in words.for (string word : words) {//Create word map.map<char, int> word_map;for (char ch : word)word_map[ch]++;//Start to judgemiss = false;for (auto k_v : word_map) {if (k_v.second > chars_map[k_v.first]) {miss = true;break;}}if (!miss) res_len += word.size();}return res_len;}
};int main()
{/** 解题思路:本题转换成查找模型,对words中的每个单词,查找chars中是否存在一一对应;* 思路1:为了快速进行查找,对chars建立索引,由于chars中的字母不能重复使用,可以建立<字母,个数>的map<char,int>遍历一遍chars,建立索引map,每找到对应的字母,将对应的int+1;判断一个Word时,拷贝一份map,每当遇到一个字母,map中对应可用个数-1;当遇到剩余可用个数不足时,认为不会,若全部字母个数都够,则认为会。* 思路2:对每一个word也建立相同类型的map,同时无需再次拷贝chars的map,“会这个单词”则转换为条件:每一个字母(即word_map中的每一个key),chars中都有足够的“可用量”(即chars_map[key]>word_map[key])。*/cout << "Process start." << endl;Solution s;vector<string> words1 = { "cat","bt","hat","tree" };string chars1 = "atach";cout << "Excepted result1 is : 6    Actually result1 is : " << s.countCharacters(words1, chars1) << endl;vector<string> words2 = { "hello","world","leetcode" };string chars2 = "welldonehoneyr";cout << "Excepted result2 is : 10    Actually result2 is : " << s.countCharacters(words2, chars2) << endl;cout<< "Process finish." << endl;return 0;
}

四、运行结果

1、一个map多次拷贝

2、分别建立map

五、时间与空间复杂度对比

        优化方式2 和 优化方式3 的时间复杂度是O(n1 + n2),空间复杂度是O(2 * n2),n1是words的总长度,n2 是chars的长度。但两种方式运行时的区别在哪呢?到底哪个更快?

        从运行信息中显示,对于给定的测试用例,优化方式2 比 优化方式3 内存占用更高,但运行速度更快(这是在我预料之外的),倘若把 3 中的 map 改为 unordered_map,内存占用将会进一步降低,而耗时则会更高。

        分析具体执行的过程,发现有两个点:

        1、给定的大部分用例中,chars的长度都是比单个word长的,这导致方式2 (拷贝chars_map)比方式3 (创建word_map)空间占用更高;

        2、对于一个不会的word,对方式2 而言,并不一定会完整地遍历整个word,只要中间有一个字母出现了问题,这个word的后续字母都将不会被遍历。而对于方式3,一定是完整地遍历这个word的(构建map在判定之前)。这是二者判定耗时上的差别。

        此外,方式2 使用了map 的拷贝构造函数(构建了更多元素),方式3使用insert 方法(插入的元素更少), 这两种方法到底哪个的性能更高仍待查找资料或者实验确认。

 

 

这篇关于拼写单词 - LeetCode 1160 - 附C++代码的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/328871

相关文章

Django开发时如何避免频繁发送短信验证码(python图文代码)

《Django开发时如何避免频繁发送短信验证码(python图文代码)》Django开发时,为防止频繁发送验证码,后端需用Redis限制请求频率,结合管道技术提升效率,通过生产者消费者模式解耦业务逻辑... 目录避免频繁发送 验证码1. www.chinasem.cn避免频繁发送 验证码逻辑分析2. 避免频繁

精选20个好玩又实用的的Python实战项目(有图文代码)

《精选20个好玩又实用的的Python实战项目(有图文代码)》文章介绍了20个实用Python项目,涵盖游戏开发、工具应用、图像处理、机器学习等,使用Tkinter、PIL、OpenCV、Kivy等库... 目录① 猜字游戏② 闹钟③ 骰子模拟器④ 二维码⑤ 语言检测⑥ 加密和解密⑦ URL缩短⑧ 音乐播放

Python使用Tenacity一行代码实现自动重试详解

《Python使用Tenacity一行代码实现自动重试详解》tenacity是一个专为Python设计的通用重试库,它的核心理念就是用简单、清晰的方式,为任何可能失败的操作添加重试能力,下面我们就来看... 目录一切始于一个简单的 API 调用Tenacity 入门:一行代码实现优雅重试精细控制:让重试按我

C++11范围for初始化列表auto decltype详解

《C++11范围for初始化列表autodecltype详解》C++11引入auto类型推导、decltype类型推断、统一列表初始化、范围for循环及智能指针,提升代码简洁性、类型安全与资源管理效... 目录C++11新特性1. 自动类型推导auto1.1 基本语法2. decltype3. 列表初始化3

C++11右值引用与Lambda表达式的使用

《C++11右值引用与Lambda表达式的使用》C++11引入右值引用,实现移动语义提升性能,支持资源转移与完美转发;同时引入Lambda表达式,简化匿名函数定义,通过捕获列表和参数列表灵活处理变量... 目录C++11新特性右值引用和移动语义左值 / 右值常见的左值和右值移动语义移动构造函数移动复制运算符

C++中detach的作用、使用场景及注意事项

《C++中detach的作用、使用场景及注意事项》关于C++中的detach,它主要涉及多线程编程中的线程管理,理解detach的作用、使用场景以及注意事项,对于写出高效、安全的多线程程序至关重要,下... 目录一、什么是join()?它的作用是什么?类比一下:二、join()的作用总结三、join()怎么

Python实现MQTT通信的示例代码

《Python实现MQTT通信的示例代码》本文主要介绍了Python实现MQTT通信的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录1. 安装paho-mqtt库‌2. 搭建MQTT代理服务器(Broker)‌‌3. pytho

MySQL进行数据库审计的详细步骤和示例代码

《MySQL进行数据库审计的详细步骤和示例代码》数据库审计通过触发器、内置功能及第三方工具记录和监控数据库活动,确保安全、完整与合规,Java代码实现自动化日志记录,整合分析系统提升监控效率,本文给大... 目录一、数据库审计的基本概念二、使用触发器进行数据库审计1. 创建审计表2. 创建触发器三、Java

C++中全局变量和局部变量的区别

《C++中全局变量和局部变量的区别》本文主要介绍了C++中全局变量和局部变量的区别,全局变量和局部变量在作用域和生命周期上有显著的区别,下面就来介绍一下,感兴趣的可以了解一下... 目录一、全局变量定义生命周期存储位置代码示例输出二、局部变量定义生命周期存储位置代码示例输出三、全局变量和局部变量的区别作用域

C++中assign函数的使用

《C++中assign函数的使用》在C++标准模板库中,std::list等容器都提供了assign成员函数,它比操作符更灵活,支持多种初始化方式,下面就来介绍一下assign的用法,具有一定的参考价... 目录​1.assign的基本功能​​语法​2. 具体用法示例​​​(1) 填充n个相同值​​(2)