Here’s a very fast way:
result = ((((12 * n + 45) * n + 50) * n + 15) * n - 2) * n // 120
How I got there:
- Rewrite the inner sum as the well-known
x*(x+1)//2
. So the whole thing becomessum(x**2 * x*(x+1)//2 for x in range(n+1))
. - Rewrite to
sum(x**4 + x**3 for x in range(n+1)) // 2
. - Look up formulas for
sum(x**4)
andsum(x**3)
. - Simplify the resulting mess to
(12*n**5 + 45*n**4 + 50*n**3 + 15*n**2 - 2*n) // 120
. - Horner it.
Another way to derive it if after steps 1. and 2. you know it’s a polynomial of degree 5:
- Compute six values with a naive implementation.
- Compute the polynomial from the six equations with six unknowns (the polynomial coefficients). I did it similarly to this, but my matrix
A
is left-right mirrored compared to that, and I called my y-vectorb
.
Code:
from fractions import Fraction
import math
from functools import reduce
def naive(n):
return sum(x**2 * sum(range(x+1)) for x in range(n+1))
def lcm(ints):
return reduce(lambda r, i: r * i // math.gcd(r, i), ints)
def polynomial(xys):
xs, ys = zip(*xys)
n = len(xs)
A = [[Fraction(x**i) for i in range(n)] for x in xs]
b = list(ys)
for _ in range(2):
for i0 in range(n):
for i in range(i0 + 1, n):
f = A[i][i0] / A[i0][i0]
for j in range(i0, n):
A[i][j] -= f * A[i0][j]
b[i] -= f * b[i0]
A = [row[::-1] for row in A[::-1]]
b.reverse()
coeffs = [b[i] / A[i][i] for i in range(n)]
denominator = lcm(c.denominator for c in coeffs)
coeffs = [int(c * denominator) for c in coeffs]
horner = str(coeffs[-1])
for c in coeffs[-2::-1]:
horner += ' * n'
if c:
horner = f"({horner} {'+' if c > 0 else '-'} {abs(c)})"
return f'{horner} // {denominator}'
print(polynomial((x, naive(x)) for x in range(6)))
Output (Try it online!):
((((12 * n + 45) * n + 50) * n + 15) * n - 2) * n // 120