LeetCode初级算法之字符串:387 字符串中的第一个唯一字符
字符串中的第一个唯一字符
题目地址:https://leetcode-cn.com/problems/first-unique-character-in-a-string/
给定一个字符串,找到它的第一个不重复的字符,并返回它的索引。如果不存在,则返回 -1。
示例:
s = "leetcode"
返回 0
s = "loveleetcode"
返回 2
提示:你可以假定该字符串只包含小写字母。
解法一:双指针
双指针比较,每指定一个值都要全文扫描有无重复
public int firstUniqChar(String s) {
char[] arr = s.toCharArray();
int n = arr.length;
for(int i = 0; i < n; i++){
boolean flag = true;
//遍历每个值与当前值比较,有重复改false
for(int j = 0; j < n; j++){
if(i != j && arr[i] == arr[j]){
flag = false;
}
}
//遍历完没有和当前值相同的
if(flag){
return i;
}
}
return -1;
}

然后LeetCode的测试用例字符串也是真的长(只截取了部分下面还可以翻页),所以在n^2的情况下超时。
解法二:细节优化
上面的解法是有可优化的点的。我们去查找第一个只出现一次的,那么一个值找到相同的后我们就不必要再往后了遍历因为不需要看它有几个相同的,它不满足就应该看下一个值也就是应该加上break。
for(int j = 0; j < n; j++){
if(i != j && arr[i] == arr[j]){
flag = false;
break;
}
}
这一点是影响还是比较大的,第二点就是我们去判断遍历完了没有重复也就是找到了可以直接返回的处理我们除了定义flag在循环体标记,到循环去判断处理。其实我们去表达循环完后的处理也可以在循环体里面,也就是循环到最后了仍然不满足相等。
for(int j = 0; j < n; j++){
if(i != j && arr[i] == arr[j]){
break;
}else if(j == n-1){
return i;
}
}
这样和前面的就是同一个意思,并且少定义一个变量flag,并且最后我们的代码不会超时
public int firstUniqChar(String s) {
char[] arr = s.toCharArray();
int n = arr.length;
for(int i = 0; i < n; i++){
for(int j = 0; j < n; j++){
if(i != j && arr[i] == arr[j]){
break;
}else if(j == n - 1){
return i;
}
}
}
return -1;
}
时间O(n^2),空间O(1)

解法三:Hash表
那么使用hash表存信息,那么就不用重复遍历了
public int firstUniqChar(String s) {
char[] arr = s.toCharArray();
int n = arr.length;
HashMap<Character, Integer> count = new HashMap<Character, Integer>();
//遍历存下所有信息
for (int i = 0; i < n; i++) {
char c = arr[i];
count.put(c, count.getOrDefault(c, 0) + 1);
}
//遍历查看key值为1的
for (int i = 0; i < n; i++) {
if (count.get(arr[i]) == 1)
return i;
}
return -1;
}
时间O(n),空间O(n),虽然如此但效率反而比双指针要差,这主要是它的这些方法。

解法四:数组
用Hash表能存,那用数组也应该是可以的,一样的key位索引值判断是不是1。同一个字母就是同一个地方对应值就加一。统计完之后遍历字符串按字符串的顺序去数组查率先等于1的就返回
public int firstUniqChar(String s) {
int[] chars = new int[26];
char[] arr = s.toCharArray();
for (char c : arr) {
chars[c - 'a'] += 1;
}
for (int i = 0; i < arr.length; ++i) {
if (chars[arr[i] - 'a'] == 1) {
return i;
}
}
return -1;
}
和解法二同样的一个思路选取数组这种数据结构,效率就直接上去了。

解法五:细节优化
上述数组解法在效率上仍然是有可优化点,因为我们去比较两个容器的时候谁短我们就遍历谁。更何况这里只需要拿值到另一个容器参考只需要一次遍历,那我们更应该遍历短的。那么当字符串长度小于26和上面一样遍历字符串到数组去记录,最后再遍历数组看结果,如果字符串长于26那么我们就遍历a-z这26个字母
int result = -1;
for (char i = 'a'; i <= 'z'; ++i) {
int begin = s.indexOf(i);
int end = s.lastIndexOf(i)
// 在字符串中存在该字符并且唯一
if (begin != -1 && begin == end {
// 不仅要唯一,且索引还要小。遍历完成拿到字符串最前的唯一
result = (result == -1 || result > begin) ? begin : result;
}
}
那么在字符串长度很大的情况下也只需要完整遍历26次就能找到首个唯一,完整代码如下:
public int firstUniqChar(String s) {
// 字符串长度不超过26
if (s.length() <= 26) {
int[] chars = new int[26];
char[] arr = s.toCharArray();
for (char c : arr) {
chars[c - 'a'] += 1;
}
for (int i = 0; i < arr.length; ++i) {
if (chars[arr[i] - 'a'] == 1) {
return i;
}
}
}
//只遍历26个字母
int result = -1;
for (char i = 'a'; i <= 'z'; ++i) {
int begin = s.indexOf(i);
int end = s.lastIndexOf(i);
if (begin != -1 && begin == end) {
result = (result == -1 || result > begin) ? begin : result;
}
}
return result;
}

总结
题目难度呢属于简单,双指针、hash表这样成对的解法就出来了,主要是通过此题去回顾一些注意点比如双循环的优化,循环中字符串的方法频繁的进出也是有一定的浪费,可以先拿数组出来操作会好一点。适合体会解题迭代的一个流程。
LeetCode初级算法之字符串:387 字符串中的第一个唯一字符的更多相关文章
- C#LeetCode刷题之#387-字符串中的第一个唯一字符(First Unique Character in a String)
问题 该文章的最新版本已迁移至个人博客[比特飞],单击链接 https://www.byteflying.com/archives/3939 访问. 给定一个字符串,找到它的第一个不重复的字符,并返回 ...
- 前端与算法 leetcode 387. 字符串中的第一个唯一字符
目录 # 前端与算法 leetcode 387. 字符串中的第一个唯一字符 题目描述 概要 提示 解析 解法一:双循环 解法二:Set法单循环 算法 传入测试用例的运行结果 执行结果 GitHub仓库 ...
- LeetCode初级算法--字符串02:字符串中的第一个唯一字符
LeetCode初级算法--字符串02:字符串中的第一个唯一字符 搜索微信公众号:'AI-ming3526'或者'计算机视觉这件小事' 获取更多算法.机器学习干货 csdn:https://blog. ...
- LeetCode初级算法的Python实现--字符串
LeetCode初级算法的Python实现--字符串 # 反转字符串 def reverseString(s): return s[::-1] # 颠倒数字 def reverse(x): if x ...
- Java实现 LeetCode 387 字符串中的第一个唯一字符
387. 字符串中的第一个唯一字符 给定一个字符串,找到它的第一个不重复的字符,并返回它的索引.如果不存在,则返回 -1. 案例: s = "leetcode" 返回 0. s = ...
- Leecode刷题之旅-C语言/python-387 字符串中的第一个唯一字符
/* * @lc app=leetcode.cn id=387 lang=c * * [387] 字符串中的第一个唯一字符 * * https://leetcode-cn.com/problems/f ...
- leecode刷题(13) -- 字符串中的第一个唯一字符
leecode刷题(13) -- 字符串中的第一个唯一字符 字符串中的第一个唯一字符 描述: 给定一个字符串,找到它的第一个不重复的字符,并返回它的索引.如果不存在,则返回 -1. 案例: s = & ...
- LeetCode 387: 字符串中的第一个唯一字符 First Unique Character in a String
题目: 给定一个字符串,找到它的第一个不重复的字符,并返回它的索引.如果不存在,则返回 -1. Given a string, find the first non-repeating charact ...
- LeetCode 387. First Unique Character in a String (字符串中的第一个唯一字符)
题目标签:String, HashMap 题目给了我们一个 string,让我们找出 第一个 唯一的 char. 设立一个 hashmap,把 char 当作 key,char 的index 当作va ...
随机推荐
- 比特魔方原创,用十分钟在Cocos-BCX上发行了自己的NFT
比特魔方原创 作者 | 第二个区块 出品 |比特魔方 NFT正在积累越来越多的共识.每看到人们讨论NFT,我隐约就能联想到2019年人们谈论DeFi的时候.隐约让我感到欠缺的是,相对2019年的DeF ...
- GitLab集成Jenkins、Harborn构建pipeline流水线任务
一.计划 在jenkins中构建流水线任务时,从GitLab当中拉取代码,通过maven打包,然后构建dokcer镜像,并将镜像推送至harbor当中.Jenkins中含开发.测试.生产视图,开发人员 ...
- 《.NET 5.0 背锅案》第3集-剧情反转:EnyimMemcachedCore 无罪,.NET 5.0 继续背锅
今天晚上基于第2集中改进版的 EnyimMemcachedCore 进行了发布,发布过程中故障重现,最大的嫌犯 EnyimMemcachedCore 被证明无罪,暂时委屈 .NET 5.0 继续背锅. ...
- Android状态栏与布局重叠解决方案
问题起因: 同组的同事将项目全局设置成了沉浸式,对于我这个半路过来开发的人 可真是头疼呵~ 没办法,那就我自己添加一个头吧.也可以在布局中取消沉浸式,不过我这个是在fragment中,为了不修改之前的 ...
- UNP第11章——名字与地址转换
1.域名系统 程序中只使用主机名和服务名的好处是,如果IP或端口变化,只需要改变映射关系,不需要重新编译程序. 1.1 资源记录 DNS的条目为资源记录,有用的项如下: A IPv4地址 AAAA I ...
- NO.A.0009——day04——idea的安装及配置教程
概述: 集成开发环境:IDE.开发工具Integrated Development Environment,IDE, 1.如果自己手洗衣服: 1. 准备一盆水 2. 放入衣服浸泡30分钟 3. 搓洗衣 ...
- mysql语句的书写顺序和执行顺序
mysql语句的书写顺序和执行顺序有很大差异. 书写顺序,mysql的一般书写顺写为: select <要返回的数据列> from <表名> <join, left jo ...
- 用JavaScript实现全选-反选
实现全选-反选 在日常生活我们会遇到需要全选-反选的地方,其实用JavaScript也能实现. 样式如下所示: 样式代码如下所示: <!DOCTYPE html PUBLIC "-// ...
- IP地址分类的计算方法
IP地址由四段组成,每个字段是一个字节,8位,最大值是255,但实际中我们用点分十进制记法. IP地址由两部分组成,即网络地址和主机地址.网络地址表示其属于互联网的哪一个网络(常见ABC三类,以固定网 ...
- webug第十六关:明天双十一
---恢复内容开始--- 第十六关:明天双十一 不说了...只能看着源码做出来 ---恢复内容结束---