剑指offer - 54 -- 字符流中第一个不重复的字符

题目描述

请实现一个函数用来找出字符流中第一个只出现一次的字符。例如,当从字符流中只读出前两个字符"go"时,第一个只出现一次的字符是"g"。当从该字符流中读出前六个字符“google"时,第一个只出现一次的字符是"l"。

返回值描述:

如果当前字符流没有存在出现一次的字符,返回#字符。

题解

Q1. 给定一个字符串(只不过这里的字符串是可变的),如果快速判断一个字符是否存在于字符串中,如果存在,也就是重复?
Q2. 这里先不考虑重复,如果快速返回第一个字符?有没有感觉有点像先来先服务?

A1:对于“重复问题”,惯性思维应该想到哈希或者set。对于“字符串问题”,大多会用到哈希。因此一结合,应该可以想到,判断一个字符是否重复,可以选择用哈希,在c++中,可以选择用map<char, int>

A2:对于字符流,源源不断的往池子中添加字符,然后还要返回第一个满足什么条件的字符,显然设计到了“顺序”,也就是先来的先服务,这种先进先出的数据结构不就是队列嘛。因此,这里可以用队列。

假如你已经知道了要用hash 和 queue 这两个数据结构,你可以试着自己想一想,接下来的算法过程是怎么样的?
这里我提供一个算法过程,如下:

  1. 初始化一个map<char, int> mp, queue<char> q

  2. 对于Insert(char ch)操作, 如果ch是第一次出现,则添加到q中,然后在mp中记录一下次数,如果不是第一次出现,也就是重复了,那么我们就没必要添加到q中,但是还是需要在mp中更新一下次数,因为之后要根据次数来判断是否重复。

  3. 对于FirstAppearingOnce()操作,我们直接判断q的头部,然后在mp中检查一下,是否重复,如果没有重复,那就是我们想要的数据。否则,如果重复了,那就应该弹出头部,然后判断下一个头部是否满足要求。

class Solution
{
public:
  //Insert one char from stringstream
    queue<char> q;
    map<char, int> mp;
    void Insert(char ch)
    {
         if(mp.find(ch) == mp.end()){// 如果是第一次出现, 则添加到队列中
             q.push(ch);
         }
        ++mp[ch];// 不管是不是第一次出现,都进行计数
    }
  //return the first appearence once char in current stringstream
    char FirstAppearingOnce()
    {
        while(!q.empty()){
            char ch = q.front();
            // 拿出头部,如果是第一次出现,则返回
            if(mp[ch] == 1){
                return ch;
            }
            else{   // 不是第一次出现,则弹出,然后继续判断下一个头部
                q.pop();
            }
        }
        return '#';
    }

};

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值