• vijos


    P1302连续自然数和

    描写叙述

    对一个给定的自然数M,求出所有的连续的自然数段(连续个数大于1)。这些连续的自然数段中的所有数之和为M。

    样例:1998+1999+2000+2001+2002 = 10000,所以从1998到2002的一个自然数段为M=10000的一个解。

    格式

    输入格式

    包括一个整数的单独一行给出M的值(10 <= M <= 2,000,000)

    输出格式

    每行两个自然数,给出一个满足条件的连续自然数段中的第一个数和最后一个数,两数之间用一个空格隔开,全部输出行的第一个按从小到大的升序排列。对于给定的输入数据,保证至少有一个解。

    例子1

    例子输入1[复制]

    10000

    例子输出1[复制]

    18 142 
    297 328 
    388 412 
    1998 2002
    这道题目假设用C++能够直接枚举,非常快就能够过,并且时间,可是这样对我们学习数论知识没有一点帮助。由于数论不仅仅是简单的枚举很多其它的是公式的推导,所以我对于数论题目尽可能的使用耗时长一点的语言。来让我将代码变得更加简短,高速,比方这道题目。用一种方法python超时,可是c++46ms就能够过了,可是假设我用python将这道题目过了,用c++直接就是0ms。

    
    
    我使用了一个公式推导式针对開始的前后两个数之差进行枚举计算
    m = math.sqrt(float(2 * n) + pow(a * 0.5,2.0)) - a * 0.5
    
    if m == int(m):
            print i + 1,i + int(m)
    
    这个会超时,原因是,无论这个数符不符合条件,你都要进行这个式子的运算
    会导致这种结果,最后一个数据会超时:
    
    
    如此进行代码优化:
    对于等差数列公式得:(2a + m)(m + 1) = 2n -> 2a(m + 1) = 2n - m(m + 1) - > 2a = 2n / (k + 1) - m
    又由于a为整数所以。2n % (k + 1)不为零的直接排除,接着是(2n / (k + 1) - m) % 2不为零的能够排除
    这样非常多情况仅仅要推断一下就能够了,根本不须要进行什么计算。复杂度自然会降低非常多
    接着就是答案输出了
    这里提供pythonAC代码:
    #!/usr/bin/env python3
    # -*- coding: utf-8 -*-
    
    import math
    n = int(raw_input())
    cnt = int(math.sqrt(2 * n))
    i = cnt
    while cnt > 0:
        if not ((2 * n) % (cnt + 1)):
            m = 2 * n / (cnt + 1)
            m -= cnt
            m >>= 1
            if (2 * m + cnt) * (cnt + 1) / 2 == n and m >= 0:
                print m,m + cnt
        cnt -= 1
                
    


     
    
    

    限制

  • 相关阅读:
    如何阅读修改代码
    C C++ TDD单元测试非常好的书
    应用代理 socket TCP协议 的资料
    闲聊桌面应用开发[Win16->Win32->ATL/WTL/MFC->WinForm->WPF/Silverlight/WinRT]
    MySQL sharding的几个参考地址
    ubuntu环境变量
    Angular JS | Closure | Google Web Toolkit | Dart | Polymer 概要汇集
    Linux下的应用程序性能分析 总结
    ubuntu处理中文时设置locale
    Tomcat https自制证书和浏览器配置
  • 原文地址:https://www.cnblogs.com/wzzkaifa/p/6937039.html
Copyright © 2020-2023  润新知