难度:中等
输入一个非负整数数组,把数组里所有数字拼接起来排成一个数,打印能拼接出的所有数字中最小的一个。
输入: [10,2]
输出: “102”
输入: [3,30,34,5,9]
输出: “3033459”
提示:
0 < nums.length <= 100
说明:
0
,最后结果不需要去掉前导 0可以看成是一个排序问题,在比较两个字符串
s1
和s2
的大小时,应该比较的是s1+s2
和s2+s1
的大小:
- 如果
s1+s2 < s2+s1
,那么应该把s1
排在前面,否则应该把s2
排在前面。
总体流程:
strs
,保存各数字的字符串格式;strs
执行排序;strs
中的所有字符串,并返回。法一:快速排序
需修改快速排序函数中的排序判断规则。字符串大小(字典序)对比的实现方法:
<
, >
;A.compareTo(B)
;法二:内置函数
(string& x, string& y){ return x + y < y + x; } ;
(x, y) -> (x + y).compareTo(y + x);
法一:快速排序
C++
class Solution {
private:
void quickSort(vector<string>& strs, int l, int r){
if(l >= r) return;
int i = l + 1, j = r;
while(i <= j){
//从前往后找第一个比str[l] 大的字符串
while(i <= j && strs[i] + strs[l] <= strs[l] + strs[i])i++;
//从后往前找第一个比str[l] 小的字符串
while(i <= j && strs[j] + strs[l] >= strs[l] + strs[j])j--;
//交换
if(i < j)
swap(strs[i++], strs[j--]);
}
swap(strs[l], strs[j]);
quickSort(strs, l, j - 1);
quickSort(strs, j + 1, r);
}
public:
string minNumber(vector<int>& nums) {
// 1. 初始化
vector<string> strs;
for(int i = 0; i < nums.size(); i++){
strs.push_back(to_string(nums[i]));
}
// 2. 排序
quickSort(strs, 0, strs.size() - 1);
// 3. 返回结果
string ans;
for(string s : strs){
ans.append(s);
}
return ans;
}
};
Java
class Solution {
private void quickSort(String[] strs, int l, int r){
if(l >= r) return;
int i = l + 1, j = r;
String temp;
while(i <= j){
//从前往后找第一个比str[l] 大的字符串
while(i <= j && (strs[i] + strs[l]).compareTo(strs[l] + strs[i]) <= 0)i++;
//从后往前找第一个比str[l] 小的字符串
while(i <= j && (strs[j] + strs[l]).compareTo(strs[l] + strs[j]) >= 0)j--;
//交换
if(i < j){
temp = strs[i];
strs[i++] = strs[j];
strs[j--] = temp;
}
}
//此时j + 1 = i,strs[i]左边的字符串一定比strs[l]小,strs[j]右边的字符串一定比strs[l]大
temp = strs[l];
strs[l] = strs[j];
strs[j] = temp;
quickSort(strs, l, j - 1);
quickSort(strs, j + 1, r);
}
public String minNumber(int[] nums) {
// 1. 初始化
String[] strs = new String[nums.length];
for(int i = 0; i < nums.length; i++){
strs[i] = String.valueOf(nums[i]);
}
// 2. 排序
quickSort(strs, 0, strs.length - 1);
// 3. 返回结果
StringBuilder ans = new StringBuilder();
for(String s : strs){
ans.append(s);
}
return ans.toString();
}
}
法二:内置函数
C++
class Solution {
public:
string minNumber(vector<int>& nums) {
// 1. 初始化
vector<string> strs;
for(int i = 0; i < nums.size(); i++){
strs.push_back(to_string(nums[i]));
}
// 2. 内置函数排序
sort(strs.begin(), strs.end(), [](string& x, string& y){return x + y < y + x;});
// 3. 返回结果
string ans;
for(string s : strs){
ans.append(s);
}
return ans;
}
};
Java
class Solution {
public String minNumber(int[] nums) {
// 1. 初始化
String[] strs = new String[nums.length];
for(int i = 0; i < nums.length; i++){
strs[i] = String.valueOf(nums[i]);
}
// 2. 内置函数排序
Arrays.sort(strs, (x, y) -> (x + y).compareTo(y + x));
// 3. 返回结果
StringBuilder ans = new StringBuilder();
for(String s : strs){
ans.append(s);
}
return ans.toString();
}
}
n
为数组的长度,使用快排或内置函数的平均时间复杂度为 O ( n l o g n ) O(nlogn) O(nlogn) ,最差为 O ( n 2 ) O(n^2) O(n2)。strs
占用线性大小的额外空间。题目来源:力扣。
放弃一件事很容易,每天能坚持一件事一定很酷,一起每日一题吧!
关注我LeetCode主页 / CSDN—力扣专栏,每日更新!