English
Languages
English
Japanese
German
Korean
Portuguese, Brazilian
French
Shortcuts

Source code for qiskit.aqua.operators.converters.abelian_grouper

# This code is part of Qiskit.
#
# (C) Copyright IBM 2020.
#
# This code is licensed under the Apache License, Version 2.0. You may
# obtain a copy of this license in the LICENSE.txt file in the root directory
# of this source tree or at http://www.apache.org/licenses/LICENSE-2.0.
#
# Any modifications or derivative works of this code must retain this
# copyright notice, and modified files need to carry a notice indicating
# that they have been altered from the originals.

"""AbelianGrouper Class"""

import warnings
from typing import List, Tuple, Dict, cast, Optional

import numpy as np
import retworkx as rx

from qiskit.aqua import AquaError
from .converter_base import ConverterBase
from ..list_ops.list_op import ListOp
from ..list_ops.summed_op import SummedOp
from ..operator_base import OperatorBase
from ..primitive_ops.pauli_op import PauliOp
from ..state_fns.operator_state_fn import OperatorStateFn


[docs]class AbelianGrouper(ConverterBase): """The AbelianGrouper converts SummedOps into a sum of Abelian sums. Meaning, it will traverse the Operator, and when it finds a SummedOp, it will evaluate which of the summed sub-Operators commute with one another. It will then convert each of the groups of commuting Operators into their own SummedOps, and return the sum-of-commuting-SummedOps. This is particularly useful for cases where mutually commuting groups can be handled similarly, as in the case of Pauli Expectations, where commuting Paulis have the same diagonalizing circuit rotation, or Pauli Evolutions, where commuting Paulis can be diagonalized together. """
[docs] def __init__(self, traverse: bool = True) -> None: """ Args: traverse: Whether to convert only the Operator passed to ``convert``, or traverse down that Operator. """ self._traverse = traverse
[docs] def convert(self, operator: OperatorBase) -> OperatorBase: """Check if operator is a SummedOp, in which case covert it into a sum of mutually commuting sums, or if the Operator contains sub-Operators and ``traverse`` is True, attempt to convert any sub-Operators. Args: operator: The Operator to attempt to convert. Returns: The converted Operator. """ # pylint: disable=cyclic-import,import-outside-toplevel from ..evolutions.evolved_op import EvolvedOp if isinstance(operator, ListOp): if isinstance(operator, SummedOp) and all(isinstance(op, PauliOp) for op in operator.oplist): # For now, we only support graphs over Paulis. return self.group_subops(operator) elif self._traverse: return operator.traverse(self.convert) else: return operator elif isinstance(operator, OperatorStateFn) and self._traverse: return OperatorStateFn(self.convert(operator.primitive), is_measurement=operator.is_measurement, coeff=operator.coeff) elif isinstance(operator, EvolvedOp) and self._traverse: return EvolvedOp(self.convert(operator.primitive), coeff=operator.coeff) # type: ignore else: return operator
[docs] @classmethod def group_subops(cls, list_op: ListOp, fast: Optional[bool] = None, use_nx: Optional[bool] = None) -> ListOp: """Given a ListOp, attempt to group into Abelian ListOps of the same type. Args: list_op: The Operator to group into Abelian groups fast: Ignored - parameter will be removed in future release use_nx: Ignored - parameter will be removed in future release Returns: The grouped Operator. Raises: AquaError: If any of list_op's sub-ops is not ``PauliOp``. """ if fast is not None or use_nx is not None: warnings.warn('Options `fast` and `use_nx` of `AbelianGrouper.group_subops` are ' 'no longer used and are now deprecated and will be removed no ' 'sooner than 3 months following the 0.8.0 release.') for op in list_op.oplist: if not isinstance(op, PauliOp): raise AquaError( 'Cannot determine Abelian groups if any Operator in list_op is not ' '`PauliOp`. E.g., {} ({})'.format(op, type(op))) edges = cls._commutation_graph(list_op) nodes = range(len(list_op)) graph = rx.PyGraph() graph.add_nodes_from(nodes) graph.add_edges_from_no_data(edges) # Keys in coloring_dict are nodes, values are colors coloring_dict = rx.graph_greedy_color(graph) groups = {} # type: Dict # sort items so that the output is consistent with all options (fast and use_nx) for idx, color in sorted(coloring_dict.items()): groups.setdefault(color, []).append(list_op[idx]) group_ops = [list_op.__class__(group, abelian=True) for group in groups.values()] if len(group_ops) == 1: return group_ops[0] * list_op.coeff # type: ignore return list_op.__class__(group_ops, coeff=list_op.coeff) # type: ignore
@staticmethod def _commutation_graph(list_op: ListOp) -> List[Tuple[int, int]]: """Create edges (i, j) if i and j are not commutable. Note: This method is applicable to only PauliOps. Args: list_op: list_op Returns: A list of pairs of indices of the operators that are not commutable """ # convert a Pauli operator into int vector where {I: 0, X: 2, Y: 3, Z: 1} mat1 = np.array([op.primitive.z + 2 * op.primitive.x for op in list_op], dtype=np.int8) mat2 = mat1[:, None] # mat3[i, j] is True if i and j are commutable with TPB mat3 = (((mat1 * mat2) * (mat1 - mat2)) == 0).all(axis=2) # return [(i, j) if mat3[i, j] is False and i < j] return cast(List[Tuple[int, int]], list(zip(*np.where(np.triu(np.logical_not(mat3), k=1)))))

© Copyright 2020, Qiskit Development Team. Last updated on 2021/05/25.

Built with Sphinx using a theme provided by Read the Docs.