面试题35:第一个只出现一次的字符

题目:在一个字符串中找到第一个只出现一次的字符。如输入abaccdeff,则输出b。(2006年google的一道笔试题。)

分析:

首先应向确认一下是ASCII字符串,而不是Unicode字符串。用hash表求解即可,由于需要先遍历一次,时间复杂度为O(n),空间复杂度为O(1) (256个ASCII字符).

满足题意的代码如下:

#include<cstdio>
#include<string>
#include<unordered_map>
using namespace std;
class Solution {
public:
char FirstNotRepeatingChar(string str) {
if(str.size() == 0) return '\0';
unordered_map<char, int> countMap; // 使用C++中的map的insert时,返回结果是自动按key排序(增序)后的结果
for(int i = 0;i < str.size();i++){
if(countMap.find(str[i]) == countMap.end())
countMap[str[i]]=1;
// countMap.insert({str[i], 1}); // 映射表中没找到相关记录,加到映射表,次数设置为1
else countMap[str[i]]++; // 映射表中找到了相关记录,将映射表中的次数+1
}
// int pos = -1;
char ch;
for(int i = 0;i < str.size();i++){
if(countMap[str[i]] == 1){
// pos = i;
ch=str[i];
break;
}
}
return ch;
}
};
// 以下为测试
int main()
{
Solution sol;
string str1="abaccdeff";
string str2="cbacnba";
char res1 = sol.FirstNotRepeatingChar(str1);
char res2 = sol.FirstNotRepeatingChar(str2);
printf("%c\n", res1);
printf("%c\n", res2);
return 0;
}

也可简化为:

#include<cstdio>
#include<string>
#include<unordered_map>
using namespace std;
class Solution {
public:
char FirstNotRepeatingChar(string str) {
int hash[256] = {0};
for(auto c : str){ // 遍历一次,复杂度为O(n)
hash[c] ++;
}
char ch;
for(int i=0;i<str.size();i++){
if(hash[str[i]] == 1){
//return i;
return str[i];
}
}
return '\0'; // if(str.size() == 0) return '\0';
}
};
// 以下为测试
int main()
{
Solution sol;
string str1="ABACCDEFF";
string str2="cbacnba";
char res1 = sol.FirstNotRepeatingChar(str1);
char res2 = sol.FirstNotRepeatingChar(str2);
printf("%c\n", res1);
printf("%c\n", res2);
return 0;
}

相比而言,前一种方法更高效,256个字符可能只出现很少的一部分,后面这种方法在空间上消耗多一点...

九度OJ 提交网址 http://ac.jobdu.com/problem.php?pid=1283

牛客网OJ 改编: 在一个字符串(1<=字符串长度<=10000,全部由字母组成)中找到第一个只出现一次的字符的位置。若为空串,返回-1。位置索引从0开始。

提交网址: http://www.nowcoder.com/practice/1c82e8cf713b4bbeb2a5b31cf5b0417c?tpId=13&tqId=11187

输入:

一个字符串。

输出:

输出第一个只出现一次的字符下标,没有只出现一次的字符则输出-1。

样例输入:
ABACCDEFF
AA
样例输出:
1
-1

牛客网 AC代码:
class Solution {
public:
int FirstNotRepeatingChar(string str) {
if(str.size() == 0) return -1;
unordered_map<char, int> countMap; // 使用C++中的map的insert时,返回结果是自动按key排序(增序)后的结果
for(int i = 0;i < str.size();i++){
if(countMap.find(str[i]) == countMap.end())
countMap[str[i]]=1; // 映射表中没找到相关记录,加到映射表,次数设置为1
else countMap[str[i]]++; // 映射表中找到了相关记录,将映射表中的次数+1
}
int pos = -1;
for(int i = 0;i < str.size();i++){
if(countMap[str[i]] == 1){
pos = i;
break;
}
}
return pos;
}
};

精简之后:

class Solution {
public:
int FirstNotRepeatingChar(string str) {
int hash[256] = {0};
for(auto c : str){ // 遍历一次,复杂度为O(n)
hash[c] ++;
}
for(int i=0;i<str.size();i++){
if(hash[str[i]] == 1){
return i;
}
}
return -1; // if(str.size() == 0) return -1;
}
};

华为OJ034-找出字符串中第一个只出现一次的字符

时间限制:1秒  空间限制:32768K 参与人数:157
本题知识点: 字符串

题目描述
找出字符串中第一个只出现一次的字符

接口说明
原型:
char FindChar(InputString);
输入参数:字符串InputString

输出参数:
如果无此字符 请输出该字符;如果无此字符,请输出'.'。

输入描述
输入一串字符

输出描述
输出一个字符

输入例子
asdfasdfo

输出例子
o

AC代码(C++风格):

#include<iostream>
#include<string>
#include<unordered_map>
using namespace std;
char FirstNotRepeatingChar(string str)
{
if(str.size() == 0) return '.';
unordered_map<char, int> countMap;
for(int i = 0;i < str.size();i++){
if(countMap.find(str[i]) == countMap.end())
countMap[str[i]]=1;
else countMap[str[i]]++; // 映射表中找到了相关记录,将映射表中的次数+1
}
char ch;
for(int i = 0;i < str.size();i++){
if(countMap[str[i]] == 1){
ch=str[i];
break;
}
}
return ch;
}
int main() {
string str;
while(cin>>str) {
char ch=FirstNotRepeatingChar(str);
cout<<ch<<endl;
}
return 0;
}

#include<iostream>
#include<string>
#include<unordered_map>
using namespace std;
char FindChar(string str)
{
unordered_map<char,int> m;
char ch;
for(unsigned int i=0; i<str.size(); i++)
m[str[i]]++;
unsigned int i;
for(i=0; i<str.size(); i++) {
if(m[str[i]]==1) {
ch=str[i]; break;
}
}
if(i==str.size()) ch='.';
return ch;
}
int main() {
string str;
while(cin>>str) {
char ch=FindChar(str);
cout<<ch<<endl;
}
return 0;
}

AC代码(使用指针):

#include<iostream>
#include<string>
using namespace std;
char FindChar(char *input)
{
char *p=input;
int hash[256];
for(int i=0;i!=256;i++)
hash[i]=0; while(*p!='\0')
{
hash[*p]++;
p++;
}
char *p1=input; while(*p1!='\0')
{
if(hash[*p1]==1)
{
return *p1;
}
p1++;
}
return '.';
}
int main()
{
char str[1000];
while(cin>>str)
{
char ch=FindChar(str);
cout<<ch<<endl;
}
return 0;
}

C++版 - 剑指Offer 面试题35:第一个只出现一次的字符 解题报告(华为OJ034-找出字符串中第一个只出现一次的字符)的更多相关文章

  1. C++版 - 剑指Offer 面试题45:圆圈中最后剩下的数字(约瑟夫环问题,ZOJ 1088:System Overload类似)题解

    剑指Offer 面试题45:圆圈中最后剩下的数字(约瑟夫环问题) 原书题目:0, 1, - , n-1 这n个数字排成一个圈圈,从数字0开始每次从圆圏里删除第m个数字.求出这个圈圈里剩下的最后一个数字 ...

  2. C++版 - 剑指offer之面试题37:两个链表的第一个公共结点[LeetCode 160] 解题报告

    剑指offer之面试题37 两个链表的第一个公共结点 提交网址: http://www.nowcoder.com/practice/6ab1d9a29e88450685099d45c9e31e46?t ...

  3. C++版 - 剑指offer 面试题23:从上往下打印二叉树(二叉树的层次遍历BFS) 题解

    剑指offer  面试题23:从上往下打印二叉树 参与人数:4853  时间限制:1秒  空间限制:32768K 提交网址: http://www.nowcoder.com/practice/7fe2 ...

  4. C++版 - 剑指offer 面试题39:判断平衡二叉树(LeetCode 110. Balanced Binary Tree) 题解

    剑指offer 面试题39:判断平衡二叉树 提交网址:  http://www.nowcoder.com/practice/8b3b95850edb4115918ecebdf1b4d222?tpId= ...

  5. C++版 - 剑指Offer 面试题39:二叉树的深度(高度)(二叉树深度优先遍历dfs的应用) 题解

    剑指Offer 面试题39:二叉树的深度(高度) 题目:输入一棵二叉树的根结点,求该树的深度.从根结点到叶结点依次经过的结点(含根.叶结点)形成树的一条路径,最长路径的长度为树的深度.例如:输入二叉树 ...

  6. C++版 - 剑指offer 面试题24:二叉搜索树BST的后序遍历序列(的判断) 题解

    剑指offer 面试题24:二叉搜索树的后序遍历序列(的判断) 题目:输入一个整数数组,判断该数组是不是某二叉搜索树的后序遍历的结果.如果是则返回true.否则返回false.假设输入的数组的任意两个 ...

  7. C++版 - 剑指offer 面试题22:栈的压入、弹出序列 题解

    剑指offer 面试题22:栈的压入.弹出序列 提交网址: http://www.nowcoder.com/practice/d77d11405cc7470d82554cb392585106?tpId ...

  8. C#版 - 剑指offer 面试题9:斐波那契数列及其变形(跳台阶、矩形覆盖) 题解

    面试题9:斐波那契数列及其变形(跳台阶.矩形覆盖) 提交网址: http://www.nowcoder.com/practice/c6c7742f5ba7442aada113136ddea0c3?tp ...

  9. C++版-剑指offer 面试题6:重建二叉树(Leetcode105. Construct Binary Tree from Preorder and Inorder Traversal) 解题报告

    剑指offer 重建二叉树 提交网址: http://www.nowcoder.com/practice/8a19cbe657394eeaac2f6ea9b0f6fcf6?tpId=13&tq ...

随机推荐

  1. Linux-硬件

    1.服务器 计算节点服务器-用于后台逻辑运算,所以cpu,磁盘读写性能要求较高 web服务器-用于用户请求访问一些页面,如果高并发,磁盘读写性能要好,可以使用raid0或raid1或raid5技术(r ...

  2. AddIn 中当前完整文件名的获取

    Me.Application.ActiveWorkbook.Name 需要注意的是:只有当前文件已经存档的情况下,才能获得后缀名.

  3. python基础其他

    目录: 一.Python为什么受欢迎的本质,知道的人寥寥无几? 二.PyCharm的模板设置 三.pycharm的一些快捷键 四.PyCharm最新2018激活 五.python虚拟环境--virtu ...

  4. Java中的异步通知

    在我们的日常开发中,经常会遇到这样的问题--"我让你做一件事情,但是你做得很慢,并不能够立马返回给我结果,害我一直在那儿等着你给我返回结果,什么都做不了". 程序是自上而下顺序执行 ...

  5. leetcode-只出现一次的数字合并两个有序数组

    题目:合并两个有序数组 给定两个有序整数数组 nums1 和 nums2,将 nums2 合并到 nums1 中,使得 num1 成为一个有序数组. 说明: 初始化 nums1 和 nums2 的元素 ...

  6. mysql学习2

    1.视图 视图是一个虚拟表(非真实存在),其本质是[根据sql语句获取动态的数据集,并为其命名],用户使用时只需要使用[名称]即可获取结果集,并可以将其当作表来使用. 搜索临时表 SELECT * F ...

  7. 浅谈C++ STL

    C++ STL(标准模板库)是一套功能强大的 C++ 模板类,提供了通用的模板类和函数,这些模板类和函数可以实现多种流行和常用的算法和数据结构,如向量.链表.队列.栈. C++ 标准模板库的核心包括以 ...

  8. Java_集合

    定义: 是一种工具,就像是容器,能存储任意数量的具有共同属性的对象. 与数组比较优点: (1)数组定义后长度不可变,集合长度可变: (2)数组只能通过下标访问,且下标类型只能是数字型,而有的集合(ma ...

  9. Centos服务器上NFS灾备环境及KVM的搭建及使用

    1.概述 由于在单台服务器上搭建灾备环境需要KVM和NFS的支持,下面先列出KVM的搭建流程,再列出使用NFS实现单台服务器灾备的流程. A.搭建KVM环境 1>.主机环境准备 Linux Sy ...

  10. Python函数式编程之闭包

    -------------------------函数式编程之*******闭包------------------------ Note: 一:简介 函数式编程不是程序必须要的,但是对于简化程序有很 ...