zoukankan      html  css  js  c++  java
  • 等差数列偶数被除2删除后的恢复问题(2018小马智行秋招计算机视觉第三道编程题)

    本文持续更新地址:https://haoqchen.site/2018/11/08/arithmetic-progression-recovery/

    题目描述

    小P写下了一个等差数列,然后小Q将等差数列中的所有偶数都除以2(一直除以2,直到变成奇数为止)。然后小P发现等差数列被改了,现在他要还原出原来的等差数列。如果有多种可能的答案,请输出首项最小的等差数列。

    输入描述

    第一行一个整数N(4 <= N <= 50),表示等差数列的整数数量。

    之后N行,每行一个奇数A[i],依次表示小Q除以2之后得到的奇数列,保证1 <= A[i] <= 1000.

    输出描述

    N行,每行一个整数,表示小P原来写下的等差数列。

    示例

    6
    1
    1
    3
    1
    5
    3

    代码(没有提交系统,只在本地测试了样例)

    #include <iostream>
    #include <algorithm>
    
    using namespace std;
    
    typedef short int T;
    
    
    struct ARI_PRO{
    	T begin;
    	T step;
    };
    
    
    bool is_double(T a, T b)
    {
    	while (a < b){
    		a = a << 1;
    	}
    	if (a == b)
    		return true;
    	return false;
    }
    
    ARI_PRO get_ari_pro(T* A, T N)
    {
    	int i = 0;
    	int temp = 0;
    	ARI_PRO result;
    	T step = A[1] - A[0];
    	for (i = 2; i < N; ++i){//begin with odd, step by even
    		if ((A[i] - A[i - 1]) != step)
    			break;
    	}
    	if (i == N){
    		result.begin = A[0];
    		result.step = step;
    		return result;
    	}
    	temp = A[0];
    	step = (A[2] - A[0]) / 2;//begin with odd, step by odd
    	for (i = 1; i < N; ++i){
    		temp = temp + step;
    		if ((i & 1) && !is_double(A[i], temp))
    			break;
    		else if ((!(i & 1)) && A[i] != temp)
    			break;		
    	}
    	if (i == N){
    		result.begin = A[0];
    		result.step = step;
    		return result;
    	}
    	step = (A[3] - A[1]) / 2;//begin with even, step by odd
    	temp = A[1] - step;
    	for (i = 0; i < N; ++i){
    		if ((!(i & 1)) && !is_double(A[i], temp))
    			break;
    		else if ((i & 1) && A[i] != temp)
    			break;
    		temp = temp + step;
    	}
    	if (i == N){
    		result.begin = A[0];
    		result.step = step;
    		return result;
    	}
    }
    
    int main(int argc, char** argv)
    {
    	T N;
    	cin >> N;
    	if (N > 50 || N < 4)
    		return 0;
    	T* A = new T[N];
    	T temp;
    	for (int i = 0; i < N; ++i)
    		cin >> A[i];
    	ARI_PRO result = get_ari_pro(A, N);
    	temp = result.begin;
    	for (int i = 0; i < N; ++i){
    		cout << temp << endl;
    		temp += result.step;
    	}
    	delete[] A;
    	return 0;
    }

    思路

    很多人可能会想用动态规划去求解。。。这个复杂度是指数级别的~~~~

    其实仔细分析一下可以知道,实际只有以下四种情况

    startstep odd(奇数) even(偶数)
    odd(奇数) (2)第0,2个数没变,作差/2就可以求出step,然后构建正确的等差数列。对奇数数列进行遍历,如果是双数的索引(从0开始)则判断是否相等,如是奇数索引则判断奇数数列中的数一直乘以2能否等于正确的等差数列。 (1)原来就所有数都是奇数,所以整个数列并没有改变,直接判断是否正确即可
    even(偶数) (3)第1,3个数没有变,作差/2即可求出step。然后剩下步奏与(2)相似 其实偶偶的话是会退化为剩下三种的,所以不必考虑

    考虑到如果有多个结果,输出首项最小的,所以安排了(1)(2)(3)的判断顺序。理论上负责度还是n*K,K很小?

    不过只通过了测试用例,没上传到系统检验

  • 相关阅读:
    Spring AOP中文教程《转》
    VBScript 转义字符
    Spring工作原理
    Struts+Hibernate+Spring工作原理及使用理由
    SQL Server中,varchar和nvarchar如何选择?<转>
    TSQL 游标使用
    oracle 与sql server临时表 比较《转》
    Oracle查询语法大全<转>
    B/S 模式 套打 的一种实现方式<转>
    基于Web的套打方案集粹<转>
  • 原文地址:https://www.cnblogs.com/HaoQChen/p/11048598.html
Copyright © 2011-2022 走看看