From f03c84dea0dc0d170d793ce010620784a8dfc347 Mon Sep 17 00:00:00 2001 From: Ricardo Olenewa Date: Wed, 13 Oct 2021 14:59:05 -0400 Subject: [PATCH 1/3] Removes sections related to experimental features. --- docs/noise.ipynb | 752 +++++++++++++++++++---------------------------- 1 file changed, 308 insertions(+), 444 deletions(-) diff --git a/docs/noise.ipynb b/docs/noise.ipynb index 68423929ec5..da17da1e843 100644 --- a/docs/noise.ipynb +++ b/docs/noise.ipynb @@ -2,21 +2,16 @@ "cells": [ { "cell_type": "markdown", - "metadata": { - "id": "LnV7yIFGeryz" - }, "source": [ "##### Copyright 2020 The Cirq Developers" - ] + ], + "metadata": { + "id": "LnV7yIFGeryz" + } }, { "cell_type": "code", "execution_count": null, - "metadata": { - "cellView": "form", - "id": "1kkohf-9esQm" - }, - "outputs": [], "source": [ "#@title Licensed under the Apache License, Version 2.0 (the \"License\");\n", "# you may not use this file except in compliance with the License.\n", @@ -29,22 +24,24 @@ "# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n", "# See the License for the specific language governing permissions and\n", "# limitations under the License." - ] + ], + "outputs": [], + "metadata": { + "cellView": "form", + "id": "1kkohf-9esQm" + } }, { "cell_type": "markdown", - "metadata": { - "id": "7zgataJVe0mU" - }, "source": [ "# Noise" - ] + ], + "metadata": { + "id": "7zgataJVe0mU" + } }, { "cell_type": "markdown", - "metadata": { - "id": "HYkRhx2pe2XX" - }, "source": [ "\n", " \n", "
\n", @@ -60,24 +57,23 @@ " Download notebook\n", "
" - ] + ], + "metadata": { + "id": "HYkRhx2pe2XX" + } }, { "cell_type": "markdown", - "metadata": { - "id": "ea381f53cf89" - }, "source": [ "Note: this notebook relies on unreleased Cirq features. If you want to try these features, make sure you install cirq via `pip install cirq --pre`." - ] + ], + "metadata": { + "id": "ea381f53cf89" + } }, { "cell_type": "code", "execution_count": null, - "metadata": { - "id": "846b32703c5c" - }, - "outputs": [], "source": [ "try:\n", " import cirq\n", @@ -85,13 +81,14 @@ " print(\"installing cirq...\")\n", " !pip install --quiet cirq --pre\n", " print(\"installed cirq.\")" - ] + ], + "outputs": [], + "metadata": { + "id": "846b32703c5c" + } }, { "cell_type": "markdown", - "metadata": { - "id": "7d3daa944d6b" - }, "source": [ "For simulation, it is useful to have `Gate` objects that enact noisy quantum evolution. Cirq supports modeling noise via *operator sum* representations of noise (these evolutions are also known as quantum operations or quantum dynamical maps). \n", "\n", @@ -108,22 +105,22 @@ "$$\n", "\n", "A channel with $m = 1$ unitary Kraus operator is called *coherent* (and is equivalent to a unitary gate operation), otherwise the channel is called *incoherent*. For a given noisy channel, Kraus operators are not necessarily unique. For more details on these operators, see [John Preskill's lecture notes](http://theory.caltech.edu/~preskill/ph219/chap3_15.pdf)." - ] + ], + "metadata": { + "id": "7d3daa944d6b" + } }, { "cell_type": "markdown", - "metadata": { - "id": "b989869680d4" - }, "source": [ "## Common channels" - ] + ], + "metadata": { + "id": "b989869680d4" + } }, { "cell_type": "markdown", - "metadata": { - "id": "708b5b720a74" - }, "source": [ "Cirq defines many commonly used quantum channels in [`ops/common_channels.py`](https://github.com/quantumlib/Cirq/blob/master/cirq/ops/common_channels.py). For example, the single-qubit bit-flip channel\n", "\n", @@ -132,59 +129,58 @@ "$$\n", "\n", "with parameter $p = 0.1$ can be created as follows." - ] + ], + "metadata": { + "id": "708b5b720a74" + } }, { "cell_type": "code", "execution_count": null, - "metadata": { - "id": "386a49be9dc7" - }, - "outputs": [], "source": [ "import cirq\n", "\n", "\"\"\"Get a single-qubit bit-flip channel.\"\"\"\n", "bit_flip = cirq.bit_flip(p=0.1)" - ] + ], + "outputs": [], + "metadata": { + "id": "386a49be9dc7" + } }, { "cell_type": "markdown", - "metadata": { - "id": "f1f501177e53" - }, "source": [ "To see the Kraus operators of a channel, the `cirq.channel` protocol can be used. (See the [protocols guide](./protocols.ipynb).)" - ] + ], + "metadata": { + "id": "f1f501177e53" + } }, { "cell_type": "code", "execution_count": null, - "metadata": { - "id": "7635588faafe" - }, - "outputs": [], "source": [ "for i, kraus in enumerate(cirq.kraus(bit_flip)):\n", " print(f\"Kraus operator {i + 1} is:\\n\", kraus, end=\"\\n\\n\")" - ] + ], + "outputs": [], + "metadata": { + "id": "7635588faafe" + } }, { "cell_type": "markdown", - "metadata": { - "id": "3f9ff40f980f" - }, "source": [ "As mentioned, all channels are subclasses of `cirq.Gate`s. As such, they can act on qubits and be used in circuits in the same manner as gates." - ] + ], + "metadata": { + "id": "3f9ff40f980f" + } }, { "cell_type": "code", "execution_count": null, - "metadata": { - "id": "91187990b395" - }, - "outputs": [], "source": [ "\"\"\"Example of using channels in a circuit.\"\"\"\n", "# See the number of qubits a channel acts on.\n", @@ -196,24 +192,24 @@ " bit_flip.on_each(cirq.LineQubit.range(3))\n", ")\n", "print(circuit)" - ] + ], + "outputs": [], + "metadata": { + "id": "91187990b395" + } }, { "cell_type": "markdown", - "metadata": { - "id": "59651802c358" - }, "source": [ "Channels can even be controlled." - ] + ], + "metadata": { + "id": "59651802c358" + } }, { "cell_type": "code", "execution_count": null, - "metadata": { - "id": "9be5383a7cd7" - }, - "outputs": [], "source": [ "\"\"\"Example of controlling a channel.\"\"\"\n", "# Get the controlled channel.\n", @@ -224,22 +220,23 @@ " controlled_bit_flip(*cirq.LineQubit.range(2))\n", ")\n", "print(circuit)" - ] + ], + "outputs": [], + "metadata": { + "id": "9be5383a7cd7" + } }, { "cell_type": "markdown", - "metadata": { - "id": "755ba122d550" - }, "source": [ "In addition to the bit-flip channel, other common channels predefined in Cirq are shown below. Definitions of these channels can be found in their docstrings - e.g., `help(cirq.depolarize)`." - ] + ], + "metadata": { + "id": "755ba122d550" + } }, { "cell_type": "markdown", - "metadata": { - "id": "6a9c9c76fb8f" - }, "source": [ "* `cirq.phase_flip`\n", "* `cirq.phase_damp`\n", @@ -247,13 +244,13 @@ "* `cirq.depolarize`\n", "* `cirq.asymmetric_depolarize`\n", "* `cirq.reset`" - ] + ], + "metadata": { + "id": "6a9c9c76fb8f" + } }, { "cell_type": "markdown", - "metadata": { - "id": "8b22bb323731" - }, "source": [ "For example, the asymmetric depolarizing channel is defined by\n", "\n", @@ -262,15 +259,14 @@ "$$\n", "\n", "and can be instantiated as follows." - ] + ], + "metadata": { + "id": "8b22bb323731" + } }, { "cell_type": "code", "execution_count": null, - "metadata": { - "id": "b6a0966a29f6" - }, - "outputs": [], "source": [ "\"\"\"Get an asymmetric depolarizing channel.\"\"\"\n", "depo = cirq.asymmetric_depolarize(\n", @@ -283,22 +279,23 @@ " depo.on_each(cirq.LineQubit(0))\n", ")\n", "print(circuit)" - ] + ], + "outputs": [], + "metadata": { + "id": "b6a0966a29f6" + } }, { "cell_type": "markdown", - "metadata": { - "id": "ddbc622c98da" - }, "source": [ "## The `channel` and `mixture` protocols" - ] + ], + "metadata": { + "id": "ddbc622c98da" + } }, { "cell_type": "markdown", - "metadata": { - "id": "2dee355d2ae8" - }, "source": [ "We have seen the `cirq.channel` protocol which returns the Kraus operators of a channel. Some channels have the interpretation of randomly applying a single unitary Kraus operator $U_k$ with probability $p_k$, namely\n", "\n", @@ -313,26 +310,26 @@ "$$\n", "\n", "can be interpreted as doing nothing (applying identity) with probability $1 - p$ and flipping the bit (applying $X$) with probability $p$. Channels with these interpretations support the `cirq.mixture` protocol. This protocol returns the probabilities and unitary Kraus operators of the channel." - ] + ], + "metadata": { + "id": "2dee355d2ae8" + } }, { "cell_type": "code", "execution_count": null, - "metadata": { - "id": "d4d84c4d4fa7" - }, - "outputs": [], "source": [ "\"\"\"Example of using the mixture protocol.\"\"\"\n", "for prob, kraus in cirq.mixture(bit_flip):\n", " print(f\"With probability {prob}, apply\\n\", kraus, end=\"\\n\\n\")" - ] + ], + "outputs": [], + "metadata": { + "id": "d4d84c4d4fa7" + } }, { "cell_type": "markdown", - "metadata": { - "id": "1ca339c3cbab" - }, "source": [ "Channels that do not have this interpretation do not support the `cirq.mixture` protocol. Such channels apply Kraus operators with probabilities that depend on the state $\\rho$. \n", "\n", @@ -343,15 +340,14 @@ "\\text{and }\n", "M_1 = \\begin{bmatrix} 0 & \\sqrt{\\gamma} \\cr 0 & 0 \\end{bmatrix} .\n", "$$" - ] + ], + "metadata": { + "id": "1ca339c3cbab" + } }, { "cell_type": "code", "execution_count": null, - "metadata": { - "id": "8e377332d2a7" - }, - "outputs": [], "source": [ "\"\"\"The amplitude damping channel is an example of a channel without a mixture.\"\"\"\n", "channel = cirq.amplitude_damp(0.1)\n", @@ -360,13 +356,14 @@ " print(f\"Channel {channel} has a _mixture_ or _unitary_ method.\")\n", "else:\n", " print(f\"Channel {channel} does not have a _mixture_ or _unitary_ method.\")" - ] + ], + "outputs": [], + "metadata": { + "id": "8e377332d2a7" + } }, { "cell_type": "markdown", - "metadata": { - "id": "6f3ae4337219" - }, "source": [ "To summarize:\n", "\n", @@ -375,13 +372,13 @@ "* A subset of channels which support `cirq.channel` also support the `cirq.mixture` protocol.\n", " - If magic method `_mixture_` is not defined, `cirq.mixture` looks for `_unitary_`.\n", "* A subset of channels which support `cirq.mixture` also support the `cirq.unitary` protocol." - ] + ], + "metadata": { + "id": "6f3ae4337219" + } }, { "cell_type": "markdown", - "metadata": { - "id": "67c8ddbc10f3" - }, "source": [ "For concrete examples, consider `cirq.X`, `cirq.BitFlipChannel`, and `cirq.AmplitudeDampingChannel` which are all subclasses of `cirq.Gate`.\n", "\n", @@ -391,22 +388,22 @@ " - As a result, it only supports the `cirq.mixture` protocol and the `cirq.channel` protocol.\n", "* `cirq.AmplitudeDampingChannel` defines the `_kraus_` method, but not the `_mixture_` method or the `_unitary_` method.\n", " - As a result, it only supports the `cirq.channel` protocol." - ] + ], + "metadata": { + "id": "67c8ddbc10f3" + } }, { "cell_type": "markdown", - "metadata": { - "id": "0f5b77825a8a" - }, "source": [ "## Custom channels" - ] + ], + "metadata": { + "id": "0f5b77825a8a" + } }, { "cell_type": "markdown", - "metadata": { - "id": "7614e0351462" - }, "source": [ "There are two configurable channel types for channels not defined in `cirq.ops.common_channels`: `MixedUnitaryChannel` and `KrausChannel`.\n", "\n", @@ -417,15 +414,14 @@ "Both types also accept a measurement key as an optional parameter. This key will be used to store the index of the selected unitary or Kraus operator in the measurement results.\n", "\n", "An example of each type is shown below." - ] + ], + "metadata": { + "id": "7614e0351462" + } }, { "cell_type": "code", "execution_count": null, - "metadata": { - "id": "7cbd723fb567" - }, - "outputs": [], "source": [ "import numpy as np\n", "q0 = cirq.LineQubit(0)\n", @@ -457,13 +453,14 @@ " cirq.measure(q0, key='x'),\n", " cirq.H(q0),\n", ")" - ] + ], + "outputs": [], + "metadata": { + "id": "7cbd723fb567" + } }, { "cell_type": "markdown", - "metadata": { - "id": "360d5a316769" - }, "source": [ "Alternatively, users can define their own channel types. Defining custom channels is similar to defining [custom gates](./custom_gates.ipynb).\n", "\n", @@ -474,15 +471,14 @@ "$$\n", "\n", "is shown below." - ] + ], + "metadata": { + "id": "360d5a316769" + } }, { "cell_type": "code", "execution_count": null, - "metadata": { - "id": "55240879394e" - }, - "outputs": [], "source": [ "\"\"\"Minimal example of defining a custom channel.\"\"\"\n", "class BitAndPhaseFlipChannel(cirq.SingleQubitGate):\n", @@ -499,304 +495,168 @@ " \n", " def _circuit_diagram_info_(self, args) -> str:\n", " return f\"BitAndPhaseFlip({self._p})\"" - ] + ], + "outputs": [], + "metadata": { + "id": "55240879394e" + } }, { "cell_type": "markdown", - "metadata": { - "id": "8ffff44ff2b2" - }, "source": [ "Note: The `_has_mixture_` magic method is not strictly required but is recommended." - ] + ], + "metadata": { + "id": "8ffff44ff2b2" + } }, { "cell_type": "markdown", - "metadata": { - "id": "2dea6bd62cfe" - }, "source": [ "We can now instantiate this channel and get its mixture:" - ] + ], + "metadata": { + "id": "2dea6bd62cfe" + } }, { "cell_type": "code", "execution_count": null, - "metadata": { - "id": "0350d9e193ca" - }, - "outputs": [], "source": [ "\"\"\"Custom channels can be used like any other channels.\"\"\"\n", "bit_phase_flip = BitAndPhaseFlipChannel(p=0.05)\n", "\n", "for prob, kraus in cirq.mixture(bit_phase_flip):\n", " print(f\"With probability {prob}, apply\\n\", kraus, end=\"\\n\\n\")" - ] + ], + "outputs": [], + "metadata": { + "id": "0350d9e193ca" + } }, { "cell_type": "markdown", - "metadata": { - "id": "72486a155131" - }, "source": [ "Note: Since `_mixture_` is defined, the `cirq.channel` protocol can also be used." - ] + ], + "metadata": { + "id": "72486a155131" + } }, { "cell_type": "markdown", - "metadata": { - "id": "ada775055c3f" - }, "source": [ "The custom channel can be used in a circuit just like other predefined channels." - ] + ], + "metadata": { + "id": "ada775055c3f" + } }, { "cell_type": "code", "execution_count": null, - "metadata": { - "id": "1af5c9f60cab" - }, - "outputs": [], "source": [ "\"\"\"Example of using a custom channel in a circuit.\"\"\"\n", "circuit = cirq.Circuit(\n", " bit_phase_flip.on_each(*cirq.LineQubit.range(3))\n", ")\n", "circuit" - ] + ], + "outputs": [], + "metadata": { + "id": "1af5c9f60cab" + } }, { "cell_type": "markdown", - "metadata": { - "id": "46eb7a01e30d" - }, "source": [ "Note: If a custom channel does not have a mixture, it should instead define the `_kraus_` magic method to return a sequence of Kraus operators (as `numpy.ndarray`s). Defining a `_has_kraus_` method which returns `True` is optional but recommended." - ] + ], + "metadata": { + "id": "46eb7a01e30d" + } }, { "cell_type": "markdown", - "metadata": { - "id": "b537c43e078c" - }, "source": [ "This method of defining custom channels is the most general, but simple channels such as the custom `BitAndPhaseFlipChannel` can also be created directly from a `Gate` with the convenient `Gate.with_probability` method." - ] + ], + "metadata": { + "id": "b537c43e078c" + } }, { "cell_type": "code", "execution_count": null, - "metadata": { - "id": "0937f8dad808" - }, - "outputs": [], "source": [ "\"\"\"Create a channel with Gate.with_probability.\"\"\"\n", "channel = cirq.Y.with_probability(probability=0.05)" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "0461f377ac46" - }, - "source": [ - "This produces the same mixture as the custom `BitAndPhaseFlip` channel above." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "id": "f0a85b33680a" - }, + ], "outputs": [], - "source": [ - "for prob, kraus in cirq.mixture(channel):\n", - " print(f\"With probability {prob}, apply\\n\", kraus, end=\"\\n\\n\")" - ] - }, - { - "cell_type": "markdown", "metadata": { - "id": "c73ec3e63e23" - }, - "source": [ - "Note that the order of Kraus operators is reversed from above, but this of course does not affect the action of the channel." - ] + "id": "0937f8dad808" + } }, { "cell_type": "markdown", - "metadata": { - "id": "c0b3d1faa7ca" - }, "source": [ - "## Noise Properties" - ] - }, - { - "cell_type": "markdown", + "This produces the same mixture as the custom `BitAndPhaseFlip` channel above." + ], "metadata": { - "id": "740b0efc2bbc" - }, - "source": [ - "The `NoiseProperties` object describes the behavior of the noise to be added to a circuit. These properties include p00, p11, Pauli error, depolarization decay constant, T1 constant, and XEB fidelity. For more information on these metrics, see [this paper](https://arxiv.org/pdf/1910.11333.pdf)." - ] + "id": "0461f377ac46" + } }, { "cell_type": "code", "execution_count": null, - "metadata": { - "id": "d451328e34ab" - }, - "outputs": [], - "source": [ - "from cirq.devices.noise_properties import NoiseProperties, NoiseModelFromNoiseProperties\n", - "noise_prop = NoiseProperties(p00 = 0.2, pauli_error = 0.1)" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "289d782f599a" - }, "source": [ - "We can then use the `noise_model_from_noise_properties` method to turn a NoiseProperties object into a NoiseModel, that can then be use with a simulator." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "id": "91b211c9b61f" - }, + "for prob, kraus in cirq.mixture(channel):\n", + " print(f\"With probability {prob}, apply\\n\", kraus, end=\"\\n\\n\")" + ], "outputs": [], - "source": [ - "noise_model = NoiseModelFromNoiseProperties(noise_prop)" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "b9b8ddcbe207" - }, - "source": [ - "This `NoiseModel` object can be applied to a `Simulator` (described below)." - ] - }, - { - "cell_type": "markdown", "metadata": { - "id": "935e20561b7c" - }, - "source": [ - "## Converting calibration metrics to noise" - ] + "id": "f0a85b33680a" + } }, { "cell_type": "markdown", - "metadata": { - "id": "cf14dc6703db" - }, "source": [ - "[Calibration metrics](quantumai.google/cirq/google/calibration) measure the behavior of Google's quantum computers. The `cirq_google.experiments.noise_models.calibration_to_noise_properties` turns a `Calibration` object into a `NoiseProperties` object, that can then be turned into a `NoiseModel`." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "id": "d8d51f3601d5" - }, - "outputs": [], - "source": [ - "import cirq_google\n", - "from google.protobuf.text_format import Merge\n", - "\n", - "t1 = 0.5 # microseconds\n", - "p00 = 0.1\n", - "\n", - "_CALIBRATION_DATA = Merge(\n", - "f\"\"\"\n", - "timestamp_ms: 1579214873,\n", - "metrics: [{{\n", - " name: 'single_qubit_idle_t1_micros',\n", - " targets: ['0_0'],\n", - " values: [{{\n", - " double_val: {t1}\n", - " }}]\n", - "}}, {{\n", - " name: 'single_qubit_p00_error',\n", - " targets: ['0_0'],\n", - " values: [{{\n", - " double_val: {p00}\n", - " }}]\n", - "}}]\n", - "\"\"\",\n", - " cirq_google.api.v2.metrics_pb2.MetricsSnapshot(),\n", - ")\n", - "\n", - "calibration = cirq_google.Calibration(_CALIBRATION_DATA)" - ] - }, - { - "cell_type": "markdown", + "Note that the order of Kraus operators is reversed from above, but this of course does not affect the action of the channel." + ], "metadata": { - "id": "da9d5f3e2c72" - }, - "source": [ - "This Calibration object can then be passed to the `compare_generated_noise_to_metrics` method in `cirq_google` to compare actual metrics with those generated from the noise model created to match the calibration data. Sample output from this method is shown below:" - ] + "id": "c73ec3e63e23" + } }, { "cell_type": "markdown", - "metadata": { - "id": "3d203521ff55" - }, "source": [ - "```\n", - " Metric Initial value Measured value\n", - "0 p00 0.1 0.104000\n", - "1 T1 500.0 575.445484\n", - "```" - ] - }, - { - "cell_type": "markdown", + "## Simulating noisy circuits" + ], "metadata": { "id": "96a696b543f7" - }, - "source": [ - "## Simulating noisy circuits" - ] + } }, { "cell_type": "markdown", - "metadata": { - "id": "d51d993ae869" - }, "source": [ "### Density matrix simulation" - ] + ], + "metadata": { + "id": "d51d993ae869" + } }, { "cell_type": "markdown", - "metadata": { - "id": "38dfcb60ef79" - }, "source": [ "The `cirq.DensityMatrixSimulator` can simulate any noisy circuit (i.e., can apply any quantum channel) because it stores the full density matrix $\\rho$. This simulation strategy updates the state $\\rho$ by directly applying the Kraus operators of each quantum channel." - ] + ], + "metadata": { + "id": "38dfcb60ef79" + } }, { "cell_type": "code", "execution_count": null, - "metadata": { - "id": "a2973a8faef0" - }, - "outputs": [], "source": [ "\"\"\"Simulating a circuit with the density matrix simulator.\"\"\"\n", "# Get a circuit.\n", @@ -817,44 +677,44 @@ "# Display the final density matrix.\n", "print(\"\\nFinal density matrix:\")\n", "print(rho)" - ] + ], + "outputs": [], + "metadata": { + "id": "a2973a8faef0" + } }, { "cell_type": "markdown", - "metadata": { - "id": "ac8a991a426b" - }, "source": [ "Note that the density matrix simulator supports the `run` method which only gives access to measurements as well as the `simulate` method (used above) which gives access to the full density matrix." - ] + ], + "metadata": { + "id": "ac8a991a426b" + } }, { "cell_type": "markdown", - "metadata": { - "id": "0c659bb20098" - }, "source": [ "### Monte Carlo wavefunction simulation" - ] + ], + "metadata": { + "id": "0c659bb20098" + } }, { "cell_type": "markdown", - "metadata": { - "id": "6f3caf9cea92" - }, "source": [ "Noisy circuits with arbitrary channels can also be simulated with the `cirq.Simulator`. When simulating such a channel, a single Kraus operator is randomly sampled (according to the probability distribution) and applied to the wavefunction. This method is known as \"Monte Carlo (wavefunction) simulation\" or \"quantum trajectories.\"\n", "\n", "Note: For channels which do not support the `cirq.mixture` protocol, the probability of applying each Kraus operator depends on the state. In contrast, for channels which do support the `cirq.mixture` protocol, the probability of applying each Kraus operator is independent of the state." - ] + ], + "metadata": { + "id": "6f3caf9cea92" + } }, { "cell_type": "code", "execution_count": null, - "metadata": { - "id": "d80a2bc7ce14" - }, - "outputs": [], "source": [ "\"\"\"Simulating a noisy circuit via Monte Carlo simulation.\"\"\"\n", "# Get a circuit.\n", @@ -872,24 +732,24 @@ "# Display the final wavefunction.\n", "print(\"\\nFinal wavefunction:\")\n", "print(psi)" - ] + ], + "outputs": [], + "metadata": { + "id": "d80a2bc7ce14" + } }, { "cell_type": "markdown", - "metadata": { - "id": "4c485bbc1dfd" - }, "source": [ "To see that the output is stochastic, you can run the cell above multiple times. Since $p = 0.5$ in the bit-flip channel, you should get $|0\\rangle$ roughly half the time and $|1\\rangle$ roughly half the time. The `run` method with many repetitions can also be used to see this behavior." - ] + ], + "metadata": { + "id": "4c485bbc1dfd" + } }, { "cell_type": "code", "execution_count": null, - "metadata": { - "id": "8143ae0c7a34" - }, - "outputs": [], "source": [ "\"\"\"Example of Monte Carlo wavefunction simulation with the `run` method.\"\"\"\n", "circuit = cirq.Circuit(\n", @@ -898,42 +758,42 @@ ")\n", "res = sim.run(circuit, repetitions=100)\n", "print(res.histogram(key=qbit))" - ] + ], + "outputs": [], + "metadata": { + "id": "8143ae0c7a34" + } }, { "cell_type": "markdown", - "metadata": { - "id": "0f123f0e3c55" - }, "source": [ "## Adding noise to circuits" - ] + ], + "metadata": { + "id": "0f123f0e3c55" + } }, { "cell_type": "markdown", - "metadata": { - "id": "473ac025a226" - }, "source": [ "Often circuits are defined with just unitary operations, but we want to simulate them with noise. There are several methods for inserting noise in Cirq." - ] + ], + "metadata": { + "id": "473ac025a226" + } }, { "cell_type": "markdown", - "metadata": { - "id": "d0de2c475f12" - }, "source": [ "For any circuit, the `with_noise` method can be called to insert a channel after every moment." - ] + ], + "metadata": { + "id": "d0de2c475f12" + } }, { "cell_type": "code", "execution_count": null, - "metadata": { - "id": "568530c8707b" - }, - "outputs": [], "source": [ "\"\"\"One method to insert noise in a circuit.\"\"\"\n", "# Define some noiseless circuit.\n", @@ -951,33 +811,33 @@ "# Display it.\n", "print(\"\\nCircuit with noise:\")\n", "print(noisy)" - ] + ], + "outputs": [], + "metadata": { + "id": "568530c8707b" + } }, { "cell_type": "markdown", - "metadata": { - "id": "c2206a7b7dd4" - }, "source": [ "This circuit can then be simulated using the methods described above." - ] + ], + "metadata": { + "id": "c2206a7b7dd4" + } }, { "cell_type": "markdown", - "metadata": { - "id": "029509de9787" - }, "source": [ "The `with_noise` method creates a `cirq.NoiseModel` from its input and adds noise to each moment. A `cirq.NoiseModel` can be explicitly created and used to add noise to a single operation, single moment, or series of moments as follows." - ] + ], + "metadata": { + "id": "029509de9787" + } }, { "cell_type": "code", "execution_count": null, - "metadata": { - "id": "10a2cd41bfe4" - }, - "outputs": [], "source": [ "\"\"\"Add noise to an operation, moment, or sequence of moments.\"\"\"\n", "# Create a noise model.\n", @@ -997,86 +857,86 @@ "# Add noise to a sequence of moments.\n", "circuit = cirq.Circuit(cirq.H(qreg[0]), cirq.CNOT(*qreg))\n", "noisy_circuit = noise_model.noisy_moments(circuit, system_qubits=qreg)" - ] + ], + "outputs": [], + "metadata": { + "id": "10a2cd41bfe4" + } }, { "cell_type": "markdown", - "metadata": { - "id": "a440a2aefdcf" - }, "source": [ "Note: In the last two examples, the argument `system_qubits` can be a subset of the qubits in the moment(s)." - ] + ], + "metadata": { + "id": "a440a2aefdcf" + } }, { "cell_type": "markdown", - "metadata": { - "id": "61cf807b3737" - }, "source": [ "The output of each \"noisy method\" is a `cirq.OP_TREE` which can be converted to a circuit by passing it into the `cirq.Circuit` constructor. For example, we create a circuit from the `noisy_moment` below." - ] + ], + "metadata": { + "id": "61cf807b3737" + } }, { "cell_type": "code", "execution_count": null, - "metadata": { - "id": "3e84e3e17421" - }, - "outputs": [], "source": [ "\"\"\"Creating a circuit from a noisy cirq.OP_TREE.\"\"\"\n", "cirq.Circuit(noisy_moment)" - ] + ], + "outputs": [], + "metadata": { + "id": "3e84e3e17421" + } }, { "cell_type": "markdown", - "metadata": { - "id": "11b18c640767" - }, "source": [ "Another technique is to pass a noise channel to the density matrix simulator as shown below." - ] + ], + "metadata": { + "id": "11b18c640767" + } }, { "cell_type": "code", "execution_count": null, - "metadata": { - "id": "3d16924842bf" - }, - "outputs": [], "source": [ "\"\"\"Define a density matrix simulator with a noise model.\"\"\"\n", "noisy_dsim = cirq.DensityMatrixSimulator(\n", " noise=cirq.generalized_amplitude_damp(p=0.1, gamma=0.5)\n", ")" - ] + ], + "outputs": [], + "metadata": { + "id": "3d16924842bf" + } }, { "cell_type": "markdown", - "metadata": { - "id": "0dcd4cf202be" - }, "source": [ "This will not explicitly add channels to the circuit being simulated, but the circuit will be simulated as though these channels were present." - ] + ], + "metadata": { + "id": "0dcd4cf202be" + } }, { "cell_type": "markdown", - "metadata": { - "id": "99ac77a2c896" - }, "source": [ "Other than these general methods, channels can be added to circuits at any moment just as gates are. The channels can be different, be correlated, act on a subset of qubits, be custom defined, etc." - ] + ], + "metadata": { + "id": "99ac77a2c896" + } }, { "cell_type": "code", "execution_count": null, - "metadata": { - "id": "de8a285b926c" - }, - "outputs": [], "source": [ "\"\"\"Defining a circuit with multiple noisy channels.\"\"\"\n", "qreg = cirq.LineQubit.range(4)\n", @@ -1093,29 +953,33 @@ "\n", "print(\"Circuit with multiple channels:\\n\")\n", "print(circ)" - ] + ], + "outputs": [], + "metadata": { + "id": "de8a285b926c" + } }, { "cell_type": "markdown", - "metadata": { - "id": "a9bb97a17256" - }, "source": [ "Circuits can also be modified with standard methods like `insert` to add channels at any point in the circuit. For example, to model simple state preparation errors, one can add bit-flip channels to the start of the circuit as follows." - ] + ], + "metadata": { + "id": "a9bb97a17256" + } }, { "cell_type": "code", "execution_count": null, - "metadata": { - "id": "524ded251565" - }, - "outputs": [], "source": [ "\"\"\"Example of inserting channels in circuits.\"\"\"\n", "circ.insert(0, cirq.bit_flip(p=0.1).on_each(qreg))\n", "print(circ)" - ] + ], + "outputs": [], + "metadata": { + "id": "524ded251565" + } } ], "metadata": { @@ -1129,5 +993,5 @@ } }, "nbformat": 4, - "nbformat_minor": 0 -} + "nbformat_minor": 2 +} \ No newline at end of file From 91adf6b704741b1ef8b3dcb3642dc10d79a2b23b Mon Sep 17 00:00:00 2001 From: Ricardo Olenewa Date: Wed, 13 Oct 2021 15:29:12 -0400 Subject: [PATCH 2/3] Ran nbfmt. --- docs/noise.ipynb | 1992 +++++++++++++++++++++++----------------------- 1 file changed, 996 insertions(+), 996 deletions(-) diff --git a/docs/noise.ipynb b/docs/noise.ipynb index da17da1e843..d9961b77125 100644 --- a/docs/noise.ipynb +++ b/docs/noise.ipynb @@ -1,997 +1,997 @@ { - "cells": [ - { - "cell_type": "markdown", - "source": [ - "##### Copyright 2020 The Cirq Developers" - ], - "metadata": { - "id": "LnV7yIFGeryz" - } - }, - { - "cell_type": "code", - "execution_count": null, - "source": [ - "#@title Licensed under the Apache License, Version 2.0 (the \"License\");\n", - "# you may not use this file except in compliance with the License.\n", - "# You may obtain a copy of the License at\n", - "#\n", - "# https://www.apache.org/licenses/LICENSE-2.0\n", - "#\n", - "# Unless required by applicable law or agreed to in writing, software\n", - "# distributed under the License is distributed on an \"AS IS\" BASIS,\n", - "# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n", - "# See the License for the specific language governing permissions and\n", - "# limitations under the License." - ], - "outputs": [], - "metadata": { - "cellView": "form", - "id": "1kkohf-9esQm" - } - }, - { - "cell_type": "markdown", - "source": [ - "# Noise" - ], - "metadata": { - "id": "7zgataJVe0mU" - } - }, - { - "cell_type": "markdown", - "source": [ - "\n", - " \n", - " \n", - " \n", - " \n", - "
\n", - " View on QuantumAI\n", - " \n", - " Run in Google Colab\n", - " \n", - " View source on GitHub\n", - " \n", - " Download notebook\n", - "
" - ], - "metadata": { - "id": "HYkRhx2pe2XX" - } - }, - { - "cell_type": "markdown", - "source": [ - "Note: this notebook relies on unreleased Cirq features. If you want to try these features, make sure you install cirq via `pip install cirq --pre`." - ], - "metadata": { - "id": "ea381f53cf89" - } - }, - { - "cell_type": "code", - "execution_count": null, - "source": [ - "try:\n", - " import cirq\n", - "except ImportError:\n", - " print(\"installing cirq...\")\n", - " !pip install --quiet cirq --pre\n", - " print(\"installed cirq.\")" - ], - "outputs": [], - "metadata": { - "id": "846b32703c5c" - } - }, - { - "cell_type": "markdown", - "source": [ - "For simulation, it is useful to have `Gate` objects that enact noisy quantum evolution. Cirq supports modeling noise via *operator sum* representations of noise (these evolutions are also known as quantum operations or quantum dynamical maps). \n", - "\n", - "This formalism models evolution of the density matrix $\\rho$ via\n", - "\n", - "$$\n", - "\\rho \\rightarrow \\sum_{k = 1}^{m} A_k \\rho A_k^\\dagger\n", - "$$\n", - "\n", - "where $A_k$ are known as *Kraus operators*. These operators are not necessarily unitary but must satisfy the trace-preserving property\n", - "\n", - "$$\n", - "\\sum_k A_k^\\dagger A_k = I .\n", - "$$\n", - "\n", - "A channel with $m = 1$ unitary Kraus operator is called *coherent* (and is equivalent to a unitary gate operation), otherwise the channel is called *incoherent*. For a given noisy channel, Kraus operators are not necessarily unique. For more details on these operators, see [John Preskill's lecture notes](http://theory.caltech.edu/~preskill/ph219/chap3_15.pdf)." - ], - "metadata": { - "id": "7d3daa944d6b" - } - }, - { - "cell_type": "markdown", - "source": [ - "## Common channels" - ], - "metadata": { - "id": "b989869680d4" - } - }, - { - "cell_type": "markdown", - "source": [ - "Cirq defines many commonly used quantum channels in [`ops/common_channels.py`](https://github.com/quantumlib/Cirq/blob/master/cirq/ops/common_channels.py). For example, the single-qubit bit-flip channel\n", - "\n", - "$$\n", - "\\rho \\rightarrow (1 - p) \\rho + p X \\rho X\n", - "$$\n", - "\n", - "with parameter $p = 0.1$ can be created as follows." - ], - "metadata": { - "id": "708b5b720a74" - } - }, - { - "cell_type": "code", - "execution_count": null, - "source": [ - "import cirq\n", - "\n", - "\"\"\"Get a single-qubit bit-flip channel.\"\"\"\n", - "bit_flip = cirq.bit_flip(p=0.1)" - ], - "outputs": [], - "metadata": { - "id": "386a49be9dc7" - } - }, - { - "cell_type": "markdown", - "source": [ - "To see the Kraus operators of a channel, the `cirq.channel` protocol can be used. (See the [protocols guide](./protocols.ipynb).)" - ], - "metadata": { - "id": "f1f501177e53" - } - }, - { - "cell_type": "code", - "execution_count": null, - "source": [ - "for i, kraus in enumerate(cirq.kraus(bit_flip)):\n", - " print(f\"Kraus operator {i + 1} is:\\n\", kraus, end=\"\\n\\n\")" - ], - "outputs": [], - "metadata": { - "id": "7635588faafe" - } - }, - { - "cell_type": "markdown", - "source": [ - "As mentioned, all channels are subclasses of `cirq.Gate`s. As such, they can act on qubits and be used in circuits in the same manner as gates." - ], - "metadata": { - "id": "3f9ff40f980f" - } - }, - { - "cell_type": "code", - "execution_count": null, - "source": [ - "\"\"\"Example of using channels in a circuit.\"\"\"\n", - "# See the number of qubits a channel acts on.\n", - "nqubits = bit_flip.num_qubits()\n", - "print(f\"Bit flip channel acts on {nqubits} qubit(s).\\n\")\n", - "\n", - "# Apply the channel to each qubit in a circuit.\n", - "circuit = cirq.Circuit(\n", - " bit_flip.on_each(cirq.LineQubit.range(3))\n", - ")\n", - "print(circuit)" - ], - "outputs": [], - "metadata": { - "id": "91187990b395" - } - }, - { - "cell_type": "markdown", - "source": [ - "Channels can even be controlled." - ], - "metadata": { - "id": "59651802c358" - } - }, - { - "cell_type": "code", - "execution_count": null, - "source": [ - "\"\"\"Example of controlling a channel.\"\"\"\n", - "# Get the controlled channel.\n", - "controlled_bit_flip = bit_flip.controlled(num_controls=1)\n", - "\n", - "# Use it in a circuit.\n", - "circuit = cirq.Circuit(\n", - " controlled_bit_flip(*cirq.LineQubit.range(2))\n", - ")\n", - "print(circuit)" - ], - "outputs": [], - "metadata": { - "id": "9be5383a7cd7" - } - }, - { - "cell_type": "markdown", - "source": [ - "In addition to the bit-flip channel, other common channels predefined in Cirq are shown below. Definitions of these channels can be found in their docstrings - e.g., `help(cirq.depolarize)`." - ], - "metadata": { - "id": "755ba122d550" - } - }, - { - "cell_type": "markdown", - "source": [ - "* `cirq.phase_flip`\n", - "* `cirq.phase_damp`\n", - "* `cirq.amplitude_damp`\n", - "* `cirq.depolarize`\n", - "* `cirq.asymmetric_depolarize`\n", - "* `cirq.reset`" - ], - "metadata": { - "id": "6a9c9c76fb8f" - } - }, - { - "cell_type": "markdown", - "source": [ - "For example, the asymmetric depolarizing channel is defined by\n", - "\n", - "$$\n", - "\\rho \\rightarrow (1-p_x-p_y-p_z) \\rho + p_x X \\rho X + p_y Y \\rho Y + p_z Z \\rho Z\n", - "$$\n", - "\n", - "and can be instantiated as follows." - ], - "metadata": { - "id": "8b22bb323731" - } - }, - { - "cell_type": "code", - "execution_count": null, - "source": [ - "\"\"\"Get an asymmetric depolarizing channel.\"\"\"\n", - "depo = cirq.asymmetric_depolarize(\n", - " p_x=0.10,\n", - " p_y=0.05,\n", - " p_z=0.15,\n", - ")\n", - "\n", - "circuit = cirq.Circuit(\n", - " depo.on_each(cirq.LineQubit(0))\n", - ")\n", - "print(circuit)" - ], - "outputs": [], - "metadata": { - "id": "b6a0966a29f6" - } - }, - { - "cell_type": "markdown", - "source": [ - "## The `channel` and `mixture` protocols" - ], - "metadata": { - "id": "ddbc622c98da" - } - }, - { - "cell_type": "markdown", - "source": [ - "We have seen the `cirq.channel` protocol which returns the Kraus operators of a channel. Some channels have the interpretation of randomly applying a single unitary Kraus operator $U_k$ with probability $p_k$, namely\n", - "\n", - "$$\n", - "\\rho \\rightarrow \\sum_k p_k U_k \\rho U_k^\\dagger {\\rm ~where~} \\sum_k p_k =1 {\\rm ~and~ U_k U_k^\\dagger= I}.\n", - "$$\n", - "\n", - "For example, the bit-flip channel from above\n", - "\n", - "$$\n", - "\\rho \\rightarrow (1 - p) \\rho + p X \\rho X\n", - "$$\n", - "\n", - "can be interpreted as doing nothing (applying identity) with probability $1 - p$ and flipping the bit (applying $X$) with probability $p$. Channels with these interpretations support the `cirq.mixture` protocol. This protocol returns the probabilities and unitary Kraus operators of the channel." - ], - "metadata": { - "id": "2dee355d2ae8" - } - }, - { - "cell_type": "code", - "execution_count": null, - "source": [ - "\"\"\"Example of using the mixture protocol.\"\"\"\n", - "for prob, kraus in cirq.mixture(bit_flip):\n", - " print(f\"With probability {prob}, apply\\n\", kraus, end=\"\\n\\n\")" - ], - "outputs": [], - "metadata": { - "id": "d4d84c4d4fa7" - } - }, - { - "cell_type": "markdown", - "source": [ - "Channels that do not have this interpretation do not support the `cirq.mixture` protocol. Such channels apply Kraus operators with probabilities that depend on the state $\\rho$. \n", - "\n", - "An example of a channel which does not support the mixture protocol is the amplitude damping channel with parameter $\\gamma$ defined by Kraus operators\n", - "\n", - "$$\n", - "M_0 = \\begin{bmatrix} 1 & 0 \\cr 0 & \\sqrt{1 - \\gamma} \\end{bmatrix} \n", - "\\text{and }\n", - "M_1 = \\begin{bmatrix} 0 & \\sqrt{\\gamma} \\cr 0 & 0 \\end{bmatrix} .\n", - "$$" - ], - "metadata": { - "id": "1ca339c3cbab" - } - }, - { - "cell_type": "code", - "execution_count": null, - "source": [ - "\"\"\"The amplitude damping channel is an example of a channel without a mixture.\"\"\"\n", - "channel = cirq.amplitude_damp(0.1)\n", - "\n", - "if cirq.has_mixture(channel):\n", - " print(f\"Channel {channel} has a _mixture_ or _unitary_ method.\")\n", - "else:\n", - " print(f\"Channel {channel} does not have a _mixture_ or _unitary_ method.\")" - ], - "outputs": [], - "metadata": { - "id": "8e377332d2a7" - } - }, - { - "cell_type": "markdown", - "source": [ - "To summarize:\n", - "\n", - "* Every `Gate` in Cirq supports the `cirq.channel` protocol.\n", - " - If magic method `_kraus_` is not defined, `cirq.channel` looks for `_mixture_` then for `_unitary_`.\n", - "* A subset of channels which support `cirq.channel` also support the `cirq.mixture` protocol.\n", - " - If magic method `_mixture_` is not defined, `cirq.mixture` looks for `_unitary_`.\n", - "* A subset of channels which support `cirq.mixture` also support the `cirq.unitary` protocol." - ], - "metadata": { - "id": "6f3ae4337219" - } - }, - { - "cell_type": "markdown", - "source": [ - "For concrete examples, consider `cirq.X`, `cirq.BitFlipChannel`, and `cirq.AmplitudeDampingChannel` which are all subclasses of `cirq.Gate`.\n", - "\n", - "* `cirq.X` defines the `_unitary_` method. \n", - " - As a result, it supports the `cirq.unitary` protocol, the `cirq.mixture` protocol, and the `cirq.channel` protocol.\n", - "* `cirq.BitFlipChannel` defines the `_mixture_` method but not the `_unitary_` method.\n", - " - As a result, it only supports the `cirq.mixture` protocol and the `cirq.channel` protocol.\n", - "* `cirq.AmplitudeDampingChannel` defines the `_kraus_` method, but not the `_mixture_` method or the `_unitary_` method.\n", - " - As a result, it only supports the `cirq.channel` protocol." - ], - "metadata": { - "id": "67c8ddbc10f3" - } - }, - { - "cell_type": "markdown", - "source": [ - "## Custom channels" - ], - "metadata": { - "id": "0f5b77825a8a" - } - }, - { - "cell_type": "markdown", - "source": [ - "There are two configurable channel types for channels not defined in `cirq.ops.common_channels`: `MixedUnitaryChannel` and `KrausChannel`.\n", - "\n", - "`MixedUnitaryChannel` takes a list of `(probability, unitary)` tuples and uses it to define the `_matrix_` method.\n", - "\n", - "`KrausChannel` takes a list of Kraus operators and uses it to define the `_channel` method.\n", - "\n", - "Both types also accept a measurement key as an optional parameter. This key will be used to store the index of the selected unitary or Kraus operator in the measurement results.\n", - "\n", - "An example of each type is shown below." - ], - "metadata": { - "id": "7614e0351462" - } - }, - { - "cell_type": "code", - "execution_count": null, - "source": [ - "import numpy as np\n", - "q0 = cirq.LineQubit(0)\n", - "# This is equivalent to a bit-flip error with probability 0.1.\n", - "mix = [\n", - " (0.9, np.array([[1, 0], [0, 1]], dtype=np.complex64)),\n", - " (0.1, np.array([[0, 1], [1, 0]], dtype=np.complex64)),\n", - "]\n", - "bit_flip = cirq.MixedUnitaryChannel(mix)\n", - "\n", - "# This is equivalent to an X-basis measurement.\n", - "ops = [\n", - " np.array([[1, 1], [1, 1]]) * 0.5,\n", - " np.array([[1, -1], [-1, 1]]) * 0.5,\n", - "]\n", - "x_meas = cirq.KrausChannel(ops, key='x')\n", - "\n", - "# These circuits have the same behavior.\n", - "circuit = cirq.Circuit(\n", - " bit_flip.on(q0),\n", - " cirq.H(q0),\n", - " x_meas.on(q0),\n", - ")\n", - "equiv_circuit = cirq.Circuit(\n", - " cirq.bit_flip(0.1).on(q0),\n", - " cirq.H(q0),\n", - " # Measure in x-basis\n", - " cirq.H(q0),\n", - " cirq.measure(q0, key='x'),\n", - " cirq.H(q0),\n", - ")" - ], - "outputs": [], - "metadata": { - "id": "7cbd723fb567" - } - }, - { - "cell_type": "markdown", - "source": [ - "Alternatively, users can define their own channel types. Defining custom channels is similar to defining [custom gates](./custom_gates.ipynb).\n", - "\n", - "A minimal example for defining the channel\n", - "\n", - "$$\n", - "\\rho \\mapsto (1 - p) \\rho + p Y \\rho Y\n", - "$$\n", - "\n", - "is shown below." - ], - "metadata": { - "id": "360d5a316769" - } - }, - { - "cell_type": "code", - "execution_count": null, - "source": [ - "\"\"\"Minimal example of defining a custom channel.\"\"\"\n", - "class BitAndPhaseFlipChannel(cirq.SingleQubitGate):\n", - " def __init__(self, p: float) -> None:\n", - " self._p = p\n", - " \n", - " def _mixture_(self):\n", - " ps = [1.0 - self._p, self._p]\n", - " ops = [cirq.unitary(cirq.I), cirq.unitary(cirq.Y)]\n", - " return tuple(zip(ps, ops))\n", - " \n", - " def _has_mixture_(self) -> bool:\n", - " return True\n", - " \n", - " def _circuit_diagram_info_(self, args) -> str:\n", - " return f\"BitAndPhaseFlip({self._p})\"" - ], - "outputs": [], - "metadata": { - "id": "55240879394e" - } - }, - { - "cell_type": "markdown", - "source": [ - "Note: The `_has_mixture_` magic method is not strictly required but is recommended." - ], - "metadata": { - "id": "8ffff44ff2b2" - } - }, - { - "cell_type": "markdown", - "source": [ - "We can now instantiate this channel and get its mixture:" - ], - "metadata": { - "id": "2dea6bd62cfe" - } - }, - { - "cell_type": "code", - "execution_count": null, - "source": [ - "\"\"\"Custom channels can be used like any other channels.\"\"\"\n", - "bit_phase_flip = BitAndPhaseFlipChannel(p=0.05)\n", - "\n", - "for prob, kraus in cirq.mixture(bit_phase_flip):\n", - " print(f\"With probability {prob}, apply\\n\", kraus, end=\"\\n\\n\")" - ], - "outputs": [], - "metadata": { - "id": "0350d9e193ca" - } - }, - { - "cell_type": "markdown", - "source": [ - "Note: Since `_mixture_` is defined, the `cirq.channel` protocol can also be used." - ], - "metadata": { - "id": "72486a155131" - } - }, - { - "cell_type": "markdown", - "source": [ - "The custom channel can be used in a circuit just like other predefined channels." - ], - "metadata": { - "id": "ada775055c3f" - } - }, - { - "cell_type": "code", - "execution_count": null, - "source": [ - "\"\"\"Example of using a custom channel in a circuit.\"\"\"\n", - "circuit = cirq.Circuit(\n", - " bit_phase_flip.on_each(*cirq.LineQubit.range(3))\n", - ")\n", - "circuit" - ], - "outputs": [], - "metadata": { - "id": "1af5c9f60cab" - } - }, - { - "cell_type": "markdown", - "source": [ - "Note: If a custom channel does not have a mixture, it should instead define the `_kraus_` magic method to return a sequence of Kraus operators (as `numpy.ndarray`s). Defining a `_has_kraus_` method which returns `True` is optional but recommended." - ], - "metadata": { - "id": "46eb7a01e30d" - } - }, - { - "cell_type": "markdown", - "source": [ - "This method of defining custom channels is the most general, but simple channels such as the custom `BitAndPhaseFlipChannel` can also be created directly from a `Gate` with the convenient `Gate.with_probability` method." - ], - "metadata": { - "id": "b537c43e078c" - } - }, - { - "cell_type": "code", - "execution_count": null, - "source": [ - "\"\"\"Create a channel with Gate.with_probability.\"\"\"\n", - "channel = cirq.Y.with_probability(probability=0.05)" - ], - "outputs": [], - "metadata": { - "id": "0937f8dad808" - } - }, - { - "cell_type": "markdown", - "source": [ - "This produces the same mixture as the custom `BitAndPhaseFlip` channel above." - ], - "metadata": { - "id": "0461f377ac46" - } - }, - { - "cell_type": "code", - "execution_count": null, - "source": [ - "for prob, kraus in cirq.mixture(channel):\n", - " print(f\"With probability {prob}, apply\\n\", kraus, end=\"\\n\\n\")" - ], - "outputs": [], - "metadata": { - "id": "f0a85b33680a" - } - }, - { - "cell_type": "markdown", - "source": [ - "Note that the order of Kraus operators is reversed from above, but this of course does not affect the action of the channel." - ], - "metadata": { - "id": "c73ec3e63e23" - } - }, - { - "cell_type": "markdown", - "source": [ - "## Simulating noisy circuits" - ], - "metadata": { - "id": "96a696b543f7" - } - }, - { - "cell_type": "markdown", - "source": [ - "### Density matrix simulation" - ], - "metadata": { - "id": "d51d993ae869" - } - }, - { - "cell_type": "markdown", - "source": [ - "The `cirq.DensityMatrixSimulator` can simulate any noisy circuit (i.e., can apply any quantum channel) because it stores the full density matrix $\\rho$. This simulation strategy updates the state $\\rho$ by directly applying the Kraus operators of each quantum channel." - ], - "metadata": { - "id": "38dfcb60ef79" - } - }, - { - "cell_type": "code", - "execution_count": null, - "source": [ - "\"\"\"Simulating a circuit with the density matrix simulator.\"\"\"\n", - "# Get a circuit.\n", - "qbit = cirq.GridQubit(0, 0)\n", - "circuit = cirq.Circuit(\n", - " cirq.X(qbit),\n", - " cirq.amplitude_damp(0.1).on(qbit)\n", - ")\n", - "\n", - "# Display it.\n", - "print(\"Simulating circuit:\")\n", - "print(circuit)\n", - "\n", - "# Simulate with the density matrix simulator.\n", - "dsim = cirq.DensityMatrixSimulator()\n", - "rho = dsim.simulate(circuit).final_density_matrix\n", - "\n", - "# Display the final density matrix.\n", - "print(\"\\nFinal density matrix:\")\n", - "print(rho)" - ], - "outputs": [], - "metadata": { - "id": "a2973a8faef0" - } - }, - { - "cell_type": "markdown", - "source": [ - "Note that the density matrix simulator supports the `run` method which only gives access to measurements as well as the `simulate` method (used above) which gives access to the full density matrix." - ], - "metadata": { - "id": "ac8a991a426b" - } - }, - { - "cell_type": "markdown", - "source": [ - "### Monte Carlo wavefunction simulation" - ], - "metadata": { - "id": "0c659bb20098" - } - }, - { - "cell_type": "markdown", - "source": [ - "Noisy circuits with arbitrary channels can also be simulated with the `cirq.Simulator`. When simulating such a channel, a single Kraus operator is randomly sampled (according to the probability distribution) and applied to the wavefunction. This method is known as \"Monte Carlo (wavefunction) simulation\" or \"quantum trajectories.\"\n", - "\n", - "Note: For channels which do not support the `cirq.mixture` protocol, the probability of applying each Kraus operator depends on the state. In contrast, for channels which do support the `cirq.mixture` protocol, the probability of applying each Kraus operator is independent of the state." - ], - "metadata": { - "id": "6f3caf9cea92" - } - }, - { - "cell_type": "code", - "execution_count": null, - "source": [ - "\"\"\"Simulating a noisy circuit via Monte Carlo simulation.\"\"\"\n", - "# Get a circuit.\n", - "qbit = cirq.NamedQubit(\"Q\")\n", - "circuit = cirq.Circuit(cirq.bit_flip(p=0.5).on(qbit))\n", - "\n", - "# Display it.\n", - "print(\"Simulating circuit:\")\n", - "print(circuit)\n", - "\n", - "# Simulate with the cirq.Simulator.\n", - "sim = cirq.Simulator()\n", - "psi = sim.simulate(circuit).dirac_notation()\n", - "\n", - "# Display the final wavefunction.\n", - "print(\"\\nFinal wavefunction:\")\n", - "print(psi)" - ], - "outputs": [], - "metadata": { - "id": "d80a2bc7ce14" - } - }, - { - "cell_type": "markdown", - "source": [ - "To see that the output is stochastic, you can run the cell above multiple times. Since $p = 0.5$ in the bit-flip channel, you should get $|0\\rangle$ roughly half the time and $|1\\rangle$ roughly half the time. The `run` method with many repetitions can also be used to see this behavior." - ], - "metadata": { - "id": "4c485bbc1dfd" - } - }, - { - "cell_type": "code", - "execution_count": null, - "source": [ - "\"\"\"Example of Monte Carlo wavefunction simulation with the `run` method.\"\"\"\n", - "circuit = cirq.Circuit(\n", - " cirq.bit_flip(p=0.5).on(qbit),\n", - " cirq.measure(qbit),\n", - ")\n", - "res = sim.run(circuit, repetitions=100)\n", - "print(res.histogram(key=qbit))" - ], - "outputs": [], - "metadata": { - "id": "8143ae0c7a34" - } - }, - { - "cell_type": "markdown", - "source": [ - "## Adding noise to circuits" - ], - "metadata": { - "id": "0f123f0e3c55" - } - }, - { - "cell_type": "markdown", - "source": [ - "Often circuits are defined with just unitary operations, but we want to simulate them with noise. There are several methods for inserting noise in Cirq." - ], - "metadata": { - "id": "473ac025a226" - } - }, - { - "cell_type": "markdown", - "source": [ - "For any circuit, the `with_noise` method can be called to insert a channel after every moment." - ], - "metadata": { - "id": "d0de2c475f12" - } - }, - { - "cell_type": "code", - "execution_count": null, - "source": [ - "\"\"\"One method to insert noise in a circuit.\"\"\"\n", - "# Define some noiseless circuit.\n", - "circuit = cirq.testing.random_circuit(\n", - " qubits=3, n_moments=3, op_density=1, random_state=11\n", - ")\n", - "\n", - "# Display the noiseless circuit.\n", - "print(\"Circuit without noise:\")\n", - "print(circuit)\n", - "\n", - "# Add noise to the circuit.\n", - "noisy = circuit.with_noise(cirq.depolarize(p=0.01))\n", - "\n", - "# Display it.\n", - "print(\"\\nCircuit with noise:\")\n", - "print(noisy)" - ], - "outputs": [], - "metadata": { - "id": "568530c8707b" - } - }, - { - "cell_type": "markdown", - "source": [ - "This circuit can then be simulated using the methods described above." - ], - "metadata": { - "id": "c2206a7b7dd4" - } - }, - { - "cell_type": "markdown", - "source": [ - "The `with_noise` method creates a `cirq.NoiseModel` from its input and adds noise to each moment. A `cirq.NoiseModel` can be explicitly created and used to add noise to a single operation, single moment, or series of moments as follows." - ], - "metadata": { - "id": "029509de9787" - } - }, - { - "cell_type": "code", - "execution_count": null, - "source": [ - "\"\"\"Add noise to an operation, moment, or sequence of moments.\"\"\"\n", - "# Create a noise model.\n", - "noise_model = cirq.NoiseModel.from_noise_model_like(cirq.depolarize(p=0.01))\n", - "\n", - "# Get a qubit register.\n", - "qreg = cirq.LineQubit.range(2)\n", - "\n", - "# Add noise to an operation.\n", - "op = cirq.CNOT(*qreg)\n", - "noisy_op = noise_model.noisy_operation(op)\n", - "\n", - "# Add noise to a moment.\n", - "moment = cirq.Moment(cirq.H.on_each(qreg))\n", - "noisy_moment = noise_model.noisy_moment(moment, system_qubits=qreg)\n", - "\n", - "# Add noise to a sequence of moments.\n", - "circuit = cirq.Circuit(cirq.H(qreg[0]), cirq.CNOT(*qreg))\n", - "noisy_circuit = noise_model.noisy_moments(circuit, system_qubits=qreg)" - ], - "outputs": [], - "metadata": { - "id": "10a2cd41bfe4" - } - }, - { - "cell_type": "markdown", - "source": [ - "Note: In the last two examples, the argument `system_qubits` can be a subset of the qubits in the moment(s)." - ], - "metadata": { - "id": "a440a2aefdcf" - } - }, - { - "cell_type": "markdown", - "source": [ - "The output of each \"noisy method\" is a `cirq.OP_TREE` which can be converted to a circuit by passing it into the `cirq.Circuit` constructor. For example, we create a circuit from the `noisy_moment` below." - ], - "metadata": { - "id": "61cf807b3737" - } - }, - { - "cell_type": "code", - "execution_count": null, - "source": [ - "\"\"\"Creating a circuit from a noisy cirq.OP_TREE.\"\"\"\n", - "cirq.Circuit(noisy_moment)" - ], - "outputs": [], - "metadata": { - "id": "3e84e3e17421" - } - }, - { - "cell_type": "markdown", - "source": [ - "Another technique is to pass a noise channel to the density matrix simulator as shown below." - ], - "metadata": { - "id": "11b18c640767" - } - }, - { - "cell_type": "code", - "execution_count": null, - "source": [ - "\"\"\"Define a density matrix simulator with a noise model.\"\"\"\n", - "noisy_dsim = cirq.DensityMatrixSimulator(\n", - " noise=cirq.generalized_amplitude_damp(p=0.1, gamma=0.5)\n", - ")" - ], - "outputs": [], - "metadata": { - "id": "3d16924842bf" - } - }, - { - "cell_type": "markdown", - "source": [ - "This will not explicitly add channels to the circuit being simulated, but the circuit will be simulated as though these channels were present." - ], - "metadata": { - "id": "0dcd4cf202be" - } - }, - { - "cell_type": "markdown", - "source": [ - "Other than these general methods, channels can be added to circuits at any moment just as gates are. The channels can be different, be correlated, act on a subset of qubits, be custom defined, etc." - ], - "metadata": { - "id": "99ac77a2c896" - } - }, - { - "cell_type": "code", - "execution_count": null, - "source": [ - "\"\"\"Defining a circuit with multiple noisy channels.\"\"\"\n", - "qreg = cirq.LineQubit.range(4)\n", - "circ = cirq.Circuit(\n", - " cirq.H.on_each(qreg),\n", - " cirq.depolarize(p=0.01).on_each(qreg),\n", - " cirq.qft(*qreg),\n", - " bit_phase_flip.on_each(qreg[1::2]),\n", - " cirq.qft(*qreg, inverse=True),\n", - " cirq.reset(qreg[1]),\n", - " cirq.measure(*qreg),\n", - " cirq.bit_flip(p=0.07).controlled(1).on(*qreg[2:]),\n", - ")\n", - "\n", - "print(\"Circuit with multiple channels:\\n\")\n", - "print(circ)" - ], - "outputs": [], - "metadata": { - "id": "de8a285b926c" - } - }, - { - "cell_type": "markdown", - "source": [ - "Circuits can also be modified with standard methods like `insert` to add channels at any point in the circuit. For example, to model simple state preparation errors, one can add bit-flip channels to the start of the circuit as follows." - ], - "metadata": { - "id": "a9bb97a17256" - } - }, - { - "cell_type": "code", - "execution_count": null, - "source": [ - "\"\"\"Example of inserting channels in circuits.\"\"\"\n", - "circ.insert(0, cirq.bit_flip(p=0.1).on_each(qreg))\n", - "print(circ)" - ], - "outputs": [], - "metadata": { - "id": "524ded251565" - } - } - ], - "metadata": { - "colab": { - "name": "noise.ipynb", - "toc_visible": true - }, - "kernelspec": { - "display_name": "Python 3", - "name": "python3" - } - }, - "nbformat": 4, - "nbformat_minor": 2 -} \ No newline at end of file + "cells": [ + { + "cell_type": "markdown", + "metadata": { + "id": "LnV7yIFGeryz" + }, + "source": [ + "##### Copyright 2020 The Cirq Developers" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "cellView": "form", + "id": "1kkohf-9esQm" + }, + "outputs": [], + "source": [ + "#@title Licensed under the Apache License, Version 2.0 (the \"License\");\n", + "# you may not use this file except in compliance with the License.\n", + "# You may obtain a copy of the License at\n", + "#\n", + "# https://www.apache.org/licenses/LICENSE-2.0\n", + "#\n", + "# Unless required by applicable law or agreed to in writing, software\n", + "# distributed under the License is distributed on an \"AS IS\" BASIS,\n", + "# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n", + "# See the License for the specific language governing permissions and\n", + "# limitations under the License." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "7zgataJVe0mU" + }, + "source": [ + "# Noise" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "HYkRhx2pe2XX" + }, + "source": [ + "\n", + " \n", + " \n", + " \n", + " \n", + "
\n", + " View on QuantumAI\n", + " \n", + " Run in Google Colab\n", + " \n", + " View source on GitHub\n", + " \n", + " Download notebook\n", + "
" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "ea381f53cf89" + }, + "source": [ + "Note: this notebook relies on unreleased Cirq features. If you want to try these features, make sure you install cirq via `pip install cirq --pre`." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "846b32703c5c" + }, + "outputs": [], + "source": [ + "try:\n", + " import cirq\n", + "except ImportError:\n", + " print(\"installing cirq...\")\n", + " !pip install --quiet cirq --pre\n", + " print(\"installed cirq.\")" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "7d3daa944d6b" + }, + "source": [ + "For simulation, it is useful to have `Gate` objects that enact noisy quantum evolution. Cirq supports modeling noise via *operator sum* representations of noise (these evolutions are also known as quantum operations or quantum dynamical maps). \n", + "\n", + "This formalism models evolution of the density matrix $\\rho$ via\n", + "\n", + "$$\n", + "\\rho \\rightarrow \\sum_{k = 1}^{m} A_k \\rho A_k^\\dagger\n", + "$$\n", + "\n", + "where $A_k$ are known as *Kraus operators*. These operators are not necessarily unitary but must satisfy the trace-preserving property\n", + "\n", + "$$\n", + "\\sum_k A_k^\\dagger A_k = I .\n", + "$$\n", + "\n", + "A channel with $m = 1$ unitary Kraus operator is called *coherent* (and is equivalent to a unitary gate operation), otherwise the channel is called *incoherent*. For a given noisy channel, Kraus operators are not necessarily unique. For more details on these operators, see [John Preskill's lecture notes](http://theory.caltech.edu/~preskill/ph219/chap3_15.pdf)." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "b989869680d4" + }, + "source": [ + "## Common channels" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "708b5b720a74" + }, + "source": [ + "Cirq defines many commonly used quantum channels in [`ops/common_channels.py`](https://github.com/quantumlib/Cirq/blob/master/cirq/ops/common_channels.py). For example, the single-qubit bit-flip channel\n", + "\n", + "$$\n", + "\\rho \\rightarrow (1 - p) \\rho + p X \\rho X\n", + "$$\n", + "\n", + "with parameter $p = 0.1$ can be created as follows." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "386a49be9dc7" + }, + "outputs": [], + "source": [ + "import cirq\n", + "\n", + "\"\"\"Get a single-qubit bit-flip channel.\"\"\"\n", + "bit_flip = cirq.bit_flip(p=0.1)" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "f1f501177e53" + }, + "source": [ + "To see the Kraus operators of a channel, the `cirq.channel` protocol can be used. (See the [protocols guide](./protocols.ipynb).)" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "7635588faafe" + }, + "outputs": [], + "source": [ + "for i, kraus in enumerate(cirq.kraus(bit_flip)):\n", + " print(f\"Kraus operator {i + 1} is:\\n\", kraus, end=\"\\n\\n\")" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "3f9ff40f980f" + }, + "source": [ + "As mentioned, all channels are subclasses of `cirq.Gate`s. As such, they can act on qubits and be used in circuits in the same manner as gates." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "91187990b395" + }, + "outputs": [], + "source": [ + "\"\"\"Example of using channels in a circuit.\"\"\"\n", + "# See the number of qubits a channel acts on.\n", + "nqubits = bit_flip.num_qubits()\n", + "print(f\"Bit flip channel acts on {nqubits} qubit(s).\\n\")\n", + "\n", + "# Apply the channel to each qubit in a circuit.\n", + "circuit = cirq.Circuit(\n", + " bit_flip.on_each(cirq.LineQubit.range(3))\n", + ")\n", + "print(circuit)" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "59651802c358" + }, + "source": [ + "Channels can even be controlled." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "9be5383a7cd7" + }, + "outputs": [], + "source": [ + "\"\"\"Example of controlling a channel.\"\"\"\n", + "# Get the controlled channel.\n", + "controlled_bit_flip = bit_flip.controlled(num_controls=1)\n", + "\n", + "# Use it in a circuit.\n", + "circuit = cirq.Circuit(\n", + " controlled_bit_flip(*cirq.LineQubit.range(2))\n", + ")\n", + "print(circuit)" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "755ba122d550" + }, + "source": [ + "In addition to the bit-flip channel, other common channels predefined in Cirq are shown below. Definitions of these channels can be found in their docstrings - e.g., `help(cirq.depolarize)`." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "6a9c9c76fb8f" + }, + "source": [ + "* `cirq.phase_flip`\n", + "* `cirq.phase_damp`\n", + "* `cirq.amplitude_damp`\n", + "* `cirq.depolarize`\n", + "* `cirq.asymmetric_depolarize`\n", + "* `cirq.reset`" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "8b22bb323731" + }, + "source": [ + "For example, the asymmetric depolarizing channel is defined by\n", + "\n", + "$$\n", + "\\rho \\rightarrow (1-p_x-p_y-p_z) \\rho + p_x X \\rho X + p_y Y \\rho Y + p_z Z \\rho Z\n", + "$$\n", + "\n", + "and can be instantiated as follows." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "b6a0966a29f6" + }, + "outputs": [], + "source": [ + "\"\"\"Get an asymmetric depolarizing channel.\"\"\"\n", + "depo = cirq.asymmetric_depolarize(\n", + " p_x=0.10,\n", + " p_y=0.05,\n", + " p_z=0.15,\n", + ")\n", + "\n", + "circuit = cirq.Circuit(\n", + " depo.on_each(cirq.LineQubit(0))\n", + ")\n", + "print(circuit)" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "ddbc622c98da" + }, + "source": [ + "## The `channel` and `mixture` protocols" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "2dee355d2ae8" + }, + "source": [ + "We have seen the `cirq.channel` protocol which returns the Kraus operators of a channel. Some channels have the interpretation of randomly applying a single unitary Kraus operator $U_k$ with probability $p_k$, namely\n", + "\n", + "$$\n", + "\\rho \\rightarrow \\sum_k p_k U_k \\rho U_k^\\dagger {\\rm ~where~} \\sum_k p_k =1 {\\rm ~and~ U_k U_k^\\dagger= I}.\n", + "$$\n", + "\n", + "For example, the bit-flip channel from above\n", + "\n", + "$$\n", + "\\rho \\rightarrow (1 - p) \\rho + p X \\rho X\n", + "$$\n", + "\n", + "can be interpreted as doing nothing (applying identity) with probability $1 - p$ and flipping the bit (applying $X$) with probability $p$. Channels with these interpretations support the `cirq.mixture` protocol. This protocol returns the probabilities and unitary Kraus operators of the channel." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "d4d84c4d4fa7" + }, + "outputs": [], + "source": [ + "\"\"\"Example of using the mixture protocol.\"\"\"\n", + "for prob, kraus in cirq.mixture(bit_flip):\n", + " print(f\"With probability {prob}, apply\\n\", kraus, end=\"\\n\\n\")" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "1ca339c3cbab" + }, + "source": [ + "Channels that do not have this interpretation do not support the `cirq.mixture` protocol. Such channels apply Kraus operators with probabilities that depend on the state $\\rho$. \n", + "\n", + "An example of a channel which does not support the mixture protocol is the amplitude damping channel with parameter $\\gamma$ defined by Kraus operators\n", + "\n", + "$$\n", + "M_0 = \\begin{bmatrix} 1 & 0 \\cr 0 & \\sqrt{1 - \\gamma} \\end{bmatrix} \n", + "\\text{and }\n", + "M_1 = \\begin{bmatrix} 0 & \\sqrt{\\gamma} \\cr 0 & 0 \\end{bmatrix} .\n", + "$$" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "8e377332d2a7" + }, + "outputs": [], + "source": [ + "\"\"\"The amplitude damping channel is an example of a channel without a mixture.\"\"\"\n", + "channel = cirq.amplitude_damp(0.1)\n", + "\n", + "if cirq.has_mixture(channel):\n", + " print(f\"Channel {channel} has a _mixture_ or _unitary_ method.\")\n", + "else:\n", + " print(f\"Channel {channel} does not have a _mixture_ or _unitary_ method.\")" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "6f3ae4337219" + }, + "source": [ + "To summarize:\n", + "\n", + "* Every `Gate` in Cirq supports the `cirq.channel` protocol.\n", + " - If magic method `_kraus_` is not defined, `cirq.channel` looks for `_mixture_` then for `_unitary_`.\n", + "* A subset of channels which support `cirq.channel` also support the `cirq.mixture` protocol.\n", + " - If magic method `_mixture_` is not defined, `cirq.mixture` looks for `_unitary_`.\n", + "* A subset of channels which support `cirq.mixture` also support the `cirq.unitary` protocol." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "67c8ddbc10f3" + }, + "source": [ + "For concrete examples, consider `cirq.X`, `cirq.BitFlipChannel`, and `cirq.AmplitudeDampingChannel` which are all subclasses of `cirq.Gate`.\n", + "\n", + "* `cirq.X` defines the `_unitary_` method. \n", + " - As a result, it supports the `cirq.unitary` protocol, the `cirq.mixture` protocol, and the `cirq.channel` protocol.\n", + "* `cirq.BitFlipChannel` defines the `_mixture_` method but not the `_unitary_` method.\n", + " - As a result, it only supports the `cirq.mixture` protocol and the `cirq.channel` protocol.\n", + "* `cirq.AmplitudeDampingChannel` defines the `_kraus_` method, but not the `_mixture_` method or the `_unitary_` method.\n", + " - As a result, it only supports the `cirq.channel` protocol." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "0f5b77825a8a" + }, + "source": [ + "## Custom channels" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "7614e0351462" + }, + "source": [ + "There are two configurable channel types for channels not defined in `cirq.ops.common_channels`: `MixedUnitaryChannel` and `KrausChannel`.\n", + "\n", + "`MixedUnitaryChannel` takes a list of `(probability, unitary)` tuples and uses it to define the `_matrix_` method.\n", + "\n", + "`KrausChannel` takes a list of Kraus operators and uses it to define the `_channel` method.\n", + "\n", + "Both types also accept a measurement key as an optional parameter. This key will be used to store the index of the selected unitary or Kraus operator in the measurement results.\n", + "\n", + "An example of each type is shown below." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "7cbd723fb567" + }, + "outputs": [], + "source": [ + "import numpy as np\n", + "q0 = cirq.LineQubit(0)\n", + "# This is equivalent to a bit-flip error with probability 0.1.\n", + "mix = [\n", + " (0.9, np.array([[1, 0], [0, 1]], dtype=np.complex64)),\n", + " (0.1, np.array([[0, 1], [1, 0]], dtype=np.complex64)),\n", + "]\n", + "bit_flip = cirq.MixedUnitaryChannel(mix)\n", + "\n", + "# This is equivalent to an X-basis measurement.\n", + "ops = [\n", + " np.array([[1, 1], [1, 1]]) * 0.5,\n", + " np.array([[1, -1], [-1, 1]]) * 0.5,\n", + "]\n", + "x_meas = cirq.KrausChannel(ops, key='x')\n", + "\n", + "# These circuits have the same behavior.\n", + "circuit = cirq.Circuit(\n", + " bit_flip.on(q0),\n", + " cirq.H(q0),\n", + " x_meas.on(q0),\n", + ")\n", + "equiv_circuit = cirq.Circuit(\n", + " cirq.bit_flip(0.1).on(q0),\n", + " cirq.H(q0),\n", + " # Measure in x-basis\n", + " cirq.H(q0),\n", + " cirq.measure(q0, key='x'),\n", + " cirq.H(q0),\n", + ")" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "360d5a316769" + }, + "source": [ + "Alternatively, users can define their own channel types. Defining custom channels is similar to defining [custom gates](./custom_gates.ipynb).\n", + "\n", + "A minimal example for defining the channel\n", + "\n", + "$$\n", + "\\rho \\mapsto (1 - p) \\rho + p Y \\rho Y\n", + "$$\n", + "\n", + "is shown below." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "55240879394e" + }, + "outputs": [], + "source": [ + "\"\"\"Minimal example of defining a custom channel.\"\"\"\n", + "class BitAndPhaseFlipChannel(cirq.SingleQubitGate):\n", + " def __init__(self, p: float) -> None:\n", + " self._p = p\n", + " \n", + " def _mixture_(self):\n", + " ps = [1.0 - self._p, self._p]\n", + " ops = [cirq.unitary(cirq.I), cirq.unitary(cirq.Y)]\n", + " return tuple(zip(ps, ops))\n", + " \n", + " def _has_mixture_(self) -> bool:\n", + " return True\n", + " \n", + " def _circuit_diagram_info_(self, args) -> str:\n", + " return f\"BitAndPhaseFlip({self._p})\"" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "8ffff44ff2b2" + }, + "source": [ + "Note: The `_has_mixture_` magic method is not strictly required but is recommended." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "2dea6bd62cfe" + }, + "source": [ + "We can now instantiate this channel and get its mixture:" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "0350d9e193ca" + }, + "outputs": [], + "source": [ + "\"\"\"Custom channels can be used like any other channels.\"\"\"\n", + "bit_phase_flip = BitAndPhaseFlipChannel(p=0.05)\n", + "\n", + "for prob, kraus in cirq.mixture(bit_phase_flip):\n", + " print(f\"With probability {prob}, apply\\n\", kraus, end=\"\\n\\n\")" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "72486a155131" + }, + "source": [ + "Note: Since `_mixture_` is defined, the `cirq.channel` protocol can also be used." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "ada775055c3f" + }, + "source": [ + "The custom channel can be used in a circuit just like other predefined channels." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "1af5c9f60cab" + }, + "outputs": [], + "source": [ + "\"\"\"Example of using a custom channel in a circuit.\"\"\"\n", + "circuit = cirq.Circuit(\n", + " bit_phase_flip.on_each(*cirq.LineQubit.range(3))\n", + ")\n", + "circuit" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "46eb7a01e30d" + }, + "source": [ + "Note: If a custom channel does not have a mixture, it should instead define the `_kraus_` magic method to return a sequence of Kraus operators (as `numpy.ndarray`s). Defining a `_has_kraus_` method which returns `True` is optional but recommended." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "b537c43e078c" + }, + "source": [ + "This method of defining custom channels is the most general, but simple channels such as the custom `BitAndPhaseFlipChannel` can also be created directly from a `Gate` with the convenient `Gate.with_probability` method." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "0937f8dad808" + }, + "outputs": [], + "source": [ + "\"\"\"Create a channel with Gate.with_probability.\"\"\"\n", + "channel = cirq.Y.with_probability(probability=0.05)" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "0461f377ac46" + }, + "source": [ + "This produces the same mixture as the custom `BitAndPhaseFlip` channel above." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "f0a85b33680a" + }, + "outputs": [], + "source": [ + "for prob, kraus in cirq.mixture(channel):\n", + " print(f\"With probability {prob}, apply\\n\", kraus, end=\"\\n\\n\")" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "c73ec3e63e23" + }, + "source": [ + "Note that the order of Kraus operators is reversed from above, but this of course does not affect the action of the channel." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "96a696b543f7" + }, + "source": [ + "## Simulating noisy circuits" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "d51d993ae869" + }, + "source": [ + "### Density matrix simulation" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "38dfcb60ef79" + }, + "source": [ + "The `cirq.DensityMatrixSimulator` can simulate any noisy circuit (i.e., can apply any quantum channel) because it stores the full density matrix $\\rho$. This simulation strategy updates the state $\\rho$ by directly applying the Kraus operators of each quantum channel." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "a2973a8faef0" + }, + "outputs": [], + "source": [ + "\"\"\"Simulating a circuit with the density matrix simulator.\"\"\"\n", + "# Get a circuit.\n", + "qbit = cirq.GridQubit(0, 0)\n", + "circuit = cirq.Circuit(\n", + " cirq.X(qbit),\n", + " cirq.amplitude_damp(0.1).on(qbit)\n", + ")\n", + "\n", + "# Display it.\n", + "print(\"Simulating circuit:\")\n", + "print(circuit)\n", + "\n", + "# Simulate with the density matrix simulator.\n", + "dsim = cirq.DensityMatrixSimulator()\n", + "rho = dsim.simulate(circuit).final_density_matrix\n", + "\n", + "# Display the final density matrix.\n", + "print(\"\\nFinal density matrix:\")\n", + "print(rho)" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "ac8a991a426b" + }, + "source": [ + "Note that the density matrix simulator supports the `run` method which only gives access to measurements as well as the `simulate` method (used above) which gives access to the full density matrix." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "0c659bb20098" + }, + "source": [ + "### Monte Carlo wavefunction simulation" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "6f3caf9cea92" + }, + "source": [ + "Noisy circuits with arbitrary channels can also be simulated with the `cirq.Simulator`. When simulating such a channel, a single Kraus operator is randomly sampled (according to the probability distribution) and applied to the wavefunction. This method is known as \"Monte Carlo (wavefunction) simulation\" or \"quantum trajectories.\"\n", + "\n", + "Note: For channels which do not support the `cirq.mixture` protocol, the probability of applying each Kraus operator depends on the state. In contrast, for channels which do support the `cirq.mixture` protocol, the probability of applying each Kraus operator is independent of the state." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "d80a2bc7ce14" + }, + "outputs": [], + "source": [ + "\"\"\"Simulating a noisy circuit via Monte Carlo simulation.\"\"\"\n", + "# Get a circuit.\n", + "qbit = cirq.NamedQubit(\"Q\")\n", + "circuit = cirq.Circuit(cirq.bit_flip(p=0.5).on(qbit))\n", + "\n", + "# Display it.\n", + "print(\"Simulating circuit:\")\n", + "print(circuit)\n", + "\n", + "# Simulate with the cirq.Simulator.\n", + "sim = cirq.Simulator()\n", + "psi = sim.simulate(circuit).dirac_notation()\n", + "\n", + "# Display the final wavefunction.\n", + "print(\"\\nFinal wavefunction:\")\n", + "print(psi)" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "4c485bbc1dfd" + }, + "source": [ + "To see that the output is stochastic, you can run the cell above multiple times. Since $p = 0.5$ in the bit-flip channel, you should get $|0\\rangle$ roughly half the time and $|1\\rangle$ roughly half the time. The `run` method with many repetitions can also be used to see this behavior." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "8143ae0c7a34" + }, + "outputs": [], + "source": [ + "\"\"\"Example of Monte Carlo wavefunction simulation with the `run` method.\"\"\"\n", + "circuit = cirq.Circuit(\n", + " cirq.bit_flip(p=0.5).on(qbit),\n", + " cirq.measure(qbit),\n", + ")\n", + "res = sim.run(circuit, repetitions=100)\n", + "print(res.histogram(key=qbit))" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "0f123f0e3c55" + }, + "source": [ + "## Adding noise to circuits" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "473ac025a226" + }, + "source": [ + "Often circuits are defined with just unitary operations, but we want to simulate them with noise. There are several methods for inserting noise in Cirq." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "d0de2c475f12" + }, + "source": [ + "For any circuit, the `with_noise` method can be called to insert a channel after every moment." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "568530c8707b" + }, + "outputs": [], + "source": [ + "\"\"\"One method to insert noise in a circuit.\"\"\"\n", + "# Define some noiseless circuit.\n", + "circuit = cirq.testing.random_circuit(\n", + " qubits=3, n_moments=3, op_density=1, random_state=11\n", + ")\n", + "\n", + "# Display the noiseless circuit.\n", + "print(\"Circuit without noise:\")\n", + "print(circuit)\n", + "\n", + "# Add noise to the circuit.\n", + "noisy = circuit.with_noise(cirq.depolarize(p=0.01))\n", + "\n", + "# Display it.\n", + "print(\"\\nCircuit with noise:\")\n", + "print(noisy)" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "c2206a7b7dd4" + }, + "source": [ + "This circuit can then be simulated using the methods described above." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "029509de9787" + }, + "source": [ + "The `with_noise` method creates a `cirq.NoiseModel` from its input and adds noise to each moment. A `cirq.NoiseModel` can be explicitly created and used to add noise to a single operation, single moment, or series of moments as follows." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "10a2cd41bfe4" + }, + "outputs": [], + "source": [ + "\"\"\"Add noise to an operation, moment, or sequence of moments.\"\"\"\n", + "# Create a noise model.\n", + "noise_model = cirq.NoiseModel.from_noise_model_like(cirq.depolarize(p=0.01))\n", + "\n", + "# Get a qubit register.\n", + "qreg = cirq.LineQubit.range(2)\n", + "\n", + "# Add noise to an operation.\n", + "op = cirq.CNOT(*qreg)\n", + "noisy_op = noise_model.noisy_operation(op)\n", + "\n", + "# Add noise to a moment.\n", + "moment = cirq.Moment(cirq.H.on_each(qreg))\n", + "noisy_moment = noise_model.noisy_moment(moment, system_qubits=qreg)\n", + "\n", + "# Add noise to a sequence of moments.\n", + "circuit = cirq.Circuit(cirq.H(qreg[0]), cirq.CNOT(*qreg))\n", + "noisy_circuit = noise_model.noisy_moments(circuit, system_qubits=qreg)" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "a440a2aefdcf" + }, + "source": [ + "Note: In the last two examples, the argument `system_qubits` can be a subset of the qubits in the moment(s)." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "61cf807b3737" + }, + "source": [ + "The output of each \"noisy method\" is a `cirq.OP_TREE` which can be converted to a circuit by passing it into the `cirq.Circuit` constructor. For example, we create a circuit from the `noisy_moment` below." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "3e84e3e17421" + }, + "outputs": [], + "source": [ + "\"\"\"Creating a circuit from a noisy cirq.OP_TREE.\"\"\"\n", + "cirq.Circuit(noisy_moment)" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "11b18c640767" + }, + "source": [ + "Another technique is to pass a noise channel to the density matrix simulator as shown below." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "3d16924842bf" + }, + "outputs": [], + "source": [ + "\"\"\"Define a density matrix simulator with a noise model.\"\"\"\n", + "noisy_dsim = cirq.DensityMatrixSimulator(\n", + " noise=cirq.generalized_amplitude_damp(p=0.1, gamma=0.5)\n", + ")" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "0dcd4cf202be" + }, + "source": [ + "This will not explicitly add channels to the circuit being simulated, but the circuit will be simulated as though these channels were present." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "99ac77a2c896" + }, + "source": [ + "Other than these general methods, channels can be added to circuits at any moment just as gates are. The channels can be different, be correlated, act on a subset of qubits, be custom defined, etc." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "de8a285b926c" + }, + "outputs": [], + "source": [ + "\"\"\"Defining a circuit with multiple noisy channels.\"\"\"\n", + "qreg = cirq.LineQubit.range(4)\n", + "circ = cirq.Circuit(\n", + " cirq.H.on_each(qreg),\n", + " cirq.depolarize(p=0.01).on_each(qreg),\n", + " cirq.qft(*qreg),\n", + " bit_phase_flip.on_each(qreg[1::2]),\n", + " cirq.qft(*qreg, inverse=True),\n", + " cirq.reset(qreg[1]),\n", + " cirq.measure(*qreg),\n", + " cirq.bit_flip(p=0.07).controlled(1).on(*qreg[2:]),\n", + ")\n", + "\n", + "print(\"Circuit with multiple channels:\\n\")\n", + "print(circ)" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "a9bb97a17256" + }, + "source": [ + "Circuits can also be modified with standard methods like `insert` to add channels at any point in the circuit. For example, to model simple state preparation errors, one can add bit-flip channels to the start of the circuit as follows." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "524ded251565" + }, + "outputs": [], + "source": [ + "\"\"\"Example of inserting channels in circuits.\"\"\"\n", + "circ.insert(0, cirq.bit_flip(p=0.1).on_each(qreg))\n", + "print(circ)" + ] + } + ], + "metadata": { + "colab": { + "name": "noise.ipynb", + "toc_visible": true + }, + "kernelspec": { + "display_name": "Python 3", + "name": "python3" + } + }, + "nbformat": 4, + "nbformat_minor": 0 +} From e437c640fb0d3339d685acc6c378605d23334b80 Mon Sep 17 00:00:00 2001 From: Ricardo Olenewa Date: Wed, 13 Oct 2021 15:56:29 -0400 Subject: [PATCH 3/3] Formatting, again. --- docs/noise.ipynb | 1990 +++++++++++++++++++++++----------------------- 1 file changed, 995 insertions(+), 995 deletions(-) diff --git a/docs/noise.ipynb b/docs/noise.ipynb index d9961b77125..8c13992d715 100644 --- a/docs/noise.ipynb +++ b/docs/noise.ipynb @@ -1,997 +1,997 @@ { - "cells": [ - { - "cell_type": "markdown", - "metadata": { - "id": "LnV7yIFGeryz" - }, - "source": [ - "##### Copyright 2020 The Cirq Developers" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "cellView": "form", - "id": "1kkohf-9esQm" - }, - "outputs": [], - "source": [ - "#@title Licensed under the Apache License, Version 2.0 (the \"License\");\n", - "# you may not use this file except in compliance with the License.\n", - "# You may obtain a copy of the License at\n", - "#\n", - "# https://www.apache.org/licenses/LICENSE-2.0\n", - "#\n", - "# Unless required by applicable law or agreed to in writing, software\n", - "# distributed under the License is distributed on an \"AS IS\" BASIS,\n", - "# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n", - "# See the License for the specific language governing permissions and\n", - "# limitations under the License." - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "7zgataJVe0mU" - }, - "source": [ - "# Noise" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "HYkRhx2pe2XX" - }, - "source": [ - "\n", - " \n", - " \n", - " \n", - " \n", - "
\n", - " View on QuantumAI\n", - " \n", - " Run in Google Colab\n", - " \n", - " View source on GitHub\n", - " \n", - " Download notebook\n", - "
" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "ea381f53cf89" - }, - "source": [ - "Note: this notebook relies on unreleased Cirq features. If you want to try these features, make sure you install cirq via `pip install cirq --pre`." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "id": "846b32703c5c" - }, - "outputs": [], - "source": [ - "try:\n", - " import cirq\n", - "except ImportError:\n", - " print(\"installing cirq...\")\n", - " !pip install --quiet cirq --pre\n", - " print(\"installed cirq.\")" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "7d3daa944d6b" - }, - "source": [ - "For simulation, it is useful to have `Gate` objects that enact noisy quantum evolution. Cirq supports modeling noise via *operator sum* representations of noise (these evolutions are also known as quantum operations or quantum dynamical maps). \n", - "\n", - "This formalism models evolution of the density matrix $\\rho$ via\n", - "\n", - "$$\n", - "\\rho \\rightarrow \\sum_{k = 1}^{m} A_k \\rho A_k^\\dagger\n", - "$$\n", - "\n", - "where $A_k$ are known as *Kraus operators*. These operators are not necessarily unitary but must satisfy the trace-preserving property\n", - "\n", - "$$\n", - "\\sum_k A_k^\\dagger A_k = I .\n", - "$$\n", - "\n", - "A channel with $m = 1$ unitary Kraus operator is called *coherent* (and is equivalent to a unitary gate operation), otherwise the channel is called *incoherent*. For a given noisy channel, Kraus operators are not necessarily unique. For more details on these operators, see [John Preskill's lecture notes](http://theory.caltech.edu/~preskill/ph219/chap3_15.pdf)." - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "b989869680d4" - }, - "source": [ - "## Common channels" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "708b5b720a74" - }, - "source": [ - "Cirq defines many commonly used quantum channels in [`ops/common_channels.py`](https://github.com/quantumlib/Cirq/blob/master/cirq/ops/common_channels.py). For example, the single-qubit bit-flip channel\n", - "\n", - "$$\n", - "\\rho \\rightarrow (1 - p) \\rho + p X \\rho X\n", - "$$\n", - "\n", - "with parameter $p = 0.1$ can be created as follows." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "id": "386a49be9dc7" - }, - "outputs": [], - "source": [ - "import cirq\n", - "\n", - "\"\"\"Get a single-qubit bit-flip channel.\"\"\"\n", - "bit_flip = cirq.bit_flip(p=0.1)" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "f1f501177e53" - }, - "source": [ - "To see the Kraus operators of a channel, the `cirq.channel` protocol can be used. (See the [protocols guide](./protocols.ipynb).)" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "id": "7635588faafe" - }, - "outputs": [], - "source": [ - "for i, kraus in enumerate(cirq.kraus(bit_flip)):\n", - " print(f\"Kraus operator {i + 1} is:\\n\", kraus, end=\"\\n\\n\")" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "3f9ff40f980f" - }, - "source": [ - "As mentioned, all channels are subclasses of `cirq.Gate`s. As such, they can act on qubits and be used in circuits in the same manner as gates." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "id": "91187990b395" - }, - "outputs": [], - "source": [ - "\"\"\"Example of using channels in a circuit.\"\"\"\n", - "# See the number of qubits a channel acts on.\n", - "nqubits = bit_flip.num_qubits()\n", - "print(f\"Bit flip channel acts on {nqubits} qubit(s).\\n\")\n", - "\n", - "# Apply the channel to each qubit in a circuit.\n", - "circuit = cirq.Circuit(\n", - " bit_flip.on_each(cirq.LineQubit.range(3))\n", - ")\n", - "print(circuit)" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "59651802c358" - }, - "source": [ - "Channels can even be controlled." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "id": "9be5383a7cd7" - }, - "outputs": [], - "source": [ - "\"\"\"Example of controlling a channel.\"\"\"\n", - "# Get the controlled channel.\n", - "controlled_bit_flip = bit_flip.controlled(num_controls=1)\n", - "\n", - "# Use it in a circuit.\n", - "circuit = cirq.Circuit(\n", - " controlled_bit_flip(*cirq.LineQubit.range(2))\n", - ")\n", - "print(circuit)" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "755ba122d550" - }, - "source": [ - "In addition to the bit-flip channel, other common channels predefined in Cirq are shown below. Definitions of these channels can be found in their docstrings - e.g., `help(cirq.depolarize)`." - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "6a9c9c76fb8f" - }, - "source": [ - "* `cirq.phase_flip`\n", - "* `cirq.phase_damp`\n", - "* `cirq.amplitude_damp`\n", - "* `cirq.depolarize`\n", - "* `cirq.asymmetric_depolarize`\n", - "* `cirq.reset`" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "8b22bb323731" - }, - "source": [ - "For example, the asymmetric depolarizing channel is defined by\n", - "\n", - "$$\n", - "\\rho \\rightarrow (1-p_x-p_y-p_z) \\rho + p_x X \\rho X + p_y Y \\rho Y + p_z Z \\rho Z\n", - "$$\n", - "\n", - "and can be instantiated as follows." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "id": "b6a0966a29f6" - }, - "outputs": [], - "source": [ - "\"\"\"Get an asymmetric depolarizing channel.\"\"\"\n", - "depo = cirq.asymmetric_depolarize(\n", - " p_x=0.10,\n", - " p_y=0.05,\n", - " p_z=0.15,\n", - ")\n", - "\n", - "circuit = cirq.Circuit(\n", - " depo.on_each(cirq.LineQubit(0))\n", - ")\n", - "print(circuit)" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "ddbc622c98da" - }, - "source": [ - "## The `channel` and `mixture` protocols" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "2dee355d2ae8" - }, - "source": [ - "We have seen the `cirq.channel` protocol which returns the Kraus operators of a channel. Some channels have the interpretation of randomly applying a single unitary Kraus operator $U_k$ with probability $p_k$, namely\n", - "\n", - "$$\n", - "\\rho \\rightarrow \\sum_k p_k U_k \\rho U_k^\\dagger {\\rm ~where~} \\sum_k p_k =1 {\\rm ~and~ U_k U_k^\\dagger= I}.\n", - "$$\n", - "\n", - "For example, the bit-flip channel from above\n", - "\n", - "$$\n", - "\\rho \\rightarrow (1 - p) \\rho + p X \\rho X\n", - "$$\n", - "\n", - "can be interpreted as doing nothing (applying identity) with probability $1 - p$ and flipping the bit (applying $X$) with probability $p$. Channels with these interpretations support the `cirq.mixture` protocol. This protocol returns the probabilities and unitary Kraus operators of the channel." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "id": "d4d84c4d4fa7" - }, - "outputs": [], - "source": [ - "\"\"\"Example of using the mixture protocol.\"\"\"\n", - "for prob, kraus in cirq.mixture(bit_flip):\n", - " print(f\"With probability {prob}, apply\\n\", kraus, end=\"\\n\\n\")" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "1ca339c3cbab" - }, - "source": [ - "Channels that do not have this interpretation do not support the `cirq.mixture` protocol. Such channels apply Kraus operators with probabilities that depend on the state $\\rho$. \n", - "\n", - "An example of a channel which does not support the mixture protocol is the amplitude damping channel with parameter $\\gamma$ defined by Kraus operators\n", - "\n", - "$$\n", - "M_0 = \\begin{bmatrix} 1 & 0 \\cr 0 & \\sqrt{1 - \\gamma} \\end{bmatrix} \n", - "\\text{and }\n", - "M_1 = \\begin{bmatrix} 0 & \\sqrt{\\gamma} \\cr 0 & 0 \\end{bmatrix} .\n", - "$$" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "id": "8e377332d2a7" - }, - "outputs": [], - "source": [ - "\"\"\"The amplitude damping channel is an example of a channel without a mixture.\"\"\"\n", - "channel = cirq.amplitude_damp(0.1)\n", - "\n", - "if cirq.has_mixture(channel):\n", - " print(f\"Channel {channel} has a _mixture_ or _unitary_ method.\")\n", - "else:\n", - " print(f\"Channel {channel} does not have a _mixture_ or _unitary_ method.\")" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "6f3ae4337219" - }, - "source": [ - "To summarize:\n", - "\n", - "* Every `Gate` in Cirq supports the `cirq.channel` protocol.\n", - " - If magic method `_kraus_` is not defined, `cirq.channel` looks for `_mixture_` then for `_unitary_`.\n", - "* A subset of channels which support `cirq.channel` also support the `cirq.mixture` protocol.\n", - " - If magic method `_mixture_` is not defined, `cirq.mixture` looks for `_unitary_`.\n", - "* A subset of channels which support `cirq.mixture` also support the `cirq.unitary` protocol." - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "67c8ddbc10f3" - }, - "source": [ - "For concrete examples, consider `cirq.X`, `cirq.BitFlipChannel`, and `cirq.AmplitudeDampingChannel` which are all subclasses of `cirq.Gate`.\n", - "\n", - "* `cirq.X` defines the `_unitary_` method. \n", - " - As a result, it supports the `cirq.unitary` protocol, the `cirq.mixture` protocol, and the `cirq.channel` protocol.\n", - "* `cirq.BitFlipChannel` defines the `_mixture_` method but not the `_unitary_` method.\n", - " - As a result, it only supports the `cirq.mixture` protocol and the `cirq.channel` protocol.\n", - "* `cirq.AmplitudeDampingChannel` defines the `_kraus_` method, but not the `_mixture_` method or the `_unitary_` method.\n", - " - As a result, it only supports the `cirq.channel` protocol." - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "0f5b77825a8a" - }, - "source": [ - "## Custom channels" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "7614e0351462" - }, - "source": [ - "There are two configurable channel types for channels not defined in `cirq.ops.common_channels`: `MixedUnitaryChannel` and `KrausChannel`.\n", - "\n", - "`MixedUnitaryChannel` takes a list of `(probability, unitary)` tuples and uses it to define the `_matrix_` method.\n", - "\n", - "`KrausChannel` takes a list of Kraus operators and uses it to define the `_channel` method.\n", - "\n", - "Both types also accept a measurement key as an optional parameter. This key will be used to store the index of the selected unitary or Kraus operator in the measurement results.\n", - "\n", - "An example of each type is shown below." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "id": "7cbd723fb567" - }, - "outputs": [], - "source": [ - "import numpy as np\n", - "q0 = cirq.LineQubit(0)\n", - "# This is equivalent to a bit-flip error with probability 0.1.\n", - "mix = [\n", - " (0.9, np.array([[1, 0], [0, 1]], dtype=np.complex64)),\n", - " (0.1, np.array([[0, 1], [1, 0]], dtype=np.complex64)),\n", - "]\n", - "bit_flip = cirq.MixedUnitaryChannel(mix)\n", - "\n", - "# This is equivalent to an X-basis measurement.\n", - "ops = [\n", - " np.array([[1, 1], [1, 1]]) * 0.5,\n", - " np.array([[1, -1], [-1, 1]]) * 0.5,\n", - "]\n", - "x_meas = cirq.KrausChannel(ops, key='x')\n", - "\n", - "# These circuits have the same behavior.\n", - "circuit = cirq.Circuit(\n", - " bit_flip.on(q0),\n", - " cirq.H(q0),\n", - " x_meas.on(q0),\n", - ")\n", - "equiv_circuit = cirq.Circuit(\n", - " cirq.bit_flip(0.1).on(q0),\n", - " cirq.H(q0),\n", - " # Measure in x-basis\n", - " cirq.H(q0),\n", - " cirq.measure(q0, key='x'),\n", - " cirq.H(q0),\n", - ")" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "360d5a316769" - }, - "source": [ - "Alternatively, users can define their own channel types. Defining custom channels is similar to defining [custom gates](./custom_gates.ipynb).\n", - "\n", - "A minimal example for defining the channel\n", - "\n", - "$$\n", - "\\rho \\mapsto (1 - p) \\rho + p Y \\rho Y\n", - "$$\n", - "\n", - "is shown below." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "id": "55240879394e" - }, - "outputs": [], - "source": [ - "\"\"\"Minimal example of defining a custom channel.\"\"\"\n", - "class BitAndPhaseFlipChannel(cirq.SingleQubitGate):\n", - " def __init__(self, p: float) -> None:\n", - " self._p = p\n", - " \n", - " def _mixture_(self):\n", - " ps = [1.0 - self._p, self._p]\n", - " ops = [cirq.unitary(cirq.I), cirq.unitary(cirq.Y)]\n", - " return tuple(zip(ps, ops))\n", - " \n", - " def _has_mixture_(self) -> bool:\n", - " return True\n", - " \n", - " def _circuit_diagram_info_(self, args) -> str:\n", - " return f\"BitAndPhaseFlip({self._p})\"" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "8ffff44ff2b2" - }, - "source": [ - "Note: The `_has_mixture_` magic method is not strictly required but is recommended." - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "2dea6bd62cfe" - }, - "source": [ - "We can now instantiate this channel and get its mixture:" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "id": "0350d9e193ca" - }, - "outputs": [], - "source": [ - "\"\"\"Custom channels can be used like any other channels.\"\"\"\n", - "bit_phase_flip = BitAndPhaseFlipChannel(p=0.05)\n", - "\n", - "for prob, kraus in cirq.mixture(bit_phase_flip):\n", - " print(f\"With probability {prob}, apply\\n\", kraus, end=\"\\n\\n\")" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "72486a155131" - }, - "source": [ - "Note: Since `_mixture_` is defined, the `cirq.channel` protocol can also be used." - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "ada775055c3f" - }, - "source": [ - "The custom channel can be used in a circuit just like other predefined channels." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "id": "1af5c9f60cab" - }, - "outputs": [], - "source": [ - "\"\"\"Example of using a custom channel in a circuit.\"\"\"\n", - "circuit = cirq.Circuit(\n", - " bit_phase_flip.on_each(*cirq.LineQubit.range(3))\n", - ")\n", - "circuit" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "46eb7a01e30d" - }, - "source": [ - "Note: If a custom channel does not have a mixture, it should instead define the `_kraus_` magic method to return a sequence of Kraus operators (as `numpy.ndarray`s). Defining a `_has_kraus_` method which returns `True` is optional but recommended." - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "b537c43e078c" - }, - "source": [ - "This method of defining custom channels is the most general, but simple channels such as the custom `BitAndPhaseFlipChannel` can also be created directly from a `Gate` with the convenient `Gate.with_probability` method." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "id": "0937f8dad808" - }, - "outputs": [], - "source": [ - "\"\"\"Create a channel with Gate.with_probability.\"\"\"\n", - "channel = cirq.Y.with_probability(probability=0.05)" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "0461f377ac46" - }, - "source": [ - "This produces the same mixture as the custom `BitAndPhaseFlip` channel above." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "id": "f0a85b33680a" - }, - "outputs": [], - "source": [ - "for prob, kraus in cirq.mixture(channel):\n", - " print(f\"With probability {prob}, apply\\n\", kraus, end=\"\\n\\n\")" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "c73ec3e63e23" - }, - "source": [ - "Note that the order of Kraus operators is reversed from above, but this of course does not affect the action of the channel." - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "96a696b543f7" - }, - "source": [ - "## Simulating noisy circuits" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "d51d993ae869" - }, - "source": [ - "### Density matrix simulation" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "38dfcb60ef79" - }, - "source": [ - "The `cirq.DensityMatrixSimulator` can simulate any noisy circuit (i.e., can apply any quantum channel) because it stores the full density matrix $\\rho$. This simulation strategy updates the state $\\rho$ by directly applying the Kraus operators of each quantum channel." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "id": "a2973a8faef0" - }, - "outputs": [], - "source": [ - "\"\"\"Simulating a circuit with the density matrix simulator.\"\"\"\n", - "# Get a circuit.\n", - "qbit = cirq.GridQubit(0, 0)\n", - "circuit = cirq.Circuit(\n", - " cirq.X(qbit),\n", - " cirq.amplitude_damp(0.1).on(qbit)\n", - ")\n", - "\n", - "# Display it.\n", - "print(\"Simulating circuit:\")\n", - "print(circuit)\n", - "\n", - "# Simulate with the density matrix simulator.\n", - "dsim = cirq.DensityMatrixSimulator()\n", - "rho = dsim.simulate(circuit).final_density_matrix\n", - "\n", - "# Display the final density matrix.\n", - "print(\"\\nFinal density matrix:\")\n", - "print(rho)" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "ac8a991a426b" - }, - "source": [ - "Note that the density matrix simulator supports the `run` method which only gives access to measurements as well as the `simulate` method (used above) which gives access to the full density matrix." - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "0c659bb20098" - }, - "source": [ - "### Monte Carlo wavefunction simulation" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "6f3caf9cea92" - }, - "source": [ - "Noisy circuits with arbitrary channels can also be simulated with the `cirq.Simulator`. When simulating such a channel, a single Kraus operator is randomly sampled (according to the probability distribution) and applied to the wavefunction. This method is known as \"Monte Carlo (wavefunction) simulation\" or \"quantum trajectories.\"\n", - "\n", - "Note: For channels which do not support the `cirq.mixture` protocol, the probability of applying each Kraus operator depends on the state. In contrast, for channels which do support the `cirq.mixture` protocol, the probability of applying each Kraus operator is independent of the state." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "id": "d80a2bc7ce14" - }, - "outputs": [], - "source": [ - "\"\"\"Simulating a noisy circuit via Monte Carlo simulation.\"\"\"\n", - "# Get a circuit.\n", - "qbit = cirq.NamedQubit(\"Q\")\n", - "circuit = cirq.Circuit(cirq.bit_flip(p=0.5).on(qbit))\n", - "\n", - "# Display it.\n", - "print(\"Simulating circuit:\")\n", - "print(circuit)\n", - "\n", - "# Simulate with the cirq.Simulator.\n", - "sim = cirq.Simulator()\n", - "psi = sim.simulate(circuit).dirac_notation()\n", - "\n", - "# Display the final wavefunction.\n", - "print(\"\\nFinal wavefunction:\")\n", - "print(psi)" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "4c485bbc1dfd" - }, - "source": [ - "To see that the output is stochastic, you can run the cell above multiple times. Since $p = 0.5$ in the bit-flip channel, you should get $|0\\rangle$ roughly half the time and $|1\\rangle$ roughly half the time. The `run` method with many repetitions can also be used to see this behavior." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "id": "8143ae0c7a34" - }, - "outputs": [], - "source": [ - "\"\"\"Example of Monte Carlo wavefunction simulation with the `run` method.\"\"\"\n", - "circuit = cirq.Circuit(\n", - " cirq.bit_flip(p=0.5).on(qbit),\n", - " cirq.measure(qbit),\n", - ")\n", - "res = sim.run(circuit, repetitions=100)\n", - "print(res.histogram(key=qbit))" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "0f123f0e3c55" - }, - "source": [ - "## Adding noise to circuits" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "473ac025a226" - }, - "source": [ - "Often circuits are defined with just unitary operations, but we want to simulate them with noise. There are several methods for inserting noise in Cirq." - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "d0de2c475f12" - }, - "source": [ - "For any circuit, the `with_noise` method can be called to insert a channel after every moment." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "id": "568530c8707b" - }, - "outputs": [], - "source": [ - "\"\"\"One method to insert noise in a circuit.\"\"\"\n", - "# Define some noiseless circuit.\n", - "circuit = cirq.testing.random_circuit(\n", - " qubits=3, n_moments=3, op_density=1, random_state=11\n", - ")\n", - "\n", - "# Display the noiseless circuit.\n", - "print(\"Circuit without noise:\")\n", - "print(circuit)\n", - "\n", - "# Add noise to the circuit.\n", - "noisy = circuit.with_noise(cirq.depolarize(p=0.01))\n", - "\n", - "# Display it.\n", - "print(\"\\nCircuit with noise:\")\n", - "print(noisy)" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "c2206a7b7dd4" - }, - "source": [ - "This circuit can then be simulated using the methods described above." - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "029509de9787" - }, - "source": [ - "The `with_noise` method creates a `cirq.NoiseModel` from its input and adds noise to each moment. A `cirq.NoiseModel` can be explicitly created and used to add noise to a single operation, single moment, or series of moments as follows." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "id": "10a2cd41bfe4" - }, - "outputs": [], - "source": [ - "\"\"\"Add noise to an operation, moment, or sequence of moments.\"\"\"\n", - "# Create a noise model.\n", - "noise_model = cirq.NoiseModel.from_noise_model_like(cirq.depolarize(p=0.01))\n", - "\n", - "# Get a qubit register.\n", - "qreg = cirq.LineQubit.range(2)\n", - "\n", - "# Add noise to an operation.\n", - "op = cirq.CNOT(*qreg)\n", - "noisy_op = noise_model.noisy_operation(op)\n", - "\n", - "# Add noise to a moment.\n", - "moment = cirq.Moment(cirq.H.on_each(qreg))\n", - "noisy_moment = noise_model.noisy_moment(moment, system_qubits=qreg)\n", - "\n", - "# Add noise to a sequence of moments.\n", - "circuit = cirq.Circuit(cirq.H(qreg[0]), cirq.CNOT(*qreg))\n", - "noisy_circuit = noise_model.noisy_moments(circuit, system_qubits=qreg)" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "a440a2aefdcf" - }, - "source": [ - "Note: In the last two examples, the argument `system_qubits` can be a subset of the qubits in the moment(s)." - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "61cf807b3737" - }, - "source": [ - "The output of each \"noisy method\" is a `cirq.OP_TREE` which can be converted to a circuit by passing it into the `cirq.Circuit` constructor. For example, we create a circuit from the `noisy_moment` below." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "id": "3e84e3e17421" - }, - "outputs": [], - "source": [ - "\"\"\"Creating a circuit from a noisy cirq.OP_TREE.\"\"\"\n", - "cirq.Circuit(noisy_moment)" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "11b18c640767" - }, - "source": [ - "Another technique is to pass a noise channel to the density matrix simulator as shown below." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "id": "3d16924842bf" - }, - "outputs": [], - "source": [ - "\"\"\"Define a density matrix simulator with a noise model.\"\"\"\n", - "noisy_dsim = cirq.DensityMatrixSimulator(\n", - " noise=cirq.generalized_amplitude_damp(p=0.1, gamma=0.5)\n", - ")" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "0dcd4cf202be" - }, - "source": [ - "This will not explicitly add channels to the circuit being simulated, but the circuit will be simulated as though these channels were present." - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "99ac77a2c896" - }, - "source": [ - "Other than these general methods, channels can be added to circuits at any moment just as gates are. The channels can be different, be correlated, act on a subset of qubits, be custom defined, etc." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "id": "de8a285b926c" - }, - "outputs": [], - "source": [ - "\"\"\"Defining a circuit with multiple noisy channels.\"\"\"\n", - "qreg = cirq.LineQubit.range(4)\n", - "circ = cirq.Circuit(\n", - " cirq.H.on_each(qreg),\n", - " cirq.depolarize(p=0.01).on_each(qreg),\n", - " cirq.qft(*qreg),\n", - " bit_phase_flip.on_each(qreg[1::2]),\n", - " cirq.qft(*qreg, inverse=True),\n", - " cirq.reset(qreg[1]),\n", - " cirq.measure(*qreg),\n", - " cirq.bit_flip(p=0.07).controlled(1).on(*qreg[2:]),\n", - ")\n", - "\n", - "print(\"Circuit with multiple channels:\\n\")\n", - "print(circ)" - ] - }, - { - "cell_type": "markdown", - "metadata": { - "id": "a9bb97a17256" - }, - "source": [ - "Circuits can also be modified with standard methods like `insert` to add channels at any point in the circuit. For example, to model simple state preparation errors, one can add bit-flip channels to the start of the circuit as follows." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": { - "id": "524ded251565" - }, - "outputs": [], - "source": [ - "\"\"\"Example of inserting channels in circuits.\"\"\"\n", - "circ.insert(0, cirq.bit_flip(p=0.1).on_each(qreg))\n", - "print(circ)" - ] - } - ], - "metadata": { - "colab": { - "name": "noise.ipynb", - "toc_visible": true - }, - "kernelspec": { - "display_name": "Python 3", - "name": "python3" - } - }, - "nbformat": 4, - "nbformat_minor": 0 + "cells": [ + { + "cell_type": "markdown", + "metadata": { + "id": "LnV7yIFGeryz" + }, + "source": [ + "##### Copyright 2020 The Cirq Developers" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "cellView": "form", + "id": "1kkohf-9esQm" + }, + "outputs": [], + "source": [ + "#@title Licensed under the Apache License, Version 2.0 (the \"License\");\n", + "# you may not use this file except in compliance with the License.\n", + "# You may obtain a copy of the License at\n", + "#\n", + "# https://www.apache.org/licenses/LICENSE-2.0\n", + "#\n", + "# Unless required by applicable law or agreed to in writing, software\n", + "# distributed under the License is distributed on an \"AS IS\" BASIS,\n", + "# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n", + "# See the License for the specific language governing permissions and\n", + "# limitations under the License." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "7zgataJVe0mU" + }, + "source": [ + "# Noise" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "HYkRhx2pe2XX" + }, + "source": [ + "\n", + " \n", + " \n", + " \n", + " \n", + "
\n", + " View on QuantumAI\n", + " \n", + " Run in Google Colab\n", + " \n", + " View source on GitHub\n", + " \n", + " Download notebook\n", + "
" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "ea381f53cf89" + }, + "source": [ + "Note: this notebook relies on unreleased Cirq features. If you want to try these features, make sure you install cirq via `pip install cirq --pre`." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "846b32703c5c" + }, + "outputs": [], + "source": [ + "try:\n", + " import cirq\n", + "except ImportError:\n", + " print(\"installing cirq...\")\n", + " !pip install --quiet cirq --pre\n", + " print(\"installed cirq.\")" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "7d3daa944d6b" + }, + "source": [ + "For simulation, it is useful to have `Gate` objects that enact noisy quantum evolution. Cirq supports modeling noise via *operator sum* representations of noise (these evolutions are also known as quantum operations or quantum dynamical maps). \n", + "\n", + "This formalism models evolution of the density matrix $\\rho$ via\n", + "\n", + "$$\n", + "\\rho \\rightarrow \\sum_{k = 1}^{m} A_k \\rho A_k^\\dagger\n", + "$$\n", + "\n", + "where $A_k$ are known as *Kraus operators*. These operators are not necessarily unitary but must satisfy the trace-preserving property\n", + "\n", + "$$\n", + "\\sum_k A_k^\\dagger A_k = I .\n", + "$$\n", + "\n", + "A channel with $m = 1$ unitary Kraus operator is called *coherent* (and is equivalent to a unitary gate operation), otherwise the channel is called *incoherent*. For a given noisy channel, Kraus operators are not necessarily unique. For more details on these operators, see [John Preskill's lecture notes](http://theory.caltech.edu/~preskill/ph219/chap3_15.pdf)." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "b989869680d4" + }, + "source": [ + "## Common channels" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "708b5b720a74" + }, + "source": [ + "Cirq defines many commonly used quantum channels in [`ops/common_channels.py`](https://github.com/quantumlib/Cirq/blob/master/cirq/ops/common_channels.py). For example, the single-qubit bit-flip channel\n", + "\n", + "$$\n", + "\\rho \\rightarrow (1 - p) \\rho + p X \\rho X\n", + "$$\n", + "\n", + "with parameter $p = 0.1$ can be created as follows." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "386a49be9dc7" + }, + "outputs": [], + "source": [ + "import cirq\n", + "\n", + "\"\"\"Get a single-qubit bit-flip channel.\"\"\"\n", + "bit_flip = cirq.bit_flip(p=0.1)" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "f1f501177e53" + }, + "source": [ + "To see the Kraus operators of a channel, the `cirq.channel` protocol can be used. (See the [protocols guide](./protocols.ipynb).)" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "7635588faafe" + }, + "outputs": [], + "source": [ + "for i, kraus in enumerate(cirq.kraus(bit_flip)):\n", + " print(f\"Kraus operator {i + 1} is:\\n\", kraus, end=\"\\n\\n\")" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "3f9ff40f980f" + }, + "source": [ + "As mentioned, all channels are subclasses of `cirq.Gate`s. As such, they can act on qubits and be used in circuits in the same manner as gates." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "91187990b395" + }, + "outputs": [], + "source": [ + "\"\"\"Example of using channels in a circuit.\"\"\"\n", + "# See the number of qubits a channel acts on.\n", + "nqubits = bit_flip.num_qubits()\n", + "print(f\"Bit flip channel acts on {nqubits} qubit(s).\\n\")\n", + "\n", + "# Apply the channel to each qubit in a circuit.\n", + "circuit = cirq.Circuit(\n", + " bit_flip.on_each(cirq.LineQubit.range(3))\n", + ")\n", + "print(circuit)" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "59651802c358" + }, + "source": [ + "Channels can even be controlled." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "9be5383a7cd7" + }, + "outputs": [], + "source": [ + "\"\"\"Example of controlling a channel.\"\"\"\n", + "# Get the controlled channel.\n", + "controlled_bit_flip = bit_flip.controlled(num_controls=1)\n", + "\n", + "# Use it in a circuit.\n", + "circuit = cirq.Circuit(\n", + " controlled_bit_flip(*cirq.LineQubit.range(2))\n", + ")\n", + "print(circuit)" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "755ba122d550" + }, + "source": [ + "In addition to the bit-flip channel, other common channels predefined in Cirq are shown below. Definitions of these channels can be found in their docstrings - e.g., `help(cirq.depolarize)`." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "6a9c9c76fb8f" + }, + "source": [ + "* `cirq.phase_flip`\n", + "* `cirq.phase_damp`\n", + "* `cirq.amplitude_damp`\n", + "* `cirq.depolarize`\n", + "* `cirq.asymmetric_depolarize`\n", + "* `cirq.reset`" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "8b22bb323731" + }, + "source": [ + "For example, the asymmetric depolarizing channel is defined by\n", + "\n", + "$$\n", + "\\rho \\rightarrow (1-p_x-p_y-p_z) \\rho + p_x X \\rho X + p_y Y \\rho Y + p_z Z \\rho Z\n", + "$$\n", + "\n", + "and can be instantiated as follows." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "b6a0966a29f6" + }, + "outputs": [], + "source": [ + "\"\"\"Get an asymmetric depolarizing channel.\"\"\"\n", + "depo = cirq.asymmetric_depolarize(\n", + " p_x=0.10,\n", + " p_y=0.05,\n", + " p_z=0.15,\n", + ")\n", + "\n", + "circuit = cirq.Circuit(\n", + " depo.on_each(cirq.LineQubit(0))\n", + ")\n", + "print(circuit)" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "ddbc622c98da" + }, + "source": [ + "## The `channel` and `mixture` protocols" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "2dee355d2ae8" + }, + "source": [ + "We have seen the `cirq.channel` protocol which returns the Kraus operators of a channel. Some channels have the interpretation of randomly applying a single unitary Kraus operator $U_k$ with probability $p_k$, namely\n", + "\n", + "$$\n", + "\\rho \\rightarrow \\sum_k p_k U_k \\rho U_k^\\dagger {\\rm ~where~} \\sum_k p_k =1 {\\rm ~and~ U_k U_k^\\dagger= I}.\n", + "$$\n", + "\n", + "For example, the bit-flip channel from above\n", + "\n", + "$$\n", + "\\rho \\rightarrow (1 - p) \\rho + p X \\rho X\n", + "$$\n", + "\n", + "can be interpreted as doing nothing (applying identity) with probability $1 - p$ and flipping the bit (applying $X$) with probability $p$. Channels with these interpretations support the `cirq.mixture` protocol. This protocol returns the probabilities and unitary Kraus operators of the channel." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "d4d84c4d4fa7" + }, + "outputs": [], + "source": [ + "\"\"\"Example of using the mixture protocol.\"\"\"\n", + "for prob, kraus in cirq.mixture(bit_flip):\n", + " print(f\"With probability {prob}, apply\\n\", kraus, end=\"\\n\\n\")" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "1ca339c3cbab" + }, + "source": [ + "Channels that do not have this interpretation do not support the `cirq.mixture` protocol. Such channels apply Kraus operators with probabilities that depend on the state $\\rho$. \n", + "\n", + "An example of a channel which does not support the mixture protocol is the amplitude damping channel with parameter $\\gamma$ defined by Kraus operators\n", + "\n", + "$$\n", + "M_0 = \\begin{bmatrix} 1 & 0 \\cr 0 & \\sqrt{1 - \\gamma} \\end{bmatrix} \n", + "\\text{and }\n", + "M_1 = \\begin{bmatrix} 0 & \\sqrt{\\gamma} \\cr 0 & 0 \\end{bmatrix} .\n", + "$$" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "8e377332d2a7" + }, + "outputs": [], + "source": [ + "\"\"\"The amplitude damping channel is an example of a channel without a mixture.\"\"\"\n", + "channel = cirq.amplitude_damp(0.1)\n", + "\n", + "if cirq.has_mixture(channel):\n", + " print(f\"Channel {channel} has a _mixture_ or _unitary_ method.\")\n", + "else:\n", + " print(f\"Channel {channel} does not have a _mixture_ or _unitary_ method.\")" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "6f3ae4337219" + }, + "source": [ + "To summarize:\n", + "\n", + "* Every `Gate` in Cirq supports the `cirq.channel` protocol.\n", + " - If magic method `_kraus_` is not defined, `cirq.channel` looks for `_mixture_` then for `_unitary_`.\n", + "* A subset of channels which support `cirq.channel` also support the `cirq.mixture` protocol.\n", + " - If magic method `_mixture_` is not defined, `cirq.mixture` looks for `_unitary_`.\n", + "* A subset of channels which support `cirq.mixture` also support the `cirq.unitary` protocol." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "67c8ddbc10f3" + }, + "source": [ + "For concrete examples, consider `cirq.X`, `cirq.BitFlipChannel`, and `cirq.AmplitudeDampingChannel` which are all subclasses of `cirq.Gate`.\n", + "\n", + "* `cirq.X` defines the `_unitary_` method. \n", + " - As a result, it supports the `cirq.unitary` protocol, the `cirq.mixture` protocol, and the `cirq.channel` protocol.\n", + "* `cirq.BitFlipChannel` defines the `_mixture_` method but not the `_unitary_` method.\n", + " - As a result, it only supports the `cirq.mixture` protocol and the `cirq.channel` protocol.\n", + "* `cirq.AmplitudeDampingChannel` defines the `_kraus_` method, but not the `_mixture_` method or the `_unitary_` method.\n", + " - As a result, it only supports the `cirq.channel` protocol." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "0f5b77825a8a" + }, + "source": [ + "## Custom channels" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "7614e0351462" + }, + "source": [ + "There are two configurable channel types for channels not defined in `cirq.ops.common_channels`: `MixedUnitaryChannel` and `KrausChannel`.\n", + "\n", + "`MixedUnitaryChannel` takes a list of `(probability, unitary)` tuples and uses it to define the `_matrix_` method.\n", + "\n", + "`KrausChannel` takes a list of Kraus operators and uses it to define the `_channel` method.\n", + "\n", + "Both types also accept a measurement key as an optional parameter. This key will be used to store the index of the selected unitary or Kraus operator in the measurement results.\n", + "\n", + "An example of each type is shown below." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "7cbd723fb567" + }, + "outputs": [], + "source": [ + "import numpy as np\n", + "q0 = cirq.LineQubit(0)\n", + "# This is equivalent to a bit-flip error with probability 0.1.\n", + "mix = [\n", + " (0.9, np.array([[1, 0], [0, 1]], dtype=np.complex64)),\n", + " (0.1, np.array([[0, 1], [1, 0]], dtype=np.complex64)),\n", + "]\n", + "bit_flip = cirq.MixedUnitaryChannel(mix)\n", + "\n", + "# This is equivalent to an X-basis measurement.\n", + "ops = [\n", + " np.array([[1, 1], [1, 1]]) * 0.5,\n", + " np.array([[1, -1], [-1, 1]]) * 0.5,\n", + "]\n", + "x_meas = cirq.KrausChannel(ops, key='x')\n", + "\n", + "# These circuits have the same behavior.\n", + "circuit = cirq.Circuit(\n", + " bit_flip.on(q0),\n", + " cirq.H(q0),\n", + " x_meas.on(q0),\n", + ")\n", + "equiv_circuit = cirq.Circuit(\n", + " cirq.bit_flip(0.1).on(q0),\n", + " cirq.H(q0),\n", + " # Measure in x-basis\n", + " cirq.H(q0),\n", + " cirq.measure(q0, key='x'),\n", + " cirq.H(q0),\n", + ")" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "360d5a316769" + }, + "source": [ + "Alternatively, users can define their own channel types. Defining custom channels is similar to defining [custom gates](./custom_gates.ipynb).\n", + "\n", + "A minimal example for defining the channel\n", + "\n", + "$$\n", + "\\rho \\mapsto (1 - p) \\rho + p Y \\rho Y\n", + "$$\n", + "\n", + "is shown below." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "55240879394e" + }, + "outputs": [], + "source": [ + "\"\"\"Minimal example of defining a custom channel.\"\"\"\n", + "class BitAndPhaseFlipChannel(cirq.SingleQubitGate):\n", + " def __init__(self, p: float) -> None:\n", + " self._p = p\n", + " \n", + " def _mixture_(self):\n", + " ps = [1.0 - self._p, self._p]\n", + " ops = [cirq.unitary(cirq.I), cirq.unitary(cirq.Y)]\n", + " return tuple(zip(ps, ops))\n", + " \n", + " def _has_mixture_(self) -> bool:\n", + " return True\n", + " \n", + " def _circuit_diagram_info_(self, args) -> str:\n", + " return f\"BitAndPhaseFlip({self._p})\"" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "8ffff44ff2b2" + }, + "source": [ + "Note: The `_has_mixture_` magic method is not strictly required but is recommended." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "2dea6bd62cfe" + }, + "source": [ + "We can now instantiate this channel and get its mixture:" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "0350d9e193ca" + }, + "outputs": [], + "source": [ + "\"\"\"Custom channels can be used like any other channels.\"\"\"\n", + "bit_phase_flip = BitAndPhaseFlipChannel(p=0.05)\n", + "\n", + "for prob, kraus in cirq.mixture(bit_phase_flip):\n", + " print(f\"With probability {prob}, apply\\n\", kraus, end=\"\\n\\n\")" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "72486a155131" + }, + "source": [ + "Note: Since `_mixture_` is defined, the `cirq.channel` protocol can also be used." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "ada775055c3f" + }, + "source": [ + "The custom channel can be used in a circuit just like other predefined channels." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "1af5c9f60cab" + }, + "outputs": [], + "source": [ + "\"\"\"Example of using a custom channel in a circuit.\"\"\"\n", + "circuit = cirq.Circuit(\n", + " bit_phase_flip.on_each(*cirq.LineQubit.range(3))\n", + ")\n", + "circuit" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "46eb7a01e30d" + }, + "source": [ + "Note: If a custom channel does not have a mixture, it should instead define the `_kraus_` magic method to return a sequence of Kraus operators (as `numpy.ndarray`s). Defining a `_has_kraus_` method which returns `True` is optional but recommended." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "b537c43e078c" + }, + "source": [ + "This method of defining custom channels is the most general, but simple channels such as the custom `BitAndPhaseFlipChannel` can also be created directly from a `Gate` with the convenient `Gate.with_probability` method." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "0937f8dad808" + }, + "outputs": [], + "source": [ + "\"\"\"Create a channel with Gate.with_probability.\"\"\"\n", + "channel = cirq.Y.with_probability(probability=0.05)" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "0461f377ac46" + }, + "source": [ + "This produces the same mixture as the custom `BitAndPhaseFlip` channel above." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "f0a85b33680a" + }, + "outputs": [], + "source": [ + "for prob, kraus in cirq.mixture(channel):\n", + " print(f\"With probability {prob}, apply\\n\", kraus, end=\"\\n\\n\")" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "c73ec3e63e23" + }, + "source": [ + "Note that the order of Kraus operators is reversed from above, but this of course does not affect the action of the channel." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "96a696b543f7" + }, + "source": [ + "## Simulating noisy circuits" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "d51d993ae869" + }, + "source": [ + "### Density matrix simulation" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "38dfcb60ef79" + }, + "source": [ + "The `cirq.DensityMatrixSimulator` can simulate any noisy circuit (i.e., can apply any quantum channel) because it stores the full density matrix $\\rho$. This simulation strategy updates the state $\\rho$ by directly applying the Kraus operators of each quantum channel." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "a2973a8faef0" + }, + "outputs": [], + "source": [ + "\"\"\"Simulating a circuit with the density matrix simulator.\"\"\"\n", + "# Get a circuit.\n", + "qbit = cirq.GridQubit(0, 0)\n", + "circuit = cirq.Circuit(\n", + " cirq.X(qbit),\n", + " cirq.amplitude_damp(0.1).on(qbit)\n", + ")\n", + "\n", + "# Display it.\n", + "print(\"Simulating circuit:\")\n", + "print(circuit)\n", + "\n", + "# Simulate with the density matrix simulator.\n", + "dsim = cirq.DensityMatrixSimulator()\n", + "rho = dsim.simulate(circuit).final_density_matrix\n", + "\n", + "# Display the final density matrix.\n", + "print(\"\\nFinal density matrix:\")\n", + "print(rho)" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "ac8a991a426b" + }, + "source": [ + "Note that the density matrix simulator supports the `run` method which only gives access to measurements as well as the `simulate` method (used above) which gives access to the full density matrix." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "0c659bb20098" + }, + "source": [ + "### Monte Carlo wavefunction simulation" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "6f3caf9cea92" + }, + "source": [ + "Noisy circuits with arbitrary channels can also be simulated with the `cirq.Simulator`. When simulating such a channel, a single Kraus operator is randomly sampled (according to the probability distribution) and applied to the wavefunction. This method is known as \"Monte Carlo (wavefunction) simulation\" or \"quantum trajectories.\"\n", + "\n", + "Note: For channels which do not support the `cirq.mixture` protocol, the probability of applying each Kraus operator depends on the state. In contrast, for channels which do support the `cirq.mixture` protocol, the probability of applying each Kraus operator is independent of the state." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "d80a2bc7ce14" + }, + "outputs": [], + "source": [ + "\"\"\"Simulating a noisy circuit via Monte Carlo simulation.\"\"\"\n", + "# Get a circuit.\n", + "qbit = cirq.NamedQubit(\"Q\")\n", + "circuit = cirq.Circuit(cirq.bit_flip(p=0.5).on(qbit))\n", + "\n", + "# Display it.\n", + "print(\"Simulating circuit:\")\n", + "print(circuit)\n", + "\n", + "# Simulate with the cirq.Simulator.\n", + "sim = cirq.Simulator()\n", + "psi = sim.simulate(circuit).dirac_notation()\n", + "\n", + "# Display the final wavefunction.\n", + "print(\"\\nFinal wavefunction:\")\n", + "print(psi)" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "4c485bbc1dfd" + }, + "source": [ + "To see that the output is stochastic, you can run the cell above multiple times. Since $p = 0.5$ in the bit-flip channel, you should get $|0\\rangle$ roughly half the time and $|1\\rangle$ roughly half the time. The `run` method with many repetitions can also be used to see this behavior." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "8143ae0c7a34" + }, + "outputs": [], + "source": [ + "\"\"\"Example of Monte Carlo wavefunction simulation with the `run` method.\"\"\"\n", + "circuit = cirq.Circuit(\n", + " cirq.bit_flip(p=0.5).on(qbit),\n", + " cirq.measure(qbit),\n", + ")\n", + "res = sim.run(circuit, repetitions=100)\n", + "print(res.histogram(key=qbit))" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "0f123f0e3c55" + }, + "source": [ + "## Adding noise to circuits" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "473ac025a226" + }, + "source": [ + "Often circuits are defined with just unitary operations, but we want to simulate them with noise. There are several methods for inserting noise in Cirq." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "d0de2c475f12" + }, + "source": [ + "For any circuit, the `with_noise` method can be called to insert a channel after every moment." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "568530c8707b" + }, + "outputs": [], + "source": [ + "\"\"\"One method to insert noise in a circuit.\"\"\"\n", + "# Define some noiseless circuit.\n", + "circuit = cirq.testing.random_circuit(\n", + " qubits=3, n_moments=3, op_density=1, random_state=11\n", + ")\n", + "\n", + "# Display the noiseless circuit.\n", + "print(\"Circuit without noise:\")\n", + "print(circuit)\n", + "\n", + "# Add noise to the circuit.\n", + "noisy = circuit.with_noise(cirq.depolarize(p=0.01))\n", + "\n", + "# Display it.\n", + "print(\"\\nCircuit with noise:\")\n", + "print(noisy)" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "c2206a7b7dd4" + }, + "source": [ + "This circuit can then be simulated using the methods described above." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "029509de9787" + }, + "source": [ + "The `with_noise` method creates a `cirq.NoiseModel` from its input and adds noise to each moment. A `cirq.NoiseModel` can be explicitly created and used to add noise to a single operation, single moment, or series of moments as follows." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "10a2cd41bfe4" + }, + "outputs": [], + "source": [ + "\"\"\"Add noise to an operation, moment, or sequence of moments.\"\"\"\n", + "# Create a noise model.\n", + "noise_model = cirq.NoiseModel.from_noise_model_like(cirq.depolarize(p=0.01))\n", + "\n", + "# Get a qubit register.\n", + "qreg = cirq.LineQubit.range(2)\n", + "\n", + "# Add noise to an operation.\n", + "op = cirq.CNOT(*qreg)\n", + "noisy_op = noise_model.noisy_operation(op)\n", + "\n", + "# Add noise to a moment.\n", + "moment = cirq.Moment(cirq.H.on_each(qreg))\n", + "noisy_moment = noise_model.noisy_moment(moment, system_qubits=qreg)\n", + "\n", + "# Add noise to a sequence of moments.\n", + "circuit = cirq.Circuit(cirq.H(qreg[0]), cirq.CNOT(*qreg))\n", + "noisy_circuit = noise_model.noisy_moments(circuit, system_qubits=qreg)" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "a440a2aefdcf" + }, + "source": [ + "Note: In the last two examples, the argument `system_qubits` can be a subset of the qubits in the moment(s)." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "61cf807b3737" + }, + "source": [ + "The output of each \"noisy method\" is a `cirq.OP_TREE` which can be converted to a circuit by passing it into the `cirq.Circuit` constructor. For example, we create a circuit from the `noisy_moment` below." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "3e84e3e17421" + }, + "outputs": [], + "source": [ + "\"\"\"Creating a circuit from a noisy cirq.OP_TREE.\"\"\"\n", + "cirq.Circuit(noisy_moment)" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "11b18c640767" + }, + "source": [ + "Another technique is to pass a noise channel to the density matrix simulator as shown below." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "3d16924842bf" + }, + "outputs": [], + "source": [ + "\"\"\"Define a density matrix simulator with a noise model.\"\"\"\n", + "noisy_dsim = cirq.DensityMatrixSimulator(\n", + " noise=cirq.generalized_amplitude_damp(p=0.1, gamma=0.5)\n", + ")" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "0dcd4cf202be" + }, + "source": [ + "This will not explicitly add channels to the circuit being simulated, but the circuit will be simulated as though these channels were present." + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "99ac77a2c896" + }, + "source": [ + "Other than these general methods, channels can be added to circuits at any moment just as gates are. The channels can be different, be correlated, act on a subset of qubits, be custom defined, etc." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "de8a285b926c" + }, + "outputs": [], + "source": [ + "\"\"\"Defining a circuit with multiple noisy channels.\"\"\"\n", + "qreg = cirq.LineQubit.range(4)\n", + "circ = cirq.Circuit(\n", + " cirq.H.on_each(qreg),\n", + " cirq.depolarize(p=0.01).on_each(qreg),\n", + " cirq.qft(*qreg),\n", + " bit_phase_flip.on_each(qreg[1::2]),\n", + " cirq.qft(*qreg, inverse=True),\n", + " cirq.reset(qreg[1]),\n", + " cirq.measure(*qreg),\n", + " cirq.bit_flip(p=0.07).controlled(1).on(*qreg[2:]),\n", + ")\n", + "\n", + "print(\"Circuit with multiple channels:\\n\")\n", + "print(circ)" + ] + }, + { + "cell_type": "markdown", + "metadata": { + "id": "a9bb97a17256" + }, + "source": [ + "Circuits can also be modified with standard methods like `insert` to add channels at any point in the circuit. For example, to model simple state preparation errors, one can add bit-flip channels to the start of the circuit as follows." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": { + "id": "524ded251565" + }, + "outputs": [], + "source": [ + "\"\"\"Example of inserting channels in circuits.\"\"\"\n", + "circ.insert(0, cirq.bit_flip(p=0.1).on_each(qreg))\n", + "print(circ)" + ] + } + ], + "metadata": { + "colab": { + "name": "noise.ipynb", + "toc_visible": true + }, + "kernelspec": { + "display_name": "Python 3", + "name": "python3" + } + }, + "nbformat": 4, + "nbformat_minor": 0 }