HDU 5297 Y sequence Y数列

本文介绍了一种高效算法来解决数论问题:给定正整数n和r,如何找出从正整数序列中删除特定幂次方数后得到的Y数列中的第n个数。通过巧妙利用容斥原理,该算法能够有效计算出答案。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

题意:给定正整数n和r,定义Y数列为从正整数序列中删除所有能表示成a^b(2 ≤ b ≤ r)的数后的数列,求Y数列的第n个数是多少。例如n = 10, r = 3,则Y数列为2 3 5 6 7 10 11 12 13 14,第10个数是14。






很有趣的一道数论题。题目给出的范围是long long范围的。所以显然不用去枚举每个数了,更不用说判断每个数是不是某个数的某次方。那么这个题怎么下手呢。首先我们可以很直观的想到,被删去的数显然是很分散的。因为能表示成某个数的幂这样的形式的数非常少。于是我们想到一个大致的思路——从n开始计算,首先计算1~n中实际上在Y数列里的数的个数t,然后答案至少要往后再加上n - t个数(被删去的数的个数),假设当前加到了m,再计算1~m中实际上在Y数列里的数的个数,假设为x个数,看x是不是等于n,如果小于n的话,让m加上n - x,再继续这样算 ...... 直到加到某个数ans时,1~ans中实际上在Y数列里的个数恰好为N,此时ans即为答案。

思路想到这里,有两个问题需要明白。第一个,会不会加到超过正确答案呢?显然不会,因为每次加上的数都只是被删去的数的个数,而新加的数有可能还有被删去的,所以肯定不会超过n。第二个,会不会效率很低呢?当然也是不会的,因为被删去的数很分散,新加进来的数里面又含有的应该被删去的数相对于当前新加进来的数来说是很少很少的,所以效率是很高的。

那么问题只剩下一个了——如何快速求1~n里面有多少个数能表示成某个数的幂的形式。

我们这样想,首先删去所有的平方数。有n^(1/2)个。再删去所有的三次方数。有n^(1/3)个 ...... 这样枚举幂数b(从2开始),然后删去n^(1/b)个数,但是有重复的,比如某个数的6次方可能被平方数删去了一次,又被3次方数删去了一次,于是想到用容斥原理加回来。这样,只需要枚举幂为质数的b的值。枚举质数b的时候,当b超过r就不再枚举。然后再用容斥求最后结果即可。当然,用pow函数是可以的,不过要注意精度。另外容斥的时候注意先不要把1算上去,最后答案减去1(因为1无论如何都是要被删的)即可。

下午打多校的时候各种写残。。。。。各种WA各种TLE然后比赛完了就过了。。。。。晕。






#include <cstdio>
#include <iostream>
#include <cstring>
#include <string>
#include <cmath>
#include <algorithm>
#include <stack>
#include <vector>
#include <map>
#include <set>
using namespace std;

//用负数方便计算容斥的符号
const int mi[19] = {-2, -3, -5, -7, -11, -13, -17, -19, -23, -29, -31, -37, -41, -43, -47, -53, -59, -61, -67};
long long n;
int r;
vector <int> rongchi; //需要用容斥计算的幂值

void get_rongchi()
{
    rongchi.clear();
    for(int i = 0; abs(mi[i]) <= r; i++)
    {
        int temp = rongchi.size();
        for(int j = 0; j < temp; j++)
        {
            if(abs(mi[i]*rongchi[j]) <= 63)
                rongchi.push_back(mi[i]*rongchi[j]);
        }
        rongchi.push_back(mi[i]);
    }
}

long long cal(long long x) //计算1~x里面实际上在Y数列里的数的个数
{
    if(x == 1)
        return 0;
    long long ans = x;
    for(int i = 0; i < rongchi.size(); i++)
    {
        long long temp = (long long)(pow(x + 0.5, 1.0/abs(rongchi[i]))) - 1; // +0.5为了保证精度,-1是暂时不计算1
        if(rongchi[i] < 0)
            ans -= temp;
        else
            ans += temp;
    }
    return ans - 1; //减去刚才没有计算的1,1是无论如何要被删的
}

void solve()
{
    get_rongchi();
    long long ans = n;
    while(1)
    {
        long long temp = cal(ans);
        if(temp == n)
            break;
        ans += n - temp; //每次加上被删去的数的个数
    }
    printf("%I64d\n", ans);
}

int main()
{
    int T;
    scanf("%d", &T);
    while(T--)
    {
        scanf("%I64d%d", &n, &r);
        solve();
    }
    return 0;
}


HDU2019的数列有序问题通常涉及到数组排序或搜索算法。这类题目一般会给出一个未排序的整数序列,然后需要检查这个序列是否能通过某种操作变得有序。常见的操作可能是交换两个元素、删除一个元素等。 例如,你可以考虑使用二分查找或者归并排序的思想。如果序列已经是升序排列,直接返回true;如果是降序排列,也需要检查能否通过一次交换将整个序列变为升序;对于其他情况,可以尝试从中间元素开始向两边遍历,看能否通过有限次的操作使序列有序。 下面是一个简单的Python示例,假设我们有一个函数`checkSorted(nums)`,它接受一个整数列表`nums`: ```python def checkSorted(nums): n = len(nums) # 如果只有一个元素或者已经有序 if n <= 1 or nums == sorted(nums): return True # 检查是否存在逆序对 for i in range(1, n): if nums[i] < nums[i - 1]: left, right = i, n - 1 while left < right: mid = (left + right) // 2 if nums[mid] > nums[i - 1]: left = mid + 1 else: right = mid # 如果找到了逆序对并且右边界小于等于左边界,说明可以通过一次交换修复 if right <= i - 1: nums[left], nums[i - 1] = nums[i - 1], nums[left] if checkSorted(nums): return True # 否则无法修复,返回false else: return False # 所有操作都尝试过了,还是有序的 return True # 测试 nums = [4, 2, 3, 1] # 这个例子应该返回True,因为可以通过一次交换变成升序 print(checkSorted(nums)) ``` 请注意,这只是一种基本思路,实际解题时可能需要根据题目给出的具体条件进行调整。如果你遇到具体的题目,请提供题目详细描述以便我能给出更精确的帮助。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值