华为诺亚方舟实验室-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