mirror of
https://github.com/TheAlgorithms/Python.git
synced 2025-01-18 08:17:01 +00:00
4d0c830d2c
* ci(pre-commit): Add ``flake8-builtins`` additional dependency to ``pre-commit`` (#7104) * refactor: Fix ``flake8-builtins`` (#7104) * fix(lru_cache): Fix naming conventions in docstrings (#7104) * ci(pre-commit): Order additional dependencies alphabetically (#7104) * fix(lfu_cache): Correct function name in docstring (#7104) * Update strings/snake_case_to_camel_pascal_case.py Co-authored-by: Christian Clauss <cclauss@me.com> * Update data_structures/stacks/next_greater_element.py Co-authored-by: Christian Clauss <cclauss@me.com> * Update digital_image_processing/index_calculation.py Co-authored-by: Christian Clauss <cclauss@me.com> * Update graphs/prim.py Co-authored-by: Christian Clauss <cclauss@me.com> * Update hashes/djb2.py Co-authored-by: Christian Clauss <cclauss@me.com> * refactor: Rename `_builtin` to `builtin_` ( #7104) * fix: Rename all instances (#7104) * refactor: Update variable names (#7104) * [pre-commit.ci] auto fixes from pre-commit.com hooks for more information, see https://pre-commit.ci * ci: Create ``tox.ini`` and ignore ``A003`` (#7123) * revert: Remove function name changes (#7104) * [pre-commit.ci] auto fixes from pre-commit.com hooks for more information, see https://pre-commit.ci * Rename tox.ini to .flake8 * Update data_structures/heap/heap.py Co-authored-by: Dhruv Manilawala <dhruvmanila@gmail.com> * refactor: Rename `next_` to `next_item` (#7104) * ci(pre-commit): Add `flake8` plugin `flake8-bugbear` (#7127) * refactor: Follow `flake8-bugbear` plugin (#7127) * fix: Correct `knapsack` code (#7127) * [pre-commit.ci] auto fixes from pre-commit.com hooks for more information, see https://pre-commit.ci Co-authored-by: Christian Clauss <cclauss@me.com> Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com> Co-authored-by: Dhruv Manilawala <dhruvmanila@gmail.com>
117 lines
3.7 KiB
Python
117 lines
3.7 KiB
Python
"""
|
|
Project Euler Problem 203: https://projecteuler.net/problem=203
|
|
|
|
The binomial coefficients (n k) can be arranged in triangular form, Pascal's
|
|
triangle, like this:
|
|
1
|
|
1 1
|
|
1 2 1
|
|
1 3 3 1
|
|
1 4 6 4 1
|
|
1 5 10 10 5 1
|
|
1 6 15 20 15 6 1
|
|
1 7 21 35 35 21 7 1
|
|
.........
|
|
|
|
It can be seen that the first eight rows of Pascal's triangle contain twelve
|
|
distinct numbers: 1, 2, 3, 4, 5, 6, 7, 10, 15, 20, 21 and 35.
|
|
|
|
A positive integer n is called squarefree if no square of a prime divides n.
|
|
Of the twelve distinct numbers in the first eight rows of Pascal's triangle,
|
|
all except 4 and 20 are squarefree. The sum of the distinct squarefree numbers
|
|
in the first eight rows is 105.
|
|
|
|
Find the sum of the distinct squarefree numbers in the first 51 rows of
|
|
Pascal's triangle.
|
|
|
|
References:
|
|
- https://en.wikipedia.org/wiki/Pascal%27s_triangle
|
|
"""
|
|
from __future__ import annotations
|
|
|
|
|
|
def get_pascal_triangle_unique_coefficients(depth: int) -> set[int]:
|
|
"""
|
|
Returns the unique coefficients of a Pascal's triangle of depth "depth".
|
|
|
|
The coefficients of this triangle are symmetric. A further improvement to this
|
|
method could be to calculate the coefficients once per level. Nonetheless,
|
|
the current implementation is fast enough for the original problem.
|
|
|
|
>>> get_pascal_triangle_unique_coefficients(1)
|
|
{1}
|
|
>>> get_pascal_triangle_unique_coefficients(2)
|
|
{1}
|
|
>>> get_pascal_triangle_unique_coefficients(3)
|
|
{1, 2}
|
|
>>> get_pascal_triangle_unique_coefficients(8)
|
|
{1, 2, 3, 4, 5, 6, 7, 35, 10, 15, 20, 21}
|
|
"""
|
|
coefficients = {1}
|
|
previous_coefficients = [1]
|
|
for _ in range(2, depth + 1):
|
|
coefficients_begins_one = previous_coefficients + [0]
|
|
coefficients_ends_one = [0] + previous_coefficients
|
|
previous_coefficients = []
|
|
for x, y in zip(coefficients_begins_one, coefficients_ends_one):
|
|
coefficients.add(x + y)
|
|
previous_coefficients.append(x + y)
|
|
return coefficients
|
|
|
|
|
|
def get_squarefrees(unique_coefficients: set[int]) -> set[int]:
|
|
"""
|
|
Calculates the squarefree numbers inside unique_coefficients.
|
|
|
|
Based on the definition of a non-squarefree number, then any non-squarefree
|
|
n can be decomposed as n = p*p*r, where p is positive prime number and r
|
|
is a positive integer.
|
|
|
|
Under the previous formula, any coefficient that is lower than p*p is
|
|
squarefree as r cannot be negative. On the contrary, if any r exists such
|
|
that n = p*p*r, then the number is non-squarefree.
|
|
|
|
>>> get_squarefrees({1})
|
|
{1}
|
|
>>> get_squarefrees({1, 2})
|
|
{1, 2}
|
|
>>> get_squarefrees({1, 2, 3, 4, 5, 6, 7, 35, 10, 15, 20, 21})
|
|
{1, 2, 3, 5, 6, 7, 35, 10, 15, 21}
|
|
"""
|
|
|
|
non_squarefrees = set()
|
|
for number in unique_coefficients:
|
|
divisor = 2
|
|
copy_number = number
|
|
while divisor**2 <= copy_number:
|
|
multiplicity = 0
|
|
while copy_number % divisor == 0:
|
|
copy_number //= divisor
|
|
multiplicity += 1
|
|
if multiplicity >= 2:
|
|
non_squarefrees.add(number)
|
|
break
|
|
divisor += 1
|
|
|
|
return unique_coefficients.difference(non_squarefrees)
|
|
|
|
|
|
def solution(n: int = 51) -> int:
|
|
"""
|
|
Returns the sum of squarefrees for a given Pascal's Triangle of depth n.
|
|
|
|
>>> solution(1)
|
|
1
|
|
>>> solution(8)
|
|
105
|
|
>>> solution(9)
|
|
175
|
|
"""
|
|
unique_coefficients = get_pascal_triangle_unique_coefficients(n)
|
|
squarefrees = get_squarefrees(unique_coefficients)
|
|
return sum(squarefrees)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
print(f"{solution() = }")
|