题目如下:

在一个字符串(0<=字符串长度<=10000,全部由字母组成)中找到第一个只出现一次的字符,并返回它的位置, 如果没有则返回 -1(需要区分大小写).

先上代码:

 class Solution {
public:
int FirstNotRepeatingChar(string str) {
//冷静的思考这个个题目,明显我们需要完整的遍历完整个字符串才可以
int length = str.size();
if(length == )
return -;
vector<int>cnt(,);//初始化所有字母的统计为0;
for(int i = ;i < length;i++)
{
if(str[i] <= )
cnt[str[i]-]++;
else
cnt[str[i]--]++;
}
for(int i = ;i <length;i++)
{
if(str[i] <= )
{
if(cnt[str[i]-] == )
return i;
}
else
{
if(cnt[str[i]--] == )
return i;
}
}
return -;
}
};

这个问题的思路是什么?

我知道很多人要说一看就是hash,抱歉说实话,我之前学习过hash,但现在还不知道怎么用。但是并不妨碍我做这一道题!

首先,题目说找到第一个只出现一次的字符,这说明了什么?

试想?如果你不遍历完整个字符串,你知道第一个只出现一次的字符是什么吗?答案是无法知道的。这也就断了你用其他高超技巧的念头,也就是说如果你不遍历完,那么你根本无法得知。

因此,我们需要做的首先就是遍历完所有的字符。

题目中说了只有大写和小写字母,那么我们可以申请一个最小为26*2的空间大小的数组(上文已经说了你跟本不知道有哪些字符要出现,所以你必须至少申请这么多!)

然后我们要做什么?

当然事统计每个字符出现的频数。

我们的想法很简单,让A-Z对应vector的索引0-25,让a-z对应索引26-51。也就是对应索引位置的值为该字符出现的次数,比如cnt[0]表示字符A出现的次数。

也就是下面这段代码:

  for(int i = ;i < length;i++)
{
if(str[i] <= )
cnt[str[i]-]++;
else
cnt[str[i]--]++;
}

当我们统计完之后,就应该查找了,说实话,我做完了这一步并不知道怎么找第一个,并一度认为这样的做法无法找出第一个出现一次的字符。为何我会这么认为呢?这里描述一下我开始的思维过程:

比如 f和a都出现了一次,但是f先出现,a后出现,所以正确的结果中应该返回f的位置。但是我当时想:这个统计数组是按照字母表排序的,如果我按照数组cnt查找第一个,那么第一个cnt==1的必然是a,那么我当然会去原字符串中去找a所在的位置。那么此时返回的结果并不是第一次只出现一次的字符,而是只出现一次的字符在字母表中的第一个。这个结果当然是错误的。究其原因,是因为我们试图以数组cnt作为索引。去遍历cnt。

但是实际上呢?我们可以不这么做,我们应该以原字符串为索引。也就是以原字符串的字符为key,而已cnt的值为value。这样才能得到正确的结果:

 for(int i = ;i <length;i++)
{
if(str[i] <= )
{
if(cnt[str[i]-] == )
return i;
}
else
{
if(cnt[str[i]--] == )
return i;
}
}

我们可以看到第五行和第十行的代码,明显是以字符为索引,以cnt为value,这样便得到了正确的结果。

倘若我们以cnt为key,str为value。显然是得不到正确结论的

牛客网剑指offer第34题——找到第一个只出现一次的字符的更多相关文章

  1. 牛客网剑指offer第13题——调整数组顺序使得奇数位于偶数前面

    题目来源:剑指offer 题目: 输入一个整数数组,实现一个函数来调整该数组中数字的顺序,使得所有的奇数位于数组的前半部分,所有的偶数位于数组的后半部分,并保证奇数和奇数,偶数和偶数之间的相对位置不变 ...

  2. 牛客网剑指offer第19题——顺时针打印矩阵

    这个题看似很简单: 题目: 输入一个矩阵,按照从外向里以顺时针的顺序依次打印出每一个数字,例如,如果输入如下4 X 4矩阵: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 1 ...

  3. 牛客网剑指offer第21题——判断出栈序列是否是入栈序列

    题目: 输入两个整数序列,第一个序列表示栈的压入顺序,请判断第二个序列是否可能为该栈的弹出顺序.假设压入栈的所有数字均不相等.例如序列1,2,3,4,5是某栈的压入顺序,序列4,5,3,2,1是该压栈 ...

  4. 数组中只出现过一次的数字 牛客网 剑指Offer

    数组中只出现过一次的数字 牛客网 剑指Offer 题目描述 一个整型数组里除了两个数字之外,其他的数字都出现了偶数次.请写程序找出这两个只出现一次的数字. def FindNumsAppearOnce ...

  5. 数组中出现次数超过一半的数字 牛客网 剑指Offer

    数组中出现次数超过一半的数字 牛客网 剑指Offer 题目描述 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字 ...

  6. 数据流中的中位数 牛客网 剑指Offer

    数据流中的中位数 牛客网 剑指Offer 题目描述 如何得到一个数据流中的中位数?如果从数据流中读出奇数个数值,那么中位数就是所有数值排序之后位于中间的数值.如果从数据流中读出偶数个数值,那么中位数就 ...

  7. 数字在排序数组中出现的次数 牛客网 剑指Offer

    数字在排序数组中出现的次数 牛客网 剑指Offer 题目描述 统计一个数字在排序数组中出现的次数. class Solution: def GetNumberOfK(self, data, k): i ...

  8. 数值的整数次方 牛客网 剑指Offer

    数值的整数次方 牛客网 剑指Offer 题目描述 给定一个double类型的浮点数base和int类型的整数exponent.求base的exponent次方 class Solution: #run ...

  9. 按之字形顺序打印二叉树 牛客网 剑指Offer

    按之字形顺序打印二叉树 牛客网 剑指Offer 题目描述 请实现一个函数按照之字形打印二叉树,即第一行按照从左到右的顺序打印,第二层按照从右至左的顺序打印,第三行按照从左到右的顺序打印,其他行以此类推 ...

随机推荐

  1. 如何回收VCSA 6自带的vPostgres数据库空间

    最近有学生连续反应由于VCSA磁盘空间满了,导致服务无法正常启动,寻求压缩数据库空间的问题.首先说下,VCSA的数据库是没办法图形界面管理的, 它的内置vPostgres数据库的管理只能通过命令行来完 ...

  2. SpringBoot:Shiro 整合 Redis

    前言 前段时间做了一个图床的小项目,安全框架使用的是Shiro.为了使用户7x24小时访问,决定把项目由单机升级为集群部署架构.但是安全框架shiro只有单机存储的SessionDao,尽管Shrio ...

  3. GIL锁和进程/线程池

    GIL锁 1.GIL锁 全局解释器锁,就是一个把互斥锁,将并发变成串行,同一时刻只能有一个线程使用共享资源,牺牲效率,保证数据安全,也让程序员避免自己一个个加锁,减轻开发负担 带来的问题 感觉单核处理 ...

  4. 第一次作业:使用java实现word count

    github项目地址: https://github.com/changrui520/homework 作业要求: 可执行程序命名为:wc.exe. 该程序处理用户需求的模式为:wc.exe [par ...

  5. [LC] 238. Product of Array Except Self

    Given an array nums of n integers where n > 1,  return an array output such that output[i] is equ ...

  6. OpenCV 实现自己的线性滤波器

    #include "opencv2/imgproc/imgproc.hpp" #include "opencv2/highgui/highgui.hpp" #i ...

  7. C++对数组进行复制

    C++ 风格的复制操作 使用STL中的copy算法 int a[] = {1,2,3,4,5}; int b[5]; std::copy(std::begin(a),std::end(a),std:: ...

  8. mybatis generator 使用教程(生成带注释的实体类)

    引言: 最近的一个项目,由于数据库表巨多,导致需要创建N多个java实体.dao.mapper.xml映射文件,如果均使用纯手工编写,无疑需要耗费大量时间和精力.于是上网学习了mybatis gene ...

  9. 虚拟网卡 TUN/TAP 驱动程序设计原理(经典)

    盗用-收藏 简介 虚拟网卡Tun/tap驱动是一个开源项目,支持很多的类UNIX平台,OpenVPN和Vtun都是基于它实现隧道包封装.本文将介绍tun/tap驱动的使用并分析虚拟网卡tun/tap驱 ...

  10. IP地址结构分类(包括主机号和网络好计算)

    IP地址:互联网上的每个接口的唯一标识. 长度:32bit. 五类不同的互联网地址格式: 各类IP地址范围: 点分十进制:32位的地址通常写成四个十进制数,其中每个整数对应一个字节. 主机号和网络的计 ...