Noise

View on TensorFlow.org Run in Google Colab View source on GitHub Download notebook

Noise is present in modern day quantum computers. Qubits are susceptible to interference from the surrounding environment, imperfect fabrication, TLS and sometimes even gamma rays. Until large scale error correction is reached, the algorithms of today must be able to remain functional in the presence of noise. This makes testing algorithms under noise an important step for validating quantum algorithms / models will function on the quantum computers of today.

In this tutorial you will explore the basics of noisy circuit simulation in TFQ via the high level tfq.layers API.

Setup

pip install tensorflow==2.15.0 tensorflow-quantum==0.7.3
pip install -q git+https://github.com/tensorflow/docs
# Update package resources to account for version changes.
import importlib, pkg_resources

importlib.reload(pkg_resources)
/tmpfs/tmp/ipykernel_11479/2730622109.py:2: DeprecationWarning: pkg_resources is deprecated as an API. See https://setuptools.pypa.io/en/latest/pkg_resources.html
  import importlib, pkg_resources
<module 'pkg_resources' from '/tmpfs/src/tf_docs_env/lib/python3.9/site-packages/pkg_resources/__init__.py'>
import random
import cirq
import sympy
import tensorflow_quantum as tfq
import tensorflow as tf
import numpy as np
# Plotting
import matplotlib.pyplot as plt
import tensorflow_docs as tfdocs
import tensorflow_docs.plots
2025-01-10 12:14:57.167133: E external/local_xla/xla/stream_executor/cuda/cuda_dnn.cc:9261] Unable to register cuDNN factory: Attempting to register factory for plugin cuDNN when one has already been registered
2025-01-10 12:14:57.167181: E external/local_xla/xla/stream_executor/cuda/cuda_fft.cc:607] Unable to register cuFFT factory: Attempting to register factory for plugin cuFFT when one has already been registered
2025-01-10 12:14:57.168732: E external/local_xla/xla/stream_executor/cuda/cuda_blas.cc:1515] Unable to register cuBLAS factory: Attempting to register factory for plugin cuBLAS when one has already been registered
2025-01-10 12:14:59.736355: E external/local_xla/xla/stream_executor/cuda/cuda_driver.cc:274] failed call to cuInit: CUDA_ERROR_NO_DEVICE: no CUDA-capable device is detected

1. Understanding quantum noise

1.1 Basic circuit noise

Noise on a quantum computer impacts the bitstring samples you are able to measure from it. One intuitive way you can start to think about this is that a noisy quantum computer will "insert", "delete" or "replace" gates in random places like the diagram below:

Building off of this intuition, when dealing with noise, you are no longer using a single pure state \(|\psi \rangle\) but instead dealing with an ensemble of all possible noisy realizations of your desired circuit: \(\rho = \sum_j p_j |\psi_j \rangle \langle \psi_j |\) . Where \(p_j\) gives the probability that the system is in \(|\psi_j \rangle\) .

Revisiting the above picture, if we knew beforehand that 90% of the time our system executed perfectly, or errored 10% of the time with just this one mode of failure, then our ensemble would be:

\(\rho = 0.9 |\psi_\text{desired} \rangle \langle \psi_\text{desired}| + 0.1 |\psi_\text{noisy} \rangle \langle \psi_\text{noisy}| \)

If there was more than just one way that our circuit could error, then the ensemble \(\rho\) would contain more than just two terms (one for each new noisy realization that could happen). \(\rho\) is referred to as the density matrix describing your noisy system.

1.2 Using channels to model circuit noise

Unfortunately in practice it's nearly impossible to know all the ways your circuit might error and their exact probabilities. A simplifying assumption you can make is that after each operation in your circuit there is some kind of channel that roughly captures how that operation might error. You can quickly create a circuit with some noise:

def x_circuit(qubits):
    """Produces an X wall circuit on `qubits`."""
    return cirq.Circuit(cirq.X.on_each(*qubits))


def make_noisy(circuit, p):
    """Add a depolarization channel to all qubits in `circuit` before measurement."""
    return circuit + cirq.Circuit(
        cirq.depolarize(p).on_each(*circuit.all_qubits()))


my_qubits = cirq.GridQubit.rect(1, 2)
my_circuit = x_circuit(my_qubits)
my_noisy_circuit = make_noisy(my_circuit, 0.5)
my_circuit
my_noisy_circuit

You can examine the noiseless density matrix \(\rho\) with:

rho = cirq.final_density_matrix(my_circuit)
np.round(rho, 3)
array([[0.+0.j, 0.+0.j, 0.+0.j, 0.+0.j],
       [0.+0.j, 0.+0.j, 0.+0.j, 0.+0.j],
       [0.+0.j, 0.+0.j, 0.+0.j, 0.+0.j],
       [0.+0.j, 0.+0.j, 0.+0.j, 1.+0.j]], dtype=complex64)

And the noisy density matrix \(\rho\) with:

rho = cirq.final_density_matrix(my_noisy_circuit)
np.round(rho, 3)
array([[0.111+0.j, 0.   +0.j, 0.   +0.j, 0.   +0.j],
       [0.   +0.j, 0.222+0.j, 0.   +0.j, 0.   +0.j],
       [0.   +0.j, 0.   +0.j, 0.222+0.j, 0.   +0.j],
       [0.   +0.j, 0.   +0.j, 0.   +0.j, 0.444+0.j]], dtype=complex64)

Comparing the two different \( \rho \) 's you can see that the noise has impacted the amplitudes of the state (and consequently sampling probabilities). In the noiseless case you would always expect to sample the \( |11\rangle \) state. But in the noisy state there is now a nonzero probability of sampling \( |00\rangle \) or \( |01\rangle \) or \( |10\rangle \) as well:

"""Sample from my_noisy_circuit."""


def plot_samples(circuit):
    samples = cirq.sample(circuit +
                          cirq.measure(*circuit.all_qubits(), key='bits'),
                          repetitions=1000)
    freqs, _ = np.histogram(
        samples.data['bits'],
        bins=[i + 0.01 for i in range(-1, 2**len(my_qubits))])
    plt.figure(figsize=(10, 5))
    plt.title('Noisy Circuit Sampling')
    plt.xlabel('Bitstring')
    plt.ylabel('Frequency')
    plt.bar([i for i in range(2**len(my_qubits))],
            freqs,
            tick_label=['00', '01', '10', '11'])


plot_samples(my_noisy_circuit)

png

Without any noise you will always get \(|11\rangle\):

"""Sample from my_circuit."""
plot_samples(my_circuit)

png

If you increase the noise a little further it will become harder and harder to distinguish the desired behavior (sampling \(|11\rangle\) ) from the noise:

my_really_noisy_circuit = make_noisy(my_circuit, 0.75)
plot_samples(my_really_noisy_circuit)

png

2. Basic noise in TFQ

With this understanding of how noise can impact circuit execution, you can explore how noise works in TFQ. TensorFlow Quantum uses monte-carlo / trajectory based simulation as an alternative to density matrix simulation. This is because the memory complexity of density matrix simulation limits large simulations to being <= 20 qubits with traditional full density matrix simulation methods. Monte-carlo / trajectory trades this cost in memory for additional cost in time. The backend='noisy' option available to all tfq.layers.Sample, tfq.layers.SampledExpectation and tfq.layers.Expectation (In the case of Expectation this does add a required repetitions parameter).

2.1 Noisy sampling in TFQ

To recreate the above plots using TFQ and trajectory simulation you can use tfq.layers.Sample

"""Draw bitstring samples from `my_noisy_circuit`"""
bitstrings = tfq.layers.Sample(backend='noisy')(my_noisy_circuit,
                                                repetitions=1000)
numeric_values = np.einsum('ijk,k->ij',
                           bitstrings.to_tensor().numpy(), [1, 2])[0]
freqs, _ = np.histogram(numeric_values,
                        bins=[i + 0.01 for i in range(-1, 2**len(my_qubits))])
plt.figure(figsize=(10, 5))
plt.title('Noisy Circuit Sampling')
plt.xlabel('Bitstring')
plt.ylabel('Frequency')
plt.bar([i for i in range(2**len(my_qubits))],
        freqs,
        tick_label=['00', '01', '10', '11'])
<BarContainer object of 4 artists>

png

2.2 Noisy sample based expectation

To do noisy sample based expectation calculation you can use tfq.layers.SampleExpectation:

some_observables = [
    cirq.X(my_qubits[0]),
    cirq.Z(my_qubits[0]), 3.0 * cirq.Y(my_qubits[1]) + 1
]
some_observables
[cirq.X(cirq.GridQubit(0, 0)),
 cirq.Z(cirq.GridQubit(0, 0)),
 cirq.PauliSum(cirq.LinearDict({frozenset({(cirq.GridQubit(0, 1), cirq.Y)}): (3+0j), frozenset(): (1+0j)}))]

Compute the noiseless expectation estimates via sampling from the circuit:

noiseless_sampled_expectation = tfq.layers.SampledExpectation(
    backend='noiseless')(my_circuit,
                         operators=some_observables,
                         repetitions=10000)
noiseless_sampled_expectation.numpy()
array([[-0.0032, -1.    ,  0.988 ]], dtype=float32)

Compare those with the noisy versions:

noisy_sampled_expectation = tfq.layers.SampledExpectation(backend='noisy')(
    [my_noisy_circuit, my_really_noisy_circuit],
    operators=some_observables,
    repetitions=10000)
noisy_sampled_expectation.numpy()
/tmpfs/src/tf_docs_env/lib/python3.9/site-packages/keras/src/initializers/initializers.py:120: UserWarning: The initializer RandomUniform is unseeded and being called multiple times, which will return identical values each time (even if the initializer is unseeded). Please update your code to provide a seed to the initializer, or avoid using the same initializer instance more than once.
  warnings.warn(
array([[ 9.8000001e-03, -3.1739998e-01,  1.0275998e+00],
       [-4.0000069e-04,  1.0600001e-02,  1.0132000e+00]], dtype=float32)

You can see that the noise has particularly impacted the \(\langle \psi | Z | \psi \rangle\) accuracy, with my_really_noisy_circuit concentrating very quickly towards 0.

2.3 Noisy analytic expectation calculation

Doing noisy analytic expectation calculations is nearly identical to above:

noiseless_analytic_expectation = tfq.layers.Expectation(backend='noiseless')(
    my_circuit, operators=some_observables)
noiseless_analytic_expectation.numpy()
array([[ 1.9106853e-15, -1.0000000e+00,  1.0000002e+00]], dtype=float32)
noisy_analytic_expectation = tfq.layers.Expectation(backend='noisy')(
    [my_noisy_circuit, my_really_noisy_circuit],
    operators=some_observables,
    repetitions=10000)
noisy_analytic_expectation.numpy()
array([[ 1.9106853e-15, -3.4260002e-01,  1.0000000e+00],
       [ 1.9106855e-15, -6.8000001e-03,  1.0000001e+00]], dtype=float32)

3. Hybrid models and quantum data noise

Now that you have implemented some noisy circuit simulations in TFQ, you can experiment with how noise impacts quantum and hybrid quantum classical models, by comparing and contrasting their noisy vs noiseless performance. A good first check to see if a model or algorithm is robust to noise is to test under a circuit wide depolarizing model which looks something like this:

Where each time slice of the circuit (sometimes referred to as moment) has a depolarizing channel appended after each gate operation in that time slice. The depolarizing channel with apply one of \(\{X, Y, Z \}\) with probability \(p\) or apply nothing (keep the original operation) with probability \(1-p\).

3.1 Data

For this example you can use some prepared circuits in the tfq.datasets module as training data:

qubits = cirq.GridQubit.rect(1, 8)
circuits, labels, pauli_sums, _ = tfq.datasets.xxz_chain(qubits, 'closed')
circuits[0]
Downloading data from https://storage.googleapis.com/download.tensorflow.org/data/quantum/spin_systems/XXZ_chain.zip 
184449737/184449737 [==============================] - 1s 0us/step

Writing a small helper function will help to generate the data for the noisy vs noiseless case:

def get_data(qubits, depolarize_p=0.):
    """Return quantum data circuits and labels in `tf.Tensor` form."""
    circuits, labels, pauli_sums, _ = tfq.datasets.xxz_chain(qubits, 'closed')
    if depolarize_p >= 1e-5:
        circuits = [
            circuit.with_noise(cirq.depolarize(depolarize_p))
            for circuit in circuits
        ]
    tmp = list(zip(circuits, labels))
    random.shuffle(tmp)
    circuits_tensor = tfq.convert_to_tensor([x[0] for x in tmp])
    labels_tensor = tf.convert_to_tensor([x[1] for x in tmp])

    return circuits_tensor, labels_tensor

3.2 Define a model circuit

Now that you have quantum data in the form of circuits, you will need a circuit to model this data, like with the data you can write a helper function to generate this circuit optionally containing noise:

def modelling_circuit(qubits, depth, depolarize_p=0.):
    """A simple classifier circuit."""
    dim = len(qubits)
    ret = cirq.Circuit(cirq.H.on_each(*qubits))

    for i in range(depth):
        # Entangle layer.
        ret += cirq.Circuit(
            cirq.CX(q1, q2) for (q1, q2) in zip(qubits[::2], qubits[1::2]))
        ret += cirq.Circuit(
            cirq.CX(q1, q2) for (q1, q2) in zip(qubits[1::2], qubits[2::2]))
        # Learnable rotation layer.
        # i_params = sympy.symbols(f'layer-{i}-0:{dim}')
        param = sympy.Symbol(f'layer-{i}')
        single_qb = cirq.X
        if i % 2 == 1:
            single_qb = cirq.Y
        ret += cirq.Circuit(single_qb(q)**param for q in qubits)

    if depolarize_p >= 1e-5:
        ret = ret.with_noise(cirq.depolarize(depolarize_p))

    return ret, [op(q) for q in qubits for op in [cirq.X, cirq.Y, cirq.Z]]


modelling_circuit(qubits, 3)[0]

3.3 Model building and training

With your data and model circuit built, the final helper function you will need is one that can assemble both a noisy or a noiseless hybrid quantum tf.keras.Model:

def build_keras_model(qubits, depolarize_p=0.):
    """Prepare a noisy hybrid quantum classical Keras model."""
    spin_input = tf.keras.Input(shape=(), dtype=tf.dtypes.string)

    circuit_and_readout = modelling_circuit(qubits, 4, depolarize_p)
    if depolarize_p >= 1e-5:
        quantum_model = tfq.layers.NoisyPQC(*circuit_and_readout,
                                            sample_based=False,
                                            repetitions=10)(spin_input)
    else:
        quantum_model = tfq.layers.PQC(*circuit_and_readout)(spin_input)

    intermediate = tf.keras.layers.Dense(4, activation='sigmoid')(quantum_model)
    post_process = tf.keras.layers.Dense(1)(intermediate)

    return tf.keras.Model(inputs=[spin_input], outputs=[post_process])

4. Compare performance

4.1 Noiseless baseline

With your data generation and model building code, you can now compare and contrast model performance in the noiseless and noisy settings, first you can run a reference noiseless training:

training_histories = dict()
depolarize_p = 0.
n_epochs = 50
phase_classifier = build_keras_model(qubits, depolarize_p)

phase_classifier.compile(
    optimizer=tf.keras.optimizers.Adam(learning_rate=0.02),
    loss=tf.keras.losses.BinaryCrossentropy(from_logits=True),
    metrics=['accuracy'])

# Show the keras plot of the model
tf.keras.utils.plot_model(phase_classifier, show_shapes=True, dpi=70)

png

noiseless_data, noiseless_labels = get_data(qubits, depolarize_p)
training_histories['noiseless'] = phase_classifier.fit(x=noiseless_data,
                                                       y=noiseless_labels,
                                                       batch_size=16,
                                                       epochs=n_epochs,
                                                       validation_split=0.15,
                                                       verbose=1)
Epoch 1/50
4/4 [==============================] - 1s 118ms/step - loss: 0.7298 - accuracy: 0.5000 - val_loss: 0.7837 - val_accuracy: 0.3333
Epoch 2/50
4/4 [==============================] - 0s 70ms/step - loss: 0.7085 - accuracy: 0.5000 - val_loss: 0.7530 - val_accuracy: 0.3333
Epoch 3/50
4/4 [==============================] - 0s 57ms/step - loss: 0.6998 - accuracy: 0.5000 - val_loss: 0.7229 - val_accuracy: 0.3333
Epoch 4/50
4/4 [==============================] - 0s 59ms/step - loss: 0.6926 - accuracy: 0.5000 - val_loss: 0.7028 - val_accuracy: 0.3333
Epoch 5/50
4/4 [==============================] - 0s 61ms/step - loss: 0.6901 - accuracy: 0.5000 - val_loss: 0.6891 - val_accuracy: 0.3333
Epoch 6/50
4/4 [==============================] - 0s 59ms/step - loss: 0.6877 - accuracy: 0.5000 - val_loss: 0.6838 - val_accuracy: 0.3333
Epoch 7/50
4/4 [==============================] - 0s 58ms/step - loss: 0.6868 - accuracy: 0.5000 - val_loss: 0.6823 - val_accuracy: 0.3333
Epoch 8/50
4/4 [==============================] - 0s 59ms/step - loss: 0.6844 - accuracy: 0.5000 - val_loss: 0.6786 - val_accuracy: 0.3333
Epoch 9/50
4/4 [==============================] - 0s 56ms/step - loss: 0.6823 - accuracy: 0.5000 - val_loss: 0.6762 - val_accuracy: 0.3333
Epoch 10/50
4/4 [==============================] - 0s 59ms/step - loss: 0.6794 - accuracy: 0.5000 - val_loss: 0.6727 - val_accuracy: 0.3333
Epoch 11/50
4/4 [==============================] - 0s 56ms/step - loss: 0.6759 - accuracy: 0.5000 - val_loss: 0.6688 - val_accuracy: 0.3333
Epoch 12/50
4/4 [==============================] - 0s 57ms/step - loss: 0.6707 - accuracy: 0.5000 - val_loss: 0.6629 - val_accuracy: 0.3333
Epoch 13/50
4/4 [==============================] - 0s 55ms/step - loss: 0.6647 - accuracy: 0.5000 - val_loss: 0.6589 - val_accuracy: 0.3333
Epoch 14/50
4/4 [==============================] - 0s 56ms/step - loss: 0.6566 - accuracy: 0.5000 - val_loss: 0.6517 - val_accuracy: 0.3333
Epoch 15/50
4/4 [==============================] - 0s 56ms/step - loss: 0.6459 - accuracy: 0.5000 - val_loss: 0.6410 - val_accuracy: 0.3333
Epoch 16/50
4/4 [==============================] - 0s 57ms/step - loss: 0.6345 - accuracy: 0.5000 - val_loss: 0.6251 - val_accuracy: 0.3333
Epoch 17/50
4/4 [==============================] - 0s 58ms/step - loss: 0.6189 - accuracy: 0.5000 - val_loss: 0.6094 - val_accuracy: 0.3333
Epoch 18/50
4/4 [==============================] - 0s 56ms/step - loss: 0.6010 - accuracy: 0.5000 - val_loss: 0.5950 - val_accuracy: 0.3333
Epoch 19/50
4/4 [==============================] - 0s 57ms/step - loss: 0.5816 - accuracy: 0.5000 - val_loss: 0.5756 - val_accuracy: 0.3333
Epoch 20/50
4/4 [==============================] - 0s 56ms/step - loss: 0.5591 - accuracy: 0.5625 - val_loss: 0.5499 - val_accuracy: 0.5000
Epoch 21/50
4/4 [==============================] - 0s 57ms/step - loss: 0.5351 - accuracy: 0.6562 - val_loss: 0.5228 - val_accuracy: 0.6667
Epoch 22/50
4/4 [==============================] - 0s 56ms/step - loss: 0.5083 - accuracy: 0.7188 - val_loss: 0.4993 - val_accuracy: 0.6667
Epoch 23/50
4/4 [==============================] - 0s 56ms/step - loss: 0.4820 - accuracy: 0.7656 - val_loss: 0.4726 - val_accuracy: 0.7500
Epoch 24/50
4/4 [==============================] - 0s 55ms/step - loss: 0.4543 - accuracy: 0.7812 - val_loss: 0.4439 - val_accuracy: 0.9167
Epoch 25/50
4/4 [==============================] - 0s 57ms/step - loss: 0.4276 - accuracy: 0.7969 - val_loss: 0.4168 - val_accuracy: 0.9167
Epoch 26/50
4/4 [==============================] - 0s 56ms/step - loss: 0.4028 - accuracy: 0.8438 - val_loss: 0.3908 - val_accuracy: 0.9167
Epoch 27/50
4/4 [==============================] - 0s 54ms/step - loss: 0.3788 - accuracy: 0.8438 - val_loss: 0.3682 - val_accuracy: 0.9167
Epoch 28/50
4/4 [==============================] - 0s 56ms/step - loss: 0.3550 - accuracy: 0.8750 - val_loss: 0.3467 - val_accuracy: 0.9167
Epoch 29/50
4/4 [==============================] - 0s 57ms/step - loss: 0.3338 - accuracy: 0.8750 - val_loss: 0.3274 - val_accuracy: 0.9167
Epoch 30/50
4/4 [==============================] - 0s 56ms/step - loss: 0.3154 - accuracy: 0.9062 - val_loss: 0.3084 - val_accuracy: 0.9167
Epoch 31/50
4/4 [==============================] - 0s 56ms/step - loss: 0.2965 - accuracy: 0.9062 - val_loss: 0.2914 - val_accuracy: 0.9167
Epoch 32/50
4/4 [==============================] - 0s 55ms/step - loss: 0.2805 - accuracy: 0.9062 - val_loss: 0.2764 - val_accuracy: 1.0000
Epoch 33/50
4/4 [==============================] - 0s 57ms/step - loss: 0.2655 - accuracy: 0.9062 - val_loss: 0.2626 - val_accuracy: 1.0000
Epoch 34/50
4/4 [==============================] - 0s 56ms/step - loss: 0.2566 - accuracy: 0.9375 - val_loss: 0.2516 - val_accuracy: 1.0000
Epoch 35/50
4/4 [==============================] - 0s 56ms/step - loss: 0.2391 - accuracy: 0.9219 - val_loss: 0.2390 - val_accuracy: 1.0000
Epoch 36/50
4/4 [==============================] - 0s 55ms/step - loss: 0.2287 - accuracy: 0.9219 - val_loss: 0.2301 - val_accuracy: 1.0000
Epoch 37/50
4/4 [==============================] - 0s 55ms/step - loss: 0.2204 - accuracy: 0.9219 - val_loss: 0.2189 - val_accuracy: 1.0000
Epoch 38/50
4/4 [==============================] - 0s 55ms/step - loss: 0.2091 - accuracy: 0.9375 - val_loss: 0.2115 - val_accuracy: 1.0000
Epoch 39/50
4/4 [==============================] - 0s 56ms/step - loss: 0.2035 - accuracy: 0.9531 - val_loss: 0.2057 - val_accuracy: 1.0000
Epoch 40/50
4/4 [==============================] - 0s 56ms/step - loss: 0.1946 - accuracy: 0.9219 - val_loss: 0.1980 - val_accuracy: 1.0000
Epoch 41/50
4/4 [==============================] - 0s 56ms/step - loss: 0.1873 - accuracy: 0.9531 - val_loss: 0.1972 - val_accuracy: 1.0000
Epoch 42/50
4/4 [==============================] - 0s 56ms/step - loss: 0.1905 - accuracy: 0.9219 - val_loss: 0.1897 - val_accuracy: 1.0000
Epoch 43/50
4/4 [==============================] - 0s 57ms/step - loss: 0.1756 - accuracy: 0.9688 - val_loss: 0.1868 - val_accuracy: 1.0000
Epoch 44/50
4/4 [==============================] - 0s 57ms/step - loss: 0.1725 - accuracy: 0.9375 - val_loss: 0.1729 - val_accuracy: 1.0000
Epoch 45/50
4/4 [==============================] - 0s 59ms/step - loss: 0.1626 - accuracy: 0.9531 - val_loss: 0.1773 - val_accuracy: 1.0000
Epoch 46/50
4/4 [==============================] - 0s 56ms/step - loss: 0.1574 - accuracy: 0.9688 - val_loss: 0.1754 - val_accuracy: 1.0000
Epoch 47/50
4/4 [==============================] - 0s 56ms/step - loss: 0.1527 - accuracy: 0.9688 - val_loss: 0.1675 - val_accuracy: 1.0000
Epoch 48/50
4/4 [==============================] - 0s 55ms/step - loss: 0.1479 - accuracy: 0.9531 - val_loss: 0.1580 - val_accuracy: 1.0000
Epoch 49/50
4/4 [==============================] - 0s 53ms/step - loss: 0.1463 - accuracy: 0.9531 - val_loss: 0.1556 - val_accuracy: 1.0000
Epoch 50/50
4/4 [==============================] - 0s 59ms/step - loss: 0.1401 - accuracy: 0.9531 - val_loss: 0.1521 - val_accuracy: 1.0000

And explore the results and accuracy:

loss_plotter = tfdocs.plots.HistoryPlotter(metric='loss', smoothing_std=10)
loss_plotter.plot(training_histories)

png

acc_plotter = tfdocs.plots.HistoryPlotter(metric='accuracy', smoothing_std=10)
acc_plotter.plot(training_histories)

png

4.2 Noisy comparison

Now you can build a new model with noisy structure and compare to the above, the code is nearly identical:

depolarize_p = 0.001
n_epochs = 50
noisy_phase_classifier = build_keras_model(qubits, depolarize_p)

noisy_phase_classifier.compile(
    optimizer=tf.keras.optimizers.Adam(learning_rate=0.02),
    loss=tf.keras.losses.BinaryCrossentropy(from_logits=True),
    metrics=['accuracy'])

# Show the keras plot of the model
tf.keras.utils.plot_model(noisy_phase_classifier, show_shapes=True, dpi=70)

png

noisy_data, noisy_labels = get_data(qubits, depolarize_p)
training_histories['noisy'] = noisy_phase_classifier.fit(x=noisy_data,
                                                         y=noisy_labels,
                                                         batch_size=16,
                                                         epochs=n_epochs,
                                                         validation_split=0.15,
                                                         verbose=1)
Epoch 1/50
4/4 [==============================] - 8s 1s/step - loss: 0.8052 - accuracy: 0.4844 - val_loss: 0.5810 - val_accuracy: 0.7500
Epoch 2/50
4/4 [==============================] - 5s 1s/step - loss: 0.7397 - accuracy: 0.5625 - val_loss: 0.6036 - val_accuracy: 0.2500
Epoch 3/50
4/4 [==============================] - 5s 1s/step - loss: 0.7111 - accuracy: 0.5156 - val_loss: 0.6346 - val_accuracy: 0.2500
Epoch 4/50
4/4 [==============================] - 5s 1s/step - loss: 0.6924 - accuracy: 0.5156 - val_loss: 0.6691 - val_accuracy: 0.2500
Epoch 5/50
4/4 [==============================] - 5s 1s/step - loss: 0.6849 - accuracy: 0.5156 - val_loss: 0.6872 - val_accuracy: 0.2500
Epoch 6/50
4/4 [==============================] - 5s 1s/step - loss: 0.6808 - accuracy: 0.5156 - val_loss: 0.7064 - val_accuracy: 0.2500
Epoch 7/50
4/4 [==============================] - 5s 1s/step - loss: 0.6777 - accuracy: 0.5156 - val_loss: 0.7077 - val_accuracy: 0.2500
Epoch 8/50
4/4 [==============================] - 5s 1s/step - loss: 0.6684 - accuracy: 0.5156 - val_loss: 0.7053 - val_accuracy: 0.2500
Epoch 9/50
4/4 [==============================] - 5s 1s/step - loss: 0.6608 - accuracy: 0.5156 - val_loss: 0.6885 - val_accuracy: 0.2500
Epoch 10/50
4/4 [==============================] - 5s 1s/step - loss: 0.6525 - accuracy: 0.5156 - val_loss: 0.6801 - val_accuracy: 0.2500
Epoch 11/50
4/4 [==============================] - 5s 1s/step - loss: 0.6423 - accuracy: 0.5156 - val_loss: 0.6526 - val_accuracy: 0.2500
Epoch 12/50
4/4 [==============================] - 5s 1s/step - loss: 0.6366 - accuracy: 0.5156 - val_loss: 0.6339 - val_accuracy: 0.2500
Epoch 13/50
4/4 [==============================] - 5s 1s/step - loss: 0.6162 - accuracy: 0.5156 - val_loss: 0.6162 - val_accuracy: 0.2500
Epoch 14/50
4/4 [==============================] - 5s 1s/step - loss: 0.6107 - accuracy: 0.5156 - val_loss: 0.6106 - val_accuracy: 0.2500
Epoch 15/50
4/4 [==============================] - 5s 1s/step - loss: 0.6042 - accuracy: 0.5312 - val_loss: 0.5797 - val_accuracy: 0.3333
Epoch 16/50
4/4 [==============================] - 5s 1s/step - loss: 0.5780 - accuracy: 0.5938 - val_loss: 0.5645 - val_accuracy: 0.4167
Epoch 17/50
4/4 [==============================] - 5s 1s/step - loss: 0.5636 - accuracy: 0.5938 - val_loss: 0.5402 - val_accuracy: 0.3333
Epoch 18/50
4/4 [==============================] - 5s 1s/step - loss: 0.5445 - accuracy: 0.5938 - val_loss: 0.5523 - val_accuracy: 0.3333
Epoch 19/50
4/4 [==============================] - 5s 1s/step - loss: 0.5259 - accuracy: 0.6562 - val_loss: 0.5021 - val_accuracy: 0.5000
Epoch 20/50
4/4 [==============================] - 5s 1s/step - loss: 0.5248 - accuracy: 0.6875 - val_loss: 0.5108 - val_accuracy: 0.4167
Epoch 21/50
4/4 [==============================] - 5s 1s/step - loss: 0.4959 - accuracy: 0.7188 - val_loss: 0.4723 - val_accuracy: 0.5833
Epoch 22/50
4/4 [==============================] - 5s 1s/step - loss: 0.4778 - accuracy: 0.7969 - val_loss: 0.4536 - val_accuracy: 0.5833
Epoch 23/50
4/4 [==============================] - 5s 1s/step - loss: 0.4577 - accuracy: 0.7812 - val_loss: 0.4613 - val_accuracy: 0.5833
Epoch 24/50
4/4 [==============================] - 5s 1s/step - loss: 0.4446 - accuracy: 0.8125 - val_loss: 0.3879 - val_accuracy: 0.8333
Epoch 25/50
4/4 [==============================] - 5s 1s/step - loss: 0.4285 - accuracy: 0.7969 - val_loss: 0.4610 - val_accuracy: 0.6667
Epoch 26/50
4/4 [==============================] - 5s 1s/step - loss: 0.4075 - accuracy: 0.8281 - val_loss: 0.4164 - val_accuracy: 0.7500
Epoch 27/50
4/4 [==============================] - 5s 1s/step - loss: 0.3982 - accuracy: 0.8594 - val_loss: 0.3915 - val_accuracy: 0.6667
Epoch 28/50
4/4 [==============================] - 5s 1s/step - loss: 0.3637 - accuracy: 0.8594 - val_loss: 0.3836 - val_accuracy: 0.7500
Epoch 29/50
4/4 [==============================] - 5s 1s/step - loss: 0.3571 - accuracy: 0.9062 - val_loss: 0.2998 - val_accuracy: 0.8333
Epoch 30/50
4/4 [==============================] - 5s 1s/step - loss: 0.3404 - accuracy: 0.8750 - val_loss: 0.3138 - val_accuracy: 0.8333
Epoch 31/50
4/4 [==============================] - 5s 1s/step - loss: 0.3155 - accuracy: 0.9219 - val_loss: 0.3326 - val_accuracy: 0.8333
Epoch 32/50
4/4 [==============================] - 5s 1s/step - loss: 0.3263 - accuracy: 0.8438 - val_loss: 0.3270 - val_accuracy: 0.7500
Epoch 33/50
4/4 [==============================] - 5s 1s/step - loss: 0.3413 - accuracy: 0.8594 - val_loss: 0.3074 - val_accuracy: 0.7500
Epoch 34/50
4/4 [==============================] - 5s 1s/step - loss: 0.2967 - accuracy: 0.8750 - val_loss: 0.3469 - val_accuracy: 0.6667
Epoch 35/50
4/4 [==============================] - 5s 1s/step - loss: 0.2958 - accuracy: 0.9219 - val_loss: 0.2479 - val_accuracy: 0.8333
Epoch 36/50
4/4 [==============================] - 5s 1s/step - loss: 0.3244 - accuracy: 0.8281 - val_loss: 0.2516 - val_accuracy: 0.8333
Epoch 37/50
4/4 [==============================] - 5s 1s/step - loss: 0.3123 - accuracy: 0.8438 - val_loss: 0.2954 - val_accuracy: 0.7500
Epoch 38/50
4/4 [==============================] - 5s 1s/step - loss: 0.2601 - accuracy: 0.9219 - val_loss: 0.3378 - val_accuracy: 0.8333
Epoch 39/50
4/4 [==============================] - 5s 1s/step - loss: 0.3066 - accuracy: 0.8906 - val_loss: 0.1861 - val_accuracy: 0.9167
Epoch 40/50
4/4 [==============================] - 5s 1s/step - loss: 0.2860 - accuracy: 0.8594 - val_loss: 0.2130 - val_accuracy: 0.8333
Epoch 41/50
4/4 [==============================] - 5s 1s/step - loss: 0.2492 - accuracy: 0.8906 - val_loss: 0.1687 - val_accuracy: 1.0000
Epoch 42/50
4/4 [==============================] - 5s 1s/step - loss: 0.2803 - accuracy: 0.9062 - val_loss: 0.2032 - val_accuracy: 0.9167
Epoch 43/50
4/4 [==============================] - 5s 1s/step - loss: 0.2536 - accuracy: 0.8906 - val_loss: 0.2044 - val_accuracy: 0.8333
Epoch 44/50
4/4 [==============================] - 5s 1s/step - loss: 0.2425 - accuracy: 0.8906 - val_loss: 0.1940 - val_accuracy: 0.9167
Epoch 45/50
4/4 [==============================] - 5s 1s/step - loss: 0.2631 - accuracy: 0.8438 - val_loss: 0.1779 - val_accuracy: 0.9167
Epoch 46/50
4/4 [==============================] - 5s 1s/step - loss: 0.2230 - accuracy: 0.9375 - val_loss: 0.2712 - val_accuracy: 0.8333
Epoch 47/50
4/4 [==============================] - 5s 1s/step - loss: 0.2723 - accuracy: 0.8906 - val_loss: 0.1961 - val_accuracy: 0.9167
Epoch 48/50
4/4 [==============================] - 5s 1s/step - loss: 0.2308 - accuracy: 0.8906 - val_loss: 0.1890 - val_accuracy: 0.8333
Epoch 49/50
4/4 [==============================] - 5s 1s/step - loss: 0.2067 - accuracy: 0.9531 - val_loss: 0.1799 - val_accuracy: 1.0000
Epoch 50/50
4/4 [==============================] - 5s 1s/step - loss: 0.2863 - accuracy: 0.8594 - val_loss: 0.2383 - val_accuracy: 0.8333
loss_plotter.plot(training_histories)

png

acc_plotter.plot(training_histories)

png