Source code for pennylane.measurements.shots
# Copyright 2018-2023 Xanadu Quantum Technologies Inc.
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
# http://www.apache.org/licenses/LICENSE-2.0
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
"""This module contains the Shots class to hold shot-related information."""
from collections.abc import Sequence
# pylint:disable=inconsistent-return-statements
from typing import NamedTuple, Union
[docs]class ShotCopies(NamedTuple):
"""A namedtuple that represents a shot quantity being repeated some number of times.
For example, ``ShotCopies(10 shots x 2)`` indicates two executions with 10 shots each for 20 shots total.
"""
shots: int
copies: int
def __str__(self):
"""The string representation of the class"""
return f"{self.shots} shots{' x '+str(self.copies) if self.copies > 1 else ''}"
def __repr__(self):
"""The representation of the class"""
return f"ShotCopies({self.shots} shots x {self.copies})"
def valid_int(s):
"""Returns True if s is a positive integer."""
return isinstance(s, int) and s > 0
def valid_tuple(s):
"""Returns True if s is a tuple of the form (shots, copies)."""
return isinstance(s, Sequence) and len(s) == 2 and valid_int(s[0]) and valid_int(s[1])
[docs]class Shots:
"""
A data class that stores shot information.
Args:
shots (Union[None, int, Sequence[int, Tuple[int, int]]]): Raw shot information
Defining shots enables users to specify circuit executions, and the Shots class standardizes
the internal representation of shots. There are three ways to specify shot values:
* The value ``None``
* A positive integer
* A sequence consisting of either positive integers or a tuple-pair of positive integers of the form ``(shots, copies)``
The tuple-pair of the form ``(shots, copies)`` is represented internally by a NamedTuple called
:class:`~ShotCopies`. The first value is the number of shots to execute, and the second value is the
number of times to repeat a circuit with that number of shots.
The ``Shots`` class exposes two properties:
* ``total_shots``, the total number of shots to be executed
* ``shot_vector``, the tuple of :class:`~ShotCopies` to be executed
Instances of this class are static. If an instance is passed to the constructor, that same
instance is returned. If an instance is constructed with a ``None`` value, ``total_shots``
will be ``None``. This indicates analytic execution. A ``Shots`` object created with a
``None`` value is Falsy, while any other value results in a Truthy object:
>>> bool(Shots(None)), bool(Shots(1))
(False, True)
**Examples**
Example constructing a Shots instance with ``None``:
>>> shots = Shots(None)
>>> shots.total_shots, shots.shot_vector
(None, ())
Example constructing a Shots instance with an int:
>>> shots = Shots(100)
>>> shots.total_shots, shots.shot_vector
(100, (ShotCopies(100 shots),))
Example constructing a Shots instance with another instance:
>>> shots = Shots(100)
>>> Shots(shots) is shots
True
Example constructing a Shots instance with a sequence of ints:
>>> shots = Shots([100, 200])
>>> shots.total_shots, shots.shot_vector
(300, (ShotCopies(100 shots x 1), ShotCopies(200 shots x 1)))
Example constructing a Shots instance with a sequence of tuple-pairs:
>>> shots = Shots(((100, 3), (200, 4),))
>>> shots.total_shots, shots.shot_vector
(1100, (ShotCopies(100 shots x 3), ShotCopies(200 shots x 4)))
Example constructing a Shots instance with a sequence of both ints and tuple-pairs.
Note that the first stand-alone ``100`` gets absorbed into the subsequent tuple because the
shot value matches:
>>> shots = Shots((10, 100, (100, 3), (200, 4),))
>>> shots.total_shots, shots.shot_vector
(1210, (ShotCopies(10 shots x 1), ShotCopies(100 shots x 4), ShotCopies(200 shots x 4)))
Example constructing a Shots instance by multiplying an existing one by an int or float:
>>> Shots(100) * 2
Shots(total_shots=200, shot_vector=(ShotCopies(200 shots x 1),))
>>> Shots([7, (100, 2)]) * 1.5
Shots(total_shots=310, shot_vector=(ShotCopies(10 shots x 1), ShotCopies(150 shots x 2)))
Example constructing a Shots instance by adding two existing instances together:
>>> Shots(100) + Shots(((10,2),))
Shots(total_shots=120, shot_vector=(ShotCopies(100 shots x 1), ShotCopies(10 shots x 2)))
One should also note that specifying a single tuple of length 2 is considered two different
shot values, and *not* a tuple-pair representing shots and copies to avoid special behaviour
depending on the iterable type:
>>> shots = Shots((100, 2))
>>> shots.total_shots, shots.shot_vector
(102, (ShotCopies(100 shots x 1), ShotCopies(2 shots x 1)))
>>> shots = Shots(((100, 2),))
>>> shots.total_shots, shots.shot_vector
(200, (ShotCopies(100 shots x 2),))
"""
total_shots: int = None
"""The total number of shots to be executed."""
shot_vector: tuple[ShotCopies] = None
"""The tuple of :class:`~ShotCopies` to be executed. Each element is of the form ``(shots, copies)``."""
_SHOT_ERROR = ValueError(
"Shots must be a single positive integer, a tuple pair of the form (shots, copies), or a sequence of these types."
)
_frozen = False
def __new__(cls, shots=None):
return shots if isinstance(shots, cls) else object.__new__(cls)
def __setattr__(self, name, value):
if self._frozen:
raise AttributeError(
"Shots is an immutable class. Consider creating a new instance if you need different shot values."
)
return super().__setattr__(name, value)
def __init__(self, shots=None):
if shots is None:
self.total_shots = None
self.shot_vector = ()
elif isinstance(shots, int):
if shots < 1:
raise self._SHOT_ERROR
self.total_shots = shots
self.shot_vector = (ShotCopies(shots, 1),)
elif isinstance(shots, Sequence):
if not all(valid_int(s) or valid_tuple(s) for s in shots):
raise self._SHOT_ERROR
self.__all_tuple_init__([s if isinstance(s, Sequence) else (s, 1) for s in shots])
elif isinstance(shots, self.__class__):
return # self already _is_ shots as defined by __new__
else:
raise self._SHOT_ERROR
self._frozen = True
def __copy__(self):
return self
def __deepcopy__(self, memo):
memo[id(self)] = self
return self
def __str__(self):
"""The string representation of the class"""
if not self.has_partitioned_shots:
return f"Shots(total={self.total_shots})"
shot_copy_str = ", ".join([str(sc) for sc in self.shot_vector]) or None
return f"Shots(total={self.total_shots}, vector=[{shot_copy_str}])"
def __repr__(self):
"""The representation of the class"""
return f"Shots(total_shots={self.total_shots}, shot_vector={self.shot_vector})"
def __eq__(self, other):
"""Equality between Shot instances."""
return (
isinstance(other, Shots)
and self.total_shots == other.total_shots
and self.shot_vector == other.shot_vector
)
def __hash__(self):
"""Hash for a given Shot instance."""
return hash(self.shot_vector)
def __iter__(self):
for shot_copy in self.shot_vector:
for _ in range(shot_copy.copies):
yield shot_copy.shots
def __all_tuple_init__(self, shots: Sequence[tuple]):
res = []
total_shots = 0
current_shots, current_copies = shots[0]
for s in shots[1:]:
if s[0] == current_shots:
current_copies += s[1]
else:
res.append(ShotCopies(current_shots, current_copies))
total_shots += current_shots * current_copies
current_shots, current_copies = s
self.shot_vector = tuple(res + [ShotCopies(current_shots, current_copies)])
self.total_shots = total_shots + current_shots * current_copies
def __bool__(self):
return self.total_shots is not None
def __add__(self, other):
return add_shots(self, other)
def __mul__(self, scalar):
if not isinstance(scalar, (int, float)):
raise TypeError("Can't multiply Shots with non-integer or float type.")
if self.total_shots is None:
return self
scaled_shot_vector = tuple(
ShotCopies(int(i.shots * scalar), i.copies) for i in self.shot_vector
)
return self.__class__(scaled_shot_vector)
def __rmul__(self, scalar):
return self.__mul__(scalar)
@property
def has_partitioned_shots(self):
"""
Evaluates to True if this instance represents either multiple shot
quantities, or the same shot quantity repeated multiple times.
Returns:
bool: whether shots are partitioned
"""
if not self:
return False
return len(self.shot_vector) > 1 or self.shot_vector[0].copies > 1
@property
def num_copies(self):
"""The total number of copies of any shot quantity."""
return sum(s.copies for s in self.shot_vector)
[docs] def bins(self):
"""
Yields:
tuple: A tuple containing the lower and upper bounds for each shot quantity in shot_vector.
Example:
>>> shots = Shots((1, 1, 2, 3))
>>> list(shots.bins())
[(0,1), (1,2), (2,4), (4,7)]
"""
lower_bound = 0
for sc in self.shot_vector:
for _ in range(sc.copies):
upper_bound = lower_bound + sc.shots
yield lower_bound, upper_bound
lower_bound = upper_bound
ShotsLike = Union[Shots, None, int, Sequence[Union[int, tuple[int, int]]]]
[docs]def add_shots(s1: Shots, s2: Shots) -> Shots:
"""Add two :class:`~.Shots` objects by concatenating their shot vectors.
Args:
s1 (Shots): a Shots object to add
s2 (Shots): a Shots object to add
Returns:
Shots: a :class:`~.Shots` object built by concatenating the shot vectors of ``s1`` and ``s2``
Example:
>>> s1 = Shots((5, (10, 2)))
>>> s2 = Shots((3, 2, (10, 3)))
>>> print(qml.measurements.add_shots(s1, s2))
Shots(total=60, vector=[5 shots, 10 shots x 2, 3 shots, 2 shots, 10 shots x 3])
"""
if s1.total_shots is None:
return s2
if s2.total_shots is None:
return s1
shot_vector = []
for shot in s1.shot_vector + s2.shot_vector:
shot_vector.append((shot.shots, shot.copies))
return Shots(shots=shot_vector)
_modules/pennylane/measurements/shots
Download Python script
Download Notebook
View on GitHub