字符流中第一个不重复的字符(Java实现)

问题描述:

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

输出描述:

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

解题思路:

首先对题目进行分析,第一:输入的是字符流,那么就代表字符是一个一个往里输入的,由字符流是一个一个的输入很容易根据先进先出原则想到队列;第二:判断只出现一次的字符,想到hash表来存储次数和字符的关系。
接下来理理思路:

  1. 需要队列和hash表,在Java中linkedlist可以实现队列操作,而hashmap可以存放字符和次数的键值对。因此要先初始化一个LinkedList和HashMap对象。
    LinkedList<Character> qu = new LinkedList<Character> ();
    HashMap<Character, Integer> map = new HashMap<Character, Integer> ();
  1. 题目中给出了两个函数:Insert(char ch)和FirstAppearingOnce()。对于Insert函数,我们要做的是将字符放入队列中,并且判断是否在HashMap中出现过,如果没有则对应字符的值设为0(为什么设0后面解释),如果出现过则不用放入队列中了。现在解释为什么设为0,因为不管有没有出现过,HashMap中对应的值都要加1,所以我们将0表示初始化。
  2. 对于FirstAppearingOnce()函数,要做的就是在队列非空的情况下从队首开始依次判断是否出现过,如果当前字符对应的值是1的话说明是第一次出现,则返回该字符即可,否则则不是第一次出现,弹出该字符,继续判断下一个字符。如果一直到队列空都没有返回则说明此次查找没有出现一次的字符,返回#即可。

代码:

import java.util.LinkedList;
import java.util.HashMap;
public class Solution {
    //Insert one char from stringstream
    LinkedList<Character> qu = new LinkedList<Character> ();
    HashMap<Character, Integer> map = new HashMap<Character, Integer> ();

    public void Insert(char ch)
    {
        // 如果是第一次出现,则添加到队列中
        if(!map.containsKey(ch)){
            qu.push(ch);
            map.put(ch, 0);
        }
        // 不管怎样,map中也计数
        //先获取之前的次数
        Integer count = map.get(ch);
        //count++;
        //再次存入 更新
        map.put(ch, ++count);
    }
    //return the first appearence once char in current stringstream
    public char FirstAppearingOnce()
    {
        while(!qu.isEmpty()){
            char ch = qu.getLast();  // 因为是栈结构,因此取出底部
            if(map.get(ch) == 1){
                System.out.println(ch);
                return ch;
            }else{   // 弹出最后面的元素,做成列表格式
                qu.removeLast();
            }
        }
        System.out.println("#");
        return '#';
    }
}

测试类进行测试:

public class Test {

    public static void main(String[] args) {
        Solution sl = new Solution();
        Scanner sc = new Scanner(System.in);
        if(sc.hasNext()){
            String str = sc.next();
            char[] str1 = str.toCharArray();
            for (int i = 0; i <str.length() ; i++) {
                sl.Insert(str1[i]);
                sl.FirstAppearingOnce();
            }
        }
    }
}

结果:

在这里插入图片描述

相关知识点:

LinkedList相关方法:
public void addFirst(E e) :将指定元素插入此列表的开头。
public void addLast(E e) :将指定元素添加到此列表的结尾。
public E getFirst() :返回此列表的第一个元素。
public E getLast() :返回此列表的最后一个元素。
public E removeFirst() :移除并返回此列表的第一个元素。
public E removeLast() :移除并返回此列表的最后一个元素。
public E pop() :从此列表所表示的堆栈处弹出一个元素。
public void push(E e) :将元素推入此列表所表示的堆栈。
public boolean isEmpty() :如果列表不包含元素,则返回true。
HashMap常用方法:
public V put(K key, V value) : 把指定的键与指定的值添加到Map集合中。
public V remove(Object key) : 把指定的键 所对应的键值对元素 在Map集合中删除,返回被删除元素的
值。
public V get(Object key) 根据指定的键,在Map集合中获取对应的值。
public Set keySet() : 获取Map集合中所有的键,存储到Set集合中。
public Set<Map.Entry<K,V>> entrySet() : 获取到Map集合中所有的键值对对象的集合(Set集合)。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值