Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/antimonit/quantum
Kotlin framework for writing quantum algorithms using QASM-like syntax
https://github.com/antimonit/quantum
jvm kotlin quantum quantum-algorithms quantum-computing simulation
Last synced: 3 months ago
JSON representation
Kotlin framework for writing quantum algorithms using QASM-like syntax
- Host: GitHub
- URL: https://github.com/antimonit/quantum
- Owner: Antimonit
- License: apache-2.0
- Created: 2019-09-25T13:20:25.000Z (over 5 years ago)
- Default Branch: master
- Last Pushed: 2024-06-25T12:53:02.000Z (7 months ago)
- Last Synced: 2024-10-12T15:42:08.003Z (3 months ago)
- Topics: jvm, kotlin, quantum, quantum-algorithms, quantum-computing, simulation
- Language: Kotlin
- Homepage:
- Size: 313 KB
- Stars: 13
- Watchers: 2
- Forks: 1
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# Quantum computing in Kotlin
[![Maven Central](https://img.shields.io/maven-central/v/io.github.antimonit/quantum.svg?label=Maven%20Central)](https://central.sonatype.com/artifact/io.github.antimonit/quantum)
[![License](https://img.shields.io/badge/License-Apache%202.0-blue.svg)](https://opensource.org/licenses/Apache-2.0)
[![CircleCI](https://circleci.com/gh/Antimonit/Quantum/tree/master.svg?style=shield)](https://circleci.com/gh/Antimonit/Quantum/tree/main)
[![Code Coverage](https://codecov.io/gh/Antimonit/Quantum/branch/master/graph/badge.svg)](https://codecov.io/gh/Antimonit/Quantum)Kotlin framework for writing quantum algorithms using QASM-like syntax.
Because the code is still compiled by Kotlin compiler we are not limited by QASM language features.
We can use classical variables, loops, functions, etc. There is no need for binary-controlled gates
as we can utilize standard `if` condition.# Examples
## Quantum Teleportation
![Quantum Teleportation](images/Quantum%20Teleportation.svg)
```kotlin
val message = ZERO // or ONE or random()program(Register(message, ZERO, ZERO)) {
// Entangle qubits q1 and q2 to form a fully entangled bell state
Hadamard[2]
CNot[2, 1]// Entangle message/state held by q0 with the rest.
CNot[0, 1]
Hadamard[0]// Measuring the first two qubits will change the state of the third qubit because
// of the entanglement. This will teleport the message from the first qubit to the
// third qubit.
val (secret, shared) = measureAndCollapse(0, 1)// The last qubit can be in one of four superpositions now. We can use qubits
// measured in the previous step to conditionally apply some gates to put it
// into one specific superposition.
if (shared == ONE) X[2]
if (secret == ONE) Z[2]
}// At this point the third qubit will be in the same superposition as the message
// defined at the beginning.
```## Grover's Algorithm
![Grover's Algorithm](images/Grover's%20Algorithm.svg)
```kotlin
val oracle: Gate = oracleGate(ONE, ONE, ZERO)program(3) {
// Initialization
step { H[0]; H[1]; H[2] }repeat(2) {
// Oracle
oracle[0, 1, 2]// Diffusion
step { H[0]; H[1]; H[2] }
step { X[0]; X[1]; X[2] }
step { C(C(Z))[0, 1, 2] }
step { X[0]; X[1]; X[2] }
step { H[0]; H[1]; H[2] }
}// Measurement
}.measure() // Returns [ONE, ONE, ZERO] with high probability
```## Quantum Adder
![Quantum Adder](images/Quantum%20Adder.svg)
```kotlin
// We can even utilize entanglement to calculate two separate additions at the same time.program(4) {
// Prepare input
H[0]
CNot[0, 1]// Encode two combinations of input into an entangled pair.
// By passing ∣00⟩ + ∣11⟩ / sqrt(2) as input to the adder
// we are essentially calculating both 0 + 0 and 1 + 1.// Full Adder
CCNot[0, 1, 3]
CNot[0, 1]
CCNot[1, 2, 3]
CNot[1, 2]
CNot[0, 1]
}// At this point it is equally likely to observe result of 0 + 0 = 00 and 1 + 1 = 10.
```# Building blocks
## [Complex numbers](quantum/src/main/kotlin/me/khol/quantum/math/Complex.kt)
Details
Complex numbers can be created either from cartesian coordinates:
```kotlin
Complex(re: Number = 0, im: Number = 0)
```
or from polar coordinates:
```kotlin
Complex.fromPolar(theta: Number, radius: Number = 1)
```
For most commonly used Complex numbers there are three predefined values:
```kotlin
val ONE = Complex(1, 0)
val ZERO = Complex(0, 0)
val I = Complex(0, 1)
```
There are several overloaded operators that allow for easier addition, subtraction, multiplication
and division of complex numbers.
## [Matrices](quantum/src/main/kotlin/me/khol/quantum/math/Matrix.kt)
Details
Matrices can be created from number of rows and cols and 1D array of complex numbers:
```kotlin
Matrix(rows: Int, cols: Int, vararg values: Complex)
```
or directly from 2D array of complex numbers:
```kotlin
Matrix(m: List>)
```
Identity matrix n×n can be created using:
```kotlin
Matrix.identity(size: Int)
```
There are many overloaded operators that allow for easier *addition* and *subtraction* of two
matrices, *multiplication* and *division* by a number, complex number or another matrix,
*conjugate transpose* and *tensor* product.
## [Qubits](quantum/src/main/kotlin/me/khol/quantum/Qubit.kt)
Qubits are defined by two complex numbers which define their probability amplitudes:
```kotlin
Qubit(alpha: Complex, beta: Complex)
```
Two commonly used qubit states are predefined:
```kotlin
val ZERO = Qubit(Complex.ONE, Complex.ZERO)
val ONE = Qubit(Complex.ZERO, Complex.ONE)
```Qubits can be converted to *bra* ⟨φ∣ or *ket* ∣φ⟩ matrices and define *dot*, *cross*
and *tensor* products.> Because two qubits may have different alpha and beta values although they represent the same
physical state they can be *normalized* so that two qubits can be programmatically compared.
This cannot be observed in the physical world but is invaluable in testing.## [Registers](quantum/src/main/kotlin/me/khol/quantum/Register.kt)
Registers are created from a list of qubits:
```kotlin
Register(vararg qubits: Qubit)
```In contrast to a simple `List` that can only hold independent qubits, a `Register` can
also hold *entangled* qubits, such as *Bell states*. This is crucial for creation of more
complex algorithms that depend on entanglement.> Just like single qubits, registers can be *normalized* for testing purposes.
## [Gates](quantum/src/main/kotlin/me/khol/quantum/gate/Gate.kt)
Most of the commonly used gates are already predefined:| [Identity][_Identity] | [X (Not)][_X] | [Y][_Y] | [Z][_Z] | [S][_S] | [T][_T] |
|--------------------------------------------------|--------------------------------------------------|---------------------------------------------------|---------------------------------------------------|--------------------------------------------------|-----------------------------------------------------------|
| $`\begin{bmatrix} 1 & 0 \\ 0 & 1 \end{bmatrix}`$ | $`\begin{bmatrix} 0 & 1 \\ 1 & 0 \end{bmatrix}`$ | $`\begin{bmatrix} 0 & -i \\ i & 0 \end{bmatrix}`$ | $`\begin{bmatrix} 1 & 0 \\ 0 & -1 \end{bmatrix}`$ | $`\begin{bmatrix} 1 & 0 \\ 0 & i \end{bmatrix}`$ | $`\begin{bmatrix} 1 & 0 \\ 0 & e^{i\pi/4} \end{bmatrix}`$ || [Hadamard][_Hadamard] | [Rx][_Rx] | [Ry][_Ry] | [Rz][_Rz] | [Phase][_Phase] |
|----------------------------------------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------|-----------------------------------------------------------------------------------------------------------------------------------|-------------------------------------------------------------------------------------------|------------------------------------------------------------|
| $`\frac{1}{\sqrt{2}} \begin{bmatrix} 1 & 1 \\ 1 & -1 \end{bmatrix}`$ | $`\begin{bmatrix} cos(\frac{\theta}{2}) & -i sin(\frac{\theta}{2}) \\ -i sin(\frac{\theta}{2}) & cos(\frac{\theta}{2}) \end{bmatrix}`$ | $`\begin{bmatrix} cos(\frac{\theta}{2}) & -sin(\frac{\theta}{2}) \\ sin(\frac{\theta}{2}) & cos(\frac{\theta}{2}) \end{bmatrix}`$ | $`\begin{bmatrix} e^{-i\frac{\theta}{2}} & 0 \\ 0 & e^{i\frac{\theta}{2}} \end{bmatrix}`$ | $`\begin{bmatrix} 1 & 0 \\ 0 & e^{i\theta} \end{bmatrix}`$ || [CNot (CX)][_CNot] | [Swap][_Swap] | [Square Root of Swap][_SwapRoot] | [Controlled][_Controlled] |
|----------------------------------------------------------------------------------------------------|----------------------------------------------------------------------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------|----------------------------------------------------------------------------------------------------------------|
| $`\begin{bmatrix} 1 & 0 & 0 & 0 \\ 0 & 1 & 0 & 0 \\ 0 & 0 & 0 & 1 \\ 0 & 0 & 1 & 0 \end{bmatrix}`$ | $`\begin{bmatrix} 1 & 0 & 0 & 0 \\ 0 & 0 & 1 & 0 \\ 0 & 1 & 0 & 0 \\ 0 & 0 & 0 & 1 \end{bmatrix}`$ | $`\begin{bmatrix} 1 & 0 & 0 & 0 \\ 0 & \frac{1}{2}(1+i) & \frac{1}{2}(1-i) & 0 \\ 0 & \frac{1}{2}(1-i) & \frac{1}{2}(1+i) & 0 \\ 0 & 0 & 0 & 1 \end{bmatrix}`$ | $`\begin{bmatrix} 1 & 0 & 0 & 0 \\ 0 & 1 & 0 & 0 \\ 0 & 0 & u_00 & u_01 \\ 0 & 0 & u_10 & u_11 \end{bmatrix}`$ || [Toffoli (CCNot)][_CCNot] | [Fredkin (CSwap)][_CSwap] |
|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| $`\begin{bmatrix} 1 & 0 & 0 & 0 & 0 & 0 & 0 & 0 \\ 0 & 1 & 0 & 0 & 0 & 0 & 0 & 0 \\ 0 & 0 & 1 & 0 & 0 & 0 & 0 & 0 \\ 0 & 0 & 0 & 1 & 0 & 0 & 0 & 0 \\ 0 & 0 & 0 & 0 & 1 & 0 & 0 & 0 \\ 0 & 0 & 0 & 0 & 0 & 1 & 0 & 0 \\ 0 & 0 & 0 & 0 & 0 & 0 & 0 & 1 \\ 0 & 0 & 0 & 0 & 0 & 0 & 1 & 0 \end{bmatrix}`$ | $`\begin{bmatrix} 1 & 0 & 0 & 0 & 0 & 0 & 0 & 0 \\ 0 & 1 & 0 & 0 & 0 & 0 & 0 & 0 \\ 0 & 0 & 1 & 0 & 0 & 0 & 0 & 0 \\ 0 & 0 & 0 & 1 & 0 & 0 & 0 & 0 \\ 0 & 0 & 0 & 0 & 1 & 0 & 0 & 0 \\ 0 & 0 & 0 & 0 & 0 & 0 & 1 & 0 \\ 0 & 0 & 0 & 0 & 0 & 1 & 0 & 0 \\ 0 & 0 & 0 & 0 & 0 & 0 & 0 & 1 \end{bmatrix}`$ |[//]: https://alexanderrodin.com/github-latex-markdown/
[_Identity]: quantum/src/main/kotlin/me/khol/quantum/gate/GateIdentity.kt
[_Hadamard]: quantum/src/main/kotlin/me/khol/quantum/gate/GateHadamard.kt
[_X]: quantum/src/main/kotlin/me/khol/quantum/gate/GateX.kt
[_Y]: quantum/src/main/kotlin/me/khol/quantum/gate/GateY.kt
[_Z]: quantum/src/main/kotlin/me/khol/quantum/gate/GateZ.kt
[_S]: quantum/src/main/kotlin/me/khol/quantum/gate/GateS.kt
[_T]: quantum/src/main/kotlin/me/khol/quantum/gate/GateT.kt
[_Rx]: quantum/src/main/kotlin/me/khol/quantum/gate/GateRx.kt
[_Ry]: quantum/src/main/kotlin/me/khol/quantum/gate/GateRy.kt
[_Rz]: quantum/src/main/kotlin/me/khol/quantum/gate/GateRz.kt
[_Phase]: quantum/src/main/kotlin/me/khol/quantum/gate/GatePhase.kt
[_Swap]: quantum/src/main/kotlin/me/khol/quantum/gate/GateSwap.kt
[_SwapRoot]: quantum/src/main/kotlin/me/khol/quantum/gate/GateSwapRoot.kt
[_CNot]: quantum/src/main/kotlin/me/khol/quantum/gate/GateCNot.kt
[_CCNot]: quantum/src/main/kotlin/me/khol/quantum/gate/GateCCNot.kt
[_CSwap]: quantum/src/main/kotlin/me/khol/quantum/gate/GateCSwap.kt
[_Controlled]: quantum/src/main/kotlin/me/khol/quantum/gate/GateControlled.ktIt should be rather straightforward to create custom gates.
Gates can be applied to a register. The size of the gate must match the number of qubits stored by
the register.
```kotlin
GateSwap * Register(Qubit.ONE, Qubit.ZERO) // Register(Qubit.ZERO, Qubit.ONE)
```
Gates that act on a single qubit can be also applied to qubits.
```kotlin
GateX * Qubit.ONE // Qubit.ZERO
```## [Programs](quantum/src/main/kotlin/me/khol/quantum/Program.kt)
Instead of manually applying gates to registers and qubits like this:
```kotlin
val bellState = CNot * Register(Hadamard * ZERO, ZERO)
```
program classes provide a less cluttered and more natural way to combine multiple gates into one,
reorder inputs of a gate, apply gates to registers with different sizes and apply multiple gates
to a register.### Precomputed Program
Pre-computes transformations of multiple gates as a standalone gate. As we apply gates
within the program, their transformation matrices are combined.
It allows us to pre-compute a part of a larger program that is executed repeatedly and
apply the result gate instead.
```kotlin
// Swap gate made using CNot gates
val swap: Gate = gate(2) {
CNot[0, 1]
CNot[1, 0]
CNot[0, 1]
}
```### Runnable Program
Applies multiple gates to a register, changing the state of its qubits with each step.```kotlin
// Fully entangled Bell state (∣00⟩ + ∣11⟩) / sqrt(2)
val bellState: Register = program(2) {
Hadamard[0]
CNot[0, 1]
}
```Compared to `gate` that does not have any state per se, `program` has a register
that we can measure at any point.```kotlin
fun measureAndCollapse(vararg qubitIndices: Int): List
```Doing so will collapse the state of specified qubits to ∣0⟩ or ∣1⟩ based on their probabilities.
Other qubits in the register entangled with any of the measured qubits will have their probabilities
updated as well to satisfy constraints imposed by entangled states before the measurement.# Download
The project is also available as a maven dependency from MavenCentral repository:
```kotlin
implementation("io.github.antimonit:quantum:1.0.1")
```