昨天下午参加了华为的上机笔试,3道题,但是答得不是很理想,回来有好好思考了一些,现在这里把相对较难的那道题分享一下。题目是这样的:
问题描述:
在给定字符串中找出单词( “单词”由大写字母和小写字母字符构成,其他非字母字符视为单词的间隔,如空格、问号、数字等等;另外单个字母不算单词);找到单词后,按照长度进行降序排序,(排序时如果长度相同,则按出现的顺序进行排列),然后输出到一个新的字符串中;如果某个单词重复出现多次,则只输出一次;如果整个输入的字符串中没有找到单词,请输出空串。输出的单词之间使用一个“空格”隔开,最后一个单词后不加空格。
要求实现函数:
void my_word(charinput[], char output[])
【输入】 char input[], 输入的字符串
【输出】 char output[],输出的字符串
【返回】 无
示例
输入:charinput[]=”some local buses, some1234123drivers” ,
输出:charoutput[]=”drivers local buses some”
输入:charinput[]=”%A^123 t 3453i*()” ,
输出:charoutput[]=””
乍一看,题目的确不怎么难,但是其中隐藏着很多细节。我的思考是这样的:
- 首先分离其中的字符串(单词),需要注意的是单个字母不能成为单词,当到达字符串末尾时,即使没有符号,也需要分离单词。
- 其次是排除其中的重复单词(java里使用 Set实现,set中不能出现重复的内容)
- 对单词进行排序,按照单词长度逆序,如果单词的长度相同,则按照单词出现的先后顺序排序。在项目中我使用的是选择排序,时间复杂度略小于o(n*n)
- 输出数据,如果没有单词输出””
下面贴出我实现的代码,如发现有什么问题,请指出:
package com.toozhao.test;
import java.util.HashSet;
import java.util.Set;
/**
* {@link} http://toozhao.com
* @author Junv
*
*/
public class MyWord {
/**
* @param args
*/
public static void main(String[] args) {
MyWord test = new MyWord();
System.out
.println(test.process("some local buses, some1234123drivers"));
}
/**
* 将输入信息通过处理,使其满足要求。并返回
*
* @param input
* @return
*/
public String process(String input) {
char[] array = input.toCharArray();
Set<String> list = new HashSet<String>();
int mark = 0;// 用于标记当前截断位置
boolean formerIsChar = false; // 设置上一个是否是字符
for (int i = 0; i < array.length; i++) {
// 说明不是字母
if (array[i] < 65 || array[i] > 122
|| (array[i] > 90 && array[i] < 97)) {
// System.out.println(array[i]);
// 如果上一个数为字符,且单词长度不能为1
if (formerIsChar == true && (i!= mark+1)) {
char[] temp = new char[i - mark];
System.arraycopy(array, mark, temp, 0, i - mark); // 复制数组
list.add(this.getStr(temp)); // 添加到list中
}
mark = i + 1; // 标记为下一个字符开始的脚标
formerIsChar = false;
} else {
// 说明上一个是字符
formerIsChar = true;
// 处理最后的字符串,即使不是非字符也需要截断
if (i == array.length - 1) {
char[] temp = new char[i - mark + 1];
System.arraycopy(array, mark, temp, 0, i - mark + 1);
list.add(this.getStr(temp));
}
}
}
// 如果为空则返回空格
if (list.size() == 0) {
return "";
}
String[] myArray = new String[list.size()];
// 将list转为数组
list.toArray(myArray);
// 对数组排序
this.sort(myArray);
// 遍历得到结果
StringBuilder sb = new StringBuilder();
for (int j = 0; j < myArray.length; j++) {
sb.append(myArray[j]);
sb.append(" ");
}
return sb.toString();
}
/**
* 将字符数组转成字符串
*
* @param charStr
* @return
*/
public String getStr(char[] charStr) {
StringBuilder sb = new StringBuilder();
for (int i = 0; i < charStr.length; i++) {
sb.append(charStr[i]);
}
return sb.toString();
}
/**
* 按照长度降序,如果长度相同,则比较谁更先出现。<br>
* 使用的是选择排序
* @param original
* @return
*/
public String[] sort(String[] original) {
for (int i = 0; i < original.length - 1; i++) {
int temp = i;
for (int j = i + 1; j < original.length; j++) {
// 如果后面的长度大于前面
if (original[j].length() > original[i].length()) {
temp = j;
}
// 如果长度相同,则看谁出现得早
if (original[j].length() == original[i].length()) {
if (j > i) {
temp = j;
}
}
}
if (temp != i) {
String tempStr = original[i];
original[i] = original[temp];
original[temp] = tempStr;
}
}
return original;
}
}