codility 11-2 CountSemiprimes

最近看到挺有意思的一题,原题链接:https://app.codility.com/programmers/lessons/11-sieve_of_eratosthenes/count_semiprimes/

题目描述
质数是一个正整数X,它有两个不同的除数:1和X。最初的几个质数是2、3、5、7、11和13。
半质数(semiprime)是一个自然数,它是两个质数的乘积(不一定是不同的)。前几个半素数是4,6,9,10,14,15,21,22,25,26。

给定两个非空数组 P 和 Q,每个数组由 M 个整数组成。这些数组表示有关指定范围内半素数个数的查询,返回 [ P[k], Q[k] ] 左闭右闭区间半质数的个数,其中1≤ P [ K ] ≤ Q [ K ]≤ N。

例如:P = [1, 4, 16], Q = [26, 10, 20] , 返回 [10, 4, 0]

P[0], Q[0] -->  [1, 26] 中间半质数有4,6,9,10,14,15,21,22,25,26, 共 10 个
P[1], Q[1] -->  [4, 10] 中间半质数有4,6,9,10, 共 4 个
P[2], Q[2] -->  [16, 20] 中间半质数共 0 个

思路与实现
由于1 ≤ P[K] ≤ Q[K] ≤ N,所以先存 0 到 N 每个数到当前数的半质数个数: dict = {数:到当前数的半质数个数},这样对于每个 P[k] 到 Q[k] 区间,只需要计算dict[Q[k]] - dict[P[k]],当 P[k] 为半质数时需要 +1。

如何求 0 到 N 每个数到当前数 x 的半质数个数,可以简单描述为 dict[x] = dict[x-1]+1 if x是半质数 else dict[x-1]

然后问题来到如何求一个数 x 是否是半质数:将 x 拆成两个数的乘积,再判断这两个数是否都为质数。又因为将 x 拆成两个数的乘积,一定有一个数小于根号 x,所以只需要遍历到根号 x 来降低复杂度。

def solution(N, P, Q):
    res = []

    # 寻找N内半质数
    semiprime_dict = {}
    prime_nums = set()  # 存质数,不用重复判断,节省空间也可以不要
    for i in range(2, int(N**0.5)+1):
        # 不是质数
        if i not in prime_nums and not is_prime(i):
            continue
        prime_nums.add(i)
        for j in range(i, int(N/i)+1):
            if j not in prime_nums and not is_prime(j):
                continue
            # 如果i, j 都是质数
            prime_nums.add(j)
            # 半质数加入semiprime_dict中
            semiprime_dict[i*j] = 0

    # semiprime_dict={数:到当前数的半质数个数} 当前只有半质数,将 semiprime_dict 扩展到所有数
    # (实现的有点复杂
    semiprimes = sorted(semiprime_dict)  # 排序后的键值
    count = 0
    for x in semiprimes:   # 对于每个半质数,到当前的半质数个数
        count += 1 
        semiprime_dict[x] = count
    semiprime_dict[0] = 0  # 1,2,3都应该是0,加入0是解决P[i]=1的情况
    for x in range(1, N+1):		# 非半质数,到当前的半质数个数与前一个相同
        if x not in semiprime_dict:
            semiprime_dict[x] = semiprime_dict[x-1]
    
    # 找 P[i]-Q[i]之间semiprime_list的元素个数 O(M)
    for i in range(len(P)):
        count = semiprime_dict[Q[i]] - semiprime_dict[P[i]]
        if semiprime_dict[P[i]] > semiprime_dict[P[i]-1]:
            count += 1
        res.append(count)

    return res

# 判断质数
def is_prime(x):
    # 偶数一定不是质数
    if x != 2 and x % 2 == 0:
        return 0 
    # x 拆成两个数的乘积一定有一个小于根号x
    for i in range(2, int(x**0.5)+1):
        if x % i == 0:
            return 0

    return 1
上一篇:构造n以内的素数表


下一篇:wordcount案例