华为诺亚方舟实验室-CodingTest
Huawei - CodingTest
01 - 编程题01
A为一个十进制整数,k位,k<100。求B使得B为大于A的最小整数,且A各位的和等于B各位的和。请给出函数实现,测试例,并说明解题思路。
思路:
首先,以暴力算法扫描的话时间复杂度过大,我们根据规律可以发现:
从A的倒数第二位开始,从后向前搜索,直到找到一个符合条件的位数
A[i],使其满足如下条件:A[i]不等于9并且A[i]之后的所有位不全为0
这样,我们将满足以上要求的数据,按一下的策略进行修改:
A[i]加1, 计算后面的所有位数之和记作sum, 并从最后一位向前,将位数和减去9并改为9如果剩余无法补
9则继续改为sum若为0则补0直到修改到A[i]
此外 还会有不满足以上要求的数字,则我们在开头先补上一个1,再按我们的规律进行修改;
测试样例及对拍代码:
对拍代码即暴力解 时间复杂度高 以此来验证代码的正确性;
对拍代码(matcher.cpp):
/*
* @Description:
* @Versions:
* @Author: Vernon Cui
* @Github: https://github.com/vernon97
* @Date: 2021-03-26 18:48:04
* @LastEditors: Vernon Cui
* @LastEditTime: 2021-03-22 21:23:26
* @FilePath: /Projects/matcher.cpp
*/
#include <cstdlib>
#include <ctime>
#include <iostream>
const int N = 100;
const std::string bf = "../bf.cpp";
const std::string algo = "../Example.cpp";
/*
封装一下 `system`,支持 ctrl + c 退出整个对拍程序
*/
int mySystem(const char* command) {
int result = system(command);
if (WIFEXITED(result)) {
// printf("Exited normally with status %d\n", WEXITSTATUS(result));
} else if (WIFSIGNALED(result)) {
printf("Exited with signal %d\n", WTERMSIG(result));
exit(1);
} else {
printf("Not sure how we exited.\n");
}
return result;
}
int main() {
mySystem("g++ ../random.cpp -o random.out");
mySystem(
("g++ -std=c++11 -o bf.out " + bf).c_str());
mySystem(
("g++ -std=c++11 -o algo.out " + algo).c_str());
for (int T = 1; T <= N; T++) {
// 生成随机数据
mySystem("./random.out > data.in");
// 记录运行的 CPU 时间
double st = clock();
// 暴力解法输出正确答案
mySystem("./bf.out < data.in > data.ans");
double ed = clock();
// 优化解法输出待检查答案
mySystem("./algo.out < data.in > data.out");
// return 0;
if (mySystem("diff data.out data.ans")) {
puts("\033[1;31mWrong Answer\033[0m");
puts("\033[1;31m-----错误数据-------\033[0m");
printf("输入: \n");
mySystem("cat data.in");
return -1;
} else {
printf("\033[1;32mAccepted\033[0m, 测试点 #%d, 用时 %.01f\n", T, ed - st);
}
}
return 0;
}随机测试样例生成代码(random.cpp):
暴力算法(bf.cpp):
代码(Example.cpp):
测试样例:
以下测试样例为自动生成, 分别为A和B
02 - 编程题02
已知有如下二维离散点:(3.1,1)(4,1.9)(4,4)(6,2.1)(9,7)(9.6,3)(11,2)(12.2,4)(14,2)(18,6),提取其中近似在一条直线上的离散点集群(要求该集群要包含最多的离散点,集群中点到集群对应直线的距离小于0.3)。输出该离散点集群以及对应的直线(以两点表示),给出函数实现,输出结果并说明解题思路(避免使用封装好的库算法)
本题的思路是采用RANSAC来拟合直线;
RANSAC做的一件事就是先随机的选取一对点,用这些点去计算出一个直线,然后用此模型去测试剩余的点,不断重复此过程,直到找到选取的这些数据点集达到了可以接受的程度为止,此时得到的模型便可认为是对数据点的最优模型构建。 答案:
Last updated