-
Notifications
You must be signed in to change notification settings - Fork 3
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #9 from qiboteam/expectation
Divide differentiation and expectation modules
- Loading branch information
Showing
5 changed files
with
748 additions
and
53 deletions.
There are no files selected for viewing
File renamed without changes.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,142 @@ | ||
"""Compute expectation values of target observables with the freedom of setting any qibo's backend.""" | ||
|
||
from typing import List, Optional, Union | ||
|
||
import qibo | ||
from qibo.backends import construct_backend | ||
from qibo.config import raise_error | ||
|
||
from qiboml.backends import TensorflowBackend | ||
|
||
|
||
def expectation( | ||
observable: qibo.hamiltonians.Hamiltonian, | ||
circuit: qibo.Circuit, | ||
initial_state: Optional[Union[List, qibo.Circuit]] = None, | ||
nshots: int = None, | ||
backend: str = "qibojit", | ||
differentiation_rule: Optional[callable] = None, | ||
): | ||
""" | ||
Compute the expectation value of ``observable`` over the state obtained by | ||
executing ``circuit`` starting from ``initial_state``. The final state is | ||
reconstructed from ``nshots`` execution of ``circuit`` on the selected ``backend``. | ||
In addition, a differentiation rule can be set, which is going to be integrated | ||
within the used high-level framework. For example, if TensorFlow is used | ||
in the user code and one parameter shift rule is selected as differentiation | ||
rule, the expectation value is computed informing the TensorFlow graph to | ||
use as gradient the output of the parameter shift rule executed on the selected | ||
backend. | ||
Args: | ||
observable (qibo.Hamiltonian): the observable whose expectation value has | ||
to be computed. | ||
circuit (qibo.Circuit): quantum circuit returning the final state over which | ||
the expectation value of ``observable`` is computed. | ||
initial_state (Optional[Union[List, qibo.Circuit]]): initial state on which | ||
the quantum circuit is applied. | ||
nshots (int): number of times the quantum circuit is executed. Increasing | ||
the number of shots will reduce the variance of the estimated expectation | ||
value while increasing the computational cost of the operation. | ||
backend (str): backend on which the circuit is executed. This same backend | ||
is used if the chosen differentiation rule makes use of expectation | ||
values. | ||
differentiation_rule (Optional[callable]): the chosen differentiation | ||
rule. It can be selected among the methods implemented in | ||
``qiboml.differentiation``. | ||
""" | ||
|
||
# read the frontend user choice | ||
frontend = observable.backend | ||
exec_backend = construct_backend(backend) | ||
|
||
kwargs = dict( | ||
observable=observable, | ||
circuit=circuit, | ||
initial_state=initial_state, | ||
nshots=nshots, | ||
backend=backend, | ||
differentiation_rule=differentiation_rule, | ||
exec_backend=exec_backend, | ||
) | ||
|
||
if differentiation_rule is not None: | ||
if isinstance(frontend, TensorflowBackend): | ||
return _with_tf(**kwargs) | ||
|
||
elif nshots is None: | ||
return _exact(observable, circuit, initial_state, exec_backend) | ||
else: | ||
return _with_shots(observable, circuit, initial_state, nshots, exec_backend) | ||
|
||
raise_error( | ||
NotImplementedError, | ||
"Only tensorflow automatic differentiation is supported at this moment.", | ||
) | ||
|
||
|
||
def _exact(observable, circuit, initial_state, exec_backend): | ||
"""Helper function to compute exact expectation values.""" | ||
return observable.expectation( | ||
exec_backend.execute_circuit( | ||
circuit=circuit, initial_state=initial_state | ||
).state() | ||
) | ||
|
||
|
||
def _with_shots(observable, circuit, initial_state, nshots, exec_backend): | ||
"""Helper function to compute expectation values from samples.""" | ||
return exec_backend.execute_circuit( | ||
circuit=circuit, initial_state=initial_state, nshots=nshots | ||
).expectation_from_samples(observable) | ||
|
||
|
||
def _with_tf( | ||
observable, | ||
circuit, | ||
initial_state, | ||
nshots, | ||
backend, | ||
differentiation_rule, | ||
): | ||
""" | ||
Compute expectation sample integrating the custom differentiation rule with | ||
TensorFlow's automatic differentiation. | ||
""" | ||
import tensorflow as tf # pylint: disable=import-error | ||
|
||
params = circuit.get_parameters() | ||
nparams = len(params) | ||
|
||
exec_backend = construct_backend(backend) | ||
|
||
@tf.custom_gradient | ||
def _expectation(params): | ||
params = tf.Variable(params) | ||
|
||
def grad(upstream): | ||
gradients = [] | ||
for p in range(nparams): | ||
gradients.append( | ||
upstream | ||
* differentiation_rule( | ||
circuit=circuit, | ||
hamiltonian=observable, | ||
parameter_index=p, | ||
initial_state=initial_state, | ||
nshots=nshots, | ||
backend=backend, | ||
) | ||
) | ||
return gradients | ||
|
||
if nshots is None: | ||
expval = _exact(observable, circuit, initial_state, exec_backend) | ||
else: | ||
expval = _with_shots( | ||
observable, circuit, initial_state, nshots, exec_backend | ||
) | ||
|
||
return expval, grad | ||
|
||
return _expectation(params) |
Oops, something went wrong.