算法的学习笔记—(牛客JZ50)

😀前言

在处理字符串时,寻找第一个只出现一次的字符是一项常见的任务。本文将探讨几种有效的解法,包括使用 HashMap 和位集(BitSet)。

🏠个人主页:尘觉主页

文章目录

🥰第一个只出现一次的字符位置

😇题目链接

牛客网

🤔题目描述

在一个字符串中找到第一个只出现一次的字符,并返回它的位置。字符串只包含 ASCII 码字符。

Input: abacc
Output: b

💖解题思路

方法一:使用 HashMap

最直观的方法是使用 HashMap 来统计每个字符的出现次数。具体步骤如下:

  1. 遍历字符串,将字符作为键,出现次数作为值存入 HashMap。
  2. 再次遍历字符串,查找第一个出现次数为 1 的字符,返回其位置。

示例代码如下:

java 复制代码
public int FirstNotRepeatingChar(String str) {
    // 创建一个 HashMap,用于统计每个字符的出现次数
    Map<Character, Integer> countMap = new HashMap<>();
    
    // 遍历字符串,将字符作为键,出现次数作为值存入 HashMap
    for (char c : str.toCharArray()) {
        countMap.put(c, countMap.getOrDefault(c, 0) + 1);
    }
    
    // 再次遍历字符串,查找第一个出现次数为 1 的字符
    for (int i = 0; i < str.length(); i++) {
        if (countMap.get(str.charAt(i)) == 1) {
            // 返回该字符的位置
            return i;
        }
    }
    
    // 如果没有找到,返回 -1
    return -1;
}

以上实现的空间复杂度还不是最优的。考虑到只需要找到只出现一次的字符,那么需要统计的次数信息只有 0,1,更大,使用两个比特位就能存储这些信息。

方法二:使用整型数组

考虑到 ASCII 码字符有限,可以使用长度为 128 的整型数组代替 HashMap,来记录每个字符的出现次数。实现方法与上面类似,但效率更高。

java 复制代码
public int FirstNotRepeatingChar(String str) {
    // 创建一个长度为 128 的整型数组,用于统计字符出现次数
    int[] cnts = new int[128];
    
    // 遍历字符串,统计每个字符的出现次数
    for (int i = 0; i < str.length(); i++) {
        cnts[str.charAt(i)]++;
    }
    
    // 再次遍历字符串,查找第一个出现次数为 1 的字符
    for (int i = 0; i < str.length(); i++) {
        if (cnts[str.charAt(i)] == 1) {
            // 返回该字符的位置
            return i;
        }
    }
    
    // 如果没有找到,返回 -1
    return -1;
}

方法三:使用位集(BitSet)

为了进一步优化空间复杂度,可以使用 BitSet 来存储每个字符的状态,分为三种情况:未出现(0)、出现一次(1)和出现多次(2)。通过两个 BitSet,可以高效地统计字符的出现情况。

java 复制代码
public int FirstNotRepeatingChar2(String str) {
    // 创建两个 BitSet,用于记录字符的状态
    BitSet bs1 = new BitSet(128); // 用于记录字符出现一次
    BitSet bs2 = new BitSet(128); // 用于记录字符出现多次
    
    // 遍历字符串,更新字符的状态
    for (char c : str.toCharArray()) {
        if (!bs1.get(c) && !bs2.get(c)) {
            // 状态 0 -> 1(首次出现)
            bs1.set(c);
        } else if (bs1.get(c) && !bs2.get(c)) {
            // 状态 1 -> 2(再次出现)
            bs2.set(c);
        }
    }
    
    // 再次遍历字符串,查找第一个状态为 1 的字符
    for (int i = 0; i < str.length(); i++) {
        char c = str.charAt(i);
        if (bs1.get(c) && !bs2.get(c)) { // 状态 1
            // 返回该字符的位置
            return i;
        }
    }
    
    // 如果没有找到,返回 -1
    return -1;
}

😄总结

本文介绍了三种方法来找到字符串中第一个只出现一次的字符。通过不同的数据结构和算法,解决方案的效率和空间复杂度各有不同。选择合适的方法取决于具体需求和数据规模。

😁热门专栏推荐
想学习vue的可以看看这个

java基础合集

数据库合集

redis合集

nginx合集

linux合集

手写机制

微服务组件

spring_尘觉

springMVC

mybits

等等等还有许多优秀的合集在主页等着大家的光顾感谢大家的支持

🤔欢迎大家加入我的社区 尘觉社区

文章到这里就结束了,如果有什么疑问的地方请指出,诸佬们一起来评论区一起讨论😁

希望能和诸佬们一起努力,今后我们一起观看感谢您的阅读🍻

如果帮助到您不妨3连支持一下,创造不易您们的支持是我的动力🤞

相关推荐
诚丞成5 分钟前
滑动窗口篇——如行云流水般的高效解法与智能之道(1)
算法
所待.3836 分钟前
JavaEE之线程初阶(上)
java·java-ee
Winston Wood10 分钟前
Java线程池详解
java·线程池·多线程·性能
Chef_Chen13 分钟前
从0开始学习机器学习--Day33--机器学习阶段总结
人工智能·学习·机器学习
手握风云-14 分钟前
数据结构(Java版)第二期:包装类和泛型
java·开发语言·数据结构
喵叔哟34 分钟前
重构代码中引入外部方法和引入本地扩展的区别
java·开发语言·重构
尘浮生40 分钟前
Java项目实战II基于微信小程序的电影院买票选座系统(开发文档+数据库+源码)
java·开发语言·数据库·微信小程序·小程序·maven·intellij-idea
hopetomorrow1 小时前
学习路之压力测试--jmeter安装教程
学习·jmeter·压力测试
hopetomorrow1 小时前
学习路之PHP--使用GROUP BY 发生错误 SELECT list is not in GROUP BY clause .......... 解决
开发语言·学习·php
不是二师兄的八戒1 小时前
本地 PHP 和 Java 开发环境 Docker 化与配置开机自启
java·docker·php