Source code for qiskit.circuit.parameter

# This code is part of Qiskit.
#
# (C) Copyright IBM 2017, 2019.
#
# 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.
"""
Parameter Class for variable parameters.
"""

from __future__ import annotations

from uuid import uuid4, UUID

from qiskit.circuit.exceptions import CircuitError
from qiskit.utils import optionals as _optionals

from .parameterexpression import ParameterExpression


[docs]class Parameter(ParameterExpression): """Parameter Class for variable parameters. A parameter is a variable value that is not required to be fixed at circuit definition. Examples: Construct a variable-rotation X gate using circuit parameters. .. plot:: :include-source: from qiskit.circuit import QuantumCircuit, Parameter # create the parameter phi = Parameter('phi') qc = QuantumCircuit(1) # parameterize the rotation qc.rx(phi, 0) qc.draw('mpl') # bind the parameters after circuit to create a bound circuit bc = qc.assign_parameters({phi: 3.14}) bc.measure_all() bc.draw('mpl') """ __slots__ = ("_name", "_uuid", "_hash") # This `__init__` does not call the super init, because we can't construct the # `_parameter_symbols` dictionary we need to pass to it before we're entirely initialised # anyway, because `ParameterExpression` depends heavily on the structure of `Parameter`. def __init__( self, name: str, *, uuid: UUID | None = None ): # pylint: disable=super-init-not-called """Create a new named :class:`Parameter`. Args: name: name of the ``Parameter``, used for visual representation. This can be any unicode string, e.g. "ϕ". uuid: For advanced usage only. Override the UUID of this parameter, in order to make it compare equal to some other parameter object. By default, two parameters with the same name do not compare equal to help catch shadowing bugs when two circuits containing the same named parameters are spurious combined. Setting the ``uuid`` field when creating two parameters to the same thing (along with the same name) allows them to be equal. This is useful during serialization and deserialization. """ self._name = name self._uuid = uuid4() if uuid is None else uuid if not _optionals.HAS_SYMENGINE: from sympy import Symbol symbol = Symbol(name) else: import symengine symbol = symengine.Symbol(name) self._symbol_expr = symbol self._parameter_keys = frozenset((self._hash_key(),)) self._hash = hash((self._parameter_keys, self._symbol_expr)) self._parameter_symbols = {self: symbol} self._name_map = None
[docs] def assign(self, parameter, value): if parameter != self: # Corresponds to superclass calls to `subs` and `bind` that would implicitly set # `allow_unknown_parameters=False`. raise CircuitError( f"Cannot bind Parameters ({[str(parameter)]}) not present in expression." ) if isinstance(value, ParameterExpression): # This is the `super().subs` case. return value # This is the `super().bind` case, where we're required to return a `ParameterExpression`, # so we need to lift the given value to a symbolic expression. if _optionals.HAS_SYMENGINE: from symengine import sympify else: from sympy import sympify return ParameterExpression({}, sympify(value))
[docs] def subs(self, parameter_map: dict, allow_unknown_parameters: bool = False): """Substitute self with the corresponding parameter in ``parameter_map``.""" if self in parameter_map: return parameter_map[self] if allow_unknown_parameters: return self raise CircuitError( "Cannot bind Parameters ({}) not present in " "expression.".format([str(p) for p in parameter_map]) )
@property def name(self): """Returns the name of the :class:`Parameter`.""" return self._name def __str__(self): return self.name def __copy__(self): return self def __deepcopy__(self, memo=None): return self def __repr__(self): return f"{self.__class__.__name__}({self.name})" def __eq__(self, other): if isinstance(other, Parameter): return self._uuid == other._uuid elif isinstance(other, ParameterExpression): return super().__eq__(other) else: return False def _hash_key(self): # `ParameterExpression` needs to be able to hash all its contained `Parameter` instances in # its hash as part of the equality comparison but has its own more complete symbolic # expression, so its full hash key is split into `(parameter_keys, symbolic_expression)`. # This method lets containing expressions get only the bits they need for equality checks in # the first value, without wasting time re-hashing individual Sympy/Symengine symbols. return (self._name, self._uuid) def __hash__(self): # This is precached for performance, since it's used a lot and we are immutable. return self._hash # We have to manually control the pickling so that the hash is computable before the unpickling # operation attempts to put this parameter into a hashmap. def __getstate__(self): return (self._name, self._uuid, self._symbol_expr) def __setstate__(self, state): self._name, self._uuid, self._symbol_expr = state self._parameter_keys = frozenset((self._hash_key(),)) self._hash = hash((self._parameter_keys, self._symbol_expr)) self._parameter_symbols = {self: self._symbol_expr} self._name_map = None