Здавалося б, ця проблема еквівалентна цілочисленному / многочленному квадратуванню:
1. Відомо, що множення полінома еквівалентно цілому множенню .
2. Очевидно, ви вже звели цю проблему до поліноміального / цілого квадратування; отже, ця проблема є не менш важкою, як квадрати.
Тепер я зменшу цілісні квадрати до цієї проблеми:
Припустимо, у вас був алгоритм:
F(a⃗ )→P2(x),where P(x)=∑ai∈a⃗ xai
Цей алгоритм по суті є алгоритмом, про який ви запитуєте у своєму запитанні. Таким чином, якщо у мене був магічний алгоритм, який може це зробити, я можу зробити функцію, яка буде квадратним цілим числом y ( о так, я люблю mathjax: P ):SQUARE(y)y
1.:2.:3.:4.:5.:6.:7.:8.:9.:10.:11.:12.:13.:Algorithm 1 Squaringprocedure SQUARE(y):a⃗ ←()i←0while y≠0 doif y & 1 thena⃗ ←a⃗ iend ifi←i+1y←y≫1end whileP2(x)←F(a⃗ )return P2(2)end procedure▹ a⃗ starts as empty polynomial sequence▹ break y down into a polynomial of base 2▹ if lsb of y is set▹ append i to a⃗ (appending xi)▹ shift y right by one▹ obtain the squared polynomial via F(a⃗ )▹ simply sum up the polynomial
Python ( тест з кодовою панеллю ):
#/cs//q/11418/2755
def F(a):
n = len(a)
for i in range(n):
assert a[i] >= 0
# (r) => coefficient
# coefficient \cdot x^{r}
S = {}
for ai in a:
for aj in a:
r = ai + aj
if r not in S:
S[r] = 0
S[r] += 1
return list(S.items())
def SQUARE(x):
x = int(x)
a = []
i = 0
while x != 0:
if x & 1 == 1:
a += [i]
x >>= 1
i += 1
print 'a:',a
P2 = F(a)
print 'P^2:',P2
s = 0
for e,c in P2:
s += (1 << e)*c
return s
3. Таким чином, квадратик є не менш важким, як ця проблема.
4. Тому ціле квадратування рівнозначно цій задачі. (кожен з них максимум такий же важкий, як один з одним, через ( 2 , 3 , 1 ))
O(nlogn)O(nlognloglogn)O(nlogn2O(log∗n))Ω(nlogn)
O(nlogn)
5. Тепер ваша проблема не є саме множенням, це квадратом. Тож квадратування простіше? Що ж, це відкрита проблема (поки що немає) : невідомо, що квадрати мають швидший алгоритм, ніж множення. Якщо ви могли знайти кращий алгоритм для своєї проблеми, ніж використання множення; то це, ймовірно, буде проривом.
O(nlogn)O(nlogn)O(nlogn)O(nlogn) або як найкращий алгоритм множення лише наближається до цієї складності.