@@ -143,6 +143,30 @@ class AbstractCircuit(abc.ABC):
143
143
* get_independent_qubit_sets
144
144
"""
145
145
146
+ @classmethod
147
+ def from_moments (cls : Type [CIRCUIT_TYPE ], * moments : 'cirq.OP_TREE' ) -> CIRCUIT_TYPE :
148
+ """Create a circuit from moment op trees.
149
+
150
+ Args:
151
+ *moments: Op tree for each moment.
152
+ """
153
+ return cls ._from_moments (
154
+ moment if isinstance (moment , Moment ) else Moment (moment ) for moment in moments
155
+ )
156
+
157
+ @classmethod
158
+ @abc .abstractmethod
159
+ def _from_moments (cls : Type [CIRCUIT_TYPE ], moments : Iterable ['cirq.Moment' ]) -> CIRCUIT_TYPE :
160
+ """Create a circuit from moments.
161
+
162
+ This must be implemented by subclasses. It provides a more efficient way
163
+ to construct a circuit instance since we already have the moments and so
164
+ can skip the analysis required to implement various insert strategies.
165
+
166
+ Args:
167
+ moments: Moments of the circuit.
168
+ """
169
+
146
170
@property
147
171
@abc .abstractmethod
148
172
def moments (self ) -> Sequence ['cirq.Moment' ]:
@@ -225,8 +249,7 @@ def __getitem__(self: CIRCUIT_TYPE, key: Tuple[slice, Iterable['cirq.Qid']]) ->
225
249
226
250
def __getitem__ (self , key ):
227
251
if isinstance (key , slice ):
228
- sliced_moments = self .moments [key ]
229
- return self ._with_sliced_moments (sliced_moments )
252
+ return self ._from_moments (self .moments [key ])
230
253
if hasattr (key , '__index__' ):
231
254
return self .moments [key ]
232
255
if isinstance (key , tuple ):
@@ -239,17 +262,12 @@ def __getitem__(self, key):
239
262
return selected_moments [qubit_idx ]
240
263
if isinstance (qubit_idx , ops .Qid ):
241
264
qubit_idx = [qubit_idx ]
242
- sliced_moments = [moment [qubit_idx ] for moment in selected_moments ]
243
- return self ._with_sliced_moments (sliced_moments )
265
+ return self ._from_moments (moment [qubit_idx ] for moment in selected_moments )
244
266
245
267
raise TypeError ('__getitem__ called with key not of type slice, int, or tuple.' )
246
268
247
269
# pylint: enable=function-redefined
248
270
249
- @abc .abstractmethod
250
- def _with_sliced_moments (self : CIRCUIT_TYPE , moments : Iterable ['cirq.Moment' ]) -> CIRCUIT_TYPE :
251
- """Helper method for constructing circuits from __getitem__."""
252
-
253
271
def __str__ (self ) -> str :
254
272
return self .to_text_diagram ()
255
273
@@ -909,7 +927,7 @@ def map_moment(moment: 'cirq.Moment') -> 'cirq.Circuit':
909
927
"""Apply func to expand each op into a circuit, then zip up the circuits."""
910
928
return Circuit .zip (* [Circuit (func (op )) for op in moment ])
911
929
912
- return self ._with_sliced_moments (m for moment in self for m in map_moment (moment ))
930
+ return self ._from_moments (m for moment in self for m in map_moment (moment ))
913
931
914
932
def qid_shape (
915
933
self , qubit_order : 'cirq.QubitOrderOrList' = ops .QubitOrder .DEFAULT
@@ -949,18 +967,16 @@ def _measurement_key_names_(self) -> FrozenSet[str]:
949
967
return self .all_measurement_key_names ()
950
968
951
969
def _with_measurement_key_mapping_ (self , key_map : Mapping [str , str ]):
952
- return self ._with_sliced_moments (
953
- [ protocols .with_measurement_key_mapping (moment , key_map ) for moment in self .moments ]
970
+ return self ._from_moments (
971
+ protocols .with_measurement_key_mapping (moment , key_map ) for moment in self .moments
954
972
)
955
973
956
974
def _with_key_path_ (self , path : Tuple [str , ...]):
957
- return self ._with_sliced_moments (
958
- [protocols .with_key_path (moment , path ) for moment in self .moments ]
959
- )
975
+ return self ._from_moments (protocols .with_key_path (moment , path ) for moment in self .moments )
960
976
961
977
def _with_key_path_prefix_ (self , prefix : Tuple [str , ...]):
962
- return self ._with_sliced_moments (
963
- [ protocols .with_key_path_prefix (moment , prefix ) for moment in self .moments ]
978
+ return self ._from_moments (
979
+ protocols .with_key_path_prefix (moment , prefix ) for moment in self .moments
964
980
)
965
981
966
982
def _with_rescoped_keys_ (
@@ -971,7 +987,7 @@ def _with_rescoped_keys_(
971
987
new_moment = protocols .with_rescoped_keys (moment , path , bindable_keys )
972
988
moments .append (new_moment )
973
989
bindable_keys |= protocols .measurement_key_objs (new_moment )
974
- return self ._with_sliced_moments (moments )
990
+ return self ._from_moments (moments )
975
991
976
992
def _qid_shape_ (self ) -> Tuple [int , ...]:
977
993
return self .qid_shape ()
@@ -1552,9 +1568,7 @@ def factorize(self: CIRCUIT_TYPE) -> Iterable[CIRCUIT_TYPE]:
1552
1568
# the qubits from one factor belong to a specific independent qubit set.
1553
1569
# This makes it possible to create independent circuits based on these
1554
1570
# moments.
1555
- return (
1556
- self ._with_sliced_moments ([m [qubits ] for m in self .moments ]) for qubits in qubit_factors
1557
- )
1571
+ return (self ._from_moments (m [qubits ] for m in self .moments ) for qubits in qubit_factors )
1558
1572
1559
1573
def _control_keys_ (self ) -> FrozenSet ['cirq.MeasurementKey' ]:
1560
1574
controls = frozenset (k for op in self .all_operations () for k in protocols .control_keys (op ))
@@ -1719,6 +1733,12 @@ def __init__(
1719
1733
else :
1720
1734
self .append (contents , strategy = strategy )
1721
1735
1736
+ @classmethod
1737
+ def _from_moments (cls , moments : Iterable ['cirq.Moment' ]) -> 'Circuit' :
1738
+ new_circuit = Circuit ()
1739
+ new_circuit ._moments [:] = moments
1740
+ return new_circuit
1741
+
1722
1742
def _load_contents_with_earliest_strategy (self , contents : 'cirq.OP_TREE' ):
1723
1743
"""Optimized algorithm to load contents quickly.
1724
1744
@@ -1813,11 +1833,6 @@ def copy(self) -> 'Circuit':
1813
1833
copied_circuit ._moments = self ._moments [:]
1814
1834
return copied_circuit
1815
1835
1816
- def _with_sliced_moments (self , moments : Iterable ['cirq.Moment' ]) -> 'Circuit' :
1817
- new_circuit = Circuit ()
1818
- new_circuit ._moments = list (moments )
1819
- return new_circuit
1820
-
1821
1836
# pylint: disable=function-redefined
1822
1837
@overload
1823
1838
def __setitem__ (self , key : int , value : 'cirq.Moment' ):
0 commit comments