Un modo semplice sarebbe quella heapq.merge:
A = [10, 20, 30, 40]
B = (20, 60, 81, 90)
from heapq import merge
for ele in merge(A,B):
print(ele)
uscita:
10
20
20
30
40
60
81
90
Alcuni timing utilizzando l'altra O(n)
soluzione:
In [53]: A = list(range(10000))
In [54]: B = list(range(1,20000,10))
In [55]: timeit list(merge(A,B))
100 loops, best of 3: 2.52 ms per loop
In [56]: %%timeit
C = []
i = j = 0
while i < len(A) and j < len(B):
if A[i] < B[j]:
C.append(A[i])
i += 1
else:
C.append(B[j])
j += 1
C += A[i:] + B[j:]
....:
100 loops, best of 3: 4.29 ms per loop
In [58]: m =list(merge(A,B))
In [59]: m == C
Out[59]: True
Se si voleva a rotolare il proprio questo è un po 'più veloce di unione:
def merger_try(a, b):
if not a or not b:
yield chain(a, b)
iter_a, iter_b = iter(a), iter(b)
prev_a, prev_b = next(iter_a), next(iter_b)
while True:
if prev_a >= prev_b:
yield prev_b
try:
prev_b = next(iter_b)
except StopIteration:
yield prev_a
break
else:
yield prev_a
try:
prev_a = next(iter_a)
except StopIteration:
yield prev_b
break
for ele in chain(iter_b, iter_a):
yield ele
Alcune timing:
In [128]: timeit list(merge(A,B))
1 loops, best of 3: 771 ms per loop
In [129]: timeit list(merger_try(A,B))
1 loops, best of 3: 581 ms per loop
In [130]: list(merger_try(A,B)) == list(merge(A,B))
Out[130]: True
In [131]: %%timeit
C = []
i = j = 0
while i < len(A) and j < len(B):
if A[i] < B[j]:
C.append(A[i])
i += 1
else:
C.append(B[j])
j += 1
C += A[i:] + B[j:]
.....:
1 loops, best of 3: 919 ms per loop
vedere l'algoritmo di unione esterna utilizzato in mergesort. Ci vuole tempo 'O (m + n)'. – Haris
Ti suggerisco di confrontare alla versione ingenua (ma leggibile) di 'ordinati (A + B)'. Inoltre, potresti voler cercare le librerie esistenti, che probabilmente saranno più veloci dell'implementazione fatta a mano: http://www.grantjenks.com/docs/sortedcontainers/ – DainDwarf
Suggerimento: non modificare nessuna lista; creane uno nuovo. Il codice diventerà pulito e veloce. – 9000