3090 - divizori5

From Bitnami MediaWiki

Fie D, K și P trei numere naturale.

Cerința

Să se determine numărul de numere naturale, notat cu T, având următoarele proprietăți:

  • au exact D divizori;
  • descompunerea în factori primi a acestor numere conține exact K numere prime;
  • toți factorii primi din descompunerea numerelor sunt mici sau egali cu P.

Date de intrare

Fișierul de intrare divizori.in conține pe primul rând numerele D, K și P cu semnificația de mai sus, despărțite prin câte un spațiu.

Date de iesire

Fișierul de ieșire divizori.out va conține pe primul rând restul împărțirii numărului T la 3000017

Restricții și precizări

  • 2 ≤ D ≤ 1014
  • 1 ≤ K ≤ 100
  • 2 ≤ P ≤ 1.000.000

Exemplul 1:

divizori.in

6 2 5

divizori.out

6

Explicație

D = 6, K = 2, P = 5. Sunt T = 6 numere cu exact 6 divizori ce conțin în descompunerea în factori primi exact două numere prime mai mici sau egale decât 5: 2132, 2152, 2231, 2251, 3152, 3251.

Exemplul 2:

divizori.in

18 3 10

divizori.out

12

Exemplul 3:

divizori.in

10 8 17

divizori.out

0

Explicație

D = 10, K = 8, P = 17. Nu există numere cu exact 10 divizori ce conțin în descompunerea în factori primi exact 8 numere prime mai mici sau egale cu 17 deoarece sunt doar 7 numere prime mai mici sau egale decât 17.

Rezolvare

<syntaxhighlight lang="python3"> MOD = 3000017

def count_numbers_with_properties(d, k, p):

   dp = [[0 for _ in range(k + 1)] for _ in range(d + 1)]
   for i in range(k + 1):
       for j in range(1, d + 1):
           if j == 1:
               dp[j][i] = 1
   for i in range(2, d + 1):
       for j in range(1, k + 1):
           dp[i][j] = 0
           for num_divisors in range(1, i):
               if j >= 2:
                   dp[i][j] += dp[num_divisors][j - 1] * dp[i - num_divisors][j]
               else:
                   # If j = 1, all prime factors must be distinct
                   if num_divisors == 1:
                       dp[i][j] += dp[num_divisors][j]
   total_numbers = 0
   for i in range(1, d + 1):
       for j in range(1, k + 1):
           # Limit prime factors to 'p'
           if max_prime_factor_up_to_p(dp[i][j], p) == dp[i][j]:
               total_numbers += dp[i][j]
   return total_numbers % MOD

def max_prime_factor_up_to_p(n, p):

   max_prime_factor = 1
   while n % 2 == 0:
       n //= 2
       max_prime_factor = 2
   for i in range(3, int(p ** 0.5) + 1, 2):
       while n % i == 0:
           n //= i
           max_prime_factor = i
   if n > 1:
       max_prime_factor = n
   return max_prime_factor

def main():

   with open("divizori.in", "r") as input_file:
       d, k, p = map(int, input_file.readline().split())
   number_count = count_numbers_with_properties(d, k, p)
   with open("divizori.out", "w") as output_file:
       output_file.write(str(number_count))

if __name__ == "__main__":

   main()

</syntaxhighlight>