2022-09-16 03:40:00 +08:00
|
|
|
from copy import copy, deepcopy
|
|
|
|
|
2021-09-28 02:02:44 +08:00
|
|
|
@tuple
|
|
|
|
class A:
|
|
|
|
a: int
|
|
|
|
b: float
|
|
|
|
|
|
|
|
def __getitem__(self: A, n: int):
|
|
|
|
return 1
|
|
|
|
|
|
|
|
def __getitem__(self: A, x: Slice):
|
|
|
|
if x.start is None and x.stop is None:
|
|
|
|
return -1
|
|
|
|
if x.start is None:
|
|
|
|
return 2
|
|
|
|
elif x.stop is None:
|
|
|
|
return 3
|
|
|
|
else:
|
|
|
|
return self.a
|
|
|
|
|
2023-05-10 21:28:25 +08:00
|
|
|
@dataclass(init=True, order=True, eq=True)
|
|
|
|
class A0:
|
|
|
|
pass
|
|
|
|
|
|
|
|
@dataclass(init=True, order=True, eq=True)
|
|
|
|
class A1:
|
|
|
|
a: int
|
|
|
|
|
|
|
|
@dataclass(init=True, order=True, eq=True)
|
|
|
|
class A2:
|
|
|
|
a: int
|
|
|
|
b: int
|
|
|
|
|
|
|
|
@dataclass(init=True, order=True, eq=True)
|
|
|
|
class A3:
|
|
|
|
a: int
|
|
|
|
b: int
|
|
|
|
c: int
|
|
|
|
|
2021-09-28 02:02:44 +08:00
|
|
|
@test
|
|
|
|
def test_tuple():
|
|
|
|
def test_in():
|
|
|
|
for i in range(10):
|
|
|
|
yield i, i in (4, 9, 10, -1, 3, 1), i in (7,)
|
|
|
|
assert list(test_in()) == [(0, False, False), (1, True, False), (2, False, False), (3, True, False), (4, True, False), (5, False, False), (6, False, False), (7, False, True), (8, False, False), (9, True, False)]
|
|
|
|
|
|
|
|
def test_cmp[T](a: T, b: T):
|
|
|
|
yield 'EQ', a == b
|
|
|
|
yield 'NE', a != b
|
|
|
|
yield 'LT', a < b
|
|
|
|
yield 'GT', a > b
|
|
|
|
yield 'LE', a <= b
|
|
|
|
yield 'GE', a >= b
|
|
|
|
|
2023-05-10 21:28:25 +08:00
|
|
|
assert list(test_cmp((1,), (2,))) == [('EQ', False), ('NE', True), ('LT', True), ('GT', False), ('LE', True), ('GE', False)]
|
|
|
|
assert list(test_cmp((1,), (1,))) == [('EQ', True), ('NE', False), ('LT', False), ('GT', False), ('LE', True), ('GE', True)]
|
|
|
|
assert list(test_cmp((2,), (1,))) == [('EQ', False), ('NE', True), ('LT', False), ('GT', True), ('LE', False), ('GE', True)]
|
2021-09-28 02:02:44 +08:00
|
|
|
assert list(test_cmp((1,2), (1,2))) == [('EQ', True), ('NE', False), ('LT', False), ('GT', False), ('LE', True), ('GE', True)]
|
|
|
|
assert list(test_cmp((1,2,2), (1,2,3))) == [('EQ', False), ('NE', True), ('LT', True), ('GT', False), ('LE', True), ('GE', False)]
|
|
|
|
assert list(test_cmp((1,2,-1), (1,0,1))) == [('EQ', False), ('NE', True), ('LT', False), ('GT', True), ('LE', False), ('GE', True)]
|
|
|
|
assert list(test_cmp((), ())) == [('EQ', True), ('NE', False), ('LT', False), ('GT', False), ('LE', True), ('GE', True)]
|
2023-05-10 21:28:25 +08:00
|
|
|
assert list(test_cmp(A0(), A0())) == [('EQ', True), ('NE', False), ('LT', False), ('GT', False), ('LE', True), ('GE', True)]
|
|
|
|
assert list(test_cmp(A1(1), A1(2))) == [('EQ', False), ('NE', True), ('LT', True), ('GT', False), ('LE', True), ('GE', False)]
|
|
|
|
assert list(test_cmp(A1(1), A1(1))) == [('EQ', True), ('NE', False), ('LT', False), ('GT', False), ('LE', True), ('GE', True)]
|
|
|
|
assert list(test_cmp(A1(2), A1(1))) == [('EQ', False), ('NE', True), ('LT', False), ('GT', True), ('LE', False), ('GE', True)]
|
|
|
|
assert list(test_cmp(A2(1,2), A2(1,2))) == [('EQ', True), ('NE', False), ('LT', False), ('GT', False), ('LE', True), ('GE', True)]
|
|
|
|
assert list(test_cmp(A3(1,2,2), A3(1,2,3))) == [('EQ', False), ('NE', True), ('LT', True), ('GT', False), ('LE', True), ('GE', False)]
|
|
|
|
assert list(test_cmp(A3(1,2,-1), A3(1,0,1))) == [('EQ', False), ('NE', True), ('LT', False), ('GT', True), ('LE', False), ('GE', True)]
|
2021-09-28 02:02:44 +08:00
|
|
|
|
|
|
|
t = (1,2,3)
|
|
|
|
assert (t[0], t[1], t[2]) == (1, 2, 3)
|
|
|
|
assert (t[-1], t[-2], t[-3]) == (3, 2, 1)
|
|
|
|
assert t[1:3] == (2, 3)
|
|
|
|
assert t[-3:1] == (1,)
|
|
|
|
assert t[-10:2] == (1, 2)
|
|
|
|
assert t[0:] == (1, 2, 3)
|
|
|
|
assert t[-2:] == (2, 3)
|
|
|
|
assert t[3:] == ()
|
|
|
|
assert t[:-1] == (1, 2)
|
|
|
|
assert t[:1] == (1,)
|
|
|
|
assert t[:] == (1, 2, 3)
|
2022-10-14 21:31:10 +08:00
|
|
|
assert t[::] == (1, 2, 3)
|
2021-09-28 02:02:44 +08:00
|
|
|
assert t[1::1] == (2, 3)
|
|
|
|
assert t[:2:1] == (1, 2)
|
|
|
|
assert t[::2] == (1, 3)
|
|
|
|
assert t[::-1] == (3, 2, 1)
|
|
|
|
assert t[0:3:-1] == ()
|
2022-10-14 21:31:10 +08:00
|
|
|
assert t[3:0:-1] == (3, 2)
|
2021-09-28 02:02:44 +08:00
|
|
|
|
|
|
|
a = A(42, 3.14)
|
|
|
|
assert a[0] == 1
|
|
|
|
assert a[:1] == 2
|
|
|
|
assert a[0:] == 3
|
|
|
|
assert a[0:1] == 42
|
|
|
|
assert a[:] == -1
|
2024-01-03 02:35:55 +08:00
|
|
|
|
|
|
|
assert str((11, 2, 333)) == '(11, 2, 333)'
|
|
|
|
assert str(()) == '()'
|
|
|
|
assert str((42,)) == '(42,)'
|
|
|
|
assert repr((11, 2, 333)) == '(11, 2, 333)'
|
|
|
|
assert repr(()) == '()'
|
|
|
|
assert repr((42,)) == '(42,)'
|
2021-09-28 02:02:44 +08:00
|
|
|
test_tuple()
|
|
|
|
|
2022-10-14 21:31:10 +08:00
|
|
|
@test
|
|
|
|
def test_dyn_tuple():
|
|
|
|
def D(*args):
|
|
|
|
return DynamicTuple(args)
|
|
|
|
|
|
|
|
def test_in():
|
|
|
|
for i in range(10):
|
|
|
|
yield i, i in D(4, 9, 10, -1, 3, 1), i in D(7)
|
|
|
|
assert list(test_in()) == [(0, False, False), (1, True, False), (2, False, False), (3, True, False), (4, True, False), (5, False, False), (6, False, False), (7, False, True), (8, False, False), (9, True, False)]
|
|
|
|
|
|
|
|
def test_cmp(a, b):
|
|
|
|
yield 'EQ', a == b
|
|
|
|
yield 'NE', a != b
|
|
|
|
yield 'LT', a < b
|
|
|
|
yield 'GT', a > b
|
|
|
|
yield 'LE', a <= b
|
|
|
|
yield 'GE', a >= b
|
|
|
|
|
|
|
|
assert list(test_cmp(D(1,2), D(1,2))) == [('EQ', True), ('NE', False), ('LT', False), ('GT', False), ('LE', True), ('GE', True)]
|
|
|
|
assert list(test_cmp(D(1,2,2), D(1,2,3))) == [('EQ', False), ('NE', True), ('LT', True), ('GT', False), ('LE', True), ('GE', False)]
|
|
|
|
assert list(test_cmp(D(1,2,-1), D(1,0,1))) == [('EQ', False), ('NE', True), ('LT', False), ('GT', True), ('LE', False), ('GE', True)]
|
|
|
|
assert list(test_cmp(DynamicTuple[int](), DynamicTuple[int]())) == [('EQ', True), ('NE', False), ('LT', False), ('GT', False), ('LE', True), ('GE', True)]
|
|
|
|
|
|
|
|
assert list(test_cmp((1,2), D(1,2))) == [('EQ', True), ('NE', False), ('LT', False), ('GT', False), ('LE', True), ('GE', True)]
|
|
|
|
assert list(test_cmp((1,2,2), D(1,2,3))) == [('EQ', False), ('NE', True), ('LT', True), ('GT', False), ('LE', True), ('GE', False)]
|
|
|
|
assert list(test_cmp((1,2,-1), D(1,0,1))) == [('EQ', False), ('NE', True), ('LT', False), ('GT', True), ('LE', False), ('GE', True)]
|
|
|
|
assert list(test_cmp((), DynamicTuple[int]())) == [('EQ', True), ('NE', False), ('LT', False), ('GT', False), ('LE', True), ('GE', True)]
|
|
|
|
|
|
|
|
assert list(test_cmp(D(1,2), (1,2))) == [('EQ', True), ('NE', False), ('LT', False), ('GT', False), ('LE', True), ('GE', True)]
|
|
|
|
assert list(test_cmp(D(1,2,2), (1,2,3))) == [('EQ', False), ('NE', True), ('LT', True), ('GT', False), ('LE', True), ('GE', False)]
|
|
|
|
assert list(test_cmp(D(1,2,-1), (1,0,1))) == [('EQ', False), ('NE', True), ('LT', False), ('GT', True), ('LE', False), ('GE', True)]
|
|
|
|
assert list(test_cmp(DynamicTuple[int](), ())) == [('EQ', True), ('NE', False), ('LT', False), ('GT', False), ('LE', True), ('GE', True)]
|
|
|
|
|
|
|
|
t = D(1,2,3)
|
|
|
|
assert D(t[0], t[1], t[2]) == D(1, 2, 3)
|
|
|
|
assert D(t[-1], t[-2], t[-3]) == D(3, 2, 1)
|
|
|
|
assert t[1:3] == D(2, 3)
|
|
|
|
assert t[-3:1] == D(1,)
|
|
|
|
assert t[-10:2] == D(1, 2)
|
|
|
|
assert t[0:] == D(1, 2, 3)
|
|
|
|
assert t[-2:] == D(2, 3)
|
|
|
|
assert t[3:] == DynamicTuple[int]()
|
|
|
|
assert t[:-1] == D(1, 2)
|
|
|
|
assert t[:1] == D(1)
|
|
|
|
assert t[:] == D(1, 2, 3)
|
|
|
|
assert t[::] == D(1, 2, 3)
|
|
|
|
assert t[1::1] == D(2, 3)
|
|
|
|
assert t[:2:1] == D(1, 2)
|
|
|
|
assert t[::2] == D(1, 3)
|
|
|
|
assert t[::-1] == D(3, 2, 1)
|
|
|
|
assert t[0:3:-1] == DynamicTuple[int]()
|
|
|
|
assert t[3:0:-1] == D(3, 2)
|
|
|
|
|
|
|
|
assert D(t[0], t[1], t[2]) == (1, 2, 3)
|
|
|
|
assert D(t[-1], t[-2], t[-3]) == (3, 2, 1)
|
|
|
|
assert t[1:3] == (2, 3)
|
|
|
|
assert t[-3:1] == (1,)
|
|
|
|
assert t[-10:2] == (1, 2)
|
|
|
|
assert t[0:] == (1, 2, 3)
|
|
|
|
assert t[-2:] == (2, 3)
|
|
|
|
assert t[3:] == ()
|
|
|
|
assert t[:-1] == (1, 2)
|
|
|
|
assert t[:1] == (1,)
|
|
|
|
assert t[:] == (1, 2, 3)
|
|
|
|
assert t[::] == (1, 2, 3)
|
|
|
|
assert t[1::1] == (2, 3)
|
|
|
|
assert t[:2:1] == (1, 2)
|
|
|
|
assert t[::2] == (1, 3)
|
|
|
|
assert t[::-1] == (3, 2, 1)
|
|
|
|
assert t[0:3:-1] == DynamicTuple[int]()
|
|
|
|
assert t[3:0:-1] == (3, 2)
|
|
|
|
|
|
|
|
t = (1,2,3)
|
|
|
|
assert (t[0], t[1], t[2]) == D(1, 2, 3)
|
|
|
|
assert (t[-1], t[-2], t[-3]) == D(3, 2, 1)
|
|
|
|
assert t[1:3] == D(2, 3)
|
|
|
|
assert t[-3:1] == D(1,)
|
|
|
|
assert t[-10:2] == D(1, 2)
|
|
|
|
assert t[0:] == D(1, 2, 3)
|
|
|
|
assert t[-2:] == D(2, 3)
|
|
|
|
assert t[3:] == DynamicTuple[int]()
|
|
|
|
assert t[:-1] == D(1, 2)
|
|
|
|
assert t[:1] == D(1)
|
|
|
|
assert t[:] == D(1, 2, 3)
|
|
|
|
assert t[::] == D(1, 2, 3)
|
|
|
|
assert t[1::1] == D(2, 3)
|
|
|
|
assert t[:2:1] == D(1, 2)
|
|
|
|
assert t[::2] == D(1, 3)
|
|
|
|
assert t[::-1] == D(3, 2, 1)
|
|
|
|
assert t[0:3:-1] == DynamicTuple[int]()
|
|
|
|
assert t[3:0:-1] == D(3, 2)
|
|
|
|
|
|
|
|
assert hash(D(1,2,3,4,5)) == hash((1,2,3,4,5))
|
2023-04-13 06:13:54 +08:00
|
|
|
|
|
|
|
assert (1, 2) + (3,) == (1, 2, 3)
|
|
|
|
assert (1,) + (2, 3) == (1, 2, 3)
|
|
|
|
assert (1, 2) + () == (1, 2)
|
|
|
|
assert () + () == ()
|
|
|
|
assert () + (1, 2) == (1, 2)
|
|
|
|
assert (1,) + (2,) == (1, 2)
|
|
|
|
assert (1, 2) * 3 == (1, 2, 1, 2, 1, 2)
|
|
|
|
assert () * 99 == ()
|
|
|
|
assert (1, 2, 3, 4) * 1 == (1, 2, 3, 4)
|
|
|
|
assert (1, 2) * 0 == ()
|
|
|
|
assert (1, 2) * (-1) == ()
|
|
|
|
assert () * -1 == ()
|
2022-10-14 21:31:10 +08:00
|
|
|
test_dyn_tuple()
|
|
|
|
|
2021-09-28 02:02:44 +08:00
|
|
|
@test
|
|
|
|
def test_list():
|
|
|
|
l1 = [i+1 for i in range(100)]
|
|
|
|
assert len(l1) == 100
|
|
|
|
l1 = l1[98:]
|
|
|
|
l2 = [1, 2] * 2
|
|
|
|
|
|
|
|
assert [a for a in l1] == [99, 100]
|
|
|
|
assert [a for a in l2] == [1, 2, 1, 2]
|
2022-12-31 12:04:29 +08:00
|
|
|
assert 2 * [1, 2] == l2
|
2021-09-28 02:02:44 +08:00
|
|
|
|
|
|
|
l1 = [i*2 for i in range(3)]
|
|
|
|
l1.insert(0, 99)
|
|
|
|
l1[0] += 1
|
|
|
|
del l1[1]
|
|
|
|
assert [a for a in l1[0:3]] == [100, 2, 4]
|
|
|
|
|
|
|
|
l3 = [1, 2, 3]
|
|
|
|
assert l3.remove(2) == True
|
|
|
|
assert l3.remove(2) == False
|
|
|
|
assert l3 == [1, 3]
|
|
|
|
assert list[int]().remove(0) == False
|
|
|
|
|
|
|
|
# l4 = [5, 1, 4, 2, 1, -10, 10, 100, -100]
|
|
|
|
# assert sorted(l4) == [-100, -10, 1, 1, 2, 4, 5, 10, 100]
|
|
|
|
#l4.sort()
|
|
|
|
#assert l4 == [-100, -10, 1, 1, 2, 4, 5, 10, 100]
|
|
|
|
#assert str(sorted(list[int]())) == "[]"
|
|
|
|
|
|
|
|
l5 = [11, 22, 33, 44]
|
|
|
|
del l5[-1]
|
|
|
|
assert l5 == [11, 22, 33]
|
|
|
|
l5.insert(-1, 55)
|
|
|
|
l5.insert(1000, 66)
|
|
|
|
l5.insert(-100, 77)
|
|
|
|
assert l5 == [77, 11, 22, 55, 33, 66]
|
|
|
|
l5 = [11, 22, 55, 33]
|
|
|
|
assert l5 + [1,2,3] == [11, 22, 55, 33, 1, 2, 3]
|
|
|
|
l5 += [1,2,3]
|
|
|
|
assert l5 == [11, 22, 55, 33, 1, 2, 3]
|
|
|
|
assert l5.pop() == 3
|
|
|
|
assert l5 * 2 == [11, 22, 55, 33, 1, 2, 11, 22, 55, 33, 1, 2]
|
|
|
|
l5 *= 2
|
|
|
|
assert l5 == [11, 22, 55, 33, 1, 2, 11, 22, 55, 33, 1, 2]
|
|
|
|
assert l5.index(33) == 3
|
2024-01-20 00:22:20 +08:00
|
|
|
try:
|
|
|
|
l5.index(0)
|
|
|
|
assert False
|
|
|
|
except ValueError as e:
|
|
|
|
assert str(e) == '0 is not in list'
|
2021-09-28 02:02:44 +08:00
|
|
|
l5 *= 0
|
|
|
|
assert len(l5) == 0
|
|
|
|
|
|
|
|
l6 = []
|
|
|
|
l6.extend('abc')
|
|
|
|
l6.extend(['xyz'])
|
|
|
|
l6.extend('')
|
|
|
|
assert l6 == ['a', 'b', 'c', 'xyz']
|
2021-10-14 04:40:14 +08:00
|
|
|
|
|
|
|
assert List[int]().copy() == List[int]()
|
|
|
|
assert [1,2,3].copy() == [1,2,3]
|
2022-09-16 03:40:00 +08:00
|
|
|
|
|
|
|
def test_cmp[T](a: T, b: T):
|
|
|
|
yield 'EQ', a == b
|
|
|
|
yield 'NE', a != b
|
|
|
|
yield 'LT', a < b
|
|
|
|
yield 'GT', a > b
|
|
|
|
yield 'LE', a <= b
|
|
|
|
yield 'GE', a >= b
|
|
|
|
|
|
|
|
assert list(test_cmp([1,2], [1,2])) == [('EQ', True), ('NE', False), ('LT', False), ('GT', False), ('LE', True), ('GE', True)]
|
|
|
|
assert list(test_cmp([1,2,2], [1,2,3])) == [('EQ', False), ('NE', True), ('LT', True), ('GT', False), ('LE', True), ('GE', False)]
|
|
|
|
assert list(test_cmp([1,2,-1], [1,0,1])) == [('EQ', False), ('NE', True), ('LT', False), ('GT', True), ('LE', False), ('GE', True)]
|
|
|
|
assert list(test_cmp(List[int](), List[int]())) == [('EQ', True), ('NE', False), ('LT', False), ('GT', False), ('LE', True), ('GE', True)]
|
|
|
|
assert list(test_cmp([1], List[int]())) == [('EQ', False), ('NE', True), ('LT', False), ('GT', True), ('LE', False), ('GE', True)]
|
|
|
|
assert list(test_cmp(List[int](), [1])) == [('EQ', False), ('NE', True), ('LT', True), ('GT', False), ('LE', True), ('GE', False)]
|
|
|
|
assert list(test_cmp([1,2,-1], [2])) == [('EQ', False), ('NE', True), ('LT', True), ('GT', False), ('LE', True), ('GE', False)]
|
|
|
|
assert list(test_cmp([1,2,-1], [1,2,-1,3])) == [('EQ', False), ('NE', True), ('LT', True), ('GT', False), ('LE', True), ('GE', False)]
|
|
|
|
assert list(test_cmp([1,2,-1,3], [1,2,-1])) == [('EQ', False), ('NE', True), ('LT', False), ('GT', True), ('LE', False), ('GE', True)]
|
2022-09-19 04:43:42 +08:00
|
|
|
|
|
|
|
assert str([1] + [2] + [] + [3]) == '[1, 2, 3]'
|
|
|
|
assert ['a', 'b'] + ['x', 'y', 'z'] == ['a', 'b'] + ['x', 'y', 'z']
|
|
|
|
assert [(1,1), (2,2)] + [] == [(1,1), (2,2)]
|
|
|
|
assert [] + [(1,1), (2,2)] == [(1,1), (2,2)]
|
|
|
|
assert List[int]() + List[int]() == List[int]()
|
|
|
|
l7 = [3.14, 2.5]
|
|
|
|
l7 += [9.99, -1.0]
|
|
|
|
assert l7 == [3.14, 2.5, 9.99, -1.0]
|
|
|
|
l8 = []
|
|
|
|
l8 += [11, 22, 33]
|
|
|
|
assert l8 == [11, 22, 33]
|
|
|
|
l8 = [11, 22, 33]
|
|
|
|
l8 += []
|
|
|
|
assert l8 == [11, 22, 33]
|
|
|
|
l8 = List[int]()
|
|
|
|
l8 += List[int]()
|
|
|
|
assert l8 == List[int]()
|
2022-12-05 08:45:21 +08:00
|
|
|
|
|
|
|
assert list(reversed(list('abc'))) == ['c', 'b', 'a']
|
|
|
|
assert list(list('abc')[::-1]) == ['c', 'b', 'a']
|
|
|
|
assert list(reversed(List[str]())) == List[str]()
|
2023-07-12 22:19:24 +08:00
|
|
|
|
|
|
|
# https://github.com/exaloop/codon/issues/402
|
|
|
|
b1 = [1 for _ in range(0)]
|
|
|
|
b2 = [str(x) for x in b1]
|
|
|
|
b3 = List[float](capacity=0)
|
|
|
|
b4 = List[float](capacity=-1)
|
|
|
|
assert len(b1) == 0
|
|
|
|
assert len(b2) == 0
|
|
|
|
assert len(b3) == 0
|
|
|
|
assert len(b4) == 0
|
|
|
|
b1.append(42)
|
|
|
|
b2.append('a')
|
|
|
|
b3.append(4.2)
|
|
|
|
b4.append(2.4)
|
|
|
|
assert b1 == [42]
|
|
|
|
assert b2 == ['a']
|
|
|
|
assert b3 == [4.2]
|
|
|
|
assert b4 == [2.4]
|
2021-09-28 02:02:44 +08:00
|
|
|
test_list()
|
|
|
|
|
|
|
|
@test
|
|
|
|
def test_setslice():
|
|
|
|
l = [0, 1]
|
|
|
|
a = l
|
|
|
|
|
|
|
|
for i in range(-3, 4):
|
|
|
|
a[:i] = l[:i]
|
|
|
|
assert a == l
|
|
|
|
a2 = a[:]
|
|
|
|
a2[:i] = a[:i]
|
|
|
|
assert a2 == a
|
|
|
|
a[i:] = l[i:]
|
|
|
|
assert a == l
|
|
|
|
a2 = a[:]
|
|
|
|
a2[i:] = a[i:]
|
|
|
|
assert a2 == a
|
|
|
|
for j in range(-3, 4):
|
|
|
|
a[i:j] = l[i:j]
|
|
|
|
assert a == l
|
|
|
|
a2 = a[:]
|
|
|
|
a2[i:j] = a[i:j]
|
|
|
|
assert a2 == a
|
|
|
|
|
|
|
|
a2 = a[:]
|
|
|
|
aa2 = a2[:]
|
|
|
|
aa2[:0] = [-2, -1]
|
|
|
|
assert aa2 == [-2, -1, 0, 1]
|
|
|
|
aa2[0:] = list[int]()
|
|
|
|
assert aa2 == list[int]()
|
|
|
|
|
|
|
|
a = [1, 2, 3, 4, 5]
|
|
|
|
a[:-1] = a
|
|
|
|
assert a == [1, 2, 3, 4, 5, 5]
|
|
|
|
a = [1, 2, 3, 4, 5]
|
|
|
|
a[1:] = a
|
|
|
|
assert a == [1, 1, 2, 3, 4, 5]
|
|
|
|
a = [1, 2, 3, 4, 5]
|
|
|
|
a[1:-1] = a
|
|
|
|
assert a == [1, 1, 2, 3, 4, 5, 5]
|
|
|
|
|
|
|
|
a = list[int]()
|
|
|
|
a[:] = list(range(10))
|
|
|
|
assert a == list(range(10))
|
|
|
|
|
|
|
|
a = list(range(20))
|
|
|
|
try:
|
|
|
|
a[0:10:0] = [1,2,3]
|
|
|
|
assert False
|
|
|
|
except ValueError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
try:
|
|
|
|
a[0:10:2] = [1,2]
|
|
|
|
assert False
|
|
|
|
except ValueError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
a[2:10:3] = [1,2,3]
|
|
|
|
assert a == [0, 1, 1, 3, 4, 2, 6, 7, 3, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19]
|
|
|
|
test_setslice()
|
|
|
|
|
|
|
|
@test
|
|
|
|
def test_delslice():
|
|
|
|
a = [0, 1]
|
|
|
|
del a[1:2]
|
|
|
|
del a[0:1]
|
|
|
|
assert a == list[int]()
|
|
|
|
|
|
|
|
a = [0, 1]
|
|
|
|
del a[1:2]
|
|
|
|
del a[0:1]
|
|
|
|
assert a == list[int]()
|
|
|
|
|
|
|
|
a = [0, 1]
|
|
|
|
del a[-2:-1]
|
|
|
|
assert a == [1]
|
|
|
|
|
|
|
|
a = [0, 1]
|
|
|
|
del a[-2:-1]
|
|
|
|
assert a == [1]
|
|
|
|
|
|
|
|
a = [0, 1]
|
|
|
|
del a[1:]
|
|
|
|
del a[:1]
|
|
|
|
assert a == list[int]()
|
|
|
|
|
|
|
|
a = [0, 1]
|
|
|
|
del a[1:]
|
|
|
|
del a[:1]
|
|
|
|
assert a == list[int]()
|
|
|
|
|
|
|
|
a = [0, 1]
|
|
|
|
del a[-1:]
|
|
|
|
assert a == [0]
|
|
|
|
|
|
|
|
a = [0, 1]
|
|
|
|
del a[-1:]
|
|
|
|
assert a == [0]
|
|
|
|
|
|
|
|
a = [0,1]
|
|
|
|
del a[:]
|
|
|
|
assert a == list[int]()
|
|
|
|
test_delslice()
|
|
|
|
|
|
|
|
@test
|
|
|
|
def test_extendedslicing():
|
|
|
|
a = [0,1,2,3,4]
|
|
|
|
del a[::2]
|
|
|
|
assert a == [1,3]
|
|
|
|
|
|
|
|
a = list(range(5))
|
|
|
|
del a[1::2]
|
|
|
|
assert a == [0,2,4]
|
|
|
|
|
|
|
|
a = list(range(5))
|
|
|
|
del a[1::-2]
|
|
|
|
assert a == [0,2,3,4]
|
|
|
|
|
|
|
|
a = list(range(10))
|
|
|
|
del a[::1000]
|
|
|
|
assert a == [1, 2, 3, 4, 5, 6, 7, 8, 9]
|
|
|
|
|
|
|
|
a = list(range(10))
|
|
|
|
a[::2] = [-1]*5
|
|
|
|
assert a == [-1, 1, -1, 3, -1, 5, -1, 7, -1, 9]
|
|
|
|
|
|
|
|
a = list(range(10))
|
|
|
|
a[::-4] = [10]*3
|
|
|
|
assert a == [0, 10, 2, 3, 4, 10, 6, 7, 8 ,10]
|
|
|
|
|
|
|
|
a = list(range(4))
|
|
|
|
a[::-1] = a
|
|
|
|
assert a == [3, 2, 1, 0]
|
|
|
|
|
|
|
|
a = list(range(10))
|
|
|
|
b = a[:]
|
|
|
|
c = a[:]
|
|
|
|
a[2:3] = [222, 333]
|
|
|
|
b[2:3] = [222, 333]
|
|
|
|
c[2:3:] = [222, 333]
|
|
|
|
assert a == b
|
|
|
|
assert a == c
|
|
|
|
|
|
|
|
a = list(range(10))
|
|
|
|
a[::2] = (0, 1, 2, 3, 4)
|
|
|
|
assert a == [0, 1, 1, 3, 2, 5, 3, 7, 4, 9]
|
2022-09-19 21:55:56 +08:00
|
|
|
|
|
|
|
assert repr(['x', 'y', 'z']) == "['x', 'y', 'z']"
|
|
|
|
assert repr(List[int]()) == '[]'
|
2021-09-28 02:02:44 +08:00
|
|
|
test_extendedslicing()
|
|
|
|
|
|
|
|
@test
|
|
|
|
def test_set():
|
|
|
|
s1 = {a for a in range(100)}
|
|
|
|
assert len(s1) == 100
|
|
|
|
s1 = {a%8 for a in range(100)}
|
|
|
|
for a in range(8):
|
|
|
|
assert a in s1
|
|
|
|
for a in range(8,100):
|
|
|
|
assert a not in s1
|
|
|
|
|
|
|
|
assert 5 in s1
|
|
|
|
s1.remove(5)
|
|
|
|
assert 5 not in s1
|
|
|
|
assert len(s1) == 7
|
|
|
|
|
|
|
|
s1 = {1,2,3,4}
|
|
|
|
s2 = {2,3,4,5}
|
|
|
|
s3 = set[int]()
|
|
|
|
|
|
|
|
assert (s1 | s2) == {1, 2, 3, 4, 5}
|
|
|
|
assert (s1 & s2) == {4, 2, 3}
|
|
|
|
assert (s1 ^ s2) == {1, 5}
|
|
|
|
assert (s1 | s3) == {1, 2, 3, 4}
|
|
|
|
assert (s1 & s3) == set[int]()
|
|
|
|
assert (s1 ^ s3) == {1, 2, 3, 4}
|
|
|
|
assert (s1 - s2) == {1}
|
|
|
|
assert (s2 - s1) == {5}
|
|
|
|
assert (s3 - s1 - s2) == set[int]()
|
|
|
|
assert (s1 > s2) == False
|
|
|
|
assert (s1 < s2) == False
|
|
|
|
assert (s3 <= s1) == True
|
|
|
|
assert (s2 >= s1) == False
|
|
|
|
assert ((s1 | s2) > s1) == True
|
|
|
|
|
|
|
|
s1c = copy(s1)
|
|
|
|
s2c = copy(s2)
|
|
|
|
s3c = copy(s3)
|
|
|
|
|
|
|
|
assert s1c == {1, 2, 3, 4}
|
|
|
|
s1c &= s2c
|
|
|
|
assert s1c == {2, 3, 4}
|
|
|
|
s1c -= s3c
|
|
|
|
assert s1c == {2, 3, 4}
|
|
|
|
s1c ^= s1c
|
|
|
|
assert s1c == set[int]()
|
|
|
|
s1c |= s2c
|
|
|
|
assert s1c == {2, 3, 4, 5}
|
|
|
|
assert s1 == {1, 2, 3, 4}
|
|
|
|
|
|
|
|
s1 = {1, 2, 3, 999999}
|
|
|
|
s2 = {1, 2, 3, 999999}
|
|
|
|
v = s1.pop()
|
|
|
|
assert v in s2
|
|
|
|
s2.remove(v)
|
|
|
|
|
|
|
|
v = s1.pop()
|
|
|
|
assert v in s2
|
|
|
|
s2.remove(v)
|
|
|
|
|
|
|
|
v = s1.pop()
|
|
|
|
assert v in s2
|
|
|
|
s2.remove(v)
|
|
|
|
|
|
|
|
v = s1.pop()
|
|
|
|
assert v in s2
|
|
|
|
s2.remove(v)
|
|
|
|
|
|
|
|
try:
|
|
|
|
s1.pop()
|
|
|
|
assert False
|
|
|
|
except ValueError:
|
|
|
|
pass
|
2022-09-19 21:55:56 +08:00
|
|
|
|
|
|
|
assert repr({(1,2)}) == '{(1, 2)}'
|
2022-09-19 21:58:57 +08:00
|
|
|
assert repr(Set[int]()) == 'set()'
|
2021-09-28 02:02:44 +08:00
|
|
|
test_set()
|
|
|
|
|
|
|
|
@test
|
|
|
|
def test_dict():
|
|
|
|
d1 = copy({a: a*a for a in range(100)})
|
|
|
|
assert len(d1) == 100
|
|
|
|
d1 = {a: a*a for a in range(5)}
|
|
|
|
assert len(copy(dict[int,int]())) == 0
|
|
|
|
|
|
|
|
assert [d1.get(a, -1) for a in range(6)] == [0, 1, 4, 9, 16, -1]
|
|
|
|
|
|
|
|
assert 2 in d1
|
|
|
|
del d1[2]
|
|
|
|
assert 2 not in d1
|
|
|
|
d1[2] = 44
|
|
|
|
assert 2 in d1
|
|
|
|
assert d1.get(2, -1) == 44
|
|
|
|
assert d1[3] == 9
|
|
|
|
|
|
|
|
del d1[3]
|
|
|
|
del d1[4]
|
|
|
|
|
|
|
|
assert [k for k in d1] == [0, 1, 2]
|
|
|
|
assert [t for t in d1.items()] == [(0, 0), (1, 1), (2, 44)]
|
|
|
|
|
|
|
|
d2 = {'x': 10, 'y': 0}
|
|
|
|
d2.increment('x')
|
|
|
|
d2.increment('y', by=-1)
|
|
|
|
d2.increment('z', by=2)
|
|
|
|
assert d2['x'] == 11
|
|
|
|
assert d2['y'] == -1
|
|
|
|
assert d2['z'] == 2
|
|
|
|
assert d2 == {'x': 11, 'y': -1, 'z': 2}
|
2022-09-16 03:40:00 +08:00
|
|
|
|
|
|
|
d3 = {1: 2, 42: 42}
|
|
|
|
d4 = {1: 5, 2: 9}
|
|
|
|
assert d3 | d4 == {1: 5, 42: 42, 2: 9}
|
|
|
|
d3 |= d4
|
|
|
|
assert d3 == {1: 5, 42: 42, 2: 9}
|
2022-09-19 21:55:56 +08:00
|
|
|
|
|
|
|
assert repr({1: ['x']}) == "{1: ['x']}"
|
|
|
|
assert repr(Dict[int,int]()) == '{}'
|
2021-09-28 02:02:44 +08:00
|
|
|
test_dict()
|
|
|
|
|
2023-07-03 06:50:43 +08:00
|
|
|
def slice_indices(slice, length):
|
|
|
|
"""
|
|
|
|
Reference implementation for the slice.indices method.
|
|
|
|
|
|
|
|
"""
|
|
|
|
# Compute step and length as integers.
|
|
|
|
#length = operator.index(length)
|
|
|
|
step: int = 1 if slice.step is None else slice.step
|
|
|
|
|
|
|
|
# Raise ValueError for negative length or zero step.
|
|
|
|
if length < 0:
|
|
|
|
raise ValueError("length should not be negative")
|
|
|
|
if step == 0:
|
|
|
|
raise ValueError("slice step cannot be zero")
|
|
|
|
|
|
|
|
# Find lower and upper bounds for start and stop.
|
|
|
|
lower = -1 if step < 0 else 0
|
|
|
|
upper = length - 1 if step < 0 else length
|
|
|
|
|
|
|
|
# Compute start.
|
|
|
|
if slice.start is None:
|
|
|
|
start = upper if step < 0 else lower
|
|
|
|
else:
|
|
|
|
start = slice.start
|
|
|
|
start = max(start + length, lower) if start < 0 else min(start, upper)
|
|
|
|
|
|
|
|
# Compute stop.
|
|
|
|
if slice.stop is None:
|
|
|
|
stop = lower if step < 0 else upper
|
|
|
|
else:
|
|
|
|
stop = slice.stop
|
|
|
|
stop = max(stop + length, lower) if stop < 0 else min(stop, upper)
|
|
|
|
|
|
|
|
return start, stop, step
|
|
|
|
|
|
|
|
def check_indices(slice, length):
|
|
|
|
err1 = False
|
|
|
|
err2 = False
|
|
|
|
|
|
|
|
try:
|
|
|
|
actual = slice.indices(length)
|
|
|
|
except ValueError:
|
|
|
|
err1 = True
|
|
|
|
|
|
|
|
try:
|
|
|
|
expected = slice_indices(slice, length)
|
|
|
|
except ValueError:
|
|
|
|
err2 = True
|
|
|
|
|
|
|
|
if err1 or err2:
|
|
|
|
return err1 and err2
|
|
|
|
|
|
|
|
if actual != expected:
|
|
|
|
return False
|
|
|
|
|
|
|
|
if length >= 0 and slice.step != 0:
|
|
|
|
actual = range(*slice.indices(length))
|
|
|
|
expected = range(length)[slice]
|
|
|
|
if actual != expected:
|
|
|
|
return False
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
@test
|
|
|
|
def test_slice():
|
|
|
|
assert repr(slice(1, 2, 3)) == 'slice(1, 2, 3)'
|
|
|
|
|
|
|
|
s1 = slice(1, 2, 3)
|
|
|
|
s2 = slice(1, 2, 3)
|
|
|
|
s3 = slice(1, 2, 4)
|
|
|
|
|
|
|
|
assert s1 == s2
|
|
|
|
assert s1 != s3
|
|
|
|
|
|
|
|
s = slice(1)
|
|
|
|
assert s.start == None
|
|
|
|
assert s.stop == 1
|
|
|
|
assert s.step == None
|
|
|
|
|
|
|
|
s = slice(1, 2)
|
|
|
|
assert s.start == 1
|
|
|
|
assert s.stop == 2
|
|
|
|
assert s.step == None
|
|
|
|
|
|
|
|
s = slice(1, 2, 3)
|
|
|
|
assert s.start == 1
|
|
|
|
assert s.stop == 2
|
|
|
|
assert s.step == 3
|
|
|
|
|
|
|
|
# TODO
|
|
|
|
assert slice(None ).indices(10) == (0, 10, 1)
|
|
|
|
assert slice(None, None, 2).indices(10) == (0, 10, 2)
|
|
|
|
assert slice(1, None, 2).indices(10) == (1, 10, 2)
|
|
|
|
assert slice(None, None, -1).indices(10) == (9, -1, -1)
|
|
|
|
assert slice(None, None, -2).indices(10) == (9, -1, -2)
|
|
|
|
assert slice(3, None, -2).indices(10) == (3, -1, -2)
|
|
|
|
# issue 3004 tests
|
|
|
|
assert slice(None, -9).indices(10) == (0, 1, 1)
|
|
|
|
assert slice(None, -10).indices(10) == (0, 0, 1)
|
|
|
|
assert slice(None, -11).indices(10) == (0, 0, 1)
|
|
|
|
assert slice(None, -10, -1).indices(10) == (9, 0, -1)
|
|
|
|
assert slice(None, -11, -1).indices(10) == (9, -1, -1)
|
|
|
|
assert slice(None, -12, -1).indices(10) == (9, -1, -1)
|
|
|
|
assert slice(None, 9).indices(10) == (0, 9, 1)
|
|
|
|
assert slice(None, 10).indices(10) == (0, 10, 1)
|
|
|
|
assert slice(None, 11).indices(10) == (0, 10, 1)
|
|
|
|
assert slice(None, 8, -1).indices(10) == (9, 8, -1)
|
|
|
|
assert slice(None, 9, -1).indices(10) == (9, 9, -1)
|
|
|
|
assert slice(None, 10, -1).indices(10) == (9, 9, -1)
|
|
|
|
|
|
|
|
assert slice(-100, 100 ).indices(10) == slice(None).indices(10)
|
|
|
|
|
|
|
|
assert slice(100, -100, -1).indices(10) == slice(None, None, -1).indices(10)
|
|
|
|
|
|
|
|
assert slice(-100, 100, 2).indices(10) == (0, 10, 2)
|
|
|
|
|
|
|
|
import sys
|
|
|
|
assert list(range(10))[::sys.maxsize - 1] == [0]
|
|
|
|
|
|
|
|
# Check a variety of start, stop, step and length values, including
|
|
|
|
# values exceeding sys.maxsize (see issue #14794).
|
|
|
|
vals = [None, -2**100, -2**30, -53, -7, -1, 0, 1, 7, 53, 2**30, 2**100]
|
|
|
|
lengths = [0, 1, 7, 53, 2**30, 2**100]
|
|
|
|
#for slice_args in itertools.product(vals, repeat=3):
|
|
|
|
for a in vals:
|
|
|
|
for b in vals:
|
|
|
|
for c in vals:
|
|
|
|
slice_args = (a, b, c)
|
|
|
|
s = slice(*slice_args)
|
|
|
|
for length in lengths:
|
|
|
|
assert check_indices(s, length)
|
|
|
|
assert check_indices(slice(0, 10, 1), -3)
|
|
|
|
|
|
|
|
# Negative length should raise ValueError
|
|
|
|
try:
|
|
|
|
slice(None).indices(-1)
|
|
|
|
assert False
|
|
|
|
except ValueError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
# Zero step should raise ValueError
|
|
|
|
try:
|
|
|
|
slice(0, 10, 0).indices(5)
|
|
|
|
except ValueError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
# ... but it should be fine to use a custom class that provides index.
|
|
|
|
assert slice(0, 10, 1).indices(5) == (0, 5, 1)
|
|
|
|
''' # not yet supported in Codon
|
|
|
|
assert slice(MyIndexable(0), 10, 1).indices(5) == (0, 5, 1)
|
|
|
|
assert slice(0, MyIndexable(10), 1).indices(5) == (0, 5, 1)
|
|
|
|
assert slice(0, 10, MyIndexable(1)).indices(5) == (0, 5, 1)
|
|
|
|
assert slice(0, 10, 1).indices(MyIndexable(5)) == (0, 5, 1)
|
|
|
|
'''
|
|
|
|
tmp = []
|
|
|
|
class X[T](object):
|
|
|
|
tmp: T
|
|
|
|
def __setitem__(self, i, k):
|
|
|
|
self.tmp.append((i, k))
|
|
|
|
|
|
|
|
x = X(tmp)
|
|
|
|
x[1:2] = 42
|
|
|
|
assert tmp == [(slice(1, 2), 42)]
|
2024-02-26 00:42:31 +08:00
|
|
|
|
|
|
|
# Non-int elements
|
|
|
|
def check_types(s, T: type, U: type, V: type):
|
|
|
|
return (type(s.start) is Optional[T] and
|
|
|
|
type(s.stop) is Optional[U] and
|
|
|
|
type(s.step) is Optional[V])
|
|
|
|
assert check_types(slice(1j, 'x', 3.14), complex, str, float)
|
|
|
|
assert check_types(slice(None, 'x', 3.14), int, str, float)
|
|
|
|
assert check_types(slice(1j, None, 3.14), complex, int, float)
|
|
|
|
assert check_types(slice(1j, 'x', None), complex, str, int)
|
|
|
|
assert check_types(slice(1j, None, None), complex, int, int)
|
|
|
|
assert check_types(slice(None, 'x', None), int, str, int)
|
|
|
|
assert check_types(slice(None, None, 3.14), int, int, float)
|
|
|
|
assert check_types(slice(None, None, None), int, int, int)
|
|
|
|
assert check_types(slice(1j, 'x'), complex, str, int)
|
|
|
|
assert check_types(slice(None, 'x'), int, str, int)
|
|
|
|
assert check_types(slice(1j, None), complex, int, int)
|
|
|
|
assert check_types(slice(None, None), int, int, int)
|
|
|
|
assert check_types(slice(1j), int, complex, int)
|
|
|
|
assert check_types(slice(None), int, int, int)
|
|
|
|
|
|
|
|
# eq / ne
|
|
|
|
assert slice(1, 2, 3) == slice(1, 2, 3)
|
|
|
|
assert slice(0, 2, 3) != slice(1, 2, 3)
|
|
|
|
assert slice(1, 0, 3) != slice(1, 2, 3)
|
|
|
|
assert slice(1, 2, 0) != slice(1, 2, 3)
|
|
|
|
assert slice(None, None, None) == slice(None, None, None)
|
|
|
|
assert slice(None, 42, None) == slice(None, 42, None)
|
|
|
|
assert slice(None, 42, None) != slice(None, 43, None)
|
|
|
|
assert slice(1, None, 3) == slice(1, None, 3)
|
|
|
|
assert slice(1, None, 3) != slice(1, None, 0)
|
|
|
|
assert slice(1, None, 3) != slice(0, None, 3)
|
|
|
|
assert slice(1) == slice(1)
|
|
|
|
assert slice(1) != slice(2)
|
2023-07-03 06:50:43 +08:00
|
|
|
test_slice()
|
|
|
|
|
2021-09-28 02:02:44 +08:00
|
|
|
@test
|
|
|
|
def test_deque():
|
|
|
|
from collections import deque
|
|
|
|
|
|
|
|
dq = deque[int]()
|
|
|
|
dq.append(1)
|
|
|
|
dq.append(2)
|
|
|
|
dq.append(3)
|
|
|
|
dq.appendleft(11)
|
|
|
|
dq.appendleft(22)
|
2022-09-16 03:40:00 +08:00
|
|
|
assert str(dq) == 'deque([22, 11, 1, 2, 3])'
|
2021-09-28 02:02:44 +08:00
|
|
|
assert bool(dq) == True
|
|
|
|
|
|
|
|
# test cap increase:
|
|
|
|
dq.clear()
|
|
|
|
assert bool(dq) == False
|
|
|
|
for i in range(20):
|
|
|
|
dq.append(i)
|
|
|
|
dq.appendleft(i)
|
2022-09-16 03:40:00 +08:00
|
|
|
assert str(dq) == 'deque([19, 18, 17, 16, 15, 14, 13, 12, 11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1, 0, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19])'
|
2021-09-28 02:02:44 +08:00
|
|
|
assert len(dq) == 40
|
|
|
|
|
|
|
|
for i in range(19):
|
|
|
|
dq.pop()
|
|
|
|
dq.popleft()
|
2022-09-16 03:40:00 +08:00
|
|
|
assert str(dq) == 'deque([0, 0])'
|
2021-09-28 02:02:44 +08:00
|
|
|
for a in dq:
|
|
|
|
assert a == 0
|
|
|
|
|
|
|
|
assert (0 in dq) == True
|
|
|
|
assert (1 in dq) == False
|
2022-09-16 03:40:00 +08:00
|
|
|
assert str(copy(dq)) == 'deque([0, 0])'
|
2021-09-28 02:02:44 +08:00
|
|
|
|
|
|
|
# test maxlen:
|
|
|
|
dq = deque[int](5)
|
|
|
|
for i in range(100):
|
|
|
|
dq.append(i)
|
2022-09-16 03:40:00 +08:00
|
|
|
assert str(dq) == 'deque([95, 96, 97, 98, 99])'
|
2021-09-28 02:02:44 +08:00
|
|
|
|
|
|
|
for i in range(5):
|
|
|
|
dq.append(i)
|
2022-09-16 03:40:00 +08:00
|
|
|
assert str(dq) == 'deque([0, 1, 2, 3, 4])'
|
2021-09-28 02:02:44 +08:00
|
|
|
test_deque()
|
|
|
|
|
|
|
|
@test
|
|
|
|
def test_counter():
|
|
|
|
from collections import Counter
|
|
|
|
|
|
|
|
# main
|
|
|
|
c = Counter('abcaba')
|
|
|
|
assert c == Counter({'a':3 , 'b': 2, 'c': 1})
|
|
|
|
assert len(c) == 3
|
|
|
|
assert sum(c.values()) == 6
|
|
|
|
assert set(c.values()) == {1, 2, 3}
|
|
|
|
assert set(c.keys()) == {'a','b','c'}
|
|
|
|
assert set(c.items()) == {('a',3), ('b',2), ('c',1)}
|
|
|
|
assert c['b'] == 2
|
|
|
|
assert c['z'] == 0
|
|
|
|
assert c.__contains__('b')
|
|
|
|
assert not c.__contains__('z')
|
|
|
|
assert c.get('b', 10) == 2
|
|
|
|
assert c.get('z', 10) == 10
|
|
|
|
assert c == {'a':3 , 'b': 2, 'c': 1}
|
|
|
|
assert c.most_common() == [('a', 3), ('b', 2), ('c', 1)]
|
|
|
|
for i in range(5):
|
|
|
|
assert c.most_common(i) == [('a', 3), ('b', 2), ('c', 1)][:i]
|
|
|
|
|
|
|
|
c['a'] += 1 # increment an existing value
|
|
|
|
c['b'] -= 2 # sub existing value to zero
|
|
|
|
del c['c'] # remove an entry
|
|
|
|
del c['c'] # make sure that del doesn't raise KeyError
|
|
|
|
c['d'] -= 2 # sub from a missing value
|
|
|
|
c['e'] = -5 # directly assign a missing value
|
|
|
|
c['f'] += 4 # add to a missing value
|
|
|
|
|
|
|
|
assert dict(c) == {'a':4, 'b':0, 'd':-2, 'e':-5, 'f':4}
|
|
|
|
assert c.pop('f') == 4
|
|
|
|
assert 'f' not in c
|
|
|
|
for i in range(3):
|
|
|
|
elem, cnt = c.popitem()
|
|
|
|
assert elem not in c
|
|
|
|
c.clear()
|
|
|
|
assert c == {}
|
|
|
|
c.update({'a':5, 'b':3})
|
|
|
|
c.update({'c': 1})
|
|
|
|
c.update(Counter('a' * 50 + 'b' * 30))
|
|
|
|
c.update()
|
|
|
|
c.update('a' * 500 + 'b' * 300)
|
|
|
|
c.update('cdc')
|
|
|
|
assert c == {'a':555, 'b':333, 'c':3, 'd':1}
|
|
|
|
|
|
|
|
assert c.setdefault('d',5) == 1
|
|
|
|
assert c['d'] == 1
|
|
|
|
assert c.setdefault('e', 5) == 5
|
|
|
|
assert c['e'] == 5
|
|
|
|
|
|
|
|
|
|
|
|
# total
|
|
|
|
c = Counter({'a':10, 'b':5, 'c':0})
|
|
|
|
assert c.total() == 15
|
|
|
|
|
|
|
|
|
|
|
|
# conversions
|
|
|
|
s = 'she sells sea shells by the sea shore'
|
|
|
|
assert sorted(Counter(s).elements()) == sorted(s)
|
|
|
|
assert sorted(Counter(s)) == sorted(set(s))
|
|
|
|
assert dict(Counter(s)) == dict(Counter(s).items())
|
|
|
|
assert set(Counter(s)) == set(s)
|
|
|
|
|
|
|
|
|
|
|
|
# in invariant
|
|
|
|
c = Counter({'a':10, 'b':-2, 'c':0})
|
|
|
|
for elem in c:
|
|
|
|
assert elem in c
|
|
|
|
|
|
|
|
|
|
|
|
# multiset
|
|
|
|
c = Counter({'a':10, 'b':-2, 'c':0}) + Counter()
|
|
|
|
assert dict(c) == {'a':10}
|
|
|
|
|
|
|
|
from random import randrange, randint
|
|
|
|
elements = 'abcd'
|
|
|
|
for i in range(1000):
|
|
|
|
# test random pairs of multisets
|
|
|
|
p = Counter(dict((elem, randrange(-2,4)) for elem in elements))
|
|
|
|
p.update({'e':1, 'f':-1, 'g':0})
|
|
|
|
q = Counter(dict((elem, randrange(-2,4)) for elem in elements))
|
|
|
|
q.update({'h':1, 'i':-1, 'j':0})
|
|
|
|
|
|
|
|
result = p + q
|
|
|
|
for x in elements:
|
|
|
|
assert max(0, p[x] + q[x]) == result[x]
|
|
|
|
assert all(x>0 for x in result.values())
|
|
|
|
|
|
|
|
result = p - q
|
|
|
|
for x in elements:
|
|
|
|
assert max(0, p[x] - q[x]) == result[x]
|
|
|
|
assert all(x>0 for x in result.values())
|
|
|
|
|
|
|
|
result = p | q
|
|
|
|
for x in elements:
|
|
|
|
assert max(0, p[x], q[x]) == result[x]
|
|
|
|
assert all(x>0 for x in result.values())
|
|
|
|
|
|
|
|
result = p & q
|
|
|
|
for x in elements:
|
|
|
|
assert max(0, min(p[x], q[x])) == result[x]
|
|
|
|
assert all(x>0 for x in result.values())
|
|
|
|
|
|
|
|
elements = 'abcdef'
|
|
|
|
for i in range(100):
|
|
|
|
# verify that random multisets with no repeats are exactly like sets
|
|
|
|
p = Counter(dict((elem, randrange(0, 2)) for elem in elements))
|
|
|
|
q = Counter(dict((elem, randrange(0, 2)) for elem in elements))
|
|
|
|
|
|
|
|
counter_result = p - q
|
|
|
|
set_result = set(p.elements()) - set(q.elements())
|
|
|
|
assert counter_result == dict.fromkeys(set_result, 1)
|
|
|
|
|
|
|
|
counter_result = p | q
|
|
|
|
set_result = set(p.elements()) | set(q.elements())
|
|
|
|
assert counter_result == dict.fromkeys(set_result, 1)
|
|
|
|
|
|
|
|
counter_result = p & q
|
|
|
|
set_result = set(p.elements()) & set(q.elements())
|
|
|
|
assert counter_result == dict.fromkeys(set_result, 1)
|
|
|
|
|
|
|
|
|
|
|
|
# in-place
|
|
|
|
elements = 'abcd'
|
|
|
|
for i in range(1000):
|
|
|
|
# test random pairs of multisets
|
|
|
|
p = Counter(dict((elem, randrange(-2,4)) for elem in elements))
|
|
|
|
p.update({'e':1, 'f':-1, 'g':0})
|
|
|
|
q = Counter(dict((elem, randrange(-2,4)) for elem in elements))
|
|
|
|
q.update({'h':1, 'i':-1, 'j':0})
|
|
|
|
|
|
|
|
c = p.copy()
|
|
|
|
c_id = id(c)
|
|
|
|
regular_result = c + q
|
|
|
|
inplace_result = c.__iadd__(q)
|
|
|
|
assert inplace_result == regular_result
|
|
|
|
assert id(inplace_result) == c_id
|
|
|
|
|
|
|
|
c = p.copy()
|
|
|
|
c_id = id(c)
|
|
|
|
regular_result = c - q
|
|
|
|
inplace_result = c.__isub__(q)
|
|
|
|
assert inplace_result == regular_result
|
|
|
|
assert id(inplace_result) == c_id
|
|
|
|
|
|
|
|
c = p.copy()
|
|
|
|
c_id = id(c)
|
|
|
|
regular_result = c | q
|
|
|
|
inplace_result = c.__ior__(q)
|
|
|
|
assert inplace_result == regular_result
|
|
|
|
assert id(inplace_result) == c_id
|
|
|
|
|
|
|
|
c = p.copy()
|
|
|
|
c_id = id(c)
|
|
|
|
regular_result = c & q
|
|
|
|
inplace_result = c.__iand__(q)
|
|
|
|
assert inplace_result == regular_result
|
|
|
|
assert id(inplace_result) == c_id
|
|
|
|
|
|
|
|
|
|
|
|
# subtract
|
|
|
|
c = Counter({'a':-5, 'b':0, 'c':5, 'd':10, 'e':15,'g':40})
|
|
|
|
c.subtract({'a':1, 'b':2, 'c':-3, 'd':10, 'e':20, 'f':30, 'h':-50})
|
|
|
|
assert c == Counter({'a':-6, 'b':-2, 'c':8, 'd':0, 'e':-5, 'f':-30, 'g':40, 'h':50})
|
|
|
|
c = Counter({'a':-5, 'b':0, 'c':5, 'd':10, 'e':15,'g':40})
|
|
|
|
c.subtract(Counter({'a':1, 'b':2, 'c':-3, 'd':10, 'e':20, 'f':30, 'h':-50}))
|
|
|
|
assert c == Counter({'a':-6, 'b':-2, 'c':8, 'd':0, 'e':-5, 'f':-30, 'g':40, 'h':50})
|
|
|
|
|
|
|
|
c = Counter('aaabbcd')
|
|
|
|
c.subtract('aaaabbcce')
|
|
|
|
assert c == Counter({'a':-1, 'b':0, 'c':-1, 'd':1, 'e':-1})
|
|
|
|
|
|
|
|
c = Counter()
|
|
|
|
c.subtract({'self':42})
|
|
|
|
assert list(c.items()) == [('self', -42)]
|
|
|
|
c = Counter()
|
|
|
|
c.subtract({'iterable':42})
|
|
|
|
assert list(c.items()) == [('iterable', -42)]
|
|
|
|
|
|
|
|
|
|
|
|
# unary
|
|
|
|
c = Counter({'a':-5, 'b':0, 'c':5, 'd':10, 'e':15, 'g':40})
|
|
|
|
assert dict(+c) == {'c':5, 'd':10, 'e':15, 'g':40}
|
|
|
|
assert dict(-c) == {'a':5}
|
|
|
|
|
|
|
|
|
|
|
|
# equality
|
|
|
|
assert Counter({'a':3, 'b':2}) == Counter('ababa')
|
|
|
|
assert Counter({'a':3, 'b':2}) != Counter('babab')
|
|
|
|
|
|
|
|
|
|
|
|
# most common
|
|
|
|
c = Counter({v:k for k,v in enumerate('hgfedcba')})
|
|
|
|
q = [(v,k) for k,v in enumerate('hgfedcba')][::-1]
|
|
|
|
assert c.most_common() == q
|
|
|
|
for i in range(10):
|
|
|
|
assert c.most_common(i) == q[:i]
|
|
|
|
|
|
|
|
for limit in range(100):
|
|
|
|
for samples in range(100):
|
|
|
|
for most_common in range(100):
|
|
|
|
data = [randint(0, limit) for _ in range(samples)]
|
|
|
|
d = Counter(data)
|
|
|
|
exp = sorted(d.values(), reverse=True)[:most_common]
|
|
|
|
got = [v for k,v in d.most_common(most_common)]
|
|
|
|
assert exp == got
|
|
|
|
|
|
|
|
for limit in range(100):
|
|
|
|
for samples in range(100):
|
|
|
|
data = [randint(0, limit) for _ in range(samples)]
|
|
|
|
d = Counter(data)
|
|
|
|
exp = sorted(d.values(), reverse=True)
|
|
|
|
got = [v for k,v in d.most_common()]
|
|
|
|
assert exp == got
|
2022-09-16 03:40:00 +08:00
|
|
|
|
|
|
|
assert repr(Counter('abcabc')) == "Counter({'a': 2, 'b': 2, 'c': 2})"
|
2021-09-28 02:02:44 +08:00
|
|
|
test_counter()
|
2022-09-16 03:40:00 +08:00
|
|
|
|
|
|
|
@test
|
|
|
|
def test_defaultdict():
|
|
|
|
from collections import defaultdict
|
|
|
|
|
|
|
|
# basic
|
|
|
|
#d1 = defaultdict()
|
|
|
|
#self.assertEqual(d1.default_factory, None)
|
|
|
|
#d1.default_factory = list
|
|
|
|
d1 = defaultdict(list)
|
|
|
|
d1[12].append(42)
|
|
|
|
assert d1 == {12: [42]}
|
|
|
|
d1[12].append(24)
|
|
|
|
assert d1 == {12: [42, 24]}
|
|
|
|
d1[13]
|
|
|
|
d1[14]
|
|
|
|
assert d1 == {12: [42, 24], 13: [], 14: []}
|
|
|
|
assert d1[12] is not d1[13] is not d1[14]
|
|
|
|
#d2 = defaultdict(list, foo=1, bar=2)
|
|
|
|
#self.assertEqual(d2.default_factory, list)
|
|
|
|
#self.assertEqual(d2, {"foo": 1, "bar": 2})
|
|
|
|
#self.assertEqual(d2["foo"], 1)
|
|
|
|
#self.assertEqual(d2["bar"], 2)
|
|
|
|
#self.assertEqual(d2[42], [])
|
|
|
|
#self.assertIn("foo", d2)
|
|
|
|
#self.assertIn("foo", d2.keys())
|
|
|
|
#self.assertIn("bar", d2)
|
|
|
|
#self.assertIn("bar", d2.keys())
|
|
|
|
#self.assertIn(42, d2)
|
|
|
|
#self.assertIn(42, d2.keys())
|
|
|
|
#self.assertNotIn(12, d2)
|
|
|
|
#self.assertNotIn(12, d2.keys())
|
|
|
|
#d2.default_factory = None
|
|
|
|
#self.assertEqual(d2.default_factory, None)
|
|
|
|
#try:
|
|
|
|
# d2[15]
|
|
|
|
#except KeyError as err:
|
|
|
|
# self.assertEqual(err.args, (15,))
|
|
|
|
#else:
|
|
|
|
# self.fail("d2[15] didn't raise KeyError")
|
|
|
|
#self.assertRaises(TypeError, defaultdict, 1)
|
|
|
|
|
|
|
|
# missing
|
|
|
|
#d1 = defaultdict()
|
|
|
|
#self.assertRaises(KeyError, d1.__missing__, 42)
|
|
|
|
#d1.default_factory = list
|
|
|
|
#d1 = defaultdict(list)
|
|
|
|
assert d1.__missing__(42) == []
|
|
|
|
|
|
|
|
# repr
|
|
|
|
d1 = defaultdict(lambda: 0)
|
|
|
|
#self.assertEqual(d1.default_factory, None)
|
|
|
|
#self.assertEqual(repr(d1), "defaultdict(None, {})")
|
|
|
|
#self.assertEqual(eval(repr(d1)), d1)
|
|
|
|
d1[11] = 41
|
|
|
|
assert repr(d1) == "defaultdict(<default factory of 'int'>, {11: 41})"
|
|
|
|
d2 = defaultdict(lambda: 0) # TODO: use 'int' when it's fixed...
|
|
|
|
#self.assertEqual(d2.default_factory, int)
|
|
|
|
d2[12] = 42
|
|
|
|
assert repr(d2) == "defaultdict(<default factory of 'int'>, {12: 42})"
|
|
|
|
def foo(): return 43
|
|
|
|
d3 = defaultdict(foo)
|
|
|
|
#self.assertTrue(d3.default_factory is foo)
|
|
|
|
d3[13]
|
|
|
|
assert repr(d3) == "defaultdict(<default factory of 'int'>, {13: 43})"
|
|
|
|
|
|
|
|
|
|
|
|
# copy
|
|
|
|
d1 = defaultdict(list)
|
|
|
|
d2 = d1.copy()
|
|
|
|
#self.assertEqual(type(d2), defaultdict)
|
|
|
|
#self.assertEqual(d2.default_factory, None)
|
|
|
|
assert d2 == {}
|
|
|
|
#d1.default_factory = list
|
|
|
|
#d3 = d1.copy()
|
|
|
|
#self.assertEqual(type(d3), defaultdict)
|
|
|
|
#self.assertEqual(d3.default_factory, list)
|
|
|
|
#self.assertEqual(d3, {})
|
|
|
|
d1[42].append(0)
|
|
|
|
#d4 = d1.copy()
|
|
|
|
#assert d4 == {42: [0]}
|
|
|
|
#d4[12]
|
|
|
|
#assert d4 == {42: [], 12: []}
|
|
|
|
|
|
|
|
# Issue 6637: Copy fails for empty default dict
|
|
|
|
#d = defaultdict()
|
|
|
|
#d['a'] = 42
|
|
|
|
#e = d.copy()
|
|
|
|
#assert e['a'] == 42
|
|
|
|
|
|
|
|
|
|
|
|
# shallow copy
|
|
|
|
foobar = list
|
|
|
|
d1 = defaultdict(foobar)
|
|
|
|
d1[1] += [1]
|
|
|
|
d2 = copy(d1)
|
|
|
|
#self.assertEqual(d2.default_factory, foobar)
|
|
|
|
assert d2 == d1
|
|
|
|
#d1.default_factory = list
|
|
|
|
d2 = copy(d1)
|
|
|
|
#self.assertEqual(d2.default_factory, list)
|
|
|
|
assert d2 == d1
|
|
|
|
|
|
|
|
# deep copy
|
|
|
|
d1 = defaultdict(foobar)
|
|
|
|
d1[1].append(1)
|
|
|
|
d2 = deepcopy(d1)
|
|
|
|
#self.assertEqual(d2.default_factory, foobar)
|
|
|
|
assert d2 == d1
|
|
|
|
assert d1[1] is not d2[1]
|
|
|
|
#d1.default_factory = list
|
|
|
|
d2 = deepcopy(d1)
|
|
|
|
#self.assertEqual(d2.default_factory, list)
|
|
|
|
assert d2 == d1
|
|
|
|
|
|
|
|
# KeyError without factory
|
|
|
|
#d1 = defaultdict()
|
|
|
|
#try:
|
|
|
|
# d1[(1,)]
|
|
|
|
#except KeyError as err:
|
|
|
|
# self.assertEqual(err.args[0], (1,))
|
|
|
|
#else:
|
|
|
|
# self.fail("expected KeyError")
|
|
|
|
|
|
|
|
# pickling
|
|
|
|
#d = defaultdict(int)
|
|
|
|
#d[1]
|
|
|
|
#for proto in range(pickle.HIGHEST_PROTOCOL + 1):
|
|
|
|
# s = pickle.dumps(d, proto)
|
|
|
|
# o = pickle.loads(s)
|
|
|
|
# self.assertEqual(d, o)
|
|
|
|
|
|
|
|
# union
|
|
|
|
i = defaultdict(int, {1: 1, 2: 2})
|
|
|
|
s = defaultdict(int, {0: 0, 1: 111})
|
|
|
|
|
|
|
|
i_s = i | s
|
|
|
|
#self.assertIs(i_s.default_factory, int)
|
|
|
|
assert i_s == {1: 111, 2: 2, 0: 0}
|
|
|
|
assert sorted(i_s) == [0, 1, 2]
|
|
|
|
|
|
|
|
s_i = s | i
|
|
|
|
#self.assertIs(s_i.default_factory, str)
|
|
|
|
assert s_i == {0: 0, 1: 1, 2: 2}
|
|
|
|
assert sorted(s_i) == [0, 1, 2]
|
|
|
|
|
|
|
|
i_ds = i | dict(s)
|
|
|
|
#self.assertIs(i_ds.default_factory, int)
|
|
|
|
assert i_ds == {1: 111, 2: 2, 0: 0}
|
|
|
|
assert sorted(i_ds) == [0, 1, 2]
|
|
|
|
|
|
|
|
ds_i = dict(s) | i
|
|
|
|
#self.assertIs(ds_i.default_factory, int)
|
|
|
|
assert ds_i == {0: 0, 1: 1, 2: 2}
|
|
|
|
assert sorted(ds_i) == [0, 1, 2]
|
|
|
|
|
|
|
|
# We inherit a fine |= from dict, so just a few sanity checks here:
|
|
|
|
i |= list(s.items())
|
|
|
|
#self.assertIs(i.default_factory, int)
|
|
|
|
assert i == {1: 111, 2: 2, 0: 0}
|
|
|
|
assert sorted(i), [1, 2, 0]
|
|
|
|
|
|
|
|
# general
|
|
|
|
s = 'mississippi'
|
|
|
|
d = defaultdict(int)
|
|
|
|
for k in s:
|
|
|
|
d[k] += 1
|
|
|
|
assert sorted(d.items()) == [('i', 4), ('m', 1), ('p', 2), ('s', 4)]
|
|
|
|
|
|
|
|
s = 'mississippi'
|
|
|
|
d = defaultdict(int)
|
|
|
|
for k in s:
|
|
|
|
d[k] = d.get(k, 0) + 1
|
|
|
|
assert sorted(d.items()) == [('i', 4), ('m', 1), ('p', 2), ('s', 4)]
|
|
|
|
|
|
|
|
s = [('yellow', 1), ('blue', 2), ('yellow', 3), ('blue', 4), ('red', 1)]
|
|
|
|
d = defaultdict(list)
|
|
|
|
for k, v in s:
|
|
|
|
d[k].append(v)
|
|
|
|
assert sorted(d.items()) == [('blue', [2, 4]), ('red', [1]), ('yellow', [1, 3])]
|
|
|
|
|
|
|
|
def constant_factory(value):
|
|
|
|
return lambda: value
|
|
|
|
|
|
|
|
d = defaultdict(constant_factory('<missing>'))
|
|
|
|
assert d[10] == '<missing>'
|
|
|
|
test_defaultdict()
|