QuTiP-2 0 0
QuTiP-2 0 0
QuTiP-2 0 0
Release 2.0
1 Frontmatter 1
1.1 About This Documentation . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 1
1.2 Citing This Project . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 1
1.3 Funding . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 1
1.4 Contributing to QuTiP . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 2
2 About QuTiP 3
2.1 Brief Description . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 3
2.2 Whats New in QuTiP Version 2.0 . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 4
3 Installation 5
3.1 General Installation Requirements . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 5
3.2 Installation on Ubuntu Linux . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 5
3.3 Installation on Mac OS X (10.6+) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 6
3.4 Verifying the Installation . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 7
3.5 Checking Version Information via the About Box . . . . . . . . . . . . . . . . . . . . . . . . . . . . 8
i
7 Change Log 195
7.1 Version 2.0.0 [SVN-2354] (June 01, 2012): . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 195
7.2 Version 1.1.4 [fixes backported to SVN-1450] (May 28, 2012): . . . . . . . . . . . . . . . . . . . . 196
7.3 Version 1.1.3 [svn-1450] (November 21, 2011): . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 196
7.4 Version 1.1.2 [svn-1218] (October 27, 2011) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 196
7.5 Version 1.1.1 [svn-1210] (October 25, 2011) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 196
7.6 Version 1.1.0 [svn-1097] (October 04, 2011) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 197
7.7 Version 1.0.0 [svn-1021] (July 29, 2011) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 197
8 Developers 199
8.1 Lead Developers . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 199
8.2 Contributors . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 199
ii
CHAPTER
ONE
FRONTMATTER
These pages contains automatically generated API documentation for QuTiP. A PDF version of this text is included
with QuTiP, and may be found on the downloads page.
For more information see the QuTiP project web page.
Author P.D. Nation
Address Department of Physics, Korea University, Seongbuk-Gu Seoul, 136-701 South Korea
Author J.R. Johansson
Address Digital Materials Laboratory Advanced Science Institute, RIKEN, Wako-shi Saitama, 351-0198
Japan
version 2.0
status In Development
copyright This documentation is in the public domain. You may do with it as your heart desires.
1.3 Funding
The development of QuTiP has been partially supported by the Japanese Society for the Promotion of Science Foreign
Postdoctoral Fellowship Program under grants P11202 (PDN) and P11501 (JRJ). Additional funding comes from
RIKEN and Kakenhi grant Nos. 2301202 (PDN) and 2302501 (JRJ).
1
QuTiP: The Quantum Toolbox in Python, Release 2.0
Like any open-source project, we welcome anyone who is interested in helping us make QuTiP the best package for
simulating quantum optics-like systems. Anyone who contributes will be duly recognized. Even small contributions
are noted. See Contributors for a list of people who have helped in one way or another. If you are interested, please
drop us a line at the QuTiP discussion group webpage.
2 Chapter 1. Frontmatter
CHAPTER
TWO
ABOUT QUTIP
Every quantum system encountered in the real world is an open quantum system. For although much care is taken
experimentally to eliminate the unwanted influence of external interactions, there remains, if ever so slight, a coupling
between the system of interest and the external world. In addition, any measurement performed on the system nec-
essarily involves coupling to the measuring device, therefore introducing an additional source of external influence.
Consequently, developing the necessary tools, both theoretical and numerical, to account for the interactions between
a system and its environment is an essential step in understanding the dynamics of quantum systems.
In general, for all but the most basic of Hamiltonians, an analytical description of the system dynamics is not possible,
and one must resort to numerical simulations of the equations of motion. In absence of a quantum computer, these
simulations must be carried out using classical computing techniques, where the exponentially increasing dimension-
ality of the underlying Hilbert space severely limits the size of system that can be efficiently simulated. However, in
many fields such as quantum optics, trapped ions, superconducting circuit devices, and most recently nanomechanical
systems, it is possible to design systems using a small number of effective oscillator and spin components, excited by
a small number of quanta, that are amenable to classical simulation in a truncated Hilbert space.
The Quantum Toolbox in Python, or QuTiP, is a fully open-source implementation of a framework written in the
Python programming language designed for simulating the open quantum dynamics for systems such as those listed
above. This framework distinguishes itself from the other available software solutions by providing the following
advantages:
QuTiP relies on completely open-source software. You are free to modify and use it as you wish with no
licensing fees.
QuTiP is based on the Python scripting language, providing easy to read, fast code generation without the need
to compile after modification.
The numerics underlying QuTiP are time-tested algorithms that run at C-code speeds, thanks to the Numpy and
Scipy libraries, and are based on many of the same algorithms used in propriety software.
QuTiP allows for solving the dynamics of Hamiltonians with arbitrary time-dependence, including collapse
operators.
Time-dependent problems can be automatically compiled into C-code at run-time for increased performance.
3
QuTiP: The Quantum Toolbox in Python, Release 2.0
Takes advantage of the multiple processing cores found in essentially all modern computers.
QuTiP was designed from the start to require a minimal learning curve for those users who have experience
using the popular quantum optics toolbox by Sze M. Tan.
Includes the ability to create high-quality plots, and animations, using the excellent Matplotlib package.
The second version of QuTiP has seen many improvements in the performance of the original code base, as well as
the addition of several new routines supporting a wide range of functionality. Some of the highlights of this release
include:
QuTiP now includes solvers for both Floquet and Bloch-Redfield master equations.
The Lindblad master equation and monte-carlo solvers allow for time-dependent collapse operators.
It is possible to automatically compile time-dependent problems into c-code using Cython (if installed).
Python functions can be used to create arbitrary time-dependent Hamiltonians and collapse operators.
Solvers now return Odedata objects containing all simulation results and parameters, simplifying the saving of
simulation results.
Important: This breaks compatibility with QuTiP version 1.x. See The Odedata Class and Dynamical Simulation
Results for further details.
mesolve and mcsolve can reuse Hamiltonian data when only the initial state, or time-dependent arguments, need
to be changed.
QuTiP includes functions for creating random quantum states and operators.
The generation and manipulation of quantum objects is now more efficient.
Quantum objects have basis transformation and matrix element calculations as built-in methods.
The quantum object eigensolver can use sparse solvers.
The partial-trace (ptrace) function is up to 20x faster.
The Bloch sphere can now be used with the Matplotlib animation function, and embedded as a subplot in a
figure.
QuTiP has built-in functions for saving quantum objects and data arrays.
The steady-state solver has been further optimized for sparse matrices, and can handle much larger system
Hamiltonians.
The steady-state solver can use the iterative bi-conjugate gradient method instead of a direct solver.
There are three new entropy functions for concurrence, mutual information, and conditional entropy.
Correlation functions have been combined under a single function.
The operator norm can now be set to trace, Frobius, one, or max norm.
Global QuTiP settings can now be modified.
QuTiP includes a collection of unit tests for verifying the installation.
Demos window now lets you copy and paste code from each example.
THREE
INSTALLATION
Important: QuTiP is designed for use on Unix based platforms such as Linux and Mac OSX only, and is known to
break when running under the Microsoft Windows operating system. As such, we do not provide any support for these
platforms. If they ever figure out that matrices do not have floating-point dimensions, then we might reconsider.
5
QuTiP: The Quantum Toolbox in Python, Release 2.0
3.2.1 Ubuntu 11.04 and Lower (skip this step in Ubuntu 11.10+):
These releases do not have Matplotlib>=1.0, and we therefore need to add the unofficial repository:
>>> sudo add-apt-repository ppa:bgamari/matplotlib-unofficial
>>> sudo apt-get update
Regardless of Ubuntu version, we are now in a position to install the plotting library:
>>> sudo apt-get install python-matplotlib
Note: On some versions of Ubuntu you might have to configure Matplotlib to use the GTKAgg or Qt4Agg back-
ends instead of the default TkAgg backend. To do this, edit /etc/matplotlibrc, and change backend: TkAgg to
backend: GTKAgg or backend: Qt4Agg.
If you have not done so already, install the Apple XCode developer tools from the Apple App Store.
1
3.3.1 Setup Using Macports:
On the Mac, it is recommended that you install the required libraries via MacPorts. After installation, the necessary
ports for QuTiP may be installed via:
>>> sudo port install py27-scipy
>>> sudo port install py27-matplotlib +latex
or:
>>> sudo port install py27-pyqt4
1 Installing QuTiP via Macports will take a long time as each of the QuTiP dependencies is build from source code. The advantage is that, after
installation, everything is more or less guaranteed to work. However, if you have a hot date waiting for you, then we do not recommend this path.
Or course, if you are reading this guide, this may not be the case.
6 Chapter 3. Installation
QuTiP: The Quantum Toolbox in Python, Release 2.0
Note: The next step is optional, but is recommended if you plan to use the string (Cython) based time-dependent
format. See Solving Time-dependent Hamiltonians.
Finally, we want to set the macports compiler to the vanilla GCC version. From the command line type:
>>> port select gcc
We want to set the the compiler to the gcc4x compiler, where x is the highest number available, in this case mp-gcc45
(the mp- does not matter). To do this type:
>>> sudo port select gcc mp-gcc45
A second option is to install the required Python packages using the SciPy Superpack. Further information on installing
the superpack can be found on the SciPy Downloads page. Note that, if you choose this option, the GUI elements of
QuTiP will not be available without further installing either the PyQt4 or PySide packages separately.
Finally, one can also use the Enthought Python Distribution version 7.1 or higher to satisfy the QuTiP dependencies.
No matter which installation path you choose, installing QuTiP is the same as on linux. From the QuTiP directory:
>>> sudo python setup.py install
QuTiP now includes a collection of built-in test scripts to verify that the installation was indeed successful. To run the
suite of tests scripts, after installing QuTiP call:
If successful, these tests indicate that all of the QuTiP functions are working properly. If any errors occur, please check
that your have installed all of the required modules. See the next section on how to check the installed versions of
the QuTiP dependencies. If these tests still fail, then head on over to the QuTiP Discussion Board and post a message
detailing your particular issue.
To further verify that all of the QuTiP components are working, you can run the collection of examples built into
QuTiP as discussed in the QuTiP Example Scripts section of the guide.
QuTiP includes a graphical about box for viewing information about QuTiP, and the important dependencies in-
stalled on your system. To view the about box type:
>>> about()
that will pop-up a window similar to the one shown below. If instead you get command-line output, then your PyQt
or PySide graphics are not installed properly or unavailable. When running the about box, QuTiP will automatically
check for a newer version of itself from the QuTiP website. As shown below, the about box will have an update link
next to the QuTiP version number if your are not running the latest version of QuTiP
Figure 3.1: QuTiP about box window with link to updated version on the QuTiP website.
8 Chapter 3. Installation
CHAPTER
FOUR
The goal of this guide is to introduce you to the basic structures and functions that make up QuTiP. If you are familiar
with the Quantum Optics Toolbox, then you should have no problem getting used to QuTiP. The guide is broken up
into several sections, each highlighting a specific set of functionalities. In combination with the examples, this guide
should provide a more or less complete overview. In addition, API documentation for each function is located at the
end of this guide.
4.1.1 Organization
QuTiP is designed to be a general toolbox for solving quantum optics like problems. As such, the QuTiP framework
is built from a large (and ever growing) library of functions; from qutip.states.basis to qutip.wigner.
The general organization of QuTiP, highlighting the important functions available to the user is shown in the QuTiP
tree-diagram of user accessible functions and classes..
To load the qutip modules, we must first call the import statement:
>>> from qutip import *
that will load all of the user available functions. Note that, in the rest of the documentation, functions are written using
qutip.module.function() notation which links to the corresponding function in the QuTiP API: QuTiP Function List.
However, in calling import *, we have already loaded all of the QuTiP modules. Therefore, we will only need the
function name and not the complete path when calling the function from the command line or a Python script.
9
QuTiP: The Quantum Toolbox in Python, Release 2.0
Introduction
The key difference between classical and quantum mechanics lies in the use of operators instead of numbers as vari-
ables. Moreover, we need to specify state vectors and their properties. Therefore, in computing the dynamics of
quantum systems we need a data structure that is capable of encapsulating the properties of a quantum operator and
ket/bra vectors. The quantum object class, qutip.Qobj, accomplishes this using matrix representation.
To begin, let us create a blank Qobj:
>>> Qobj()
Quantum object: dims = [[1], [1]], shape = [1, 1]
Qobj data =
[[0]]
where we see the blank Qobj object with dimensions, shape, and data. Here the data corresponds to a 1x1-dimensional
matrix consisting of a single zero entry.
Hint: By convention, Class objects in Python such as Qobj() differ from functions in the use of a beginning capital
letter.
We can create a Qobj with a user defined data set by passing a list or array of data into the Qobj:
>>> Qobj([1,2,3,4,5])
Quantum object: dims = [[1], [5]], shape = [1, 5]
Qobj data =
[[1 2 3 4 5]]
>>> x=array([[1],[2],[3],[4],[5]])
>>> print Qobj(x)
Quantum object: dims = [[5], [1]], shape = [5, 1]
Qobj data =
[[1]
[2]
[3]
[4]
[5]]
>>> r=random((4,4))
>>> print Qobj(r)
Quantum object: dims = [[4], [4]], shape = [4, 4]
Qobj data =
[[ 0.76799998 0.06936066 0.10970546 0.13724402]
[ 0.70644984 0.15371775 0.90649545 0.15349102]
[ 0.69515726 0.13609801 0.52707457 0.6484309 ]
[ 0.78328543 0.87295996 0.58964046 0.3998962 ]]
Notice how both the dims and shape change according to the input data. Although dims and shape appear to have the
same function, the difference will become quite clear in the section on tensor products and partial traces.
Note: If you are running QuTiP from a python script you must use the print function to view the Qobj properties.
Now, unless you have lots of free time, specifying the data for each object is inefficient. Even more so when most ob-
jects correspond to commonly used types such as the ladder operators of a harmonic oscillator,the Pauli spin operators
for a two-level system, or state vectors such as Fock states. Therefore, QuTiP includes predefined objects for a variety
of states:
States Command (# Inputs
means optional)
Fock state ket vector basis(N,#m) / N = number of levels in Hilbert space, m = level
fock(N,#m) containing excitation (0 if no m given)
Fock density matrix (outer fock_dm(N,#p) same as basis(N,m) / fock(N,m)
product of basis)
Coherent state coherent(N,alpha) alpha = complex number (eigenvalue) for requested
coherent state
Coherent density matrix coher- same as coherent(N,alpha)
(outer product) ent_dm(N,alpha)
Thermal density matrix (for thermal_dm(N,n) n = particle number expectation value
n particles)
and operators:
>>> destroy(4)
Quantum object: dims = [[4], [4]], shape = [4, 4]
Qobj data =
[[ 0. 1. 0. 0. ]
[ 0. 0. 1.41421356 0. ]
[ 0. 0. 0. 1.73205081]
[ 0. 0. 0. 0. ]]
>>> sigmaz()
Quantum object: dims = [[2], [2]], shape = [2, 2]
Qobj data =
[[ 1. 0.]
[ 0. -1.]]
>>> jmat(5/2.0,+)
Quantum object: dims = [[6], [6]], shape = [6, 6]
Qobj data =
[[ 0. 2.23606798 0. 0. 0. 0. ]
[ 0. 0. 2.82842712 0. 0. 0. ]
[ 0. 0. 0. 3. 0. 0. ]
[ 0. 0. 0. 0. 2.82842712 0. ]
[ 0. 0. 0. 0. 0. 2.23606798]
[ 0. 0. 0. 0. 0. 0. ]]
Qobj properties
We have seen that a quantum object has three internal attributes, the data, dims, and shape properties. These can be
accessed in the following way:
>>> q=destroy(4)
>>> print q.dims
[[4], [4]]
>>> q.shape
[4, 4]
In general, the properties of a Qobj object (or any Python class) can be retrieved using the Q.property notation. In
addition to the properties shown with the print function, the Qobj class also has the following:
Property Com- Description
mand
Data Q.data Matrix representing state or operator
Dimensions Q.dims List keeping track of shapes for individual components of a multipartite system
(for tensor products and partial traces).
Shape Q.shape Dimensions of underlying data matrix.
is Hermitian? Q.isherm Is the operator Hermitian or not?
Type Q.type Is object of type ket, bra, oper, or super?
Figure 4.2: The Qobj Class viewed as a container for the properties need to characterize a quantum operator or state
vector.
For the destruction operator above:
>>> q.type
oper
>>> q.isherm
False
>>> q.data
<4x4 sparse matrix of type <type numpy.complex128>
with 3 stored elements in Compressed Sparse Row format>
The data property returns a message stating that the data is a sparse matrix. All Qobjs store their data as a sparse matrix
to save memory. To access the underlying matrix one needs to use the qutip.Qobj.full function as described in
the functions section.
Qobj Math
The rules for mathematical operations on Qobjs are similar to standard matrix arithmetic:
>>> q=destroy(4)
>>> x=sigmax()
>>> print q+5
Quantum object: dims = [[4], [4]], shape = [4, 4]
Qobj data =
[[ 5. 6. 5. 5. ]
[ 5. 5. 6.41421356 5. ]
[ 5. 5. 5. 6.73205081]
[ 5. 5. 5. 5. ]]
of course, like matrices, multiplying two objects of incompatible shape throws an error:
>>> q*x
TypeError: Incompatible Qobj shapes
In addition, the logic operators is equal == and is not equal != are also supported.
Like properties, the quantum object class has defined functions (methods) that operate only on members of the Qobj
class. For a general quantum object Q:
>>> basis(5,3).dag()
Quantum object: dims = [[1], [5]], shape = [1, 5], type = bra
Qobj data =
[[ 0. 0. 0. 1. 0.]]
>>> coherent_dm(5,1)
Quantum object: dims = [[5], [5]], shape = [5, 5], type = oper, isHerm = True
Qobj data =
[[ 0.36791117 0.36774407 0.26105441 0.14620658 0.08826704]
[ 0.36774407 0.36757705 0.26093584 0.14614018 0.08822695]
[ 0.26105441 0.26093584 0.18523331 0.10374209 0.06263061]
[ 0.14620658 0.14614018 0.10374209 0.05810197 0.035077 ]
[ 0.08826704 0.08822695 0.06263061 0.035077 0.0211765 ]]
>>> coherent_dm(5,1).diag()
array([ 0.36791117, 0.36757705, 0.18523331, 0.05810197, 0.0211765 ])
>>> coherent_dm(5,1).full()
array([[ 0.36791117, 0.36774407, 0.26105441, 0.14620658, 0.08826704],
[ 0.36774407, 0.36757705, 0.26093584, 0.14614018, 0.08822695],
[ 0.26105441, 0.26093584, 0.18523331, 0.10374209, 0.06263061],
[ 0.14620658, 0.14614018, 0.10374209, 0.05810197, 0.035077 ],
[ 0.08826704, 0.08822695, 0.06263061, 0.035077 , 0.0211765 ]])
>>> coherent_dm(5,1).norm()
1.0
>>> coherent_dm(5,1).sqrtm()
Quantum object: dims = [[5], [5]], shape = [5, 5], type = oper, isHerm = False
Qobj data =
[[ 0.36791117 +6.66013801e-09j 0.36774407 -2.87612199e-09j
0.26105441 -4.24323387e-09j 0.14620658 -1.21628628e-09j
0.08826704 -1.21357197e-09j]
[ 0.36774407 -3.87481342e-09j 0.36757705 +1.66576107e-09j
0.26093584 +2.50548614e-09j 0.14614018 +7.07508704e-10j
0.08822695 +6.28805009e-10j]
[ 0.26105441 -2.75065517e-09j 0.26093584 +1.15201146e-09j
0.18523331 +1.92733313e-09j 0.10374209 +5.01775972e-10j
0.06263061 +1.34247407e-10j]
[ 0.14620658 -1.54053667e-09j 0.14614017 +6.89127552e-10j
0.10374209 +8.65055761e-10j 0.05810198 +2.81704042e-10j
0.03507700 +5.25048476e-10j]
[ 0.08826704 -9.30044364e-10j 0.08822695 +4.99516749e-10j
0.06263061 +1.14878928e-10j 0.03507700 +1.71358232e-10j
0.02117650 +1.17185351e-09j]]
>>> coherent_dm(5,1).tr()
1.0
>>> (basis(4,2)+basis(4,1)).unit()
Quantum object: dims = [[4], [1]], shape = [4, 1], type = ket
Qobj data =
[[ 0. ]
[ 0.70710678]
[ 0.70710678]
[ 0. ]]
4.3.1 Introduction
In the previous guide section Basic Operations on Quantum Objects, we saw how to create operators and states, using
the functions built into QuTiP. In this portion of the guide, we will look at performing basic operations with states
and operators. For more detailed demonstrations on how to use and manipulate these objects, see the QuTiP Example
Scripts section.
Here we begin by creating a Fock qutip.basis vacuum state vector |0i with in a Hilbert space with 5 number
states, 0 -> 4:
>>> vec=basis(5,0)
>>> print vec
Quantum object: dims = [[5], [1]], shape = [5, 1], type = ket
Qobj data =
[[ 1.] #<-- |0>
[ 0.] #<-- |1>
[ 0.] #<-- |2>
[ 0.] #<-- |3>
[ 0.]] #<-- |4>
and then create a lowering operator (a) corresponding to 5 number states using the qutip.destroy function:
>>> a=destroy(5)
>>> print a
Quantum object: dims = [[5], [5]], shape = [5, 5], type = oper, isHerm = False
Qobj data =
[[ 0. 1. 0. 0. 0. ]
[ 0. 0. 1.41421356 0. 0. ]
[ 0. 0. 0. 1.73205081 0. ]
[ 0. 0. 0. 0. 2. ]
[ 0. 0. 0. 0. 0. ]]
Now lets apply the destruction operator to our vacuum state vec,
>>> a*vec
Quantum object: dims = [[5], [1]], shape = [5, 1], type = ket
Qobj data =
[[ 0.]
[ 0.]
[ 0.]
[ 0.]
[ 0.]]
We see that, as expected, the vacuum is transformed to the zero vector. A more interesting example comes from using
the adjoint of the lowering operator, the raising operator a :
>>> a.dag()*vec
Quantum object: dims = [[5], [1]], shape = [5, 1], type = ket
Qobj data =
[[ 0.]
[ 1.] #<-- |1>
[ 0.]
[ 0.]
[ 0.]]
The raising operator has in indeed raised the state vec from the vacuum to the |1i state. Instead of using the dagger
dag() command to raise the state, we could have also used the built in qutip.create function to make a raising
operator:
>>> c=create(5)
>>> c*vec()
Quantum object: dims = [[5], [1]], shape = [5, 1], type = ket
Qobj data =
[[ 0.]
[ 1.]
[ 0.]
[ 0.]
[ 0.]]
which obviously does the same thing. We can of course raise the vacuum state more than once:
>>> c*c*vec
Quantum object: dims = [[5], [1]], shape = [5, 1], type = ket
Qobj data =
[[ 0. ]
[ 0. ]
[ 1.41421356] #<-- |2>
[ 0. ]
[ 0. ]]
2
or just taking the square of the raising operator a :
>>> c**2*vec
Quantum object: dims = [[5], [1]], shape = [5, 1], type = ket
Qobj data =
[[ 0. ]
[ 0. ]
[ 1.41421356]
[ 0. ]
[ 0. ]]
p
Applying the raising operator twice gives the expected (n + 1) dependence. We can use the product of c a to also
apply the number operator to the state vector vec:
>>> c*a*vec
Quantum object: dims = [[5], [1]], shape = [5, 1], type = ket
Qobj data =
[[ 0.]
[ 0.]
[ 0.]
[ 0.]
[ 0.]]
in this last example, application of the number operator does notgive the expected value n = 2, but
Notice how
rather 2 2. This is because this last state is not normalized to unity as c |ni = n + 1 |n + 1i. Therefore, we should
normalize our vector first:
>>> c*a*(c**2*vec).unit()
Quantum object: dims = [[5], [1]], shape = [5, 1], type = ket
Qobj data =
[[ 0.]
[ 0.]
[ 2.]
[ 0.]
[ 0.]]
Since we are giving a demonstration of using states and operators, we have done a lot more work than we should have.
For example, we do not need to operate on the vacuum state to generate a higher number Fock state. Instead we can
use the qutip.basis (or qutip.fock) function to directly obtain the required state:
>>> vec=basis(5,2)
>>> print vec
Quantum object: dims = [[5], [1]], shape = [5, 1], type = ket
Qobj data =
[[ 0.]
[ 0.]
[ 1.]
[ 0.]
[ 0.]]
Notice how it is automatically normalized. We can also use the built in qutip.num operator:
>>> n=num(5)
>>> print n
Quantum object: dims = [[5], [5]], shape = [5, 5], type = oper, isHerm = True
Qobj data =
[[0 0 0 0 0]
[0 1 0 0 0]
[0 0 2 0 0]
[0 0 0 3 0]
[0 0 0 0 4]]
where we have used the qutip.Qobj.unit function to again normalize the state. Operating with the number
function again:
>>> n*vec
Quantum object: dims = [[5], [1]], shape = [5, 1], type = ket
Qobj data =
[[ 0. ]
[ 0.70710678]
[ 0. ]
[ 0. ]
[ 0. ]]
We can also create coherent states and squeezed states by applying the qutip.displace and qutip.squeez
functions to the vacuum state:
>>> vec=basis(5,0)
>>> d=displace(5,1j)
>>> s=squeez(5,0.25+0.25j)
>>> d*vec
Quantum object: dims = [[5], [1]], shape = [5, 1], type = ket
Qobj data =
[[ 0.60655682+0.j ]
[ 0.00000000+0.60628133j]
[-0.43038740+0.j ]
[ 0.00000000-0.24104351j]
[ 0.14552147+0.j ]]
>>> d*s*vec
Quantum object: dims = [[5], [1]], shape = [5, 1], type = ket
Qobj data =
[[ 0.65893786+0.08139381j]
[ 0.10779462+0.51579735j]
[-0.37567217-0.01326853j]
[-0.02688063-0.23828775j]
[ 0.26352814+0.11512178j]]
Of course, displacing the vacuum gives a coherent state, which can also be generated using the built in
qutip.coherent function.
The main purpose of QuTiP is to explore the dynamics of open quantum systems, where the most general state of a
system is not longer a state vector, but rather a density matrix. Since operations on density matrices operate identically
to those of vectors, we will just briefly highlight creating and using these structures.
The simplest density matrix is created by forming the outer-product :math:leftpsiright>left<psiright| of a ket vector:
>>> vec=basis(5,2)
>>> vec*vec.dag()
Quantum object: dims = [[5], [5]], shape = [5, 5], type = oper, isHerm = True
Qobj data =
[[ 0. 0. 0. 0. 0.]
[ 0. 0. 0. 0. 0.]
[ 0. 0. 1. 0. 0.]
[ 0. 0. 0. 0. 0.]
[ 0. 0. 0. 0. 0.]]
A similar task can also be accomplished via the qutip.fock_dm or qutip.ket2dm functions:
>>> fock_dm(5,2)
Quantum object: dims = [[5], [5]], shape = [5, 5], type = oper, isHerm = True
Qobj data =
[[ 0. 0. 0. 0. 0.]
[ 0. 0. 0. 0. 0.]
[ 0. 0. 1. 0. 0.]
[ 0. 0. 0. 0. 0.]
[ 0. 0. 0. 0. 0.]]
>>> ket2dm(vec)
Quantum object: dims = [[5], [5]], shape = [5, 5], type = oper, isHerm = True
Qobj data =
[[ 0. 0. 0. 0. 0.]
[ 0. 0. 0. 0. 0.]
[ 0. 0. 1. 0. 0.]
[ 0. 0. 0. 0. 0.]
[ 0. 0. 0. 0. 0.]]
If we want to create a density matrix with equal classical probability of being found in the |2i or |4i number states we
can do the following:
>>> 0.5*ket2dm(basis(5,4))+0.5*ket2dm(basis(5,2))
Quantum object: dims = [[5], [5]], shape = [5, 5], type = oper, isHerm = True
Qobj data =
[[ 0. 0. 0. 0. 0. ]
[ 0. 0. 0. 0. 0. ]
[ 0. 0. 0.5 0. 0. ]
[ 0. 0. 0. 0. 0. ]
[ 0. 0. 0. 0. 0.5]]
or use 0.5*fock_dm(5,2)+0.5*fock_dm(5,4). There are also several other built in functions for creating
predefined density matrices, for example qutip.coherent_dm and qutip.thermal_dm which create coherent
state and thermal state density matrices, respectively.
>>> coherent_dm(5,1.25)
Quantum object: dims = [[5], [5]], shape = [5, 5], type = oper, isHerm = True
Qobj data =
[[ 0.20980701 0.26141096 0.23509686 0.15572585 0.13390765]
[ 0.26141096 0.32570738 0.29292109 0.19402805 0.16684347]
[ 0.23509686 0.29292109 0.26343512 0.17449684 0.1500487 ]
[ 0.15572585 0.19402805 0.17449684 0.11558499 0.09939079]
[ 0.13390765 0.16684347 0.1500487 0.09939079 0.0854655 ]]
>>> thermal_dm(5,1.25)
Quantum object: dims = [[5], [5]], shape = [5, 5], type = oper, isHerm = True
Qobj data =
[[ 0.44444444 0. 0. 0. 0. ]
[ 0. 0.24691358 0. 0. 0. ]
[ 0. 0. 0.13717421 0. 0. ]
[ 0. 0. 0. 0.0762079 0. ]
[ 0. 0. 0. 0. 0.04233772]]
}}}
QuTiP also provides a set of distance metrics for determining how close two density matrix distributions are to each
other. Included are the trace distance qutip.tracedist and the fidelity qutip.fidelity.
>>> x=coherent_dm(5,1.25)
>>> y=coherent_dm(5,1.25j) #<-- note the j
>>> z=thermal_dm(5,0.125)
>>> fidelity(x,x)
1.0000000051410474
>>> tracedist(y,y)
0.0
We also know that for two-pure states, the trace distance (T) and the fidelity (F) are related by T = 1 F 2.
>>> tracedist(y,x)
0.9771565838870081
>>> sqrt(1-fidelity(y,x)**2)
0.97715657039974568
For a pure state and a mixed state, 1 F 2 T which can also be verified:
>>> 1-fidelity(x,z)**2
0.7784456314854065
>>> tracedist(x,z)
0.8563182215236257
Having spent a fair amount of time on basis states that represent harmonic oscillator states, we now move on to qubit,
or two-level spin systems. To create a state vector corresponding to a qubit system, we use the same qutip.basis,
or qutip.fock, function with only two levels:
>>> spin=basis(2,0)
Now at this point one may ask how this state is different than that of a harmonic oscillator in the vacuum state truncated
to two energy levels?
>>> vec=basis(2,0)
At this stage, there is no difference. This should not be surprising as we called the exact same function twice.
The difference between the two comes from the action of the spin operators qutip.sigmax, qutip.sigmay,
qutip.sigmaz, qutip.sigmap, and qutip.sigmam on these two-level states. For example, if vec corre-
sponds to the vacuum state of a harmonic oscillator, then, as we have already seen, we can use the raising operator to
get the |1i state:
>>> vec
Quantum object: dims = [[2], [1]], shape = [2, 1], type = ket
Qobj data =
[[ 1.]
[ 0.]]
>>> c=create(2)
>>> c*vec
Quantum object: dims = [[2], [1]], shape = [2, 1], type = ket
Qobj data =
[[ 0.]
[ 1.]]
For a spin system, the operator analogous to the raising operator is the sigma-plus operator qutip.sigmap. Oper-
ating on the spin state gives:
>>> spin
Quantum object: dims = [[2], [1]], shape = [2, 1], type = ket
Qobj data =
[[ 1.]
[ 0.]]
>>> sigmap()*spin
Quantum object: dims = [[2], [1]], shape = [2, 1], type = ket
Qobj data =
[[ 0.]
[ 0.]]
Now we see the difference! The qutip.sigmap operator acting on the spin state returns the zero vector. Why is
this? To see what happened, let us use the qutip.sigmaz operator:
>>> sigmaz()
Quantum object: dims = [[2], [2]], shape = [2, 2], type = oper, isHerm = True
Qobj data =
[[ 1. 0.]
[ 0. -1.]]
>>> sigmaz()*spin
Quantum object: dims = [[2], [1]], shape = [2, 1], type = ket
Qobj data =
[[ 1.]
[ 0.]]
>>> spin2=basis(2,1)
>>> spin2
Quantum object: dims = [[2], [1]], shape = [2, 1], type = ket
Qobj data =
[[ 0.]
[ 1.]]
>>> sigmaz()*spin2
Quantum object: dims = [[2], [1]], shape = [2, 1], type = ket
Qobj data =
[[ 0.]
[-1.]]
The answer is now apparent. Since the QuTiP qutip.sigmaz function uses the standard z-basis representation of
the sigma-z spin operator, the spin state corresponds to the |upi state of a two-level spin system while spin2 gives
the |downi state. Therefore, in our previous example sigmap()*spin, we raised the qubit state out of the truncated
two-level Hilbert space resulting in the zero state.
While at first glance this convention might seem somewhat odd, it is in fact quite handy. For one, the spin operators
remain in the conventional form. Second, when the spin system is in the |upi state:
>>> sigmaz()*spin
Quantum object: dims = [[2], [1]], shape = [2, 1], type = ket
Qobj data =
[[ 1.] #<--- zeroth element of matrix
[ 0.]]
the non-zero component is the zeroth-element of the underlying matrix (remember that python uses c-indexing, and
matrices start with the zeroth element). The |downi state therefore has a non-zero entry in the first index position.
This corresponds nicely with the quantum information definitions of qubit states, where the excited |upi state is label
as |0i, and the |upi state by |1i.
If one wants to create spin operators for higher spin systems, then the qutip.operators.jmat function comes
in handy.
Some of the most important information about quantum systems comes from calculating the expectation value of
operators, both Hermitian and non-Hermitian, as the state or density matrix of the system varies in time. Therefore, in
this section we demonstrate the use of the qutip.expect function. To begin:
>>> vac=basis(5,0)
>>> one=basis(5,1)
>>> c=create(5)
>>> N=num(5)
>>> expect(N,vac)
0.0
>>> expect(N,one)
1.0
>>> coh=coherent_dm(5,1.0j)
>>> expect(N,coh)
0.997055574581 #should be equal to 1, small diff. due to truncated Hilbert space
>>> cat=(basis(5,4)+1.0j*basis(5,3)).unit()
>>> expect(c,cat)
1j
The qutip.expect function also accepts lists or arrays of state vectors or density matrices for the second input:
>>> states=[(c**k*vac).unit() for k in range(5)] #must normalize
>>> expect(N,states)
[ 0. 1. 2. 3. 4.]
Notice how in this last example, all of the return values are complex numbers. Yet if we calculate just the first
expectation value,
>>> expect(c,basis(5,4))
0.0
we get a real number. This is because the qutip.expect function looks to see whether the operator is Hermitian
or not. If the operator is Hermitian, than the output will always be real. In the case of non-Hermitian operators, the
return values may be complex. Therefore, the expect function will return a array of complex values for non-Hermitian
operators when the input is a list/array of states or density matrices.
Of course, the expect function works for spin states and operators:
>>> up=basis(2,0)
>>> down=basis(2,1)
>>> expect(sigmaz(),up)
1.0
>>>expect(sigmaz(),down)
-1.0
as well as the composite objects discussed in the next section Using Tensor Products and Partial Traces:
>>> spin1=basis(2,0)
>>> spin2=basis(2,1)
>>>two_spins=tensor(spin1,spin2)
>>> sz1=tensor(sigmaz(),qeye(2))
>>> sz2=tensor(qeye(2),sigmaz())
>>> expect(sz1,two_spins)
1.0
>>> expect(sz2,two_spins)
-1.0
To describe the states of multipartite quantum systems, such as two coupled qubits, a qubit coupled to an oscillator,
etc..., we need to expand the Hilbert space by taking the tensor product of the state vectors for each of the system
components. Similarly, the operators acting on the state vectors in the combined Hilbert space (describing the coupled
system) are formed by taking the tensor product of the individual operators.
In QuTiP the function qutip.tensor.tensor is used to accomplish this task. The tensor function takes as its
argument a collection:
>>> tensor(op1,op2,op3)
or a list:
>>> tensor([op1,op2,op3])
of state vectors or operators and returns a composite quantum object for the combined Hilbert space. The returned
quantum objects type is the same as that of the input(s).
For example, the state vector describing two qubits in their ground states is formed by taking the tensor product of the
two single-qubit ground state vectors:
>>> tensor(basis(2,0), basis(2,0))
Quantum object: dims = [[2, 2], [1, 1]], shape = [4, 1], type = ket
Qobj data =
[[ 1.]
[ 0.]
[ 0.]
[ 0.]]
This is straight forward to generalize to more qubits by adding more component state vectors in the argument list to
the qutip.tensor.tensor function, as illustrated in the following example:
>>> tensor((basis(2,0)+basis(2,1)).unit(), (basis(2,0)+basis(2,1)).unit(), basis(2,0))
Quantum object: dims = [[2, 2, 2], [1, 1, 1]], shape = [8, 1], type = ket
Qobj data =
[[ 0.5]
[ 0. ]
[ 0.5]
[ 0. ]
[ 0.5]
[ 0. ]
[ 0.5]
[ 0. ]]
This state is slightly more complicated, describing two qubits in a superposition between the up and down states, while
the third qubit remains in its ground state.
To construct operators that act on an extended Hilbert space of a combined system, we similarly pass a list of operators
for each component system to the qutip.tensor.tensor function. For example, to form the operator that
represents the simultaneous action of the sigma x operator on two qubits:
>>> tensor(sigmax(), sigmax())
Quantum object: dims = [[2, 2], [2, 2]], shape = [4, 4], type = oper, isHerm = True
Qobj data =
[[ 0. 0. 0. 1.]
[ 0. 0. 1. 0.]
[ 0. 1. 0. 0.]
[ 1. 0. 0. 0.]]
To create operators in a combined Hilbert space that only act only on a single component, we take the tensor product
of the operator acting on the subspace of interest, with the identity operators corresponding to the components that
are to be unchanged. For example, the operator that represents sigma-z on the first qubit in a two-qubit system, while
leaving the second qubit unaffected:
>>> tensor(sigmaz(), qeye(2))
Quantum object: dims = [[2, 2], [2, 2]], shape = [4, 4], type = oper, isHerm = True
Qobj data =
[[ 1. 0. 0. 0.]
[ 0. 1. 0. 0.]
[ 0. 0. -1. 0.]
[ 0. 0. 0. -1.]]
The qutip.tensor.tensor function is extensively used when constructing Hamiltonians for composite systems.
Here well look at some simple examples.
First, lets consider a system of two coupled qubits. Assume that both qubit has equal energy splitting, and that the
qubits are coupled through a sigmax-sigmax interaction with strength g = 0.05 (in units where the bare qubit energy
splitting is unity). The Hamiltonian describing this system is:
>>> H = tensor(sigmaz(), qeye(2)) + tensor(qeye(2), sigmaz()) + 0.05 * tensor(sigmax(), sigmax())
>>> H
Quantum object: dims = [[2, 2], [2, 2]], shape = [4, 4], type = oper, isHerm = True
Qobj data =
[[ 2. 0. 0. 0.05]
[ 0. 0. 0.05 0. ]
[ 0. 0.05 0. 0. ]
[ 0.05 0. 0. -2. ]]
[ 0. 1. 0.25 0. 0. 0. 0. 0.5 ]
[ 0. 0.25 1. 0. 0.5 0. 0. 0. ]
[ 0.25 0. 0. -1. 0. 0.5 0. 0. ]
[ 0. 0. 0.5 0. 1. 0. 0. 0.25]
[ 0. 0. 0. 0.5 0. -1. 0.25 0. ]
[ 0.5 0. 0. 0. 0. 0.25 -1. 0. ]
[ 0. 0.5 0. 0. 0.25 0. 0. -3. ]]
The simplest possible quantum mechanical description for light-matter interaction is encapsulated in the Jaynes-
Cummings model, which describes the coupling between a two-level atom and a single-mode electromagnetic field
(a cavity mode). Denoting the energy splitting of the atom and cavity omega_a and omega_c, respectively, and the
atom-cavity interaction strength g, the Jaynes-Cumming Hamiltonian can be constructed as:
>>> N = 10
>>> omega_a = 1.0
>>> omega_c = 1.25
>>> g = 0.05
>>> a = tensor(qeye(2), destroy(N))
>>> sm = tensor(destroy(2), qeye(N))
>>> sz = tensor(sigmaz(), qeye(N))
>>> H = 0.5 * omega_a * sz + omega_c * a.dag() * a + g * (a.dag() * sm + a * sm.dag())
The partial trace is an operation that reduces the dimension of a Hilbert space by eliminating some degrees of freedom
by averaging (tracing). In this sense it is therefore the converse of the tensor product. It is useful when one is
interested in only a part of a coupled quantum system. For open quantum systems, this typically involves tracing over
the environment leaving only the system of interest. In QuTiP the function qutip.ptrace.ptrace is used to take
partial traces. It takes two arguments: rho is the density matrix (or state vector) of the composite system, and sel is
a list of integers that mark the component systems that should be kept. All other components are traced over.
For example, the density matrix describing a single qubit obtained from a coupled two-qubit system is obtained via:
>>> psi = tensor(basis(2,0), basis(2,1))
>>> ptrace(psi, 0)
Quantum object: dims = [[2], [2]], shape = [2, 2], type = oper, isHerm = True
Qobj data =
[[ 1. 0.]
[ 0. 0.]]
>>> ptrace(psi, 1)
Quantum object: dims = [[2], [2]], shape = [2, 2], type = oper, isHerm = True
Qobj data =
[[ 0. 0.]
[ 0. 1.]]
Note that the partial trace always results in a density matrix (mixed state), regardless of whether the composite system
is a pure state (described by a state vector) or a mixed state (described by a density matrix):
>>> psi = tensor((basis(2,0)+basis(2,1)).unit(), basis(2,0))
>>> psi
Quantum object: dims = [[2, 2], [1, 1]], shape = [4, 1], type = ket
Qobj data =
[[ 0.70710678]
[ 0. ]
[ 0.70710678]
[ 0. ]]
>>> ptrace(psi, 0)
Quantum object: dims = [[2], [2]], shape = [2, 2], type = oper, isHerm = True
Qobj data =
[[ 0.5 0.5]
[ 0.5 0.5]]
>>> rho = tensor(ket2dm((basis(2,0)+basis(2,1)).unit()), fock_dm(2,0))
>>> rho
Quantum object: dims = [[2, 2], [2, 2]], shape = [4, 4], type = oper, isHerm = True
Qobj data =
[[ 0.5 0. 0.5 0. ]
[ 0. 0. 0. 0. ]
[ 0.5 0. 0.5 0. ]
[ 0. 0. 0. 0. ]]
>>> ptrace(rho, 0)
Quantum object: dims = [[2], [2]], shape = [2, 2], type = oper, isHerm = True
Qobj data =
[[ 0.5 0.5]
[ 0.5 0.5]]
Important: In QuTiP 2, the results from all of the dynamics solvers are returned as Odedata objects. This signifi-
cantly simplifies the storage and saving of simulation data. However, this change also results in the loss of backward
compatibility with QuTiP version 1.x. Therefore, please read this section to avoid running into any issues.
Before embarking on simulating the dynamics of quantum systems, we will first look at the data structure used for
returning the simulation results to the user. This object is a qutip.Odedata class that stores all the crucial data
needed for analyzing and plotting the results of a simulation. Like the qutip.Qobj class, the Odedata class has
a collection of properties for storing information. However, in contrast to the Qobj class, this structure contains no
methods, and is therefore nothing but a container object. A generic Odedata object odedata contains the following
properties for storing simulation data:
Property Description
odedata.solver String indicating which solver was used to generate the data.
odedata.times List/array of times at which simulation data is calculated.
odedata.expect List/array of expectation values, if requested.
odedata.states List/array of state vectors /density matrices calculated at times , if
requested.
odedata.num_expect The number of expectation value operators in the simulation.
odedata.num_collapse The number of collapse operators in the simulation.
odedata.ntraj Number of monte-carlo trajectories run (if using mcsolve).
odedata.col_times Times at which state collapse occurred. Only for Monte-Carlo solver.
odedata.col_which Which collapse operator was responsible for each collapse in in
col_times. mcsolver only.
To understand how to access the data in a Odedata object we will use the Driven Cavity+Qubit Monte-Carlo example
as a guide, although we do not worry about the simulation details at this stage. Like all solvers, the monte-carlo solver
used in this example returns an Odedata object, here called simply data. To see what is contained inside data we
can use the print command:
>>> print data
Odedata object with mcsolve data.
---------------------------------
expect = True
num_expect = 2, num_collapse = 2, ntraj = 500
The first line tells us that this data object was generated from the Monte-Carlo solver mcsolve (discussed in Quantum
Dynamics via the Monte-Carlo Solver). The next line (not the --- line of course) indicates that this object contains
expectation value data. Finally, the last line gives the number of expectation value and collapse operators used in the
simulation, along with the number of Monte-Carlo trajectories run. Note that the number of trajectories ntraj is
only displayed when using the Monte-Carlo solver.
Now we have all the information needed to reproduce the simulation results. To access the data for the two expectation
values one can do:
>>> expt0 = data.expect[0]
>>> expt1 = data.expect[1]
Recall that Python uses C-style indexing that begins with zero ( i.e. [0] => 1st collapse operator data). Together with
the array of times at which these expectation values are calculated:
>>> times = data.times
State vectors, or density matrices, as well as col_times and col_which, are accessed in a similar manner, al-
though typically one does not need an index (i.e [0]) since there is only one list for each of these components. The
one exception to this rule is if you choose to output state vectors from the Monte-Carlo solver, in which case there are
ntraj number of state vector arrays.
Figure 4.3: Data for expectation values extracted from the data Odedata object.
The main advantage in using the Odedata class as a data storage object comes from the simplicity in which simulation
data can be stored and later retrieved. The qutip.fileio.qsave and qutip.fileio.qload functions are
designed for this task. To begin, let us save the data object from the previous section into a file called cavity+qubit-
data in the current working directory by calling:
>>> qsave(data,cavity+qubit-data)
All of the data results are then stored in a single file of the same name with a .qu extension. Therefore, everything
needed to later this data is stored in a single file. Loading the file is just as easy as saving:
>>> chicken=qload(cavity+qubit-data)
Loaded Odedata object:
Odedata object with mcsolve data.
---------------------------------
expect = True
num_expect = 2, num_collapse = 2, ntraj = 500
where chicken is the new name of the Odedata object. We can then extract the data and plot in the same manner as
before:
expt0 = chicken.expect[0]
expt1 = chicken.expect[1]
times = chicken.times
plot(times,expt0,times,expt1)
show()
Also see Saving QuTiP Objects and Data Sets for more information on saving quantum objects, as well as arrays for
use in other programs.
Figure 4.4: Data for expectation values from the chicken object loaded from the data object stored with
qutip.fileio.qsave
Unitary evolution
The dynamics of a closed (pure) quantum system is governed by the Schrdinger equation
ih = H, (4.1)
t
where is the wave function, H the Hamiltonian, and h is Plancks constant. In general, the Schrdinger equation is
a partial differential equation (PDE) where both and H are functions of space and time. For computational purposes
it is useful to expand the PDE in a set of basis functions that span the Hilbert space of the Hamiltonian, and to write
the equation in matrix and vector form
d
ih |i = H |i
dt
where |i is the state vector and H is the matrix representation of the Hamiltonian. This matrix equation can, in
principle, be solved by diagonalizing the Hamiltonian matrix H. In practice, however, it is difficult to perform this
diagonalization unless the size of the Hilbert space (dimension of the matrix H) is small. Analytically, it is a formidable
task to calculate the dynamics for systems with more than two states. If, in addition, we consider dissipation due to the
inevitable interaction with a surrounding environment, the computational complexity grows even larger, and we have
to resort to numerical calculations in all realistic situations. This illustrates the importance of numerical calculations
in describing the dynamics of open quantum systems, and the need for efficient and accessible tools for this task.
The Schrdinger equation, which governs the time-evolution of closed quantum systems, is defined by its Hamiltonian
and state vector. In the previous section, Using Tensor Products and Partial Traces, we showed how Hamiltonians
and state vectors are constructed in QuTiP. Given a Hamiltonian, we can calculate the unitary (non-dissipative) time-
evolution of an arbitrary state vector |0 i (psi0) using the QuTiP function qutip.mesolve. It evolves the state
vector and evaluates the expectation values for a set of operators expt_op_list at the points in time in the list
tlist, using an ordinary differential equation solver. Alternatively, we can use the function qutip.essolve,
which uses the exponential-series technique to calculate the time evolution of a system. The qutip.mesolve and
qutip.essolve functions take the same arguments and it is therefore easy switch between the two solvers.
For example, the time evolution of a quantum spin-1/2 system with tunneling rate 0.1 that initially is in the up state is
calculated, and the expectation values of the z operator evaluated, with the following code:
>>> H = 2 * pi * 0.1 * sigmax()
>>> psi0 = basis(2, 0)
>>> tlist = linspace(0.0, 10.0, 20.0)
>>> mesolve(H, psi0, tlist, [], [sigmaz()])
array([[ 1.00000000+0.j, 0.78914229+0.j, 0.24548596+0.j, -0.40169696+0.j,
-0.87947669+0.j, -0.98636356+0.j, -0.67728166+0.j, -0.08257676+0.j,
0.54695235+0.j, 0.94582040+0.j, 0.94581706+0.j, 0.54694422+0.j,
-0.08258520+0.j, -0.67728673+0.j, -0.98636329+0.j, -0.87947111+0.j,
-0.40168898+0.j, 0.24549302+0.j, 0.78914528+0.j, 0.99999927+0.j]])
The brackets in the fourth argument is an empty list of collapse operators, since we consider unitary evolution in this
example. See the next section for examples on how dissipation is included by defining a list of collapse operators.
The function returns an array of expectation values for the operators that are included in the list in the fifth argument.
Adding operators to this list results in a larger output array returned by the function (one list of numbers, corresponding
to the times in tlist, for each operator):
>>> mesolve(H, psi0, tlist, [], [sigmaz(), sigmay()])
array([[ 1.00000000e+00+0.j, 7.89142292e-01+0.j, 2.45485961e-01+0.j,
-4.01696962e-01+0.j, -8.79476686e-01+0.j, -9.86363558e-01+0.j,
-6.77281655e-01+0.j, -8.25767574e-02+0.j, 5.46952346e-01+0.j,
9.45820404e-01+0.j, 9.45817056e-01+0.j, 5.46944216e-01+0.j,
-8.25852032e-02+0.j, -6.77286734e-01+0.j, -9.86363287e-01+0.j,
-8.79471112e-01+0.j, -4.01688979e-01+0.j, 2.45493023e-01+0.j,
7.89145284e-01+0.j, 9.99999271e-01+0.j],
[ 0.00000000e+00+0.j, -6.14214010e-01+0.j, -9.69403055e-01+0.j,
-9.15775807e-01+0.j, -4.75947716e-01+0.j, 1.64596791e-01+0.j,
7.35726839e-01+0.j, 9.96586861e-01+0.j, 8.37166184e-01+0.j,
3.24695883e-01+0.j, -3.24704840e-01+0.j, -8.37170685e-01+0.j,
-9.96585195e-01+0.j, -7.35720619e-01+0.j, -1.64588257e-01+0.j,
4.75953748e-01+0.j, 9.15776736e-01+0.j, 9.69398541e-01+0.j,
6.14206262e-01+0.j, -8.13905967e-06+0.j]])
The resulting list of expectation values can easily be visualized using matplotlibs plotting functions:
>>> tlist = linspace(0.0, 10.0, 100)
>>> expt_list = mesolve(H, psi0, tlist, [], [sigmaz(), sigmay()])
>>>
>>> from pylab import *
>>> plot(tlist, real(expt_list[0]))
>>> plot(tlist, real(expt_list[1]))
>>> xlabel(Time)
>>> ylabel(Expectation values)
>>> legend(("Simga-Z", "Sigma-Y"))
>>> show()
If an empty list of operators is passed as fifth parameter, the qutip.mesolve function returns a list of state vectors
for the times specified in tlist:
>>> tlist = [0.0, 1.0]
>>> mesolve(H, psi0, tlist, [], [])
[
Quantum object: dims = [[2], [1]], shape = [2, 1], type = ket
Qobj data =
[[ 1.+0.j]
[ 0.+0.j]]
, Quantum object: dims = [[2], [1]], shape = [2, 1], type = ket
Qobj data =
[[ 0.80901765+0.j ]
[ 0.00000000-0.58778584j]]
, Quantum object: dims = [[2], [1]], shape = [2, 1], type = ket
Qobj data =
[[ 0.3090168+0.j ]
[ 0.0000000-0.95105751j]]
, Quantum object: dims = [[2], [1]], shape = [2, 1], type = ket
Qobj data =
[[-0.30901806+0.j ]
[ 0.00000000-0.95105684j]]
]
Non-unitary evolution
While the evolution of the state vector in a closed quantum system is deterministic, open quantum systems are stochas-
tic in nature. The effect of an environment on the system of interest is to induce stochastic transitions between energy
levels, and to introduce uncertainty in the phase difference between states of the system. The state of an open quantum
system is therefore described in terms of ensemble averaged states using the density matrix formalism. P A density
matrix describes a probability distribution of quantum states |n i, in a matrix representation = n pn |n i hn |,
where pn is the classical probability that the system is in the quantum state |n i. The time evolution of a density
matrix is the topic of the remaining portions of this section.
The standard approach for deriving the equations of motion for a system interacting with its environment is to expand
the scope of the system to include the environment. The combined quantum system is then closed, and its evolution is
governed by the von Neumann equation
i
tot (t) = [Htot , tot (t)], (4.2)
h
the equivalent of the Schrdinger equation ((4.1)) in the density matrix formalism. Here, the total Hamiltonian
Htot = Hsys + Henv + Hint ,
includes the original system Hamiltonian Hsys , the Hamiltonian for the environment Henv , and a term representing the
interaction between the system and its environment Hint . Since we are only interested in the dynamics of the system,
we can at this point perform a partial trace over the environmental degrees of freedom in Eq.~((4.2)), and thereby
obtain a master equation for the motion of the original system density matrix. The most general trace-preserving
and completely positive form of this evolution is the Lindblad master equation for the reduced density matrix =
Trenv [tot ]
i X1
2Cn (t)Cn+ (t)Cn+ Cn Cn+ Cn (t)
(t) = [H(t), (t)] + (4.3)
h n
2
where the Cn = n An are collapse operators, and An are the operators through which the environment couples to
the system in Hint , and n are the corresponding rates. The derivation of Eq.~((4.3)) may be found in several sources,
and will not be reproduced here. Instead, we emphasize the approximations that are required to arrive at the master
equation in the form of Eq.~((4.3)), and hence perform a calculation in QuTiP:
Separability: At t = 0 there are no correlations between the system and its environment such that the total
density matrix can be written as a tensor product Itot (0) = I (0) Ienv (0).
Born approximation: Requires: (1) that the state of the environment does not significantly change as a result of
the interaction with the system; (2) The system and the environment remain separable throughout the evolution.
These assumptions are justified if the interaction is weak, and if the environment is much larger than the system.
In summary, tot (t) (t) env .
Markov approximation The time-scale of decay for the environment env is much shorter than the smallest
time-scale of the system dynamics sys env . This approximation is often deemed a short-memory environ-
ment as it requires that environmental correlation functions decay on a time-scale fast compared to those of the
system.
Secular approximation Stipulates that elements in the master equation corresponding to transition frequencies
satisfy |ab cd | 1/sys , i.e., all fast rotating terms in the interaction picture can be neglected. It also
ignores terms that lead to a small renormalization of the system energy levels. This approximation is not strictly
necessary for all master-equation formalisms (e.g., the Block-Redfield master equation), but it is required for
arriving at the Lindblad form ((4.3)) which is used in QuTiP.
For systems with environments satisfying the conditions outlined above, the Lindblad master equation ((4.3)) governs
the time-evolution of the system density matrix, giving an ensemble average of the system dynamics. In order to
ensure that these approximations are not violated, it is important that the decay rates n be smaller than the minimum
energy splitting in the system Hamiltonian. Situations that demand special attention therefore include, for example,
systems strongly coupled to their environment, and systems with degenerate or nearly degenerate energy levels.
For non-unitary evolution of a quantum systems, i.e., evolution that includes incoherent processes such as relaxation
and dephasing, it is common to use master equations. In QuTiP, the same function (qutip.mesolve) is used for
evolution both according to the Schrdinger equation and to the master equation, even though these two equations
of motion are very different. The qutip.mesolve function automatically determines if it is sufficient to use the
Schrdinger equation (if no collapse operators were given) or if it has to use the master equation (if collapse operators
were given). Note that to calculate the time evolution according to the Schrdinger equation is easier and much faster
(for large systems) than using the master equation, so if possible the solver will fall back on using the Schrdinger
equation.
What is new in the master equation compared to the Schrdinger equation are processes that describe dissipation in
the quantum system due to its interaction with an environment. These environmental interactions are defined by the
operators through which the system couples to the environment, and rates that describe the strength of the processes.
In QuTiP, the product of the square root of the rate and the operator that describe the dissipation process is called a col-
lapse operator. A list of collapse operators (c_op_list) is passed as the fourth argument to the qutip.mesolve
function in order to define the dissipation processes in the master equation. When the c_op_list isnt empty, the
qutip.mesolve function will use the master equation instead of the unitary Schrdinger equation.
Using the example with the spin dynamics from the previous section, we can easily add a relaxation process (describing
the dissipation of energy from the spin to its environment), by adding sqrt(0.05) * sigmax() to the previously
empty list in the fourth parameter to the qutip.mesolve function:
>>> tlist = linspace(0.0, 10.0, 100)
>>> expt_list = mesolve(H, psi0, tlist, [sqrt(0.05) * sigmax()], [sigmaz(), sigmay()])
>>>
>>> from pylab import *
>>> plot(tlist, real(expt_list[0]))
>>> plot(tlist, real(expt_list[1]))
>>> xlabel(Time)
>>> ylabel(Expectation values)
>>> legend(("Sigma-Z", "Sigma-Y"))
>>> show()
Here, 0.05 is the rate and the operator x (qutip.operators.sigmax) describes the dissipation process.
Now a slightly more complex example: Consider a two-level atom coupled to a leaky single-mode cavity through a
dipole-type interaction, which supports a coherent exchange of quanta between the two systems. If the atom initially
is in its groundstate and the cavity in a 5-photon Fock state, the dynamics is calculated with the lines following code:
>>> tlist = linspace(0.0, 10.0, 200)
>>> psi0 = tensor(fock(2,0), fock(10, 5))
>>> a = tensor(qeye(2), destroy(10))
>>> sm = tensor(destroy(2), qeye(10))
>>> H = 2*pi * a.dag() * a + 2 * pi * sm.dag() * sm + 2*pi * 0.25 * (sm*a.dag() + sm.dag() * a)
>>> expt_list = mesolve(H, psi0, tlist, ntraj, [sqrt(0.1)*a], [a.dag()*a, sm.dag()*sm])
>>>
>>> from pylab import *
>>> plot(tlist, real(expt_list[0]))
>>> plot(tlist, real(expt_list[1]))
>>> xlabel(Time)
>>> ylabel(Expectation values)
>>> legend(("cavity photon number", "atom excitation probability"))
>>> show()
Introduction
Where as the density matrix formalism describes the ensemble average over many identical realizations of a quantum
system, the Monte-Carlo (MC), or quantum-jump approach to wave function evolution, allows for simulating an
individual realization of the system dynamics. Here, the environment is continuously monitored, resulting in a series
of quantum jumps in the system wave function, conditioned on the increase in information gained about the state of
the system via the environmental measurements. In general, this evolution is governed by the Schrdinger equation
with a non-Hermitian effective Hamiltonian
ih X +
Heff = Hsys Cn Cn , (4.4)
2 i
where again, the Cn are collapse operators, each corresponding to a separate irreversible process with rate n . Here,
the strictly negative non-Hermitian portion of Eq.((4.4)) gives rise to a reduction in the norm of the wave function, that
to first-order in a small time t, is given by h(t + t)|(t + t)i = 1 p where
X
(t)|Cn+ Cn |(t) ,
p = t (4.5)
n
and t is such that p 1. With a probability of remaining in the state |(t + t)i given by 1 p, the corresponding
quantum jump probability is thus Eq.((4.5)). If the environmental measurements register a quantum jump, say via the
emission of a photon into the environment, or a change in the spin of a quantum dot, the wave function undergoes a
jump into a state defined by projecting |(t)i using the collapse operator Cn corresponding to the measurement
1/2
|(t + t)i = Cn |(t)i / (t)|Cn+ Cn |(t)
. (4.6)
If more than a single collapse operator is present in Eq~((4.4)), the probability of collapse due to the ith-operator Ci
is given by
Evaluating the MC evolution to first-order in time is quite tedious. Instead, QuTiP uses the following algorithm to
simulate a single realization of a quantum system. Starting from a pure state |(0)i:
I: Choose a random number r between zero and one, representing the probability that a quantum jump occurs.
II: Integrate the Schrdinger equation ((??)), using the effective Hamiltonian ((4.4)) until a time such that the
norm of the wave function satisfies h( ) |( )i = r, at which point a jump occurs.
III: The resultant jump projects the system at time into one of the renormalized states given by Eq.((4.6)).
The corresponding collapse operator Cn is chosen such that n is the smallest integer satisfying:
n
X
Pn ( ) r (4.8)
i=1
where the individual Pn are given by Eq.((4.7)). Note that the left hand side of Eq.((4.8)) is, by definition, normalized
to unity.
IV: Using the renormalized state from step III as the new initial condition at time , draw a new random number,
and repeat the above procedure until the final simulation time is reached.
Monte-Carlo in QuTiP:
In QuTiP, Monto-Carlo evolution is implemented with the qutip.mcsolve function. It takes nearly the same
arguments as the qutip.mesolve function for master-equation evolution, except that the initial state must be a ket
vector, as oppose to a density matrix, and there is an optional keyword parameter ntraj that defines the number of
stochastic trajectories to be simulated. By default, ntraj=500 indicating that 500 monte-carlo trajectories will be
performed.
To illustrate the use of the Monte-Carlo evolution of quantum systems in QuTiP, lets again consider the case of a
two-level atom coupled to a leaky cavity. The only differences to the master-equation treatment is that in this case we
invoke the qutip.mcsolve function instead of qutip.mesolve:
from qutip import *
from pylab import *
The advantage of the Monte-Carlo method over the master equation approach is that only the state vector is required
to be kept in the computers memory, as opposed to the entire density matrix. For large quantum system this becomes a
significant advantage, and the Monte-Carlo solver is therefore generally recommended for such systems. For example,
simulating a Heisenberg spin-chain consisting of 10 spins with random parameters and initial states takes almost
7 times longer using the master equation rather than Monte-Carlo approach with the default number of trajectories
running on a quad-CPU machine. Furthermore, it takes about 7 times the memory as well. However, for small systems,
the added overhead of averaging a large number of stochastic trajectories to obtain the open system dynamics, as well
as starting the multiprocessing functionality, outweighs the benefit of the minor (in this case) memory saving. Master
equation methods are therefore generally more efficient when Hilbert space sizes are on the order of a couple of
hundred states or smaller.
Like the master equation solver qutip.mesolve, the Monte-Carlo solver returns a Odedata object consisting of
expectation values, if the user has defined expectation value oeprators in the 5th-arguement to mcsolve, or state
vectors if no expectation value operators are given. If state vectors are returned, then the qutip.Odedata returned
by qutip.mcsolve will be an array of length ntraj, with each element containing an array of ket-type qobjs
with the same number of elements as tlist. Furthermore, the output Odedata object will also contain a list of times
at which collapse occurred, and which collapse operators did the collapse, in the col_times and col_which
properties, respectively. See example Visualizing MC Collapse Times and Operators for an example using these
properties.
As mentioned earlier, by default, the mcsolve function runs 500 trajectories. This value was chosen because it gives
good accuracy, monte-carlo errors scale as 1/n where n is the number of trajectories, and simultaneously does not take
an excessive amount of time to run. However, like many other options in QuTiP you are free to change the number of
trajectories to fit your needs. If we want to run 1000 trajectories in the above example, we can simply modify the call
to macsolve like:
>>> data = mcsolve(H, psi0, tlist, [sqrt(0.1)*a], [a.dag()*a, sm.dag()*sm],ntraj=1000)
where we have added the keyword arguement ntraj=1000 at the end of the inputs. Now, the Monte-Carlo solver
will calculate expectation values for both operators, a.dag()*a, sm.dag()*sm averaging over 1000 trajectories.
Sometimes one is also interested in seeing how the Monte-Carlo trajectories converge to the master equation solution
by calculating expectation values over a range of trajectory numbers. If, for example, we want to average over 1, 10,
100, and 1000 trajectories, then we can input this into the solver using:
>>> ntraj = [1,10,100,1000]
Keep in mind that the input list must be in ascending order since the total number of trajectories run by mcsolve will
be calculated using the last element of ntraj. In this case, we need to use an extra index when getting the expectation
values from the ODedata object returned by mcsolve. In the above example using:
>>> data = mcsolve(H, psi0, tlist, [sqrt(0.1)*a], [a.dag()*a, sm.dag()*sm],ntraj=[1,10,100,1000])
Note: This section covers a specialized topic and may be skipped if you are new to QuTiP.
In order to solve a given simulation as fast as possible, the solvers in QuTiP take the given input operators and break
them down into simpler components before passing them on to the ODE solvers. Although these operations are
reasonably fast, the time spent organizing data can become appreciable when repeatedly solving a system over, for
example, many different initial conditions. In cases such as this, the Hamiltonian and other operators may be reused
after the initial configuration, thus speeding up calculations. Note that, unless you are planning to reuse the data many
times, this functionality will not be very useful.
To turn on the reuse functionality we must set the rhs_reuse=True flag in the qutip.Odeoptions:
>>> options=Odeoptions(rhs_reuse=True)
A full account of this feature is given in Setting Options for the Dynamics ODE Solvers. Using the previous example,
we will calculate the dynamics for two different initial states, with the Hamiltonian data being reused on the second
call:
from qutip import *
from pylab import *
#construct operators
tlist = linspace(0.0, 10.0, 200)
psi0 = tensor(fock(2,0), fock(10, 5))
a = tensor(qeye(2), destroy(10))
sm = tensor(destroy(2), qeye(10))
H = 2*pi * a.dag() * a + 2 * pi * sm.dag() * sm + 2*pi * 0.25 * (sm*a.dag() + sm.dag() * a)
#first run
data1 = mcsolve(H, psi0, tlist, [sqrt(0.1)*a], [a.dag()*a, sm.dag()*sm])
In addition to the initial state, one may reuse the Hamiltonian data when changing the number of trajectories ntraj
or simulation times tlist. The reusing of Hamiltonian data is also supported for time-dependent Hamiltonians. See
Solving Time-dependent Hamiltonians for further details.
Introduction
The Lindblad master equation introduced earlier is constructed so that it describes a physical evolution of the density
matrix (i.e., trace and positivity preserving), but it does not provide a connection to any underlaying microscopic phys-
ical model. The Lindblad operators (collapse operators) describe phenomenological processes, such as for example
dephasing and spin flips, and the rates of these processes are arbitrary parameters in the model. In many situations the
collapse operators and their corresponding rates have clear physical interpretation, such as dephasing and relaxation
rates, and in those cases the Lindblad master equation is usually the method of choice.
However, in some cases, for example systems with varying energy biases and eigenstates and that couple to an envi-
ronment in some well-defined manner (through a physically motivated system-environment interaction operator), it is
often desirable to derive the master equation from more fundamental physical principles, and relate it to for example
the noise-power spectrum of the environment.
The Bloch-Redfield formalism is one such approach to derive a master equation from a microscopic system. It starts
from a combined system-environment perspective, and derives a perturbative master equation for the system alone,
under the assumption of weak system-environment coupling. One advantage of this approach is that the dissipation
processes and rates are obtained directly from the properties of the environment. On the downside, it does not in-
trinsically guarantee that the resulting master equation unconditionally preserves the physical properties of the density
matrix (because it is a perturbative method). The Bloch-Redfield master equation must therefore be used with care, and
the assumptions made in the derivation must be honored. (The Lindblad master equation is in a sense more robust it
always results in a physical density matrix although some collapse operators might not be physically justified). For
a full derivation of the Bloch Redfield master equation, see e.g. Atom-Physics Interactions by Cohen-Tannoudji et al.
(Wiley, 1992), or Theory of open quantum systems by Breuer and Petruccione (Oxford, 2002). Here we present only a
brief version of the derivation, with the intention of introducing the notation and how it relates to the implementation
in QuTiP.
The starting point of the Bloch-Redfield formalism is the total Hamiltonian for the system and the environment (bath):
H = HS +HB +HI , where H is the total system+bath Hamiltonian, HS and HB are the system and bath Hamiltonians,
respectively, and HI is the interaction Hamiltonian.
The most general form of a master equation for the system dynamics is obtained by tracing out the bath from the
von-Neumann equation of motion for the combined system ( = ih1 [H, ]). In the interaction picture the result is
Z t
d
S (t) = h2 d TrB [HI (t), [HI ( ), S ( ) B ]], (4.9)
dt 0
where the additional assumption that the total system-bath density matrix can be factorized as (t) S (t) B .
This assumption is known as the Born approximation, and it implies that there never is any entanglement between the
system and the bath, neither in the initial state nor at any time during the evolution. It is justified for weak system-bath
interaction.
The master equation (4.9) is non-Markovian, i.e., the change in the density matrix at a time t depends on states at all
times < t, making it intractable to solve both theoretically and numerically. To make progress towards a manageable
master equation, we now introduce the Markovian approximation, in which (s) is replaced by (t) in Eq. (4.9). The
result is the Redfield equation
Z t
d
S (t) = h2 d TrB [HI (t), [HI ( ), S (t) B ]], (4.10)
dt 0
which is local in time with respect the density matrix, but still not Markovian since it contains an implicit dependence
on the initial state. By extending the integration to infinity and substituting t , a fully Markovian master
equation is obtained:
Z
d 2
S (t) = h d TrB [HI (t), [HI (t ), S (t) B ]]. (4.11)
dt 0
The two Markovian approximations introduced above are valid if the time-scale with which the system dynamics
changes is large compared to the time-scale with which correlations in the bath decays (corresponding to a short-
memory bath, which results in Markovian system dynamics).
The master equation (4.11) is still on a too general form to bePsuitable for numerical implementation. We therefore
assume that the system-bath interaction takes the form HI = A B and where A are system operators and
B are bath operators. This allows us to write master equation in terms of system operators and bath correlation
functions:
d XZ
S (t) = h2 d {g ( ) [A (t)A (t )S (t) A (t )S (t)A (t)]
dt 0
where g ( ) = TrB [B (t)B (t )B ] = hB ( )B (0)i, since the bath state B is a steady state.
In the eigenbasis of the system Hamiltonian, where Amn (t) = Amn eimn t , mn = m n and m are the
eigenfrequencies corresponding the eigenstate |mi, we obtain in matrix form in the Schrdinger picture
sec Z
( " #
d 2
XX X
icn ica
ab (t) = iab ab (t) h d g ( ) bd Aan Anc e Aac Adb e
dt
, c,d 0 n
" #)
X ind ibd
+ g ( ) ac Adn Anb e Aac Adb e cd (t),
n
where the sec above the summation symbol indicate summation of the secular terms which satisfy |ab cd |
decay . This is an almost-useful form of the master equation. The final step before arriving at the form of the Bloch-
Redfield master equation that is implemented in QuTiP, involves
R rewriting the bath correlation function g( ) in terms
of the noise-power spectrum of the environment S() = d ei g( ):
Z
1
d g ( )ei = S () + i (), (4.12)
0 2
where ab () is an energy shift that is neglected here. The final form of the Bloch-Redfield master equation is
sec
d X
ab (t) = iab ab (t) + Rabcd cd (t), (4.13)
dt
c,d
where
(
h2 X X
Rabcd = bd A
an Anc S (cn ) Aac Adb S (ca )
2 n
,
)
X
+ ac A
dn Anb S (dn ) Aac Adb S (db ) ,
n
In QuTiP, the Bloch-Redfield tensor Eq. (4.5.4) can be calculated using the function bloch_redfield_tensor.
It takes three mandatory arguments: The system Hamiltonian H, a list of operators through which to the bath A , and
a list of corresponding spectral density functions S (). The spectral density functions are callback functions that
takes the (angular) frequency as a single argument.
To illustrate how to calculate the Bloch-Redfield tensor, lets consider a two-level atom
1 1
H = x 0 z (4.14)
2 2
that couples to an Ohmic bath through the x operator. The corresponding Bloch-Redfield tensor can be calculated in
QuTiP using the following code:
>>> delta = 0.2 * 2*pi; eps0 = 1.0 * 2*pi; gamma1 = 0.5
>>> H = - delta/2.0 * sigmax() - eps0/2.0 * sigmaz()
>>> def ohmic_spectrum(w):
>>> if w == 0.0: # dephasing inducing noise
>>> return gamma1
For convenience, the function bloch_redfield_tensor also returns a list of eigenkets ekets, since they are
calculated in the process of calculating the Bloch-Redfield tensor R, and the ekets are usually needed again later when
transforming operators between the computational basis and the eigenbasis.
The evolution of a wavefunction or density matrix, according to the Bloch-Redfield master equation (4.13), can be
calculated using the QuTiP function bloch_redfield_solve. It takes five mandatory arguments: the Bloch-
Redfield tensor R, the list of eigenkets ekets, the initial state psi0 (as a ket or density matrix), a list of times tlist for
which to evaluate the expectation values, and a list of expectation values to evaluate at each time-step defined by tlist.
For example, to evaluate the expectation values of the x , y , and z operators for the example above, we can use the
following code:
>>> tlist = linspace(0, 15.0, 1000)
>>> psi0 = rand_ket(2)
>>> e_ops = [sigmax(), sigmay(), sigmaz()]
>>> expt_values = bloch_redfield_solve(R, ekets, psi0, tlist, e_ops)
>>>
>>> sphere = Bloch()
>>> sphere.add_points([expt_values[0], expt_values[1], expt_values[2]])
>>> sphere.vector_color = [r]
>>> # Hamiltonian axis
>>> sphere.add_vectors(array([delta, 0, eps0]) / sqrt(delta**2 + eps0**2))
>>> sphere.make_sphere()
>>> show()
The two steps of calculating the Bloch-Redfield tensor and evolve the corresponding master equation can be combined
into one by using the function brmesolve, which takes same arguments as mesolve and mcsolve expect for the
additional list of spectral callback functions.
>>> output = brmesolve(H, psi0, tlist, [sigmax()], e_ops, [ohmic_spectrum])
In the previous examples of quantum evolution, we assumed that the systems under consideration were described by
time-independent Hamiltonians. However, many systems have explicit time-dependence in either the Hamiltonian, or
the collapse operators describing coupling to the environment, and sometimes both components might depend on time.
The two main evolution solvers in QuTiP, qutip.mesolve and qutip.mcsolve, discussed in Lindblad Master
Equation Solver and Quantum Dynamics via the Monte-Carlo Solver respectively, are capable of handling time-
dependent Hamiltonians and collapse terms. There are, in general, three different ways to implement time-dependent
problems in QuTiP 2:
1. Function based: Hamiltonian / collapse operators expressed using [qobj,func] pairs, where the time-dependent
coefficients of the Hamiltonian (or collapse operators) are expresed in the Python functions.
2. String (Cython) based: The Hamiltonian and/or collapse operators are expressed as a list of [qobj,string] pairs,
where the time-dependent coefficients are represented as strings. The resulting Hamiltonian is then compiled
into c-code using Cython and executed.
3. Hamiltonian function (outdated): The Hamiltonian is itself a Python function with time-dependence. Collapse
operators must be time-independent using this input format.
Give the multiple choices of input style, the first question that arrises is which option to choose? In short, the function
based method (option #1) is the most general, allowing for essentially arbitrary coefficients expressed via user defined
functions. However, by automatically compiling your system into c-code, the second option (string based) tends to be
more efficient and will run faster. Of course, for small system sizes and evolution times, the difference will be minor.
Although this method does not support all time-dependent coefficients that one can think of, it does support essentially
all problems that one would typically encounter. If you can write you time-dependent coefficients using any of the
following functions, or combinations thereof (including constants) then you may use this method:
[acos, acosh, asin, asinh, atan, atan2, atanh, ceil
, copysign, cos, cosh, degrees, erf, erfc, exp, expm1
, fabs, factorial, floor, fmod, frexp, fsum, gamma
, hypot, isinf, isnan, ldexp, lgamma, log, log10, log1p
, modf, pow, radians, sin, sinh, sqrt, tan, tanh, trunc]
Finally option #3, expressing the Hamiltonian as a Python function, is the original method for time-dependence in
QuTiP 1.x. However, this method is somewhat less efficient then the previously mentioned methods, and does not
allow for time-dependent collapse operators.
A collection of examples demonstrating the simulation of time-dependent problems can be found here: Dynamics of
time-dependent systems
The most general way to write a time-dependent Hamiltonian or collapse operator is by using Python functions as the
time-dependent coefficients. To accomplish this, we need to write a python function that returns the time-dependent
coefficient. Additionally, we need to tell QuTiP that a given Hamiltonian or collapse operator should be associated with
a given Python function. To do this, one needs to specify operator-function pairs in list format: [Op,py_coeff],
where Op is a given Hamiltonian or collapse operator and py_coeff is the name of the Python function representing
the coefficient. With this format, the form of the Hamiltonian for both mesolve and mcsolve is:
>>> H = [H0,[H1,py_coeff1],[H2,py_coeff2],...]
where H0 is a time-INDEPENDENT Hamiltonian, while H1,H2, are time-dependent. Keep in mind that there must
always be at least one time-independent Hamiltonian term. The same format can be used for collapse operators:
>>> c_op_list = [[C0,py_coeff0],C1,[C2,py_coeff2],...]
Here we have demonstrated that the ordering of time-dependent and time-independent terms does not matter. In
addition, any or all of the collapse operators may be time-dependent.
Note: While, in general, you can arrange time-dependent and independent terms in any order you like, it is best to
place all time-independent terms first.
As an example, we will look at Single photon source based on a three level atom strongly coupled to a cavity that has
i of the form H = H0 f (t)H1 where f (t) is the time-dependent driving strength given
a time-dependenthHamiltonian
2
as f (t) = A exp (t/) . The follow code sets up the problem:
from qutip import *
# Define atomic states. Use ordering from paper
ustate = basis(3,0)
excited = basis(3,1)
ground = basis(3,2)
Given that we have a single time-dependent Hamiltonian term, and constant collapse terms, we need to specify a single
Python function for the coefficient f (t). In this case, one can simply do:
def H1_coeff(t, args):
return 9 * exp(-(t/5.)**2)
In this case, the return value dependents only on time. However, when specifying Python functions for coefficients,
the function must have (t,args) as the input variables, in that order. Having specified our coefficient function, we
can now specify the Hamiltonian in list format and call the solver (in this case mesolve):
H=[H0,[H1,H1_coeff]]
output = mesolve(H, psi0, t, c_op_list,[ada, sigma_UU, sigma_GG])
We can call the Monte-Carlo solver in the exact same way (if using the default ntraj=500):
>>> output = mcsolve(H, psi0, t, c_op_list,[ada, sigma_UU, sigma_GG])
The output from the master equation solver is identical to that shown in the examples, the monte-carlo however will
be noticeably off, suggesting we should increase the number of trajectories for this example. In addition, we can also
consider the decay of a simple Harmonic oscillator with time-varying decay rate:
from qutip import *
kappa=0.5
def col_coeff(t,args): #coefficient function
return sqrt(kappa*exp(-t))
N=10 #number of basis states
a=destroy(N)
H=a.dag()*a #simple HO
psi0=basis(N,9) #initial state
c_op_list=[[a,col_coeff]] #time-depdendent collapse term
tlist=linspace(0,10,100)
output=mesolve(H,psi0,tlist,c_op_list,[a.dag()*a])
A comparison of this time-dependent damping, with that of a constant decay term is presented below.
In the previous example we hardcoded all of the variables, driving amplitude A and width , with their numerical
values. This is fine for problems that are specialized, or that we only want to run once. However, in many cases, we
would like to change the parameters of the problem in only one location (usually at the top of the script), and not have
to worry about manually changing the values on each run. QuTiP allows you to accomplish this using the keyword
args as an input to the solvers. For instance, instead of explicitly writing 9 for the amplitude and 5 for the width of
the gaussian driving term, we can make us of the args variable:
or equivalently:
def H1_coeff(t, args):
A=args[A]
sig=args[sigma]
return A * exp(-(t/sig)**2)
where args is a Python dictionary of key:value pairs args = {A:a,sigma:b} where a and b are the two
parameters for the amplitude and width, respectively. Of course, we can always hardcode the values in the dictionary
as well args = {A:9,sigma:5}, but there is much more flexibility by using variables in args. To let the
solvers know that we have a set of args to pass we append the args to the end of the solver input:
>>> output=mesolve(H,psi0,tlist,c_op_list,[a.dag()*a],args={A:9,sigma:5})
Note: You must have Cython installed on your computer to use this format. See Installation for instructions on
installing Cython.
The string-based time-dependent format works in a similar manner as the previously discussed Python function
method. That being said, the underlying code does something completely different. When using this format, the
strings used to represent the time-dependent coefficients, as well as Hamiltonian and collapse operators, are rewritten
as Cython code using a code generator class and then compiled into c-code. The details of this meta-programming
will be published in due course. however, in short, this can lead to a substantial reduction in time for complex time-
dependent problems, or when simulating over long intervals. We remind the reader that the types of functions that can
be used with this method is limited to:
[acos, acosh, asin, asinh, atan, atan2, atanh, ceil
, copysign, cos, cosh, degrees, erf, erfc, exp, expm1
, fabs, factorial, floor, fmod, frexp, fsum, gamma
, hypot, isinf, isnan, ldexp, lgamma, log, log10, log1p
, modf, pow, radians, sin, sinh, sqrt, tan, tanh, trunc]
Like the previous method, the string based format uses a list pair format [Op,str] where str is now a string rep-
resenting the time-dependent coefficient. For our first example, this string would be 9 * exp(-(t/5.)**2).
The Hamiltonian in this format would take the form:
>>> H=[H0,[H1,9 * exp(-(t/5.)**2)]]
Notice that this is a valid Hamiltonian for the string based format as exp is included in the above list of suitable
functions. Once again, you need at least one time-independent Hamiltonian term. Calling the solvers is the same as
before:
>>> output=mesolve(H,psi0,tlist,c_op_list,[a.dag()*a])
We can also use the args variable in the same manner as before, however we must rewrite our string term to read: A
* exp(-(t/sig)**2):
H=[H0,[H1,A * exp(-(t/sig)**2)]]
args = {A:9,sig:5}
output=mesolve(H,psi0,tlist,c_op_list,[a.dag()*a],args=args)
Important: Naming your args variables e or pi will mess things up when using the string based format.
In the previous version of QuTiP, the simulation of time-dependent problems required writing the Hamiltonian itself
as a Python function. This is in fact the method used in our example Single photon source based on a three level
atom strongly coupled to a cavity. However, this method does not allow for time-dependent collapse operators, and
is therefore more restrictive. Furthermore, it is less efficient than the other methods for all but the most basic of
Hamiltonians (see the next section for a comparison of times.). In this format, the entire Hamiltonian is written as a
Python function:
def Hfunc(t, args):
H0 = args[0]
H1 = args[1]
w = 9 * exp(-(t/5.)**2)
return H0 - w * H1
where the args variable must always be given, and is now a list of Hamiltonian terms: args=[H0,H1]. In this
format, our call to the master equation is now:
>>> output=mesolve(Hfunc,psi0,tlist,c_op_list,[a.dag()*a],args=[H0,H1])
We cannot evaluate time-dependent collapse operators in this format, so we can not simulate the previous harmonic
oscillator decay example.
Here we give a table of simulation times for the single-photon example using the different time-dependent formats and
both the master equation and monte-carlo solver.
Format Master Equation Monte-Carlo
Python Function 2.1 sec 27 sec
Cython String 1.4 sec 9 sec
Hamiltonian Function 1.0 sec 238 sec
For the current example, the table indicates that the Hamiltonian function method is in fact the fastest when using the
master equation solver. This is because the simulation is quite small. In contrast, the Hamiltonian function is over 26x
slower than the compiled string version when using the monte-carlo solver. In this case, the 500 trajectories needed in
the simulation highlights the inefficient nature of the Python function calls.
Note: This section covers a specialized topic and may be skipped if you are new to QuTiP.
When repeatedly simulating a system where only the time-dependent variables, or initial state change, it is possible to
reuse the Hamiltonian data stored in QuTiP and there by avoid spending time needlessly preparing the Hamiltonian
and collapse terms for simulation. To turn on the the reuse features, we must pass a qutip.Odeoptions object
with the rhs_resue flag turned on. Instructions on setting flags are found in Setting Options for the Dynamics ODE
Solvers. For example, we can do:
H=[H0,[H1,A * exp(-(t/sig)**2)]]
args = {A:9,sig:5}
output=mcsolve(H,psi0,tlist,c_op_list,[a.dag()*a],args=args)
opts=Odeoptions(rhs_reuse=True)
args = {A:10,sig:3}
output=mcsolve(H,psi0,tlist,c_op_list,[a.dag()*a],args=args,options=opts)
In this case, the second call to mcsolve takes 3 seconds less than the first. Of course our parameters are different,
but this also shows how much time one can save by not reorganizing the data, and in the case of the string format, not
recompiling the code. If you need to call the solvers many times for different parameters, this savings will obviously
start to add up.
Note: This section covers a specialized topic and may be skipped if you are new to QuTiP.
In this section we discuss running string based time-dependent problems using the qutip.parfor function. Be-
cause the parfor function uses the built-in Python multiprocessing functionality, the solvers will try to gen-
erate the same compiled code over and over and will crash. To get around this problem you can call the
qutip.rhs_generate function to compile simulation into c-code before calling parfor. You must then set the
qutip.Odedata object rhs_reuse=True for all solver calls inside the parfor loop.
Introduction
Many time-dependent problems of interest are periodic. The dynamics of such systems can be solved for directly by
numerical integration of the Schrdinger or Master equation, using the time-dependent Hamiltonian. But they can
also be transformed into time-independent problems using the Floquet formalism. Time-independent problems can be
solve much more efficiently, so such a transformation is often very desirable.
In the standard derivations of the Lindblad and Bloch-Redfield master equations the Hamiltonian describing the system
under consideration is assumed to be time independent. Thus, strictly speaking, the standard forms of these master
equation formalisms should not blindly be applied to system with time-dependent Hamiltonians. However, in many
relevant cases, in particular for weak driving, the standard master equations still turns out to be useful for many time-
dependent problems. But a more rigorous approach would be to rederive the master equation taking the time-dependent
nature of the Hamiltonian into account from the start. The Floquet-Markov Master equation is one such a formalism,
with important applications for strongly driven systems [see e.g., Grifoni et al., Physics Reports 304, 299 (1998)].
Here we give an overview of how the Floquet and Floquet-Markov formalisms can be used for solving time-dependent
problems in QuTiP. To introduce the terminology and naming conventions used in QuTiP we first give a brief summary
of quantum Floquet theory.
where (t) are the Floquet states (i.e., the set of wave function solutions to the Schrdinger equation), (t) =
(t + T ) are the periodic Floquet modes, and are the quasienergy levels. The quasienergy levels are constants in
time, but only uniquely defined up to multiples of 2/T (i.e., unique value in the interval [0, 2/T ]).
If we know the Floquet modes (for t [0, T ]) and the quasienergies for a particular H(t), we can easily decompose
any initial wavefunction (t = 0) in the Floquet states and immediately obtain the solution for arbitrary t
X X
(t) = c (t) = c exp(i t/h) (t), (4.17)
P
where the coefficients c are determined by the initial wavefunction (0) = c (0).
This formalism is useful for finding (t) for a given H(t) only if we can obtain the Floquet modes a (t) and
quasienergies more easily than directly solving (4.15). By substituting (4.16) into the Schrdinger equation (4.15)
we obtain an eigenvalue equation for the Floquet modes and quasienergies
where H(t) = H(t) iht . This eigenvalue problem could be solved analytically or numerically, but in QuTiP we
use an alternative approach for numerically finding the Floquet states and quasienergies [see e.g. Creffield et al., Phys.
Rev. B 67, 165301 (2003)]. Consider the propagator for the time-dependent Schrdinger equation (4.15), which by
definition satisfies
U (T + t, t)(t) = (T + t).
Inserting the Floquet states from (4.16) into this expression results in
which shows that the Floquet modes are eigenstates of the one-period propagator. We can therefore find the Floquet
modes and quasienergies = h arg( )/T by numerically calculating U (T +t, t) and diagonalizing it. In particular
this method is useful to find (0) by calculating and diagonalize U (T, 0).
The Floquet modes at arbitrary time t can then be found by propagating (0) to (t) using the wave function
propagator U (t, 0) (0) = (t), which for the Floquet modes yields
so that (t) = exp(i t/h)U (t, 0) (0). Since (t) is periodic we only need to evaluate it for t [0, T ], and
from (t [0, T ]) we can directly evaluate (t), (t) and (t) for arbitrary large t.
QuTiP provides a family of functions to calculate the Floquet modes and quasi energies, Floquet state decomposition,
etc., given a time-dependent Hamiltonian on the callback format, list-string format and list-callback format (see, e.g.,
qutip.mesolve for details).
Consider for example the case of a strongly driven two-level atom, described by the Hamiltonian
1 1 1
H(t) = x 0 z + A sin(t)z . (4.19)
2 2 2
In QuTiP we can define this Hamiltonian as follows
>>> delta = 0.2 * 2*pi; eps0 = 1.0 * 2*pi; A = 2.5 * 2*pi; omega = 1.0 * 2*pi
>>> H0 = - delta/2.0 * sigmax() - eps0/2.0 * sigmaz()
>>> H1 = A/2.0 * sigmaz()
>>> args = {w: omega}
>>> H = [H0, [H1, sin(w * t)]]
The t = 0 Floquet modes corresponding to the Hamiltonian (4.19) can then be calculated using the
qutip.floquet.floquet_modes function, which returns lists containing the Floquet modes and the quasiener-
gies
>>> T = 2*pi / omega
>>> f_modes, f_energies = floquet_modes(H, T, args)
>>> f_energies
array([ 2.83131211, -2.83131211])
>>> f_modes0
[Quantum object: dims = [[2], [1]], shape = [2, 1], type = ket
Qobj data =
[[ 0.39993745+0.554682j]
[ 0.72964232+0.j ]],
Quantum object: dims = [[2], [1]], shape = [2, 1], type = ket
Qobj data =
[[ 0.72964232+0.j ]
[-0.39993745+0.554682j]]]
For some problems interesting observations can be draw from the quasienergy levels alone. Consider for example the
quasienergies for the driven two-level system introduced above as a function of the driving amplitude, calculated and
plotted in the following example. For certain driving amplitudes the quasienergy levels cross. Since the the quasiener-
gies can be associated with the time-scale of the long-term dynamics due that the driving, degenerate quasienergies
indicates a freezing of the dynamics (sometimes known as coherent destruction of tunneling).
from qutip import *
Given the Floquet modes at t = 0, we obtain the Floquet mode at some later time t using the function
qutip.floquet.floquet_mode_t:
>>> f_modes_t = floquet_modes_t(f_modes_0, f_energies, 2.5, H, T, args)
>>> f_modes_t
[Quantum object: dims = [[2], [1]], shape = [2, 1], type = ket
Qobj data =
[[-0.03189259+0.6830849j ]
[-0.61110159+0.39866357j]],
Quantum object: dims = [[2], [1]], shape = [2, 1], type = ket
Qobj data =
[[-0.61110159-0.39866357j]
[ 0.03189259+0.6830849j ]]]
The purpose of calculating the Floquet modes is to find the wavefunction solution to the original problem (4.19) given
some initial state |0 i. To do that, we first need to decompose the initial state in the Floquet states, using the function
qutip.floquet.floquet_state_decomposition
>>> psi0 = rand_ket(2)
>>> f_coeff = floquet_state_decomposition(f_modes_0, f_energies, psi0)
[(0.81334464307183041-0.15802444453870021j),
(-0.17549465805005662-0.53169576969399113j)]
and given this decomposition of the initial state in the Floquet states we can easily evaluate the wavefunction that is
the solution to (4.19) at an arbitrary time t using the function qutip.floquet.floquet_wavefunction_t
>>> t = 10 * rand()
>>> psi_t = floquet_wavefunction_t(f_modes_0, f_energies, f_coeff, t, H, T, args)
>>> psi_t
Quantum object: dims = [[2], [1]], shape = [2, 1], type = ket
Qobj data =
[[-0.29352582+0.84431304j]
[ 0.30515868+0.32841589j]]
The following example illustrates how to use the functions introduced above to calculate and plot the time-evolution
of (4.19).
from qutip import *
When evaluating the Floquet states or the wavefunction at many points in time it is useful to pre-compute
the Floquet modes for the first period of the driving with the required resolution. In QuTiP the function
qutip.floquet.floquet_modes_table calculates a table of Floquet modes which later can be used together
with the function qutip.floquet.floquet_modes_t_lookup to efficiently lookup the Floquet mode at an
arbitrary time. The following example illustrates how the example from the previous section can be solved more
efficiently using these functions for pre-computing the Floquet modes.
from qutip import *
Note that the parameters and the Hamiltonian used in this example is not the same as in the previous section, and hence
the different appearance of the resulting figure.
A driven system that is interacting with its environment is not necessarily well described by the standard Lindblad
master equation, since its dissipation process could be time-dependent due to the driving. In such cases a rigorious
approach would be to take the driving into account when deriving the master equation. This can be done in many
different ways, but one way common approach is to derive the master equation in the Floquet basis. That approach
results in the so-called Floquet-Markov master equation, see Grifoni et al., Physics Reports 304, 299 (1998) for details.
The QuTiP function qutip.floquet.fmmesolve implements the Floquet-Markov master equation. It calculates
the dynamics of a system given its initial state, a time-dependent hamiltonian, a list of operators through which the
system couples to its environment and a list of corresponding spectral density functions that describes the environment.
In contrast to the qutip.mesolve and qutip.mcsolve, and the qutip.floquet.fmmesolve does char-
acterize the environment with dissipation rates, but extract the strength of the coupling to the environment from the
noise spectral-density functions and the instantaneous Hamiltonian parameters (similar to the Bloch-Redfield master
equation solver qutip.bloch_redfield.brmesolve).
Note: Currently the qutip.floquet.fmmesolve can only accept a single environment coupling operator and
spectral density function.
The noise spectral density function of the environment is implemented as a python callback function that is passed to
the solver. For example:
>>> gamma1 = 0.1
>>> def noise_spectrum(omega):
>>> return 0.5 * gamma1 * omega/(2*pi)
The other parameters are similar to the qutip.mesolve and qutip.mcsolve, and the same format for the
return value is used qutip.Odedata. The following example extends the example studied above, and uses
qutip.floquet.fmmesolve to introduce dissipation into the calculation
from qutip import *
Here we compare the performance of the master-equation and monte-Carlo solvers to their quantum optics toolbox
counterparts.
In this example, we calculate the time-evolution of the density matrix for a coupled oscillator system using the mesolve
function, and compare it to the quantum optics toolbox (qotoolbox). Here, we see that the QuTiP solver out performs
its qotoolbox counterpart by a substantial margin as the system size increases.
To test the Monte-Carlo solvers, here we simulate a trilinear Hamiltonian over a range of Hilbert space sizes. Since
QuTiP uses multiprocessing, we can measure the performance gain when using several CPUs. In contrast, the qotool-
box is limited to a single-processor only. In the legend, we show the speed-up factor in the parenthesis, which should
ideally be equal to the number of processors. Finally, we have included the results using hyperthreading, written here
as 4+(x) where x is the number of hyperthreads, found in some newer Intel processors. We see however that the
performance benefits from hyperthreading are marginal at best.
Occasionally it is necessary to change the built in parameters of the ODE solvers used by both the mesolve and
mcsolve functions. The ODE options for either of these functions may be changed by calling the Odeoptions class
qutip.Odeoptions
>>> opts=Odeoptions()
the properties and default values of this class can be view via the print command:
Note that the order in which you input the options does not matter. Using either method, the resulting opts variable is
now:
>>> print opts
Odeoptions properties:
----------------------
atol: 1e-10
rtol: 1e-06
method: adams
order: 12
nsteps: 1000
first_step: 0
min_step: 0
max_step: 0
tidy: True
num_cpus: 3
rhs_filename: None
rhs_reuse: False
gui: False
expect_avg: True
To use these new settings we can use the keyword argument options in either the mesolve or mcsolve function.
We can modify the last example as:
>>> mesolve(H0, psi0, tlist, c_op_list, [sigmaz()],options=opts)
>>> mesolve(hamiltonian_t, psi0, tlist, c_op_list, [sigmaz()], H_args,options=opts)
or:
>>> mcsolve(H0, psi0, tlist, ntraj,c_op_list, [sigmaz()],options=opts)
>>> mcsolve(hamiltonian_t, psi0, tlist, ntraj, c_op_list, [sigmaz()], H_args,options=opts)
4.6.1 Introduction
For open quantum systems with decay rates larger than the corresponding excitation rate, the system will tend toward
a steady-state as t . For many these systems, solving for the asymptotic state vector can be achieved using
an iterative method faster than master equation or monte-carlo methods. In QuTiP, the steady-state solution for a
system Hamiltonian or Louivillian is given by qutip.steady.steadystate or qutip.steady.steady,
respectively. Both of these functions use an inverse power method with a random initial state. (Details of these methods
may be found in any iterative linear algebra text.) In general, it is best to use the qutip.steady.steadystate
function with a given Hamiltonian and list of collapse operators. This function will automatically build the Louivillian
for you and then call the qutip.steady.steady function.
A general call to the steady-state solver qutip.steady.steadystate may be accomplished using the command:
>>> ss_ket=steadystate(H,c_op_list)
where H is a quantum object representing the system Hamiltonian, and c_op_list is a list of quantum objects for
the system collapse operators. The output, labeled as ss_ket, is the steady-state solution for the system dynamics.
Note that the output here is a ket-vector and not a density matrix. Although this method will produce the required
solution in most situations, there are additional options that may be set if the algorithm does not converge properly.
These optional parameters may be used by calling the steady-state solver as:
>>> ss_ket=steadystate(H,c_op_list,maxiter,tol,method)
where maxiter is the maximum number of iterations performed by the solver, tol is the requested solution tol-
erance, and method indicates whether the linear equation solver uses a direct or iterative solution method. More
information on these options may be found in the qutip.steady.steadystate section of the API.
This solver can also use a Louvillian constructed from a Hamiltonian and collapse operators as the input variable when
called using the qutip.steady.steady function:
>>> ss_ket=steady(L)
where L is the Louvillian. This function also takes the previously mentioned optional parameters. We do however
recommend using the qutip.steady.steadystate function as it will automatically build the system Louvillian
and call qutip.steady.steady for you.
A simple example of a system that reaches a steady-state is a harmonic oscillator coupled to a thermal environment.
Below we consider a harmonic oscillator, initially in a |10i number state, and weakly coupled to a thermal environment
characterized by an average particle expectation value of hni = 2. We calculate the evolution via master equation and
monte-carlo methods, and see that they converge to the steady-state solution. Here we choose to perform only a few
Monte Carlo trajectories so we can distinguish this evolution from the master equation solution.
from qutip import *
from pylab import *
# Define paramters
N=20 #number of basis states to consider
a=destroy(N)
H=a.dag()*a
psi0=basis(N,10) #initial state
kappa=0.1 #coupling to oscillator
# collapse operators
c_op_list = []
n_th_a = 2 # temperature with average of 2 excitations
rate = kappa * (1 + n_th_a)
if rate > 0.0:
c_op_list.append(sqrt(rate) * a) #decay operators
rate = kappa * n_th_a
if rate > 0.0:
c_op_list.append(sqrt(rate) * a.dag()) #excitation operators
tlist=linspace(0,50,100)
#monte-carlo
mcdata = mcsolve(H,psi0,tlist,c_op_list, [a.dag()*a],ntraj=100)
#master eq.
medata = mesolve(H,psi0,tlist,c_op_list, [a.dag()*a])
plot(tlist,mcdata.expect[0],tlist,medata.expect[0],lw=2)
#plot steady-state expt. value as horizontal line (should be = 2)
axhline(y=fexpt,color=r,lw=1.5)
ylim([0,10])
xlabel(Time,fontsize=14)
ylabel(Number of excitations,fontsize=14)
legend((Monte-Carlo,Master Equation,Steady State))
title(Decay of Fock state $\left|10\\rangle\\right.$+
in a thermal environment with $\langle n\\rangle=2$)
show()
The eseries object in QuTiP is a representation of an exponential-series expansion of time-dependent quantum objects
(a concept borrowed from the quantum optics toolbox).
An exponential series is parameterized by its amplitude coefficients ci and rates ri , so that the series takes the form
E(t) = i ci eri t . The coefficients are typically quantum objects (type Qobj: states, operators, etc.), so that the value
P
of the eseries also is a quantum object, and the rates can be either real or complex numbers (describing decay rates
and oscillation frequencies, respectively). Note that all amplitude coefficients in an exponential series must be of the
same dimensions and composition.
In QuTiP, an exponential series object is constructed by creating an instance of the class qutip.eseries:
>>> es1 = eseries(sigmax(), 1j)
where the first argument is the amplitude coefficient (here, the sigma-X operator), and the second argument is the rate.
The eseries in this example represents the time-dependent operator x eit .
To add more terms to an qutip.eseries object we simply add objects using the + operator:
>>> es2 = eseries(0.5 * sigmax(), 1j * omega) + eseries(0.5 * sigmax(), -1j * omega)
The qutip.eseries in this example represents the operator 0.5x eit + 0.5x eit , which is the exponen-
tial series representation of x cos(t). Alternatively, we can also specify a list of amplitudes and rates when the
qutip.eseries is created:
>>> es2 = eseries([0.5 * sigmax(), 0.5 * sigmax()], [1j * omega, -1j * omega])
We can inspect the structure of an qutip.eseries object by printing it to the standard output console:
>>> print es2
ESERIES object: 2 terms
Hilbert space dimensions: [[2], [2]]
Exponent #0 = 1j
Quantum object: dims = [[2], [2]], shape = [2, 2], type = oper, isHerm = True
Qobj data =
[[ 0. 0.5]
[ 0.5 0. ]]
Exponent #1 = -1j
Quantum object: dims = [[2], [2]], shape = [2, 2], type = oper, isHerm = True
Qobj data =
[[ 0. 0.5]
[ 0.5 0. ]]
To calculate the expectation value of an time-dependent operator represented by an qutip.eseries, we use the
qutip.expect function. For example, consider the operator x cos(t) + z sin(t), and say we would like to
know the expectation value of this operator for a spin in its excited state (rho = fock_dm(2,1) produce this
state):
>>> es3 = eseries([0.5*sigmaz(), 0.5*sigmaz()], [1j, -1j]) + eseries([-0.5j*sigmax(), 0.5j*sigmax()],
>>> rho = fock_dm(2, 1)
>>> es3_expect = expect(rho, es3)
>>> print es3_expect
ESERIES object: 2 terms
Hilbert space dimensions: [[1, 1]]
Exponent #0 = -1j
(-0.5+0j)
Exponent #1 = 1j
(-0.5+0j)
>>> print es3_expect.value([0.0, pi/2])
[ -1.00000000e+00+0.j -6.12303177e-17+0.j]
Note the expectation value of the qutip.eseries object, expect(rho, es3), itself is an qutip.eseries,
but with amplitude coefficients that are C-numbers instead of quantum operators. To evaluate the C-
number qutip.eseries at the times tlist we use esval(es3_expect, tlist), or, equivalently,
es3_expect.value(tlist).
The exponential series formalism can be useful for the time-evolution of quantum systems. One approach to calculat-
ing the time evolution of a quantum system is to diagonalize its Hamiltonian (or Liouvillian, for dissipative systems)
and to express the propagator (e.g., exp(iHt) exp(iHt)) as an exponential series.
The QuTiP function qutip.es2ode and qutip.essolve use this method to evolve quantum systems in time.
The exponential series approach is particularly suitable for cases when the same system is to be evolved for many
different initial states, since the diagonalization only needs to be performed once (as opposed to e.g. the ode solver
that would need to be ran independently for each initial state).
As an example, consider a spin-1/2 with a Hamiltonian pointing in the z direction, and that is subject to noise causing
relaxation. For a spin originally is in the up state, we can create an qutip.eseries object describing its dynamics
by using the qutip.es2ode function:
>>> psi0 = basis(2,1)
>>> H = sigmaz()
>>> L = liouvillian(H, [sqrt(1.0) * destroy(2)])
>>> es = ode2es(L, psi0)
The qutip.eseries.ode2es function diagonalizes the Liouvillian L and creates an exponential series with the
correct eigenfrequencies and amplitudes for the initial state 0 (psi0).
We can examine the resulting qutip.eseries object by printing a text representation:
>>> print es
ESERIES object: 2 terms
Hilbert space dimensions: [[2], [2]]
Exponent #0 = 0j
Quantum object: dims = [[2], [2]], shape = [2, 2], type = oper, isHerm = True
Qobj data =
[[ 1.+0.j 0.+0.j]
[ 0.+0.j 0.+0.j]]
Exponent #1 = (-1+0j)
Quantum object: dims = [[2], [2]], shape = [2, 2], type = oper, isHerm = True
Qobj data =
[[-1.+0.j 0.+0.j]
[ 0.+0.j 1.+0.j]]
and the expectation value of the exponential series can be calculated using the qutip.expect function:
>>> es_expect = expect(sigmaz(), es)
The result es_expect is now an exponential series with C-numbers as amplitudes, which easily can be evaluated at
arbitrary times:
Figure 4.5: The expectation value of the z operator, calculated using the exponential-series expansion of the Liouvil-
lian.
4.8.1 Introduction
When studying the dynamics of a two-level system, it is often convent to visualize the state of the system by plotting
the state-vector or density matrix on the Bloch sphere. In QuTiP, we have created the Bloch class to allow for easy
creation and manipulation of data sets, both vectors and data points, on the Bloch sphere.
which will load an instance of the qutip.Bloch class. Before getting into the details of this object, we can simply
plot the blank Bloch sphere associated with this instance via:
>>> b.show()
Figure 4.6: Empty Bloch sphere generated by calling the Bloch() command.
In addition to the show() command, the Bloch class has the following functions:
Name Input Parameters (#=optional) Description
add_points(pnts,#meth) pnts list/array of (x,y,z) points, meth=m (default Adds a single or set of data points to be
meth=s) will plot a collection of points as multi- plotted on the sphere.
colored data points.
add_states(state,#kind) state Qobj or list/array of Qobjs representing state Input multiple states as a list or array
or density matrix of a two-level system, kind (op-
tional) string specifying if state should be plotted
as point (point) or vector (default).
add_vectors(vec) vec list/array of (x,y,z) points giving direction and adds single or multiple vectors to plot.
length of state vectors.
clear() Removes all data from Bloch sphere.
Keeps customized figure properties.
save(#format,#dirc) format format (default=png) of output file, dirc Saves Bloch sphere to a file.
(default=cwd) output directory
show() Generates Bloch sphere with given data.
As an example, we can add a single data point:
>>> pnt=[1/sqrt(3),1/sqrt(3),1/sqrt(3)]
>>> b.add_points(pnt)
>>> b.show()
Notice that when we add more than a single vector (or data point), a different color will automatically be applied to
the later data set (mod 4). In total, the code for constructing our Bloch sphere with one vector, one state, and a single
data point is:
Figure 4.7: Bloch sphere with single bloch vector represented as a data point.
Figure 4.8: Bloch sphere with a single data point and one vector.
Figure 4.9: Bloch sphere demonstrating multiple vectors and points on a single sphere.
>>> b=Bloch()
>>> pnt=[1/sqrt(3),1/sqrt(3),1/sqrt(3)]
>>> b.add_points(pnt)
>>> #b.show()
>>> vec=[0,1,0]
>>> b.add_vectors(vec)
>>> #b.show()
>>> up=basis(2,0)
>>> b.add_states(up)
>>> b.show()
where we have commented out the extra show() commands. We can also plot multiple points, vectors, and states at the
same time by passing list or arrays instead of individual elements. Before giving an example, we can use the clear()
command to remove the current data from our Bloch sphere instead of creating a new instance:
>>> b.clear()
>>> b.show()
Figure 4.10: The data in a Bloch sphere object is removed by the clear() method. Note that other properties, such
as the sphere color and axes labels are not affected by this command.
Now on the same Bloch sphere, we can plot the three states associated with the x, y, and z directions:
>>> x=(basis(2,0)+(1+0j)*basis(2,1)).unit()
>>> y=(basis(2,0)+(0+1j)*basis(2,1)).unit()
>>> z=(basis(2,0)+(0+0j)*basis(2,1)).unit()
>>> b.add_states([x,y,z])
>>> b.show()
Adding multiple points to the Bloch sphere works slightly differently than adding multiple states or vectors. For
example, lets add a set of 20 points around the equator (after calling clear()):
>>> xp=[cos(th) for th in linspace(0,2*pi,20)]
>>> yp=[sin(th) for th in linspace(0,2*pi,20)]
>>> zp=zeros(20)
>>> pnts=[xp,yp,zp]
>>> b.add_points(pnts)
>>> b.show()
Notice that, in contrast to states or vectors, each point remains the same color as the initial point. This is because
adding multiple data points using the add_points function is interpreted, by default, to correspond to a single data
point (single qubit state) plotted at different times. This is very useful when visualizing the dynamics of a qubit. An
example of this is given in the example . If we want to plot additional qubit states we can call additional add_points
functions:
>>> xz=zeros(20)
>>> yz=[sin(th) for th in linspace(0,pi,20)]
>>> zz=[cos(th) for th in linspace(0,pi,20)]
>>> b.add_points([xz,yz,zz])
>>> b.show()
b=Bloch()
xp=[cos(th) for th in linspace(0,2*pi,20)]
yp=[sin(th) for th in linspace(0,2*pi,20)]
zp=zeros(20)
xz=zeros(20)
yz=[sin(th) for th in linspace(0,pi,20)]
zz=[cos(th) for th in linspace(0,pi,20)]
b.add_points([xp,yp,zp])
b.add_points([xz,yz,zz])
b.show()
The color and shape of the data points is varied automatically by the Bloch class. Notice how the color and point
markers change for each set of data. Again, we have had to call add_points twice because adding more than one
set of multiple data points is not supported by the add_points function.
What if we want to vary the color of our points. We can tell the qutip.Bloch class to vary the color of each point
according to the colors listed in the b.point_color list (see Configuring the Bloch sphere below). Again after
clear():
>>> xp=[cos(th) for th in linspace(0,2*pi,20)]
>>> yp=[sin(th) for th in linspace(0,2*pi,20)]
>>> zp=zeros(20)
>>> pnts=[xp,yp,zp]
>>> b.add_points(pnts,m) # <-- add a m string to signify multi colored points
>>> b.show()
Now, the data points cycle through a variety of predefined colors. Now lets add another set of points, but this time we
want the set to be a single color, representing say a qubit going from the |upi state to the |downi state in the y-z plane:
>>> xz=zeros(20)
>>> yz=[sin(th) for th in linspace(0,pi,20)]
>>> zz=[cos(th) for th in linspace(0,pi,20)]
>>> b.add_points([xz,yz,zz]) # no m
>>> b.show()
A more slick way of using this multi color feature is also given in the example, where we set the color of the markers
as a function of time.
At the end of the last section we saw that the colors and marker shapes of the data plotted on the Bloch sphere are
automatically varied according to the number of points and vectors added. But what if you want a different choice of
color, or you want your sphere to be purple with different axes labels? Well then you are in luck as the Bloch class has
20 attributes which one can control. Assuming b=Bloch():
The Bloch class was designed from the outset to generate animations. To animate a set of vectors or data points
the basic idea is: plot the data at time t1, save the sphere, clear the sphere, plot data at t2,... The Bloch sphere will
automatically number the output file based on how many times the object has been saved (this is stored in b.savenum).
The easiest way to animate data on the Bloch sphere is to use the save() method and generate a series of images to
convert into an animation. However, as of Matplotlib version 1.1, creating animations is built-in. We will demonstrate
both methods by looking at the decay of a qubit on the bloch sphere.
The code for calculating the expectation values for the Pauli spin operators of a qubit decay is given below. This code
is common to both animation examples.
from qutip import *
An example of generating images for generating an animation outside of Python is given below:
b=Bloch()
b.vector_color = [r]
b.view=[-40,30]
for i in xrange(len(sx)):
b.clear()
b.add_vectors([sin(theta),0,cos(theta)])
b.add_points([sx[:i+1],sy[:i+1],sz[:i+1]])
b.save(dirc=temp) #saving images to temp directory in current working directory
Important: Generating animations directly from Matplotlib requires installing either mencoder or ffmpeg. While
either choice works on linux, it is best to choose ffmpeg when running on the Mac. If using macports just do: sudo
port install ffmpeg.
To the code to directly generate an mp4 movie of the Qubit decay is as follows:
from pylab import *
import matplotlib.animation as animation
from mpl_toolkits.mplot3d import Axes3D
fig = figure()
ax = Axes3D(fig,azim=-40,elev=30)
sphere=Bloch(axes=ax)
def animate(i):
sphere.clear()
sphere.add_vectors([sin(theta),0,cos(theta)])
sphere.add_points([sx[:i+1],sy[:i+1],sz[:i+1]])
sphere.make_sphere()
return ax
def init():
sphere.vector_color = [r]
return ax
With the QuTiP time-evolution functions (for example qutip.odesolve.mesolve and qutip.mcsolve), a
state vector or density matrix can be evolved from an initial state at t0 to an arbitrary time t, (t) = V (t, t0 ) {(t0 )},
where V (t, t0 ) is the propagator defined by the equation of motion. The resulting density matrix can then be used to
evaluate the expectation values of arbitrary combinations of same-time operators.
To calculate two-time correlation functions on the form hA(t + )B(t)i, we can use the quantum regression theorem
[see, e.g., Gardineer and Zoller, Quantum Noise, Springer, 2004] to write
We therefore first calculate (t) = V (t, 0) {(0)} using one of the QuTiP evolution solvers with (0) as initial state,
and then again use the same solver to calculate V (t + , t) {B(t)} using B(t) as initial state.
Note that if the intial state is the steady state, then (t) = V (t, 0) {ss } = ss and
The following code demonstrates how to calculate the hx(t)x(0)i correlation for a leaky cavity with three different
relaxation rates.
from qutip import *
tlist = linspace(0,10.0,200)
a = destroy(10)
x = a.dag() + a
H = a.dag()*a
More generally, we can also calculate correlation functions of the kind hA(t1 + t2 )B(t1 )i, i.e., the correlation func-
tion of a system that is not in its steadystate. In QuTiP, we can evoluate such correlation functions using the function
qutip.correlation.correlation. The default behavior of this function is to return a matrix with the corre-
lations as a function of the two time coordinates (t1 and t2 ).
from qutip import *
tlist = linspace(0,10.0,200)
a = destroy(10)
x = a.dag() + a
H = a.dag()*a
alpha = 2.5
corr = correlation(H, coherent_dm(10, alpha), tlist, tlist, [sqrt(0.25)*a], x, x)
However, in some cases we might be interested in the correlation functions on the form hA(t1 + t2 )B(t1 )i, but only
as a function of time coordinate t2 . In this case we can also use the qutip.correlation.correlation
function, if we pass the density matrix at time t1 as second argument, and None as third argument. The
qutip.correlation.correlation function then returns a vector with the correlation values corresponding to
the times in taulist (the fourth argument).
This example demonstrates how to calculate a correlation function on the form hA( )B(0)i for a non-steady initial
state. Consider an oscillator that is interacting with a thermal environment. If the oscillator initially is in a coherent
state, it will gradually decay to a thermal (incoherent) state. The amount of coherence can be quantified using the
ha ( )a(0)i
first-order optical coherence function g (1) ( ) =
. For a coherent state |g (1) ( )| = 1, and for a
ha ( )a( )iha (0)a(0)i
(1)
completely incoherent (thermal) state g ( ) = 0. The following code calculates and plots g (1) ( ) as a function of .
from qutip import *
N = 15
taulist = linspace(0,10.0,200)
a = destroy(N)
H = 2*pi*a.dag()*a
# collapse operator
G1 = 0.75
n_th = 2.00 # bath temperature in terms of excitation number
c_ops = [sqrt(G1*(1+n_th)) * a, sqrt(G1*n_th) * a.dag()]
Often one is interested in the output of a given function as a single-parameter is varied. For instance, in the Driven
steady-state cavity example, we calculate the steady-state response as the driving frequency is varied. In cases such as
this, where each iteration is independent of the others, we may speedup the calculations by performing the iterations in
parallel. In QuTiP, parallel computations may be performed using the qutip.parfor (parallel-for-loop) function.
To use the parfor function we need to define a function of a single-variable, and the range over which this variable is
to be iterated. For example:
>>> def func1(x):
... return x,x**2,x**3
>>> [a,b,c]=parfor(func1,range(10))
>>> print a
[0 1 2 3 4 5 6 7 8 9]
>>> print b
[ 0 1 4 9 16 25 36 49 64 81]
>>> print c
[ 0 1 8 27 64 125 216 343 512 729]
[ 0 1 4 9 16 25 36 49 64 81]
>>> print x[2]
[ 0 1 8 27 64 125 216 343 512 729]
The qutip.parfor function is not limited to just numbers, but also works for a variety of outputs:
>>> def func2(x):
... return x,Qobj(x),a*x
Although qutip.parfor allows functions with only one input, we can in fact pass more an a single variable by
using a list of lists. Sounds confusing, but it is quite easy.
>>> def func1(args):
... index,x=args #<--sets the index variable to args[0], and x to args[1]
... print index #<-- print which element in sequence is being calculated
... return x,x**2,x**3
>>> args=[[k,2*k] for k in range(10)] #<-- create list of lists with more than one variable
>>> print args
[[0, 0], [1, 2], [2, 4], [3, 6], [4, 8], [5, 10], [6, 12], [7, 14], [8, 16], [9, 18]]
>>> [a,b,c]=parfor(func1,args)
0
1
2
3
4
5
6
8
9
7
>>> print a
[ 0 2 4 6 8 10 12 14 16 18]
This example also highlights the fact that the parfor function does not evaluate the sequence of elements in order.
Therefore, passing an index variable, as done in the previous example, is useful if one needs to keep track of individ-
ual function evaluations, for example when plotting. Parfor is also useful for repeated tasks such as generating plots
corresponding to the dynamical evolution of your system, or simultaneously simulating different parameter configu-
rations.
With time-consuming calculations it is often necessary to store the results to files on disk, so it can be post-processed
and archived. In QuTiP there are two facilities for storing data: Quantum objects can be stored to files and later read
back as python pickles, and numerical data (vectors and matrices) can be exported as plain text files in for example CSV
(comma-separated values), TSV (tab-separated values), etc. The former method is preferred when further calculations
will be performed with the data, and the latter when the calculations are completed and data is to be imported into a
post-processing tool (e.g. for generating figures).
To store and load arbitrary QuTiP related objects (Qobj, Odedata, etc.) there are two functions: qutip.qsave
and qutip.qload. The function qutip.qsave takes an arbitrary object as first parameter and an optional file-
name as second parameter (default filename is qutip_data.qu). The filename extension is always .qu. The function
qutip.qload takes a mandatory filename as first argument and loads and returns the objects in the file.
To illustrate how these functions can be used, consider a simple calculation of the steadystate of the harmonic oscilla-
tor:
>>> a = destroy(10); H = a.dag() * a ; c_ops = [sqrt(0.5) * a, sqrt(0.25) * a.dag()]
>>> rho_ss = steadystate(H, c_ops)
The steadystate density matrix rho_ss is an instance of qutip.Qobj. It can be stored to a file steadystate.qu using
>>> qsave(rho_ss, steadystate)
>>> ls *.qu
steadystate.qu
The nice thing about the qutip.qsave and qutip.qload functions is that almost any object can be stored and
load again later on. We can for example store a list of density matrices as returned by qutip.mesolve:
>>> a = destroy(10); H = a.dag() * a ; c_ops = [sqrt(0.5) * a, sqrt(0.25) * a.dag()]
>>> psi0 = rand_ket(10)
>>> tlist = linspace(0, 10, 10)
>>> dm_list = mesolve(H, psi0, tlist, c_ops, [])
>>> qsave(dm_list, density_matrix_vs_time)
And it can then be loaded and used again, for example in an other program:
>>> dm_list_loaded = qload(density_matrix_vs_time)
Loaded list object.
>>> # use it in some calculation
>>> a = destroy(10)
>>> array([expect(a.dag() * a, dm) for dm in dm_list_loaded])
The qutip.qsave and qutip.qload are great, but the file format used is only understood by QuTiP (python)
programs. When data must be exported to other programs the prefered method is to store the data in the commonly used
plain-text file formats. With the QuTiP functions qutip.file_data_store and qutip.file_data_read
we can store and load numpy arrays and matrices to files on disk using a deliminator-separated value format (for
example comma-separated values CSV). Almost any program can handle this file format.
The qutip.file_data_store takes two mandatory and three optional arguments:
>>> file_data_store(filename, data, numtype="complex", numformat="decimal", sep=",")
where filename is the name of the file, data is the data to be written to the file (must be a numpy array), numtype
(optional) is a flag indicating numerical type that can take values complex or real, numformat (optional) specifies the
numerical format that can take the values exp for the format 1.0e1 and decimal for the format 10.0, and sep (optional)
is an arbitrary single-character field separator (usually a tab, space, comma, semicolon, etc.).
A common use for the qutip.file_data_store function is to store the expectation values of a set of operators
for a sequence of times, e.g., as returned by the qutip.mesolve function, which is what the following example
does:
>>> a = destroy(10); H = a.dag() * a ; c_ops = [sqrt(0.5) * a, sqrt(0.25) * a.dag()]
>>> psi0 = rand_ket(10)
>>> tlist = linspace(0, 100, 100)
>>> expt_values = mesolve(H, psi0, tlist, c_ops, [a.dag() * a, a+a.dag(), -1j(a-a.dag())])
>>> shape(expt_values)
(3, 100)
>>> shape(tlist)
(100,1)
>>> output_data = vstack((tlist, expt_values)) # join time and expt data
>>> file_data_store(expect.dat, output_data.T) # Note the .T for transpose !
>>> ls *.dat
expect.dat
>>> !head expect.dat
# Generated by QuTiP: 100x4 complex matrix in decimal format [, separated values].
0.0000000000+0.0000000000j,4.2424841416+0.0000000000j,2.3200046262+0.0000000000j,0.1937084248+0.00000
0.1010101010+0.0000000000j,4.1379975175+0.0000000000j,2.2803919588+0.0000000000j,-0.0289188081+0.0000
0.2020202020+0.0000000000j,4.0424499915+0.0000000000j,2.2243729051+0.0000000000j,-0.2486713739+0.0000
0.3030303030+0.0000000000j,3.9527450545+0.0000000000j,2.1495725421+0.0000000000j,-0.4605913338+0.0000
0.4040404040+0.0000000000j,3.8674018874+0.0000000000j,2.0562127752+0.0000000000j,-0.6616078826+0.0000
0.5050505051+0.0000000000j,3.7856377679+0.0000000000j,1.9454779063+0.0000000000j,-0.8495893849+0.0000
0.6060606061+0.0000000000j,3.7069902520+0.0000000000j,1.8189312038+0.0000000000j,-1.0229166838+0.0000
0.7070707071+0.0000000000j,3.6311546972+0.0000000000j,1.6783060144+0.0000000000j,-1.1802965424+0.0000
0.8080808081+0.0000000000j,3.5579106202+0.0000000000j,1.5254272690+0.0000000000j,-1.3206823679+0.0000
In this case we didnt really need to store both the real and imaginary parts, so instead we could use the num-
type=real option:
>>> file_data_store(expect.dat, output_data.T, numtype="real")
>>> !head -n5 expect.dat
# Generated by QuTiP: 100x4 real matrix in decimal format [, separated values].
0.0000000000,4.2424841416,2.3200046262,0.1937084248
0.1010101010,4.1379975175,2.2803919588,-0.0289188081
0.2020202020,4.0424499915,2.2243729051,-0.2486713739
0.3030303030,3.9527450545,2.1495725421,-0.4605913338
and if we prefer scientific notation we can request that using the numformat=exp option
>>> file_data_store(expect.dat, output_data.T, numtype="real", numformat="exp")
>>> !head -n 5 expect.dat
# Generated by QuTiP: 100x4 real matrix in exp format [, separated values].
0.0000000000e+00,4.2424841416e+00,2.3200046262e+00,1.9370842484e-01
1.0101010101e-01,4.1379975175e+00,2.2803919588e+00,-2.8918808147e-02
2.0202020202e-01,4.0424499915e+00,2.2243729051e+00,-2.4867137392e-01
3.0303030303e-01,3.9527450545e+00,2.1495725421e+00,-4.6059133382e-01
Loading data previously stored using qutip.file_data_store (or some other software) is a even easier. Re-
gardless of which deliminator was used, if data was stored as complex or real numbers, if it is in decimal or exponential
form, the data can be loaded using the qutip.file_data_read, which only takes the filename as mandatory ar-
gument.
>>> input_data = file_data_read(expect.dat)
>>> shape(input_data)
(100, 4)
>>> # do something with the data, e.g.
>>> plot(input_data[:,0],input_data[:,1]); show()
(If a particularly obscure choice of deliminator was used it might be necessary to use the optional second argument,
for example sep=_ if _ is the deliminator).
QuTiP includes a collection of random state generators for simulations, theorem evaluation, and code testing:
Function Description
rand_ket Random ket-vector
rand_dm Random density ma-
trix
rand_herm Random Hermitian
matrix
rand_unitary Random Unitary ma-
trix
See the API documentation: Random Operators and States for details.
In all cases, these functions can be called with a single parameter N that indicates a N xN matrix (rand_dm,
rand_herm, rand_unitary), or a N x1 vector (rand_ket), should be generated. For example:
>>> rand_ket(5)
Quantum object: dims = [[5], [1]], shape = [5, 1], type = ket
Qobj data =
[[-0.69257444+0.07716822j]
[-0.48903635+0.00547509j]
[-0.04434023-0.04301746j]
[-0.27839376+0.00130696j]
[-0.35384822+0.26204821j]]
or:
>>> rand_herm(5)
Quantum object: dims = [[5], [5]], shape = [5, 5], type = oper, isHerm = True
Qobj data =
[[ 0.00000000+0.j -0.06185503-0.20132642j -0.14510832-0.02572751j
-0.59958603-0.13581992j -0.11296987-0.19364997j]
[-0.06185503+0.20132642j 0.00000000+0.j -0.67138090+0.38714512j
-0.14847184+0.13814627j -0.34759237-0.09025718j]
[-0.14510832+0.02572751j -0.67138090-0.38714512j 0.00000000+0.j
-0.45128021+0.16790562j 0.00000000+0.j ]
[-0.59958603+0.13581992j -0.14847184-0.13814627j -0.45128021-0.16790562j
-0.33788000+0.j 0.00000000+0.j ]
[-0.11296987+0.19364997j -0.34759237+0.09025718j 0.00000000+0.j
0.00000000+0.j 0.00000000+0.j ]]
In this previous example, we see that the generated Hermitian operator contains a faction of elements that are identi-
cally equal to zero. The number of nonzero elements is called the density and can be controlled by calling any of the
random state/operator generators with a second argument between 0 and 1. By default, the density for the operators is
0.75 where as ket vectors are completely dense (1). For example:
>>> rand_dm(5,0.5)
Quantum object: dims = [[5], [5]], shape = [5, 5], type = oper, isHerm = True
Qobj data =
[[ 0.21266414+0.j 0.02943826+0.04209834j 0.16825067-0.08486911j
0.05243364-0.08772405j 0.00000000+0.j ]
[ 0.02943826-0.04209834j 0.78733586+0.j 0.02000972+0.05611193j
0.19337602+0.08780273j 0.00000000+0.j ]
[ 0.16825067+0.08486911j 0.02000972-0.05611193j 0.00000000+0.j
0.00000000+0.j 0.00000000+0.j ]
[ 0.05243364+0.08772405j 0.19337602-0.08780273j 0.00000000+0.j
0.00000000+0.j 0.00000000+0.j ]
[ 0.00000000+0.j 0.00000000+0.j 0.00000000+0.j
0.00000000+0.j 0.00000000+0.j ]]
Important: In the case of a density matrix, setting the density too low will result in not enough diagonal elements to
satisfy T r() = 1.
In many cases, one is interested in generating random quantum objects that correspond to composite systems generated
using the qutip.tensor.tensor function. Specifying the tensor structure of a quantum object is done using the
dims keyword argument in the same fashion as one would do for a qutip.Qobj object:
>>> rand_unitary(4,dims=[[2,2],[2,2]])
Quantum object: dims = [[2, 2], [2, 2]], shape = [4, 4], type = oper, isHerm = False
Qobj data =
[[ 0.63224397+0.03961913j -0.48934943+0.5741797j -0.11171375+0.12333134j
-0.01673677-0.03977414j]
[-0.02160731+0.75410763j 0.51216688+0.40252671j -0.06994174-0.0380775j
0.00990143-0.00850615j]
[ 0.02132803+0.16503241j -0.07581063-0.02438219j 0.77718792+0.3962325j
-0.35597928+0.27968763j]
[-0.04216640+0.00917044j 0.00197097-0.0129038j -0.16907045+0.41995429j
0.61014364+0.64864923j]]
Note: This section deals with modifying the internal QuTiP settings. Use only if you know what you are doing.
In this section we show how to modify a few of the internal parameters used by QuTiP. The settings that can be
modified are given in the following table:
Setting Description Options
qutip_graphics Use matplotlib True / False
qutip_gui Pick GUI library, or disable. PYSIDE,PYQT4,NONE
auto_tidyup Automatically tidyup quantum ob- True / False
jects.
auto_tidyup_atol Tolerance used by tidyup any float value > 0
num_cpus Number of CPUs used for multi- int between 1 and # cpus
processing.
The two most important settings are auto_tidyup and auto_tidyup_atol as they control whether the small elements of a
quantum object should be removed, and what number should be considered as the cutoff tolerance. Modifying these,
or any other parameters, is quite simple:
>>> qutip.settings.auto_tidyup=False
>>> qutip.settings.qutip_gui="NONE"
These settings will be used for the current QuTiP session only and will need to be modified again when restarting
QuTiP. If running QuTiP from a script file, then place the qutip.setings.xxxx commands immediately after from qutip
import * at the top of the script file. If you want to reset the parameters back to their default values then call the reset
command:
>>> qutip.settings.reset()
FIVE
QuTip includes 20 built in demos from the examples below that demonstrate the usage of the built in functions for
solving a variety of problems. To run the demos, load the QuTiP package:
>>> from qutip import *
This will generate the examples GUI, or a command line list of demos, depending on the availability of the graphics
libraries:
If you do not have any graphics libraries installed, or they are disabled, then the demos function must be run from the
terminal.
85
QuTiP: The Quantum Toolbox in Python, Release 2.0
Calculates the Wigner distribution for a Schrodinger Cat-state composed of two coherent states 1 = 2 2j, and
2 = 2 + 2j.
#
#Calculates the Schrodinger cat state for
#a superposition of two coherent states.
#
from qutip import *
from pylab import * #loads matplotlib
def run():
#Number of basis states
N = 20
if __name__ == "__main__":
run()
Calculates the Q distribution for a Schrodinger Cat-state composed of two coherent states 1 = 2 2j, and 2 =
2 + 2j.
#
#Calculates the Q-function of Schrodinger cat state
#formed from a superposition of two coherent states.
#
from qutip import *
from pylab import * #loads matplotlib
def run():
#Number of basis states
N = 20
ylim([-6,6])
title(Q - function of Schrodinger cat)
if __name__ == "__main__":
run()
Squeezed State
3D Wigner and Q-functions for a squeezed coherent state where = 1.0 is the coherent state amplitude and = 0.5j
is the squeezing parameter.
#
# 3D Wigner and Q-functions for
# a squeezed coherent state.
#
from qutip import *
from mpl_toolkits.mplot3d import Axes3D
from matplotlib import cm
from pylab import *
def run():
#setup constants:
N = 20
alpha = -1.0 # Coherent amplitude of field
epsilon = 0.5j# Squeezing parameter
a = destroy(N)
D = displace(N,alpha) # Displacement
S = squeez(N,epsilon) # Squeezing
psi = D*S*basis(N,0); # Apply to vacuum state
xvec = linspace(-6,6,150)
X,Y = meshgrid(xvec, xvec)
W=wigner(psi,xvec,xvec)
Q = qfunc(psi,xvec,xvec)
fig =figure()
ax = Axes3D(fig,azim=-62,elev=25)
ax.plot_surface(X, Y, W, rstride=2, cstride=2, cmap=cm.jet,lw=.1)
ax.set_xlim3d(-6,6)
ax.set_xlim3d(-6,6)
ax.set_zlim3d(0,0.4)
title(Wigner function of squeezed coherent state)
show()
fig =figure()
ax2 = Axes3D(fig,azim=-43,elev=37)
ax2.plot_surface(X, Y, Q, rstride=2, cstride=2, cmap=cm.jet,lw=.1)
ax2.set_xlim3d(-6,6)
ax2.set_xlim3d(-6,6)
ax2.set_zlim3d(0,0.2)
title(Q-function of squeezed coherent state)
show()
if __name__ == "__main__":
run()
Groundstate properties of an ultra-strongly coupled atom-cavity system as the coupling strength is varied.
#
# Textbook example: Groundstate properties of an
# ultra-strongly coupled atom-cavity system.
#
#
from qutip import *
from pylab import *
from mpl_toolkits.mplot3d import Axes3D
idx = 0
na_expt = zeros(len(glist))
nc_expt = zeros(len(glist))
grnd_kets= zeros(len(glist),dtype=object)
for g in glist:
def run():
#
# set up the calculation
#
wc = 1.0 * 2 * pi # cavity frequency
wa = 1.0 * 2 * pi # atom frequency
N = 25 # number of cavity fock states
use_rwa = False # Set to True to see that non-RWA is necessary
#run computation
nc, na, grnd_kets = compute(N, wc, wa, glist, use_rwa)
#
# plot the cavity and atom occupation numbers as a function of
# coupling strength
figure(1)
plot(glist/(2*pi), nc,lw=2)
plot(glist/(2*pi), na,lw=2)
legend(("Cavity", "Atom excited state"),loc=0)
xlabel(Coupling strength (g))
ylabel(Occupation Number)
title(# of Photons in the Groundstate)
show()
if __name__ == "__main__":
run()
Shows the relationship 1 F 2 T between fidelity F and trace distance T for pure state density matrices .
#
# Prove that 1-F**2 <= T for pure state density matricies
# where F and T are the fidelity and trace distance metrics,
# respectively using randomly generated ket vectors.
#
from qutip import *
from pylab import *
def run():
N=21#number of kets to generate
#create arrays of pure density matrices from random kets using ket2dm
x=array([ket2dm(rand_ket(10)) for k in range(N)])
if __name__ == "__main__":
run()
#qubit 1 operators
sz1 = tensor(sigmaz(), qeye(2), qeye(2))
sx1 = tensor(sigmax(), qeye(2), qeye(2))
#qubit 2 operators
sz2 = tensor(qeye(2), sigmaz(), qeye(2))
sx2 = tensor(qeye(2), sigmax(), qeye(2))
#qubit 3 operators
sz3 = tensor(qeye(2), qeye(2), sigmaz())
sx3 = tensor(qeye(2), qeye(2), sigmax())
idx = 0
#preallocate output array
evals_mat = zeros((len(w1list),2*2*2))
for w1 in w1list:
idx += 1
return evals_mat
def run():
#
# set up the calculation
#
w1 = 1.0 * 2 * pi # atom 1 frequency: sweep this one
w2 = 0.9 * 2 * pi # atom 2 frequency
w3 = 1.1 * 2 * pi # atom 3 frequency
g12 = 0.05 * 2 * pi # atom1-atom2 coupling strength
g13 = 0.05 * 2 * pi # atom1-atom3 coupling strength
# run computation
evals_mat = compute(w1list, w2, w3, g12, g13)
#
# plot the energy eigenvalues
#
figure(1)
colors=[b,r,g] #list of colors for plotting
for n in [1,2,3]:
plot(w1list / (2*pi), (evals_mat[:,n]-evals_mat[:,0]) / (2*pi), colors[n-1],lw=2)
show()
if __name__=="__main__":
run()
Bell State
Creation and manipulation of a Bell state density matrix. The Bell state is given as |i = 1 (|00i + |11i )
2
#
# Creation and manipulation of a Bell state with
# 3D histogram plot output.
#
def qubit_hist(Q,xlabels,ylabels,title):
# Plots density matrix 3D histogram from Qobj
# xlabels and ylabels are list of strings for axes tick labels
num_elem=prod(Q.shape) #num. of elements to plot
xpos,ypos=meshgrid(range(Q.shape[0]),range(Q.shape[1]))
xpos=xpos.T.flatten()-0.5 #center bars on integer value of x-axis
ypos=ypos.T.flatten()-0.5 #center bars on integer value of y-axis
zpos = zeros(num_elem) #all bars start at z=0
dx =0.8*ones(num_elem) #width of bars in x-direction
dy = dx.copy() #width of bars in y-direction (same as x-dir here)
dz = real(Q.full().flatten()) #height of bars from density matrix elements.
#plot figure
fig = plt.figure()
ax = Axes3D(fig,azim=-47,elev=85)
#remove z-axis tick labels by moving them outside the plot range
ax.axes.w_zaxis.set_major_locator(IndexLocator(2,2)) #set z-ticks to integers
#set the plot range in the z-direction to fit data
ax.set_zlim3d([min(dz)-0.1,max(dz)+0.1])
plt.title(title)
#add colorbar with color range normalized to data
cax,kw=mpl.colorbar.make_axes(ax,shrink=.75,pad=.02)
cb1=mpl.colorbar.ColorbarBase(cax,cmap=cm.jet,norm=nrm)
cb1.set_label("Probability",fontsize=14)
show()
def run():
#create Bell state
up = basis(2,0)
dn = basis(2,1)
bell = (tensor([up,up])+tensor([dn,dn])).unit()
rho_bell=ket2dm(bell)
x_bell_labels=[$\left|00\\rangle\\right.$,$\left|01\\rangle\\right.$,
$\left|10\\rangle\\right.$,$\left|11\\rangle\\right.$]
y_bell_labels=x_bell_labels
title=Bell state density matrix
#plot Bell state density matrix
qubit_hist(rho_bell,x_bell_labels,y_bell_labels,title)
after_trace=ptrace(after,1)
title="Completely mixed state after partial trace over qubit 0"
qubit_hist(after_trace,xlabels,ylabels,title)
if __name__==__main__:
run()
Cavity-Qubit Steadystate
Find the steady state of a cavity-qubit system as a function of the cavity driving frequency.
#
# Steady-state density matrix of a two-level atom in a high-Q
# cavity for various driving frequencies calculated using
# iterative steady solver.
#
# Adapted from probss example in the qotoolbox by Sze M. Tan.
#
from qutip import *
from pylab import *
def probss(E,kappa,gamma,g,wc,w0,wl,N):
#construct composite operators
ida=qeye(N)
idatom=qeye(2)
a=tensor(destroy(N),idatom)
sm=tensor(ida,sigmam())
#Hamiltonian
H=(w0-wl)*sm.dag()*sm+(wc-wl)*a.dag()*a+1j*g*(a.dag()*sm-sm.dag()*a)+E*(a.dag()+a)
#Collapse operators
C1=sqrt(2*kappa)*a
C2=sqrt(gamma)*sm
C1dC1=C1.dag() * C1
C2dC2=C2.dag() * C2
def run():
show()
if __name__=="__main__":
run()
Binary Entropy
Entropy of a binary system a|0ih0| + (1 a)|1ih1| as probability of being in the excited state a is varied.
#
# Entropy of binary system as probability
# of being in the excited state is varied.
#
from qutip import *
from pylab import *
def run():
a=linspace(0,1,20)
out=zeros(len(a)) #preallocate output array
for k in range(len(a)):
# a*|0><0|
x=a[k]*ket2dm(basis(2,0))
# (1-a)*|1><1|
y=(1-a[k])*ket2dm(basis(2,1))
rho=x+y
# Von-Neumann entropy (base 2) of rho
out[k]=entropy_vn(rho,2)
fig=figure()
plot(a,out,lw=2)
xlabel(rProbability of being in excited state $(a)$)
ylabel(rEntropy)
title("Entropy of $a|0\\rangle\langle0|+(1-a)|1\\rangle\langle1|$")
show()
if __name__==__main__:
run()
Plot the density matrix for the 3-qubit GHZ state 1 (| , , i + | , , i) found by simultaneous diagonalization.
2
#
# Plots the entangled superposition
# 3-qubit GHZ eigenstate |up,up,up>+|dn,dn,dn>
#
# From the xGHZ qotoolbox example by Sze M. Tan
#
from qutip import *
from pylab import *
from matplotlib import pyplot, mpl,cm
from mpl_toolkits.mplot3d import Axes3D
import matplotlib.pyplot as plt
def run():
#create spin operators for the three qubits.
sx1=tensor(sigmax(),qeye(2),qeye(2))
sy1=tensor(sigmay(),qeye(2),qeye(2))
sx2=tensor(qeye(2),sigmax(),qeye(2))
sy2=tensor(qeye(2),sigmay(),qeye(2))
sx3=tensor(qeye(2),qeye(2),sigmax())
sy3=tensor(qeye(2),qeye(2),sigmay())
#Calculate products
op1=sx1*sy2*sy3
op2=sy1*sx2*sy3
op3=sy1*sy2*sx3
opghz=sx1*sx2*sx3
#plot figure
fig = plt.figure()
ax = Axes3D(fig,azim=-15,elev=75)
ax.bar3d(xpos, ypos, zpos, dx, dy, dz, color=colors)
#remove z-axis tick labels by moving them outside the plot range
ax.axes.w_zaxis.set_major_locator(IndexLocator(2,2)) #set z-ticks to integers
#set the plot range in the z-direction to fit data
ax.set_zlim3d([min(dz)-0.1,max(dz)+0.1])
plt.title(title)
#add colorbar with color range normalized to data
cax,kw=mpl.colorbar.make_axes(ax,shrink=.75,pad=.02)
cb1=mpl.colorbar.ColorbarBase(cax,cmap=cm.jet,norm=nrm)
show()
if __name__==__main__:
run()
Dissipative i-Swap Gate vs. ideal gate. The accuracy of dissipative gate given by the fidelity.
This example demonstrates how to create a composite system of two-qubits, define dissipation processor for each
qubit, and solve for the dynamics of the system using the standard Lindblad master equation. It also shows how to
obtain expectation values of select operators from the dynamics solver.
#
# Dissipative $i$-SWAP gate vs ideal gate.
# Accuracy of gate given by Fidelity of
# final state and ideal final state.
#
from qutip import *
from pylab import *
def run():
# setup system parameters
g = 1.0 * 2 * pi # coupling strength
g1 = 0.75 # relaxation rate
g2 = 0.05 # dephasing rate
n_th = 0.75 # bath temperature
T = pi/(4*g)
# construct Hamiltonian
H = g * (tensor(sigmax(), sigmax()) +
tensor(sigmay(), sigmay()))
# construct inital state
psi0 = tensor(basis(2,1), basis(2,0))
if __name__==__main__:
run()
Illustrates the vacuum Rabi oscillations in the Jaynes-Cummings model with dissipation.
This examples introduces the use of bosonic operators (destroy, for the annihilation operator) and how to setup the
Jaynes-Cummings model.
#
# Vacuum Rabi oscillations in the Jaynes-Cummings model with dissipation
#
from qutip import *
from pylab import *
def run():
# Configure parameters
wc = 1.0 * 2 * pi # cavity frequency
wa = 1.0 * 2 * pi # atom frequency
g = 0.05 * 2 * pi # coupling strength
kappa = 0.005 # cavity dissipation rate
gamma = 0.05 # atom dissipation rate
N = 5 # number of cavity fock states
use_rwa = True
# intial state
psi0 = tensor(basis(N,0), basis(2,1)) # start with an excited atom
# Hamiltonian
a = tensor(destroy(N), qeye(2))
sm = tensor(qeye(N), destroy(2))
if use_rwa:
# use the rotating wave approxiation
H = wc * a.dag() * a + wa * sm.dag() * sm + g * (a.dag() * sm + a * sm.dag())
else:
H = wc * a.dag() * a + wa * sm.dag() * sm + g * (a.dag() + a) * (sm + sm.dag())
# collapse operators
c_op_list = []
rate = gamma
if rate > 0.0:
c_op_list.append(sqrt(rate) * sm)
if __name__==__main__:
run()
This example implements a simple model for single-atom lasing in a Jaynes-Cumming-like system. In addition to the
standard Jaynes-Cumming model, there is an incoherent pumping that strive to create a population inversion in the
atom.
This examples demonstrates how reversed relaxation processes (note the atomic collapse operators) can be used to
introduce incoherent pump processes a system.
#
# Single-atom lasing in a Jaynes-Cumming-like system
#
from qutip import *
from pylab import *
from mpl_toolkits.mplot3d import Axes3D
from matplotlib import cm
def run():
# Configure parameters
N = 12 # number of cavity fock states
wc = 2*pi*1.0 # cavity frequency
wa = 2*pi*1.0 # atom frequency
g = 2*pi*0.1 # coupling strength
kappa = 0.05 # cavity dissipation rate
gamma = 0.0 # atom dissipation rate
pump = 0.4 # atom pump rate
use_rwa = True
# Hamiltonian
a = tensor(destroy(N), qeye(2))
sm = tensor(qeye(N), destroy(2))
# collapse operators
c_op_list = []
rate = gamma
if rate > 0.0:
c_op_list.append(sqrt(rate) * sm)
rate = pump
if rate > 0.0:
c_op_list.append(sqrt(rate) * sm.dag())
#
# plot the time-evolution of the cavity and atom occupation
#
figure(1)
plot(tlist, real(nc), r-, tlist, real(na), b-)
xlabel(Time);
ylabel(Occupation probability);
legend(("Cavity occupation", "Atom occupation"))
#
# plot the final photon distribution in the cavity
#
rho_final = output.states[-1]
rho_cavity = ptrace(rho_final, 0)
figure(2)
bar(range(0, N), real(rho_cavity.diag()))
xlabel("Photon number")
ylabel("Occupation probability")
title("Photon distribution in the cavity")
#
# plot the wigner function
#
xvec = linspace(-5, 5, 100)
W = wigner(rho_cavity, xvec, xvec)
X,Y = meshgrid(xvec, xvec)
figure(3)
contourf(X, Y, W, 100)
colorbar()
show()
if __name__==__main__:
run()
This example demonstrates the generation of Wigner distributions from the master equation evolution of the Jaynes-
Cummings model. In particular, it introduce the use of the wigner to calculate the Wigner distribution from a density
matrix for a bosonic mode, and the use of ptrace to trace out the qubit from the total density matrix (to obtain the
density matrix for the cavity alone).
#
# Dynamics of the Wigner distributions for the Jaynes-Cummings model
#
from qutip import *
from pylab import *
from mpl_toolkits.mplot3d import Axes3D
from matplotlib import cm
def run():
# Configure parameters
N = 10 # number of cavity fock states
wc = 2*pi*1.0 # cavity frequency
wa = 2*pi*1.0 # atom frequency
g = 2*pi*0.1 # coupling strength
kappa = 0.05 # cavity dissipation rate
gamma = 0.15 # atom dissipation rate
use_rwa = True
# Hamiltonian
a = tensor(destroy(N), qeye(2))
sm = tensor(qeye(N), destroy(2))
if use_rwa:
# use the rotating wave approxiation
H = wc * a.dag() * a + wa * sm.dag() * sm + g * (a.dag() * sm + a * sm.dag())
else:
# collapse operators
c_op_list = []
rate = gamma
if rate > 0.0:
c_op_list.append(sqrt(rate) * sm)
#for idx, rho in enumerate(output.states): # suggestion: try to loop over all rho
for idx, rho in enumerate([output.states[44]]): # for a selected time t=4.4
rho_cavity = ptrace(rho, 0)
W = wigner(rho_cavity, xvec, xvec)
show()
if __name__==__main__:
run()
The saved images may be used to generate a animation of the dynamics of the Wigner distribution.
This example demonstrates how to calculate the dynamics of a spin-1/2 Heisenberg chain, i.e., a sequence of two-level
system (spin up or spin down) that are coupled to its nearest neighbors. The Hamiltonian for this system is
N N 1
1X 1 X n n n+1
hn zn Jx x x + Jyn yn yn+1 + Jzn zn zn+1 ,
H= (5.1)
2 n 2 n
#
# Hamiltonian
#
# H = - 0.5 sum_n^N h_n sigma_z(n)
# - 0.5 sum_n^(N-1) [ Jx_n sigma_x(n) sigma_x(n+1) +
# Jy_n sigma_y(n) sigma_y(n+1) +
# Jz_n sigma_z(n) sigma_z(n+1)]
#
si = qeye(2)
sx = sigmax()
sy = sigmay()
sz = sigmaz()
sx_list = []
sy_list = []
sz_list = []
for n in range(N):
op_list = []
for m in range(N):
op_list.append(si)
op_list[n] = sx
sx_list.append(tensor(op_list))
op_list[n] = sy
sy_list.append(tensor(op_list))
op_list[n] = sz
sz_list.append(tensor(op_list))
# interaction terms
for n in range(N-1):
H += - 0.5 * Jx[n] * sx_list[n] * sx_list[n+1]
H += - 0.5 * Jy[n] * sy_list[n] * sy_list[n+1]
H += - 0.5 * Jz[n] * sz_list[n] * sz_list[n+1]
# collapse operators
c_op_list = []
# spin dephasing
for n in range(N):
if gamma[n] > 0.0:
c_op_list.append(sqrt(gamma[n]) * sz_list[n])
return output.expect
def run():
#
# set up the calculation
#
# dephasing rate
gamma = 0.01 * ones(N)
# intial state, first spin in state |1>, the rest in state |0>
psi_list = []
psi_list.append(basis(2,1))
for n in range(N-1):
psi_list.append(basis(2,0))
psi0 = tensor(psi_list)
show()
if __name__=="__main__":
run()
In this example we look at how to calculate the steady state for a master equation. To demonstrate a steady state
calculation we look at an example from nanomechanics: Consider a low-frequency nanomechanical resonator (with
frequency m lower than temperature) coupled to a high-frequency (optical) resonantor (with frequency r higher
than temperature). In the steady state the mechanical resonator is highly excited and the optical resonator is near its
quantum ground state.
By applying a driving field to the high-frequency resonator with a frequency matching the frequency difference be-
tween the resonator, the two oscilators can effectively be brought into resonance in a rotating frame, allowing for
excitation transfer from the low-frequency mechanical resonator to the high-frequency resonator (i.e., sideband cool-
ing of the mechanical resonator).
The Hamiltonian considered for this problem is
which in the rotating frame that eliminates the time-dependence of the driving term becomes
1
H = (r d )a a + m b b + ga a(b + b ) + A(a + a ).
2
In the following code we look at the state state of this system as a function of the ambient temperature T .
#
# Steady state and photon occupation number for a sideband-cooled
# nanomechanical resonator, as a function of the ambient temperature.
#
from qutip import *
from pylab import *
from mpl_toolkits.mplot3d import Axes3D
from matplotlib import cm
# constants
hbar = 6.626e-34/(2*pi)
kB = 1.38e-23
#
# calculate the steadystate average photon count in the two resonators as a
# function of Temperature, with and without sideband cooling driving
#
def compute(T_vec, N_r, N_m, w_r, w_m, g, w_d, A, kappa_r, kappa_m):
# pre-calculate operators
a = tensor(destroy(N_r), qeye(N_m)) # for high-freq. mode
b = tensor(qeye(N_r), destroy(N_m)) # for mechanical mode
# collapse operators
c_ops = []
return photon_count
def run():
#A = w_d = 0 # no cooling
plot(T_vec, photon_count[:,0], b)
plot(T_vec, photon_count[:,1], b:)
plot(T_vec, photon_count[:,2], r)
plot(T_vec, photon_count[:,3], r:)
xlabel(rTemperature [mK],fontsize=14)
ylabel(rOccupation number,fontsize=14)
title("Average photon occupation number\n" +
"in a sideband-cooled mechanical resonator")
if __name__==__main__:
run()
Here we measure the distance of a single mode (mode #1) of a trilinear Hamiltonian from that of a thermal density
matrix characterized by the expectation value of the number of excitations in the mode at time t. Here the pump mode
(mode #0) is assumed to be in a initial coherent state with the given excitation number.
#
# Measuring the distance between density matrices via the fidelity
#
from qutip import *
from pylab import *
def run():
fids=zeros((3,60)) #initialize data matrix
hilbert=[4,5,6] #list of Hilbert space sizes
num_sizes=[1,2,3] #list of <n>s for initial state of pump mode #0
#define operators
a0=tensor(destroy(N0),qeye(N1),qeye(N2))
a1=tensor(qeye(N0),destroy(N1),qeye(N2))
a2=tensor(qeye(N0),qeye(N1),destroy(N2))
#trilinear Hamiltonian
H=1.0j*(a0*a1.dag()*a2.dag()-a0.dag()*a1*a2)
#run odesolver
tlist=linspace(0,3,60)
output=mesolve(H,psi0,tlist,[],[])
if __name__=="__main__":
run()
This example demonstrates how to visualize the dynamics of a two-level system on the Bloch sphere.
#
# Qubit dynamics shown in a Bloch sphere.
#
from qutip import *
from pylab import *
def run():
# initial state
a = .5
psi0 = (a* basis(2,0) + (1-a)*basis(2,1)).unit()
tlist = linspace(0,3,500)
sx, sy, sz = qubit_integrate(w, theta, gamma1, gamma2, psi0, tlist)
sphere=Bloch()
sphere.add_points([sx,sy,sz])
sphere.point_color=[r]
sphere.vector_color = [b]
sphere.size=[4,4]
sphere.font_size=14
sphere.add_vectors([sin(theta),0,cos(theta)])
sphere.show()
if __name__=="__main__":
run()
Monte Carlo evoution of a coherently driven cavity with a two-level atom initially in the ground state and no photons
in the cavity.
Adapted from qotoolbox example probqmc3 by Sze M. Tan.
#load qutip and matplotlib
from qutip import *
from pylab import *
def run():
# set system parameters
kappa=2.0 #mirror coupling
gamma=0.2 #spontaneous emission rate
g=1 #atom/cavity coupling strength
wc=0 #cavity frequency
w0=0 #atom frequency
wl=0 #driving frequency
E=0.5 #driving amplitude
N=4 #number of cavity energy levels (0->3 Fock states)
tlist=linspace(0,10,101) #times for expectation values
# construct Hamiltonian
ida=qeye(N)
idatom=qeye(2)
a=tensor(destroy(N),idatom)
sm=tensor(ida,sigmam())
H=(w0-wl)*sm.dag()*sm+(wc-wl)*a.dag()*a+1j*g*(a.dag()*sm-sm.dag()*a)+E*(a.dag()+a)
#collapse operators
C1=sqrt(2*kappa)*a
C2=sqrt(gamma)*sm
C1dC1=C1.dag()*C1
C2dC2=C2.dag()*C2
#intial state
psi0=tensor(basis(N,0),basis(2,1))
if __name__==__main__:
run()
Occupation number of two coupled osciilators with oscillator A driven by an external classical drive. Both oscillators
are assumed to start in the ground state.
#
# Occupation number of two coupled osciilators with
# oscillator A driven by an external classical drive.
# Both oscillators are assumed to start in the ground
# state.
#
from qutip import *
def run():
wa = 1.0 * 2 * pi # frequency of system a
wb = 1.0 * 2 * pi # frequency of system a
wab = 0.2 * 2 * pi # coupling frequency
ga = 0.2 * 2 * pi # dissipation rate of system a
gb = 0.1 * 2 * pi # dissipation rate of system b
Na = 10 # number of states in system a
Nb = 10 # number of states in system b
E = 1.0 * 2 * pi # Oscillator A driving strength
a = tensor(destroy(Na), qeye(Nb))
b = tensor(qeye(Na), destroy(Nb))
na = a.dag() * a
nb = b.dag() * b
H = wa*na + wb*nb + wab*(a.dag()*b+a*b.dag()) + E*(a.dag()+a)
c_op_list = []
c_op_list.append(sqrt(ga) * a)
c_op_list.append(sqrt(gb) * b)
#run simulation
data = mcsolve(H,psi0,tlist,c_op_list,[na,nb])
#plot results
plot(tlist,data.expect[0],b,tlist,data.expect[1],r,lw=2)
xlabel(Time,fontsize=14)
ylabel(Excitations,fontsize=14)
legend((Oscillator A, Oscillator B))
show()
if __name__==__main__:
run()
This is a Monte-Carlo simulation showing the decay of a cavity Fock state |1i in a thermal environment with an
average occupation number of n = 0.063. Here, the coupling strength is given by the inverse of the cavity ring-down
time Tc = 0.129.
The parameters chosen here correspond to those from S. Gleyzes, et al., Nature 446, 297 (2007).
#load qutip and matplotlib
from qutip import *
from pylab import *
def run():
# define parameters
N=4 # number of basis states to consider
kappa=1.0/0.129 # coupling to heat bath
nth= 0.063 # temperature with <n>=0.063
# collapse operators
c_op_list = []
# decay operator
c_op_list.append(sqrt(kappa * (1 + nth)) * a)
# excitation operator
c_op_list.append(sqrt(kappa * nth) * a.dag())
#
# plot results using vertically stacked plots
#
f = figure(figsize=(6,9))
subplots_adjust(hspace=0.001) #no space between plots
# subplot 1 (top)
ax1 = subplot(411)
ax1.plot(tlist,ex1,b,lw=2)
ax1.axhline(y=fexpt,color=k,lw=1.5)
yticks(linspace(0,2,5))
ylim([-0.1,1.5])
ylabel($\left< N \\right>$,fontsize=14)
title("Ensemble Averaging of Monte Carlo Trajectories")
legend((Single trajectory,steady state),prop=leg_prop)
# subplot 2
ax2=subplot(412,sharex=ax1) #share x-axis of subplot 1
ax2.plot(tlist,ex5,b,lw=2)
ax2.axhline(y=fexpt,color=k,lw=1.5)
yticks(linspace(0,2,5))
ylim([-0.1,1.5])
ylabel($\left< N \\right>$,fontsize=14)
legend((5 trajectories,steadystate),prop=leg_prop)
# subplot 3
ax3=subplot(413,sharex=ax1) #share x-axis of subplot 1
ax3.plot(tlist,ex15,b,lw=2)
ax3.plot(tlist,me.expect[0],r--,lw=1.5)
ax3.axhline(y=fexpt,color=k,lw=1.5)
yticks(linspace(0,2,5))
ylim([-0.1,1.5])
ylabel($\left< N \\right>$,fontsize=14)
legend((15 trajectories,master equation,steady state),prop=leg_prop)
# subplot 4 (bottom)
ax4=subplot(414,sharex=ax1) #share x-axis of subplot 1
ax4.plot(tlist,ex904,b,lw=2)
ax4.plot(tlist,me.expect[0],r--,lw=1.5)
ax4.axhline(y=fexpt,color=k,lw=1.5)
yticks(linspace(0,2,5))
ylim([-0.1,1.5])
ylabel($\left< N \\right>$,fontsize=14)
legend((904 trajectories,master equation,steady state),prop=leg_prop)
ax1.xaxis.set_major_locator(MaxNLocator(4))
xlabel(Time (sec),fontsize=14)
show()
if __name__=="__main__":
run()
Demonstrates the deviation from a thermal distribution for a single oscillator mode of the trilinear Hamiltonian.
def run():
#number of states for each mode
N0=15
N1=15
N2=15
#define operators
a0=tensor(destroy(N0),qeye(N1),qeye(N2))
a1=tensor(qeye(N0),destroy(N1),qeye(N2))
a2=tensor(qeye(N0),qeye(N1),destroy(N2))
#trilinear Hamiltonian
H=1.0j*(a0*a1.dag()*a2.dag()-a0.dag()*a1*a2)
#run Monte-Carlo
tlist=linspace(0,2.5,50)
output=mcsolve(H,psi0,tlist,[],[],ntraj=1)
#plot results
from mpl_toolkits.mplot3d import Axes3D
from matplotlib import cm
colors=[m, g,orange,b, y,pink]
x=arange(N1)
params = {axes.labelsize: 14,text.fontsize: 14,legend.fontsize: 12,xtick.labelsize: 14,y
rcParams.update(params)
fig = plt.figure()
ax = Axes3D(fig)
for j in range(5):
ax.bar(x, diags1[10*j], zs=tlist[10*j], zdir=y,color=colors[j],linewidth=1.0,alpha=0.6,alig
ax.plot(x,thermal[10*j],zs=tlist[10*j],zdir=y,color=r,linewidth=3,alpha=1)
ax.set_zlabel(rProbability)
ax.set_xlabel(rNumber State)
ax.set_ylabel(rTime)
ax.set_zlim3d(0,1)
show()
if __name__==__main__:
run()
Example showing which times and operators were responsible for wave function collapse in a monte-carlo simulation
of a dissipative trilinear Hamiltonian. Operators are color coded for clarity.
#
# Example showing which times and operators
# were responsible for wave function collapse
# in the monte-carlo simulation of a dissipative
# trilinear Hamiltonian.
#
def run():
#number of states for each mode
N0=6
N1=6
N2=6
#damping rates
gamma0=0.1
gamma1=0.4
gamma2=0.1
alpha=sqrt(2)#initial coherent state param for mode 0
tlist=linspace(0,4,200)
ntraj=500#number of trajectories
#define operators
a0=tensor(destroy(N0),qeye(N1),qeye(N2))
a1=tensor(qeye(N0),destroy(N1),qeye(N2))
a2=tensor(qeye(N0),qeye(N1),destroy(N2))
#trilinear Hamiltonian
H=1j*(a0*a1.dag()*a2.dag()-a0.dag()*a1*a2)
#run Monte-Carlo
data=mcsolve(H,psi0,tlist,[C0,C1,C2],[num0,num1,num2])
#plot results
fig = plt.figure()
ax = fig.add_subplot(111)
cs=[b,r,g] #set three colors, one for each operator
for k in xrange(ntraj):
if len(data.col_times[k])>0:#just in case no collapse
colors=[cs[j] for j in data.col_which[k]]#set color
xdat=[k for x in xrange(len(data.col_times[k]))]
ax.scatter(xdat,data.col_times[k],marker=o,c=colors)
ax.set_xlim([-1,ntraj+1])
ax.set_ylim([0,tlist[-1]])
ax.set_xlabel(Trajectory,fontsize=14)
ax.set_ylabel(Collpase Time,fontsize=14)
ax.set_title(Blue = C0, Red = C1, Green= C2)
show()
if __name__==__main__:
run()
Monte Carlo evoution of a coherently driven cavity with a two-level atom initially in the ground state and no photons
in the cavity.
Adapted from qotoolbox example probqmc3 by Sze M. Tan.
#
# Calculate the correlation and power spectrum of a cavity,
# with and without coupling to a two-level atom.
#
from qutip import *
from pylab import *
def run():
# Hamiltonian
a = tensor(destroy(N), qeye(2))
sm = tensor(qeye(N), destroy(2))
H = wc * a.dag() * a + wa * sm.dag() * sm + g * (a.dag() * sm + a * sm.dag())
# collapse operators
c_op_list = []
n_th_a = 0.5
rate = kappa * (1 + n_th_a)
if rate > 0.0:
c_op_list.append(sqrt(rate) * a)
rate = gamma
if rate > 0.0:
c_op_list.append(sqrt(rate) * sm)
A = a.dag() + a
B = A
#
# setup the calcualtion
#
N = 4 # number of cavity fock states
wc = 1.00 * 2 * pi # cavity frequency
wa = 1.00 * 2 * pi # atom frequency
g = 0.20 * 2 * pi # coupling strength
kappa = 1.0 # cavity dissipation rate
gamma = 0.2 # atom dissipation rate
if __name__==__main__:
run()
The state of an atom subject to a classical driving field oscillates between its ground and excited states, a phenomena
known as Rabi oscillations. This example gives a numerical demonstration of this effect by solving for the dynamics
of the time-dependent two-level Hamiltonian. This example also demonstrates how the list-string format is used to
def run():
#
# problem parameters:
#
delta = 0.0 * 2 * pi # qubit sigma_x coefficient
eps0 = 1.0 * 2 * pi # qubit sigma_z coefficient
A = 0.25 * 2 * pi # driving amplitude (reduce to make the RWA more accurate)
w = 1.0 * 2 * pi # driving frequency
gamma1 = 0.0 # relaxation rate
n_th = 0.0 # average number of excitations ("temperature")
psi0 = basis(2,1) # initial state
#
# Hamiltonian
#
sx = sigmax()
sz = sigmaz()
sm = destroy(2)
H0 = - delta/2.0 * sx - eps0/2.0 * sz
H1 = - A * sx
#
# collapse operators
#
c_op_list = []
#
# Plot the solution
#
plot(tlist, real(output1.expect[0]), b, tlist, real(output2.expect[0]), r)
xlabel(Time)
ylabel(Occupation probability)
title(Excitation probabilty of qubit)
legend(("Time-dependent Hamiltonian", "Corresponding RWA"))
show()
if __name__==__main__:
run()
Single photon source based on a three level atom strongly coupled to a cavity
This example demonstrates a single photon source based on a three level atom strongly coupled to a cavity. Contributed
by Markus Baden.
#
# Single photon source based on a three level atom strongly coupled to a cavity
#
# We follow the treatment presented in Kuhn et al.,
# Appl. Phys. B 69, 373-377 (1999),
# http://www.mpq.mpg.de/qdynamics/publications/library/APB69p373_Kuhn.pdf,
# for more details see M. Hennrichs thesis,
# http://mediatum2.ub.tum.de/node?id=602970.
#
# We study the following lambda system,
#
# |e>
# --------
# / \
# Omega / \ g
# / \
# / -------
# ------- |g>
# |u>
#
# where |u> and |g> are the ground states and |e> is the exicted state.
# |u> and |e> are coupled by a classical laser field with Rabi frequency
# Omega, and |g> and |e> by a cavity field with 2g being the single-photon
# Rabi frequency.
#
from qutip import *
from pylab import *
def run():
subplot(212)
plot(t, exp_uu, k-, label=$P{\mathrm{uu}}$)
plot(t, exp_gg, k:, label=$P{\mathrm{gg}}$)
ylabel(Population)
xlabel(Time [$\mu s$])
legend()
show()
if __name__==__main__:
run()
The Landau-Zener problem is a simple yet nontrivial example of a time-dependent problem in quantum mechanics.
It concerns the occupation probabilities of the states of a two-level atom when its energy bias is linearly swept from
negative to positive infinity, through an avoided-level crossing. The Hamiltonian for the problem is
1 1
H(t) = x + vtz ,
2 2
where is the tunneling rate at t = 0, v is the sweep rate of the bare energy splitting, and time t goes from
to . The Landau-Zener formula gives the final occupation probabilities at t , e.g., for the final ground state:
P = 1 exp(2 /2v). However, there is no analytic formala for the occupation probabilities at intermediate times.
In QuTiP it is easy to calculate the time-evolution of the Landau-Zener problem numerically, which is demonstrated
here. This example also shows how to use the function-callback format to define a time-dependent Hamiltonian.
#
# Landau-Zener transitions in a quantum two-level system
#
from qutip.states import *
from qutip.Qobj import *
from qutip.tensor import *
from qutip.ptrace import *
from qutip.operators import *
from qutip.expect import *
from qutip.correlation import *
def run():
#
# set up the parameters
#
delta = 0.5 * 2 * pi # qubit sigma_x coefficient
eps0 = 0.0 * 2 * pi # qubit sigma_z coefficient
A = 2.0 * 2 * pi # sweep rate
gamma1 = 0.0 # relaxation rate
n_th = 0.0 # average number of thermal photons
psi0 = basis(2,0) # initial state
#
# Hamiltonian
#
sx = sigmax()
sz = sigmaz()
sm = destroy(2)
H0 = - delta/2.0 * sx - eps0/2.0 * sz
H1 = - A/2.0 * sz
args = (H0, H1)
#
# collapse operators, only active if gamma1 > 0
#
c_ops = []
#
# Plot the results
#
plot(tlist, real(output.expect[0]), b, tlist, real(1-output.expect[0]), r)
plot(tlist, 1 - exp( - pi * delta **2 / (2 * A)) * ones(shape(tlist)), k)
xlabel(Time)
ylabel(Occupation probability)
title(Excitation probabilty the two-level system)
legend(("Excited state", "Ground state", "Landau-Zener formula"), loc=0)
show()
if __name__==__main__:
run()
In this example we consider a strongly driven two-level system where the driving field couples to the z operator. The
system is subject to repeated Landau-Zener-like transitions:
1 1 1
H(t) = x z A cos(t)z .
2 2 2
Here, is the tunneling rate, is the energy-bias in the absense of the driving field, A is the (strong) driving
amplitude, and is the driving frequency.
In the following code we evolve the system for a few driving periods and plot the results, to get an idea of how the
state of the two-level system changes at the avoided-level crossing points (where the z coefficient in the Hamiltonian
is zero).
Next, we use the qutip.propagator function to find the propagator for the system for one driving period, and
then we use the qutip.propagator_steadystate function to find the pseudo steady state density matrix that
follows from infinitely many applications of the one-period propagotor.
This examples demonstrates how to use the list-callback format to define a time-dependent Hamiltonian.
#
# Using the propagator to find the steady state of a driven system.
#
from qutip import *
from pylab import *
from mpl_toolkits.mplot3d import Axes3D
from matplotlib import cm
def run():
#
# configure the parameters
#
delta = 0.075 * 2 * pi # qubit sigma_x coefficient
eps0 = 0.0 * 2 * pi # qubit sigma_z coefficient
A = 2.0 * 2 * pi # sweep rate
gamma1 = 0.0001 # relaxation rate
#
# Hamiltonian
#
sx = sigmax()
sz = sigmaz()
sm = destroy(2)
H0 = - delta/2.0 * sx - eps0/2.0 * sz
H1 = - A/2.0 * sz
#
# collapse operators
#
c_op_list = []
rate = gamma2
if rate > 0.0:
c_op_list.append(sqrt(rate) * sz) # dephasing
#
# evolve for five driving periods
#
tlist = linspace(0.0, 5 * T, 1500)
output = mesolve(hamiltonian_t, psi0, tlist, c_op_list, [sm.dag() * sm], args)
#
# find the propagator for one driving period
#
T = 2*pi / omega
U = propagator(hamiltonian_t, T, c_op_list, args)
#
# find the steady state of repeated applications of the propagator
# (i.e., t -> inf)
#
rho_ss = propagator_steadystate(U)
p_ex_ss = real(expect(sm.dag() * sm, rho_ss))
#
# plot the results
#
figure(1)
subplot(211)
plot(tlist, real(output.expect[0]), b)
plot(tlist, real(1-output.expect[0]), r)
plot(tlist, ones(shape(tlist)) * p_ex_ss, k, linewidth=2)
xlabel(Time)
ylabel(Probability)
title(Occupation probabilty of qubit [NEW])
legend((r"$\left|1\right>$", r"$\left|0\right>$", r"$\left|1\right>$ steady state"), loc=0)
subplot(212)
plot(tlist, -delta/2.0 * ones(shape(tlist)), r)
plot(tlist, -(eps0/2.0 + A/2.0 * cos(omega * tlist)), b)
legend(("$\sigma_x$ coefficient", "$\sigma_z$ coefficient"))
xlabel(Time)
ylabel(Coefficients in the Hamiltonian)
show()
if __name__==__main__:
run()
This example demonstrates how to calculate the Floquet quasienergies for a driven system. The example is taken from
Creffield et al., Phys. Rev. B 67, 165301 (2003), see Fig. 1(a) in that paper. The Hamiltonian is
E
H(t) = z + cos(t)z ,
2 2
and we use the QuTiP function qutip.floquet.floquet_modes to obtain the Floquet modes and the quasiener-
gies.
#
# Calculate the quasienergies for a driven two-level system as a function of
# driving amplitude. See Creffield et al., Phys. Rev. B 67, 165301 (2003).
#
def run():
sx = sigmax()
sz = sigmaz()
sm = destroy(2)
H0 = delta/2.0 * sz - epsilon/2.0 * sx
for idx, A in enumerate(E_vec):
H1 = A/2.0 * sx
q_energies[idx,:] = f_energies
if __name__==__main__:
run()
In this example we demonstrate how to use the Floquet-Markov master equation solver in QuTiP by revisting the vac-
uum Rabi oscillation problem: i.e., a simple two level system subject to a driving field (classical in this example) and
dissipation due to its interaction with the environment. We use the QuTiP function qutip.floquet.fmmesolve
to obtain the system dynamics. For comparison we also calculate the dynamics using the standard Lindblad master
eqaution. For weak driving and dissipation the two solvers should give similar results, but not necessarily when the
driving amplitude or dissipation rates are large compared to the two-level energy splitting.
#
# Calculate the dynamics of a driven two-level system with according to the
# Floquet-Markov master equation. For compari
#
from qutip import *
from pylab import *
import qutip.odeconfig
def J_cb(omega):
""" Noise spectral density """
return 0.5 * gamma1 * omega/(2*pi)
def run():
# --------------------------------------------------------------------------
# Standard lindblad master equation with time-dependent hamiltonian
#
c_op_list = [sqrt(gamma1) * sigmax(), sqrt(gamma2) * sigmaz()]
p_ex_me = mesolve(H, psi0, tlist, c_op_list, [num(2)], args=args).expect[0]
# --------------------------------------------------------------------------
# Floquet markov master equation dynamics
#
qutip.odeconfig.tdfunc = None # reset td func flag
show()
if __name__==__main__:
run()
#
# Nonadiabatic sweep: Gradually transform a simple decoupled spin chain
# hamiltonian to a complicated interacting spin chain.
#
# pre-allocate operators
si = qeye(2)
sx = sigmax()
sy = sigmay()
sz = sigmaz()
sx_list = []
sy_list = []
sz_list = []
for n in range(N):
op_list = []
for m in range(N):
op_list.append(si)
op_list[n] = sx
sx_list.append(tensor(op_list))
op_list[n] = sy
sy_list.append(tensor(op_list))
op_list[n] = sz
sz_list.append(tensor(op_list))
#
# Construct the initial hamiltonian and state vector
#
psi_list = [basis(2,0) for n in range(N)]
psi0 = tensor(psi_list)
H0 = 0
for n in range(N):
H0 += - 0.5 * 2.5 * sz_list[n]
#
# Construct the target hamiltonian
#
H1 = 0
for n in range(N-1):
# interaction terms
H1 += - 0.5 * Jx[n] * sx_list[n] * sx_list[n+1]
H1 += - 0.5 * Jy[n] * sy_list[n] * sy_list[n+1]
H1 += - 0.5 * Jz[n] * sz_list[n] * sz_list[n+1]
#
# callback function for each time-step
#
evals_mat = zeros((len(taulist),M))
occupation_mat = zeros((len(taulist),M))
idx = [0]
def process_rho(tau, psi):
evals_mat[idx[0],:] = real(evals)
idx[0] += 1
#
# Evolve the system, request the solver to call process_rho at each time
# step.
#
mesolve(h_t, psi0, taulist, [], process_rho, args)
def run():
#
# set up the paramters
#
N = 6 # number of spins
M = 20 # number of eigenenergies to solve for
#---------------------------------------------------------------------------
# plots
#
rc(text, usetex=True)
rc(font, family=serif)
figure(figsize=(9,12))
#
# plot the energy eigenvalues
#
subplot(2,1,1)
# second, draw line that encode the occupation probability of each corresponding
# state in the linewidth. thicker line => high occupation probability.
for idx in range(len(taulist)-1):
for n in range(len(occ_mat[0,:])):
lw = 0.5 + 4*occ_mat[idx,n]
if lw > 0.55:
plot(array([taulist[idx], taulist[idx+1]])/taumax, array([evals_mat[idx,n], evals_mat
xlabel(r$\tau$)
ylabel(Eigenenergies)
title("Energyspectrum (%d lowest values) of a chain of %d spins.\nThe occupation probabilities ar
legend(("Ground state",))
#
# plot the occupation probabilities for the few lowest eigenstates
#
subplot(2,1,2)
for n in range(len(occ_mat[0,:])):
if n == 0:
plot(taulist/max(taulist), 0 + occ_mat[:,n], r, linewidth=2)
else:
plot(taulist/max(taulist), 0 + occ_mat[:,n])
xlabel(r$\tau$)
ylabel(Occupation probability)
title("Occupation probability of the %d lowest eigenstates for a chain of %d spins" % (M, N))
legend(("Ground state",))
show()
if __name__==__main__:
run()
Landau-Zener-Stckelberg interferometry
This is an other example of how to use the propagator of a driven system to calculate its steadystate. The system is the
same as considered in Using the propagator to find the steady state of a driven system, for which the Hamiltonian is
1 1 1
H(t) = x z A cos(t)z .
2 2 2
This example also illustrates how to use the qutip.parfor to parallelize the loop over the elements of a matrix.
#
# Landau-Zener-Stuckelberg interferometry: steady state of repeated Landau-Zener
# like avoided-level crossing, as a function of driving amplitude and bias.
#
from qutip import *
import os
import time
# performance tuning
qutip.settings.auto_herm=False
qutip.settings.auto_tidyup=False
# collapse operators
c_op_list = [sqrt(gamma1) * sm, sqrt(gamma2) * sz] # relaxation and dephasing
m, eps = args
#H0 = - delta/2.0 * sx - eps/2.0 * sz
p_mat_m = zeros(len(A_list))
for n, A in enumerate(A_list):
#H1 = (A/2) * sz
# list-str format
args = {w: w}; H_td = [[sz, -delta/2.0], [sx, -eps/2.0 + A/2 *sin(w * t)]]
# list-function format
#args = w; H_td = [H0, [H1, lambda t, w: sin(w * t)]]
# start a parallel for loop over the list bias point values (eps_list)
start_time = time.time()
p_mat_list = parfor(task, enumerate(eps_list))
print time elapsed = + str(time.time() - start_time)
fig = figure()
ax = fig.add_axes([0.1, 0.1, 0.9, 0.8])
c = ax.pcolor(eps_mat, A_mat, p_mat, antialiased=False)
c.set_cmap(RdYlBu_r)
cbar = fig.colorbar(c)
cbar.set_label("Probability")
ax.set_xlabel(rBias point $\epsilon$)
ax.set_ylabel(rAmplitude $A$)
ax.autoscale(tight=True)
title(Steadystate excitation probability\n +
$H = -\\frac{1}{2}\\Delta\\sigma_x -\\frac{1}{2}\\epsilon\\sigma_z - \\frac{1}{2}A\\sin(\\omeg
show()
Comparison of Lindblad vs. Bloch-Redfield Master Equation for Coupled Qubit System
#
# Comparing the Lindblad and Bloch-Redfield master equations
#
from qutip import *
from pylab import *
def run():
def qubit_integrate(w, theta, g, gamma1, gamma2, psi0, tlist):
#
# Hamiltonian
#
sx1 = tensor(sigmax(),qeye(2))
sy1 = tensor(sigmay(),qeye(2))
sz1 = tensor(sigmaz(),qeye(2))
sm1 = tensor(sigmam(),qeye(2))
sx2 = tensor(qeye(2),sigmax())
sy2 = tensor(qeye(2),sigmay())
sz2 = tensor(qeye(2),sigmaz())
sm2 = tensor(qeye(2),sigmam())
#
# Lindblad master equation
#
c_op_list = []
n_th = 0.0 # zero temperature
rate = gamma1[0] * (n_th + 1)
if rate > 0.0: c_op_list.append(sqrt(rate) * sm1)
rate = gamma1[1] * (n_th + 1)
if rate > 0.0: c_op_list.append(sqrt(rate) * sm2)
#
# Bloch-Redfield tensor
#
def ohmic_spectrum1(w):
if w == 0.0:
# dephasing inducing noise
return gamma1[0]
else:
# relaxation inducing noise
return gamma1[0] * w / (2*pi) * (w > 0.0)
def ohmic_spectrum2(w):
if w == 0.0:
# dephasing inducing noise
return gamma1[1]
else:
# relaxation inducing noise
return gamma1[1] * w / (2*pi) * (w > 0.0)
brme_results = brmesolve(H, psi0, tlist, [sx1, sx2], [sx1, sy1, sz1], [ohmic_spectrum1, ohmic
fig = figure(figsize=(10,10))
ax = fig.add_subplot(2,2,1)
title(Lindblad master equation)
ax.plot(tlist, lme_results[0], r)
ax.plot(tlist, lme_results[1], g)
ax.plot(tlist, lme_results[2], b)
ax.legend(("sx1", "sy1", "sz1"))
xlabel(Time)
ax = fig.add_subplot(2,2,2)
title(Bloch-Redfield master equation)
ax.plot(tlist, brme_results[0], r)
ax.plot(tlist, brme_results[1], g)
ax.plot(tlist, brme_results[2], b)
ax.legend(("sx1", "sy1", "sz1"))
xlabel(Time)
show()
if __name__ == "__main__":
run()
SIX
API DOCUMENTATION
This chapter contains automatically generated API documentation, including a complete list of QuTiPs public classes
and functions.
153
QuTiP: The Quantum Toolbox in Python, Release 2.0
type : str
Type of quantum object: bra, ket, oper, or super.
Methods conj() :
Conjugate of quantum object.
dag() :
Adjoint (dagger) of quantum object.
eigenenergies(sparse=False, sort=low, eigvals=0, tol=0, maxiter=100000) :
Returns eigenenergies (eigenvalues) of a quantum object.
eigenstates(sparse=False, sort=low, eigvals=0, tol=0, maxiter=100000) :
Returns eigenenergies and eigenstates of quantum object.
expm() :
Matrix exponential of quantum object.
full() :
Returns dense array of quantum object data attribute.
groundstate(sparse=False,tol=0,maxiter=100000) :
Returns eigenvalue and eigenket for the groundstate of a quantum object.
matrix_element(bra,ket) :
Returns the matrix element of operator between bra and ket vectors.
norm(oper_norm=tr,sparse=False,tol=0,maxiter=100000) :
Returns norm of operator.
ptrace(sel) :
Returns quantum object for selected dimensions after performing partial trace.
sqrtm() :
Matrix square root of quantum object.
tidyup(atol=1e-15) :
Removes small elements from quantum object.
tr() :
Trace of quantum object.
trans() :
Transpose of quantum object.
transform(inpt,inverse=False) :
Performs a basis transformation defined by inpt matrix.
unit(oper_norm=tr,sparse=False,tol=0,maxiter=100000) :
Returns normalized quantum object.
checkherm()
Explicitly check if the Qobj is hermitian
Notes
The sparse eigensolver is much slower than the dense version. Use sparse only if memory requirements
demand it.
eigenstates(sparse=False, sort=low, eigvals=0, tol=0, maxiter=100000)
Find the eigenstates and eigenenergies.
Eigenstates and Eigenvalues are defined for operators and superoperators only.
Parameters sparse : bool
Use sparse Eigensolver
sort : str
Sort eigenvalues (and vectors) low to high, or high to low.
eigvals : int
Number of requested eigenvalues. Default is all eigenvalues.
tol : float
Tolerance used by sparse Eigensolver (0 = machine precision).The sparse solver may
not converge if the tolerance is set too low.
maxiter : int
Maximum number of iterations performed by sparse solver (if used).
Returns eigvals : array
Array of eigenvalues for operator.
eigvecs : array
Array of quantum operators representing the oprator eigenkets. Order of eigenkets is
determined by order of eigenvalues.
Notes
The sparse eigensolver is much slower than the dense version. Use sparse only if memory requirements
demand it.
expm()
Matrix exponential of quantum operator.
Input operator must be square.
Returns oper : qobj
Exponentiated quantum operator.
Raises TypeError :
Quantum operator is not square.
full()
Returns a dense array from quantum object.
Returns data : array
Array of complex data from quantum objects data attribute.
groundstate(sparse=False, tol=0, maxiter=100000)
Finds the ground state Eigenvalue and Eigenvector.
Defined for quantum operators or superoperators only.
Parameters sparse : bool
Use sparse Eigensolver
tol : float
Tolerance used by sparse Eigensolver (0 = machine precision). The sparse solver may
not converge if the tolerance is set too low.
maxiter : int
Maximum number of iterations performed by sparse solver (if used).
Returns eigval : float
Notes
The sparse eigensolver is much slower than the dense version. Use sparse only if memory requirements
demand it.
matrix_element(bra, ket)
Calculates a matrix element.
Gives matrix for the Qobj sandwiched between a bra and ket vector.
Parameters bra : qobj
Quantum object of type bra.
ket : qobj
Quantum object of type ket.
Returns elem : complex
Complex valued matrix element.
Raises TypeError :
Can only calculate matrix elements between a bra and ket quantum object.
norm(oper_norm=tr, sparse=False, tol=0, maxiter=100000)
Returns the norm of a quantum object.
Norm is L2-norm for kets and trace-norm (by default) for operators. Other operator norms may be specified
using the oper_norm argument.
Parameters oper_norm : str
Which norm to use for operators: trace tr, Frobius fro,one one, or max max. This
parameter does not affect the norm of a state vector.
sparse : bool
Use sparse eigenvalue solver for trace norm. Other norms are not affected by this pa-
rameter.
tol : float
Tolerance for sparse solver (if used) for trace norm. The sparse solver may not converge
if the tolerance is set too low.
maxiter : int
Maximum number of iterations performed by sparse solver (if used) for trace norm.
Returns norm : float
The requested norm of the operator or state quantum object.
Notes
The sparse eigensolver is much slower than the dense version. Use sparse only if memory requirements
demand it.
ptrace(sel)
Partial trace of the Qobj with selected components remaining.
Parameters sel : int/list
An int or list of components to keep after partial trace.
Returns oper: qobj :
Quantum object representing partial trace with selected components remaining.
Notes
This function is identical to the qutip.Qobj.ptrace function that has been depreciated.
sqrtm(sparse=False, tol=0, maxiter=100000)
The sqrt of a quantum operator.
Operator must be square.
Parameters sparse : bool
Use sparse eigenvalue/vector solver.
tol : float
Tolerance used by sparse solver (0 = machine precision).
maxiter : int
Maximum number of iterations used by sparse solver.
Returns oper: qobj :
Matrix square root of operator.
Raises TypeError :
Quantum object is not square.
Notes
The sparse eigensolver is much slower than the dense version. Use sparse only if memory requirements
demand it.
tidyup(atol=1e-15)
Removes small elements from a quantum object.
Parameters atol : float
Absolute tolerance used by tidyup. Default is set via qutip global settings parameters.
Returns oper: qobj :
Quantum object with small elements removed.
tr()
The trace of a quantum object.
Notes
tidy(*args)
return a tidier version of self
value(tlist)
Evaluate an exponential series at the times listed in tlist.
class Odedata
Class for storing simulation results from any of the dynamics solvers.
Attributes solver : str
Which solver was used [mesolve,mcsolve,brsolve,floquet]
times : list/array
Times at which simulation data was collected.
expect : list/array
Expectation values (if requested) for simulation.
states : array
State of the simulation (density matrix or ket) evaluated at times.
num_expect : int
Number of expectation value operators in simulation.
num_collapse : int
Number of collapse operators in simualation.
ntraj : int/list
Number of monte-carlo trajectories (if using mcsolve). List indicates that averaging of
expectation values was done over a subset of total number of trajectories.
col_times : list
Times at which state collpase occurred. Only for Monte-Carlo solver.
col_which : list
Which collapse operator was responsible for each collapse in col_times. mcsolver
only.
Quantum States
basis(N, *args)
Generates the vector representation of a Fock state.
Parameters N : int
Number of Fock states in Hilbert space.
args : int
int corresponding to desired number state, defaults to 0 if omitted.
Returns state : qobj
Qobj representing the requested number state |args>.
Notes
but in qotoolbox:
basis(N, 1) = ground state
Examples
>>> basis(5,2)
Quantum object: dims = [[5], [1]], shape = [5, 1], type = ket
Qobj data =
[[ 0.+0.j]
[ 0.+0.j]
[ 1.+0.j]
[ 0.+0.j]
[ 0.+0.j]]
coherent(N, alpha)
Generates a coherent state with eigenvalue alpha.
Constructed using displacement operator on vacuum state.
Parameters N : int
Number of Fock states in Hilbert space.
alpha : float/complex
Eigenvalue of coherent state.
Returns state : qobj
Qobj quantum object for coherent state
Examples
>>> coherent(5,0.25j)
Quantum object: dims = [[5], [1]], shape = [5, 1], type = ket
Qobj data =
[[ 9.69233235e-01+0.j ]
[ 0.00000000e+00+0.24230831j]
[ -4.28344935e-02+0.j ]
[ 0.00000000e+00-0.00618204j]
[ 7.80904967e-04+0.j ]]
coherent_dm(N, alpha)
Density matrix representation of a coherent state.
Constructed via outer product of qutip.states.coherent
Parameters N : int
Number of Fock states in Hilbert space.
alpha : float/complex
Eigenvalue for coherent state.
Returns dm : qobj
Density matrix representation of coherent state.
Examples
>>> coherent_dm(3,0.25j)
Quantum object: dims = [[3], [3]], shape = [3, 3], type = oper, isHerm = True
Qobj data =
[[ 0.93941695+0.j 0.00000000-0.23480733j -0.04216943+0.j ]
[ 0.00000000+0.23480733j 0.05869011+0.j 0.00000000-0.01054025j]
[-0.04216943+0.j 0.00000000+0.01054025j 0.00189294+0.j ]]
fock(N, *args)
Bosonic Fock (number) state.
Same as qutip.states.basis.
Parameters N : int
Number of states in the Hilbert space.
m : int
int for desired number state, defaults to 0 if omitted.
Returns Requested number state :math:left|mathrm{args}right>. :
Examples
>>> fock(4,3)
Quantum object: dims = [[4], [1]], shape = [4, 1], type = ket
Qobj data =
[[ 0.+0.j]
[ 0.+0.j]
[ 0.+0.j]
[ 1.+0.j]]
fock_dm(N, *args)
Density matrix representation of a Fock state
Constructed via outer product of qutip.states.fock.
Parameters N : int
Number of Fock states in Hilbert space.
m : int
int for desired number state, defaults to 0 if omitted.
Returns dm : qobj
Density matrix representation of Fock state.
Examples
>>> fock_dm(3,1)
Quantum object: dims = [[3], [3]], shape = [3, 3], type = oper, isHerm = True
Qobj data =
[[ 0.+0.j 0.+0.j 0.+0.j]
[ 0.+0.j 1.+0.j 0.+0.j]
[ 0.+0.j 0.+0.j 0.+0.j]]
ket2dm(Q)
Takes input ket or bra vector and returns density matrix formed by outer product.
Parameters Q : qobj
Ket or bra type quantum object.
Returns dm : qobj
Density matrix formed by outer product of Q.
Examples
>>> x=basis(3,2)
>>> ket2dm(x)
Quantum object: dims = [[3], [3]], shape = [3, 3], type = oper, isHerm = True
Qobj data =
[[ 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]]
qutrit_basis()
Basis states for a three level system (qutrit)
Returns qstates : array
Array of qutrit basis vectors
thermal_dm(N, n)
Density matrix for a thermal state of n particles
Parameters N : int
Examples
>>> thermal_dm(5,1)
Quantum object: dims = [[5], [5]], shape = [5, 5], type = oper, isHerm = True
Qobj data =
[[ 0.50000+0.j 0.00000+0.j 0.00000+0.j 0.00000+0.j 0.00000+0.j]
[ 0.00000+0.j 0.25000+0.j 0.00000+0.j 0.00000+0.j 0.00000+0.j]
[ 0.00000+0.j 0.00000+0.j 0.12500+0.j 0.00000+0.j 0.00000+0.j]
[ 0.00000+0.j 0.00000+0.j 0.00000+0.j 0.06250+0.j 0.00000+0.j]
[ 0.00000+0.j 0.00000+0.j 0.00000+0.j 0.00000+0.j 0.03125+0.j]]
qstate(string)
Creates a tensor product for a set of qubits in either the up |0 > or down |1 > state.
Parameters string : str
String containing u or d for each qubit (ex. ududd)
Returns qstate : qobj
Qobj for tensor product corresponding to input string.
Examples
>>> qstate(udu)
Quantum object: dims = [[2, 2, 2], [1, 1, 1]], shape = [8, 1], type = ket
Qobj data =
[[ 0.]
[ 0.]
[ 0.]
[ 0.]
[ 0.]
[ 1.]
[ 0.]
[ 0.]]
Quantum Operators
create(N)
Creation (raising) operator.
Parameters N : int
Dimension of Hilbert space.
Returns oper : qobj
Qobj for raising operator.
Examples
>>> create(4)
Quantum object: dims = [[4], [4]], shape = [4, 4], type = oper, isHerm = False
Qobj data =
[[ 0.00000000+0.j 0.00000000+0.j 0.00000000+0.j 0.00000000+0.j]
[ 1.00000000+0.j 0.00000000+0.j 0.00000000+0.j 0.00000000+0.j]
[ 0.00000000+0.j 1.41421356+0.j 0.00000000+0.j 0.00000000+0.j]
[ 0.00000000+0.j 0.00000000+0.j 1.73205081+0.j 0.00000000+0.j]]
destroy(N)
Destruction (lowering) operator.
Parameters N : int
Dimension of Hilbert space.
Returns oper : qobj
Qobj for lowering operator.
Examples
>>> destroy(4)
Quantum object: dims = [[4], [4]], shape = [4, 4], type = oper, isHerm = False
Qobj data =
[[ 0.00000000+0.j 1.00000000+0.j 0.00000000+0.j 0.00000000+0.j]
[ 0.00000000+0.j 0.00000000+0.j 1.41421356+0.j 0.00000000+0.j]
[ 0.00000000+0.j 0.00000000+0.j 0.00000000+0.j 1.73205081+0.j]
[ 0.00000000+0.j 0.00000000+0.j 0.00000000+0.j 0.00000000+0.j]]
displace(N, alpha)
Single-mode displacement operator.
Parameters N : int
Dimension of Hilbert space.
alpha : float/complex
Displacement amplitude.
Returns oper : qobj
Displacement operator.
Examples
>>> displace(4,0.25)
Quantum object: dims = [[4], [4]], shape = [4, 4], type = oper, isHerm = False
Qobj data =
[[ 0.96923323+0.j -0.24230859+0.j 0.04282883+0.j -0.00626025+0.j]
[ 0.24230859+0.j 0.90866411+0.j -0.33183303+0.j 0.07418172+0.j]
[ 0.04282883+0.j 0.33183303+0.j 0.84809499+0.j -0.41083747+0.j]
[ 0.00626025+0.j 0.07418172+0.j 0.41083747+0.j 0.90866411+0.j]]
jmat(j, *args)
Higher-order spin operators:
Parameters j : float
Spin of operator
args : str
Which operator to return x,y,z,+,-. If no args given, then output is [x,y,z]
Returns jmat : qobj/list
qobj for requested spin operator(s).
Notes
Examples
>>> jmat(1)
[ Quantum object: dims = [[3], [3]], shape = [3, 3], type = oper, isHerm = True
Qobj data =
[[ 0. 0.70710678 0. ]
[ 0.70710678 0. 0.70710678]
[ 0. 0.70710678 0. ]]
Quantum object: dims = [[3], [3]], shape = [3, 3], type = oper, isHerm = True
Qobj data =
[[ 0.+0.j 0.+0.70710678j 0.+0.j ]
[ 0.-0.70710678j 0.+0.j 0.+0.70710678j]
[ 0.+0.j 0.-0.70710678j 0.+0.j ]]
Quantum object: dims = [[3], [3]], shape = [3, 3], type = oper, isHerm = True
Qobj data =
[[ 1. 0. 0.]
[ 0. 0. 0.]
[ 0. 0. -1.]]]
num(N)
Quantum object for number operator.
Parameters N : int
The dimension of the Hilbert space.
Returns oper: qobj :
Qobj for number operator.
Examples
>>> num(4)
Quantum object: dims = [[4], [4]], shape = [4, 4], type = oper, isHerm = True
Qobj data =
[[0 0 0 0]
[0 1 0 0]
[0 0 2 0]
[0 0 0 3]]
qeye(N)
Identity operator
Parameters N : int
Dimension of Hilbert space.
Returns oper : qobj
Identity operator Qobj.
Examples
>>> qeye(3)
Quantum object: dims = [[3], [3]], shape = [3, 3], type = oper, isHerm = True
Qobj data =
[[ 1. 0. 0.]
[ 0. 1. 0.]
[ 0. 0. 1.]]
qutrit_ops()
Operators for a three level system (qutrit).
Returns opers: array :
array of qutrit operators.
sigmam()
Annihilation operator for Pauli spins.
Examples
>>> sigmam()
Quantum object: dims = [[2], [2]], shape = [2, 2], type = oper, isHerm = False
Qobj data =
[[ 0. 0.]
[ 1. 0.]]
sigmap()
Creation operator for Pauli spins.
Examples
>>> sigmam()
Quantum object: dims = [[2], [2]], shape = [2, 2], type = oper, isHerm = False
Qobj data =
[[ 0. 1.]
[ 0. 0.]]
sigmax()
Pauli spin 1/2 sigma-x operator
Examples
>>> sigmax()
Quantum object: dims = [[2], [2]], shape = [2, 2], type = oper, isHerm = False
Qobj data =
[[ 0. 1.]
[ 1. 0.]]
sigmay()
Pauli spin 1/2 sigma-y operator.
Examples
>>> sigmay()
Quantum object: dims = [[2], [2]], shape = [2, 2], type = oper, isHerm = True
Qobj data =
[[ 0.+0.j 0.-1.j]
[ 0.+1.j 0.+0.j]]
sigmaz()
Pauli spin 1/2 sigma-z operator.
Examples
>>> sigmaz()
Quantum object: dims = [[2], [2]], shape = [2, 2], type = oper, isHerm = True
Qobj data =
[[ 1. 0.]
[ 0. -1.]]
squeez(N, sp)
Single-mode Squeezing operator.
Parameters N : int
Dimension of hilbert space.
sp : float/complex
Squeezing parameter.
Returns oper : qobj
Squeezing operator.
Examples
>>> squeez(4,0.25)
Quantum object: dims = [[4], [4]], shape = [4, 4], type = oper, isHerm = False
Qobj data =
[[ 0.98441565+0.j 0.00000000+0.j 0.17585742+0.j 0.00000000+0.j]
[ 0.00000000+0.j 0.95349007+0.j 0.00000000+0.j 0.30142443+0.j]
[-0.17585742+0.j 0.00000000+0.j 0.98441565+0.j 0.00000000+0.j]
[ 0.00000000+0.j -0.30142443+0.j 0.00000000+0.j 0.95349007+0.j]]
isbra(Q)
Determines if given quantum object is a bra-vector.
Parameters Q : qobj
Quantum object
Returns isbra : bool
True if Qobj is bra-vector, False otherwise.
Examples
>>> psi=basis(5,2)
>>> isket(psi)
False
isequal(A, B, tol=1e-15)
Determines if two qobj objects are equal to within given tolerance.
Parameters A : qobj
Qobj one
B : qobj
Qobj two
tol : float
Tolerence for equality to be valid
Returns isequal : bool
True if qobjs are equal, False otherwise.
isherm(Q)
Determines if given operator is Hermitian.
Parameters Q : qobj
Quantum object
Returns isherm : bool
True if operator is Hermitian, False otherwise.
Examples
>>> a=destroy(4)
>>> isherm(a)
False
isket(Q)
Determines if given quantum object is a ket-vector.
Parameters Q : qobj
Quantum object
Returns isket : bool
Examples
>>> psi=basis(5,2)
>>> isket(psi)
True
isoper(Q)
Determines if given quantum object is a operator.
Parameters Q : qobj
Quantum object
Returns isoper : bool
True if Qobj is operator, False otherwise.
Examples
>>> a=destroy(5)
>>> isoper(a)
True
issuper(Q)
Determines if given quantum object is a super-operator.
Parameters Q : qobj
Quantum object
Returns issuper : bool
True if Qobj is superoperator, False otherwise.
Liouvillian
liouvillian(H, c_op_list)
Assembles the Liouvillian superoperator from a Hamiltonian and a list of collapse operators.
Parameters H : qobj
System Hamiltonian.
c_op_list : array_like
A list or array of collpase operators.
Returns L : qobj
Louvillian superoperator.
spost(A)
Superoperator formed from post-multiplication by operator A
Parameters A : qobj
Quantum operator for post multiplication.
Returns super : qobj
Tensor
Module for the creation of composite quantum objects via the tensor product.
tensor(*args)
Calculates the tensor product of input operators.
Parameters args : array_like
list or array of quantum objects for tensor product.
Returns obj : qobj
A composite quantum object.
Examples
Expectation Values
expect(oper, state)
Calculates the expectation value for operator and state(s).
Parameters oper : qobj
Operator for expectation value.
state : qobj/list
A single or list of quantum states or density matricies.
Returns expt : float
Expectation value. real if oper is Hermitian, complex otherwise.
Examples
>>> expect(num(4),basis(4,3))
3
Pseudoprobability Functions
Three-level atoms
This module provides functions that are useful for simulating the three level atom with QuTiP. A three level atom
(qutrit) has three states, which are linked by dipole transitions so that 1 <-> 2 <-> 3. Depending on there relative
energies they are in the ladder, lambda or vee configuration. The structure of the relevant operators is the same for any
of the three configurations:
Ladder: Lambda: Vee:
|two> |three>
-------|three> ------- -------
| / \ |one> /
| / \ ------- /
| / \ \ /
-------|two> / \ \ /
| / \ \ /
| / \ \ /
| / -------- \ /
-------|one> ------- |three> -------
|one> |two>
References
Notes
Master Equation
Examples
H = [[H0, sin(w*t)], [H1, sin(2*w*t)]]
H = [[H0, f0_t], [H1, f1_t]]
where f0_t and f1_t are python functions with signature f_t(t, args).
In the list string format and list callback format, the string expression and the callback function must evaluate
to a real or complex number (coefficient for the corresponding operator).
In all cases of time-dependent operators, args is a dictionary of parameters that is used when evaluating opera-
tors. It is passed to the callback functions as second argument
Note: On using callback function: mesolve transforms all qutip.Qobj objects to sparse matrices before
handing the problem to the integrator function. In order for your callback function to work correctly, pass
all qutip.Qobj objects that are used in constructing the Hamiltonian via args. odesolve will check for
qutip.Qobj in args and handle the conversion to sparse matrices. All other qutip.Qobj objects that
are not passed via args will be passed on to the integrator to scipy who will raise an NotImplemented exception.
Parameters H : qutip.Qobj
system Hamiltonian, or a callback function for time-dependent Hamiltonians.
rho0 : qutip.Qobj
initial density matrix or state vector (ket).
tlist : list / array
list of times for t.
c_ops : list of qutip.Qobj
list of collapse operators.
expt_ops : list of qutip.Qobj / callback function
list of operators for which to evaluate expectation values.
args : dictionary
dictionary of parameters for time-dependent Hamiltonians and collapse operators.
options : qutip.Qdeoptions
with options for the ODE solver.
Returns output: :class:qutip.Odedata :
An instance of the class qutip.Odedata, which contains either an array of expecta-
tion values for the times specified by tlist, or an array or state vectors or density matrices
corresponding to the times in tlist [if expt_ops is an empty list], or nothing if a callback
function was given inplace of operators for which to calculate the expectation values.
For problems with time-dependent Hamiltonians, H can be a callback function that takes two arguments, time
and H_args, and returns the Hamiltonian at that point in time. H_args is a list of parameters that is passed to the
callback function H (only used for time-dependent Hamiltonians).
Parameters H : qutip.Qobj
system Hamiltonian, or a callback function for time-dependent Hamiltonians.
rho0 : qutip.Qobj
initial density matrix or state vector (ket).
tlist : list / array
list of times for t.
c_op_list : list of qutip.Qobj
list of collapse operators.
expt_ops : list of qutip.Qobj / callback function
list of operators for which to evaluate expectation values.
H_args : dictionary
dictionary of parameters for time-dependent Hamiltonians and collapse operators.
options : qutip.Qdeoptions
with options for the ODE solver.
Returns output :array :
Expectation values of wavefunctions/density matrices :
for the times specified by tlist. :
Notes
On using callback function: odesolve transforms all qutip.Qobj objects to sparse matrices before handing the
problem to the integrator function. In order for your callback function to work correctly, pass all qutip.Qobj
objects that are used in constructing the Hamiltonian via H_args. odesolve will check for qutip.Qobj in
H_args and handle the conversion to sparse matrices. All other qutip.Qobj objects that are not passed via
H_args will be passed on to the integrator to scipy who will raise an NotImplemented exception.
Depreciated in QuTiP 2.0.0. Use mesolve instead.
As an example of a time-dependent problem, consider a Hamiltonian with two terms H0 and H1, where H1 is
time-dependent with coefficient sin(w*t), and collapse operators C0 and C1, where C1 is time-dependent
with coeffcient exp(-a*t). Here, w and a are constant arguments with values W and A.
Using the Python function time-dependent format requires two Python functions, one for each collapse coeffi-
cient. Therefore, this problem could be expressed as:
def H1_coeff(t,args):
return sin(args[w]*t)
def C1_coeff(t,args):
return exp(-args[a]*t)
H=[H0,[H1,H1_coeff]]
c_op_list=[C0,[C1,C1_coeff]]
args={a:A,w:W}
c_op_list=[C0,[C1,exp(-a*t)]]
args={a:A,w:W}
Constant terms are preferably placed first in the Hamiltonian and collapse operator lists.
Parameters H : qobj
System Hamiltonian.
psi0 : qobj
Initial state vector
tlist : array_like
Times at which results are recorded.
ntraj : int
Number of trajectories to run.
c_ops : array_like
list or array of collapse operators.
e_ops : array_like
list or array of operators for calculating expectation values.
args : dict
Arguments for time-dependent Hamiltonian and collapse operator terms.
options : Odeoptions
Instance of ODE solver options.
Returns results : Odedata
Object storing all results from simulation.
Note: This solver does not currently support time-dependent Hamiltonian or collapse operators.
Parameters H : qutip.Qobj
system Hamiltonian.
rho0 / psi0: :class:qutip.Qobj :
initial density matrix or state vector (ket).
tlist : list / array
list of times for t.
c_ops : list of qutip.Qobj
list of collapse operators.
expt_ops : list of qutip.Qobj / callback function
list of operators for which to evaluate expectation values.
args : dictionary
dictionary of parameters for time-dependent Hamiltonians and collapse operators.
options : qutip.Qdeoptions
with options for the ODE solver.
Returns output: :class:qutip.Odedata :
An instance of the class qutip.Odedata, which contains either an array of expecta-
tion values for the times specified by tlist.
Note: This solver currently does not support multiple collapse operators.
Parameters H : qutip.Qobj
system Hamiltonian.
rho0 / psi0 : qutip.Qobj
initial density matrix or state vector (ket).
tlist : list / array
list of times for t.
options : qutip.Odeoptions
options for the ODE solver.
Returns output : qutip.Odedata
An instance of the class qutip.Odedata, which contains either an array of expecta-
tion values for the times specified by tlist.
f_energies : list
Floquet energies.
t : float
The time at which to evaluate the floquet modes.
H : qutip.Qobj
system Hamiltonian, time-dependent with period T
args : dictionary
dictionary with variables required to evaluate H
T : float
The period of the time-dependence of the hamiltonian.
Returns output : list of kets
The Floquet modes as kets at time t
floquet_modes_table(f_modes_0, f_energies, tlist, H, T, args=None)
Pre-calculate the Floquet modes for a range of times spanning the floquet period. Can later be used as a table to
look up the floquet modes for any time.
floquet_modes_t_lookup(f_modes_table_t, t, T)
Lookup the floquet mode at time t in the pre-calculated table of floquet modes in the first period of the time-
dependence.
floquet_states_t(f_modes_0, f_energies, t, H, T, args=None)
Evaluate the floquet states at time t.
Returns a list of the wavefunctions.
floquet_wavefunction_t(f_modes_0, f_energies, f_coeff, t, H, T, args=None)
Evaluate the wavefunction for a time t using the Floquet states decompositon.
Returns the wavefunction.
floquet_state_decomposition(f_modes_0, f_energies, psi0)
Decompose the wavefunction psi in the Floquet states, return the coefficients in the decomposition as an array
of complex amplitudes.
Correlation Functions
Parameters H : qutip.Qobj
system Hamiltonian.
wlist : list / array
list of frequencies for .
c_op_list : list of qutip.Qobj
list of collapse operators.
a_op : qutip.Qobj
operator A.
b_op : qutip.Qobj
operator B.
Returns spectrum: *array* :
An array with spectrum S() for the frequencies specified in wlist.
Exponential Series
Steady-state Solvers
Module contains functions for iteratively solving for the steady state density matrix of an open qunatum system defind
by a Louvillian.
steady(L, maxiter=100, tol=1e-06, method=solve)
Steady state for the evolution subject to the supplied Louvillian.
Parameters L : qobj
Liouvillian superoperator.
maxiter : int
Maximum number of iterations to perform, default = 100.
tol : float
Tolerance used by iterative solver, default = 1e-6.
method : str
Method for solving linear equations. Direct solver solve (default) or iterative biconju-
gate gradient method bicg.
Returns ket : qobj
Ket vector for steady state.
Notes
Uses the inverse power method. See any Linear Algebra book with an iterative methods section.
steadystate(H, c_op_list, maxiter=100, tol=1e-06, method=solve)
Calculates the steady state for the evolution subject to the supplied Hamiltonian and list of collapse operators.
This function builds the Louvillian from the Hamiltonaian and calls the qutip.steady.steady function.
Parameters H : qobj
Hamiltonian operator.
c_op_list : list
A list of collapse operators.
maxiter : int
Maximum number of iterations to perform, default = 100.
tol : float
Tolerance used by iterative solver, default = 1e-6.
method : str
Method for solving linear equations. Direct solver solve (default) or iterative biconju-
gate gradient method bicg.
Returns ket : qobj
Notes
Uses the inverse power method. See any Linear Algebra book with an iterative methods section.
Fileio Functions
file_data_read(filename, sep=None)
Retrieves an array of data from the requested file.
Parameters filename : str
Name of file containing reqested data.
Returns data : array_like
Data from selected file.
file_data_store(filename, data, numtype=complex, numformat=decimal, sep=, )
Stores a matrix of data to a file to be read by an external program.
Parameters filename : str
Name of data file to be stored, including extension.
data: array_like :
Data to be written to file.
numtype : str {complex, real}
Type of numerical data.
numformat : str {decimal,exp}
Format for written data.
qload(name)
Loads data file from file named filename.qu in current directory.
Parameters name : str
Name of data file to be loaded.
Returns qobject : instance / array_like
Object retrieved from requested file.
qsave(data, name=qutip_data)
Saves given data to file named filename.qu in current directory.
Parameters data : instance/array_like
Input Python object to be stored.
filename : str
Name of output data file.
Entropy Functions
entropy_linear(rho)
Linear entropy of a density matrix.
Parameters rho : qobj
sensity matrix or ket/bra vector.
Returns entropy : float
Linear entropy of rho.
Examples
>>> rho=0.5*fock_dm(2,0)+0.5*fock_dm(2,1)
>>> entropy_linear(rho)
0.5
Examples
>>> rho=0.5*fock_dm(2,0)+0.5*fock_dm(2,1)
>>> entropy_vn(rho,2)
1.0
cnot()
Quantum object representing the CNOT gate.
Returns cnot_gate : qobj
Quantum object representation of CNOT gate
Examples
>>> cnot()
Quantum object: dims = [[2, 2], [2, 2]], shape = [4, 4], type = oper, isHerm = True
Qobj data =
[[ 1.+0.j 0.+0.j 0.+0.j 0.+0.j]
[ 0.+0.j 1.+0.j 0.+0.j 0.+0.j]
fredkin()
Quantum object representing the Fredkin gate.
Returns fred_gate : qobj
Quantum object representation of Fredkin gate.
Examples
>>> fredkin()
Quantum object: dims = [[2, 2, 2], [2, 2, 2]], shape = [8, 8], type = oper, isHerm = True
Qobj data =
[[ 1.+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 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 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 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 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 0.+0.j]
[ 0.+0.j 0.+0.j 0.+0.j 0.+0.j 0.+0.j 1.+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]]
phasegate(theta)
Returns quantum object representing the phase shift gate.
Parameters theta : float
Phase rotation angle.
Returns phase_gate : qobj
Quantum object representation of phase shift gate.
Examples
>>> phasegate(pi/4)
Quantum object: dims = [[2], [2]], shape = [2, 2], type = oper, isHerm = False
Qobj data =
[[ 1.00000000+0.j 0.00000000+0.j ]
[ 0.00000000+0.j 0.70710678+0.70710678j]]
snot()
Quantum object representing the SNOT (Hadamard) gate.
Returns snot_gate : qobj
Quantum object representation of SNOT (Hadamard) gate.
Examples
>>> snot()
Quantum object: dims = [[2], [2]], shape = [2, 2], type = oper, isHerm = True
Qobj data =
[[ 0.70710678+0.j 0.70710678+0.j]
[ 0.70710678+0.j -0.70710678+0.j]]
swap()
Quantum object representing the SWAP gate.
Returns swap_gate : qobj
Quantum object representation of SWAP gate
Examples
>>> swap()
Quantum object: dims = [[2, 2], [2, 2]], shape = [4, 4], type = oper, isHerm = True
Qobj data =
[[ 1.+0.j 0.+0.j 0.+0.j 0.+0.j]
[ 0.+0.j 0.+0.j 1.+0.j 0.+0.j]
[ 0.+0.j 1.+0.j 0.+0.j 0.+0.j]
[ 0.+0.j 0.+0.j 0.+0.j 1.+0.j]]
toffoli()
Quantum object representing the Toffoli gate.
Returns toff_gate : qobj
Quantum object representation of Toffoli gate.
Examples
>>> toffoli()
Quantum object: dims = [[2, 2, 2], [2, 2, 2]], shape = [8, 8], type = oper, isHerm = True
Qobj data =
[[ 1.+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 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 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 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 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 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]
[ 0.+0.j 0.+0.j 0.+0.j 0.+0.j 0.+0.j 0.+0.j 1.+0.j 0.+0.j]]
Examples
>>> x=fock_dm(5,3)
>>> y=coherent_dm(5,1)
>>> fidelity(x,y)
0.24104350624628332
Examples
>>> x=fock_dm(5,3)
>>> y=coherent_dm(5,1)
>>> tracedist(x,y)
0.9705143161472971
This module is a collection of random state and operator generators. The sparsity of the ouput Qobjs is controlled by
varing the density parameter.
rand_dm(N, density=0.75, pure=False, dims=None)
Creates a random NxN density matrix.
Parameters N : int
Shape of output density matrix.
density : float
Density etween [0,1] of output density matrix.
Returns oper : qobj
NxN density matrix quantum operator.
Notes
For small density matricies, choosing a low density will result in an error as no diagonal elements will be
generated such that T r() = 1.
rand_herm(N, density=0.75, dims=None)
Creates a random NxN sparse Hermitian quantum object.
Uses H = X + X + where X is a randomly generated quantum operator with a given density.
Parameters N : int
Shape of output quantum operator.
density : float
Density etween [0,1] of output Hermitian operator.
Returns oper : qobj
NxN Hermitian quantum operator.
rand_ket(N, density=1, dims=None)
Creates a random Nx1 sparse ket vector.
Parameters N : int
Number of rows for output quantum operator.
density : float
Density between [0,1] of output ket state.
Returns oper : qobj
Nx1 ket state quantum operator.
rand_unitary(N, density=0.75, dims=None)
Creates a random NxN sparse unitary quantum object.
Uses exp(iH) where H is a randomly generated Hermitian operator.
Parameters N : int
Shape of output quantum operator.
density : float
Density between [0,1] of output Unitary operator.
Returns oper : qobj
NxN Unitary quantum operator.
Miscellaneous
about()
About box for qutip. Gives version numbers for QuTiP, NumPy, SciPy, and MatPlotLib. GUI version requires
PySide or PyQt4.
clebsch(j1, j2, j3, m1, m2, m3)
Calculates the Clebsch-Gordon coefficient for coupling (j1,m1) and (j2,m2) to give (j3,m3).
Parameters j1 : float
Total angular momentum 1.
j2 : float
Total angular momentum 2.
j3 : float
Total angular momentum 3.
m1 : float
z-component of angular momentum 1.
m2 : float
z-component of angular momentum 2.
m3 : float
z-component of angular momentum 3.
Returns cg_coeff : float
Requested Clebsch-Gordan coefficient.
demos()
Calls the demos scripts via a GUI window if PySide or PyQt4 are avaliable. Otherwise, a commandline interface
is given in the terminal.
hinton(rho)
Draws a Hinton diagram for visualizing a density matrix.
Parameters rho : qobj
Input density matrix.
Raises ValueError :
Input argument is not a quantum object.
orbital(theta, phi, *args)
Calculates an angular wave function on a sphere. psi = orbital(theta,phi,ket1,ket2,...)
calculates
P the angular wave function on a sphere at the mesh of points defined by theta and phi which is
lm clm Ylm (theta, phi) where Clm are the coefficients specified by the list of kets. Each ket has 2l+1 compo-
nents for some integer l.
Parameters theta : list/array
Polar angles
phi : list/array
Azimuthal angles
args : list/array
list of ket vectors.
Returns array for angular wave function :
parfor(func, frange)
Executes a single-variable function in parallel.
Parallel execution of a for-loop over function func for a single variable frange.
Parameters func: function_type :
A single-variable function.
frange: array_type :
An array of values to be passed on to func.
Returns ans : list
A list with length equal to number of input parameters containting the output from
func. In general, the ordering of the output variables will not be in the same order as
frange.
.. note:: :
Multiple values can be passed into the parfor function using Pythons builtin zip com-
mand, or using multidimensional lists or arrays.
rhs_generate(H, psi0, tlist, c_ops, e_ops, ntraj=500, args={}, options=<qutip.Odeoptions.Odeoptions
instance at 0x10f25cea8>, solver=me, name=None)
Used to generate the Cython functions for solving the dynamics of a given system before using the parfor
function.
Parameters H : qobj
System Hamiltonian.
psi0 : qobj
Initial state vector
tlist : array_like
Times at which results are recorded.
ntraj : int
Number of trajectories to run.
c_ops : array_like
list or array of collapse operators.
e_ops : array_like
list or array of operators for calculating expectation values.
args : dict
Arguments for time-dependent Hamiltonian and collapse operator terms.
options : Odeoptions
Instance of ODE solver options.
solver: str :
String indicating which solver me or mc
name: str :
Name of generated RHS
simdiag(ops, evals=True)
Simulateous diagonalization of communting Hermitian matricies.
Parameters ops : list/array
list or array of qobjs representing commuting Hermitian operators.
Returns eigs : tuple
Tuple of arrays representing eigvecs and eigvals of quantum objects :
corresponding to simultaneous eigenvectors and eigenvalues for each operator. :
sphereplot(theta, phi, values, save=False)
Plots a matrix of values on a sphere
Parameters theta : float
Angle with respect to z-axis
phi : float
Angle in x-y plane :
values : array
Data set to be plotted
save : bool {False , True}
Whether to save the figure or not
Returns Plots figure, returns nonthing. :
SEVEN
CHANGE LOG
QuTiP now includes solvers for both Floquet and Bloch-Redfield master equations.
The Lindblad master equation and monte-carlo solvers allow for time-dependent collapse operators.
It is possible to automatically compile time-dependent problems into c-code using Cython (if installed).
Python functions can be used to create arbitrary time-dependent Hamiltonians and collapse operators.
Solvers now return Odedata objects containing all simulation results and parameters, simplifying the saving of
simulation results.
mesolve and mcsolve can reuse Hamiltonian data when only the initial state, or time-dependent arguments, need
to be changed.
QuTiP includes functions for creating random quantum states and operators.
The generation and manipulation of quantum objects is now more efficient.
Quantum objects have basis transformation and matrix element calulations as built-in methods.
The quantum object eigensolver can use sparse solvers.
The partial-trace (ptrace) function is up to 20x faster.
The Bloch sphere can now be used with the Matplotlib animation function, and embedded as a subplot in a
figure.
QuTiP has built-in functions for saving quantum objects and data arrays.
The steady-state solver has been further optimized for sparse matrices, and can handle much larger system
Hamiltonians.
The steady-state solver can use the iterative bi-conjugate gradient method instead of a direct solver.
There are three new entropy functions for concurrence, mutual information, and conditional entropy.
Correlation functions have been combined under a single function.
The operator norm can now be set to trace, Frobius, one, or max norm.
Global QuTiP settings can now be modified.
QuTiP includes a collection of unit tests for verifying the installation.
Demos window now lets you copy and paste code from each example.
195
QuTiP: The Quantum Toolbox in Python, Release 2.0
SVN-1218: Fixed issue where Monte-Carlo states were not output properly.
Initial release.
EIGHT
DEVELOPERS
8.2 Contributors
Note: Anyone is free to contribute to QuTiP. If you are interested in helping, please let us know!
199
QuTiP: The Quantum Toolbox in Python, Release 2.0
NINE
genindex
modindex
search
201
QuTiP: The Quantum Toolbox in Python, Release 2.0
q
qutip, 191
qutip.bloch_redfield, 180
qutip.correlation, 182
qutip.entropy, 187
qutip.essolve, 184
qutip.expect, 174
qutip.fileio, 186
qutip.floquet, 180
qutip.gates, 187
qutip.istests, 172
qutip.mcsolve, 178
qutip.mesolve, 176
qutip.metrics, 189
qutip.operators, 167
qutip.qstate, 167
qutip.rand, 190
qutip.states, 164
qutip.steady, 185
qutip.superoperator, 173
qutip.tensor, 174
qutip.three_level_atom, 175
qutip.wigner, 175
203