问题描述:
请实现一个函数用来找出字符流中第一个只出现一次的字符。例如,当从字符流中只读出前两个字符"go"时,第一个只出现一次的字符是"g"。当从该字符流中读出前六个字符“google"时,第一个只出现一次的字符是"l"。
输出描述:
如果当前字符流没有存在出现一次的字符,返回#字符。
解题思路:
首先对题目进行分析,第一:输入的是字符流,那么就代表字符是一个一个往里输入的,由字符流是一个一个的输入很容易根据先进先出原则想到队列;第二:判断只出现一次的字符,想到hash表来存储次数和字符的关系。
接下来理理思路:
- 需要队列和hash表,在Java中linkedlist可以实现队列操作,而hashmap可以存放字符和次数的键值对。因此要先初始化一个LinkedList和HashMap对象。
LinkedList<Character> qu = new LinkedList<Character> ();
HashMap<Character, Integer> map = new HashMap<Character, Integer> ();
- 题目中给出了两个函数:Insert(char ch)和FirstAppearingOnce()。对于Insert函数,我们要做的是将字符放入队列中,并且判断是否在HashMap中出现过,如果没有则对应字符的值设为0(为什么设0后面解释),如果出现过则不用放入队列中了。现在解释为什么设为0,因为不管有没有出现过,HashMap中对应的值都要加1,所以我们将0表示初始化。
- 对于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集合)。