天梯赛 L1-027 出租 (20 分)
题目
下面是新浪微博上曾经很火的一张图:

一时间网上一片求救声,急问这个怎么破。其实这段代码很简单,index数组就是arr数组的下标,index[0]=2 对应 arr[2]=1,index[1]=0 对应 arr[0]=8,index[2]=3 对应 arr[3]=0,以此类推…… 很容易得到电话号码是18013820100。
本题要求你编写一个程序,为任何一个电话号码生成这段代码 —— 事实上,只要生成最前面两行就可以了,后面内容是不变的。
输入格式:
输入在一行中给出一个由11位数字组成的手机号码。
输出格式:
为输入的号码生成代码的前两行,其中arr中的数字必须按递减顺序给出。
测试样例
输入样例1:
18013820100
输出样例1:
int[] arr = new int[]{8,3,2,1,0};
int[] index = new int[]{3,0,4,3,1,0,2,4,3,4,4};
简单分析:
- 11位超过了int范围,处理最好使用字符串吧,减少麻烦,不过要记得每一位处理的时候要 - ‘0’
- 自己写了两重循环,好吧,L1做题的时候快速通过就行。
- 放一种大佬精益求精一重循环的办法。
代码:
自己写的蒟蒻两重循环
#include <iostream>
#include <algorithm>
#include <vector>
using namespace std;
bool num[10];
int main() {
string str;
cin >> str;
for (int i = 0; i < str.size(); i ++) {
if (!num[str[i] - '0']) num[str[i] - '0'] = true;
}
vector<int> arr;
for (int i = 9; i >= 0; i --) {
if (num[i]) arr.push_back(i);
}
printf("int[] arr = new int[]{");
for (int i = 0; i < arr.size(); i ++) {
if (i != 0) cout << ',';
cout << arr[i];
}
printf("};\n");
printf("int[] index = new int[]{");
for (int i = 0; i < str.size(); i ++) {
if (i != 0) cout << ',';
for (int j = 0; j < arr.size(); j ++) {
if (str[i] - '0' == arr[j]) {
cout << j;
break;
}
}
}
printf("};\n");
return 0;
}
#include<bits/stdc++.h>
using namespace std;
int arr[15],idx[15],cnt;
bool vis[15];
char a[15];
int main()
{
for(int i=0;i<11;i++)
{
cin>>a[i];
if(vis[a[i]^48])continue;
vis[a[i]^48]=1;
arr[cnt++]=a[i]^48;
}
sort(arr,arr+cnt,greater<int>());
for(int i=0;i<cnt;i++)idx[arr[i]]=i;
cout<<"int[] arr = new int[]{";
for(int i=0;i<cnt;i++)
{
if(i)cout<<',';
cout<<arr[i];
}
cout<<"};"<<endl<<"int[] index = new int[]{";
for(int i=0;i<11;i++)
{
if(i)cout<<',';
cout<<idx[a[i]^48];
}
cout<<"};"<<endl;
return 0;
}
感悟
管他是不是两重循环,是不是最优解。只要能在允许的时间复杂度内跑通,AC就行。
比赛看的是速度!加油啊。
这篇博客介绍了如何为一个11位电话号码编写生成特定格式代码的程序,重点讲解了一种更高效的一重循环解决方案,对比了传统两重循环,并分享了大佬的简洁代码实例。

被折叠的 条评论
为什么被折叠?



