文 / 李锡涵,Google Developers Expert
本文节选自《简单粗暴 TensorFlow 2》,回复 “手册” 获取合集
我们身边的经典计算机利用比特位和逻辑门进行二进制运算。在物理硬件上,这种运算主要是通过半导体的特殊导电性质实现的。经过几十年的发展,我们已经可以在一片小小的半导体芯片上集成上亿个晶体管,从而实现高性能的经典计算。
而量子计算(Quantum Computing)旨在利用具有量子特性(例如量子态叠加和量子纠缠)的 “量子比特位” 和 “量子逻辑门” 进行计算。这种新的计算模式可以在搜索和大数分解等重要领域达成指数级的加速,让当前无法实现的一些超大规模运算成为可能,从而可能在未来深远地改变世界。在物理硬件上,这类量子运算也可以通过一些具有量子特性的结构(例如超导约瑟夫森结)实现。
不幸的是,尽管量子计算的理论已经有了比较深入的发展,可在物理硬件上,我们目前仍然造不出一台超越经典计算机的通用量子计算机 [1]。IBM 和谷歌等业界巨头在通用量子计算机的物理构建上已经取得了一些成绩,但无论是量子比特的个数还是在退相干问题的解决上,都还远无法达到实用的层级。
以上是量子计算的基本背景,接下来我们讨论量子机器学习。量子机器学习的一种最直接的思路是使用量子计算加速传统的机器学习任务,例如量子版本的 PCA、SVM 和 K-Means 算法,然而这些算法目前都尚未达到可实用的程度。我们在本章讨论的量子机器学习则采用另一种思路,即构建参数化的量子线路(Parameterized Quantum Circuits, PQCs)。PQC 可以作为深度学习模型中的层而被使用,如果我们在普通深度学习模型的基础上加入 PQC,即称为混合量子 — 经典机器学习(Hybrid Quantum-Classical Machine Learning)。这种混合模型尤其适合于量子数据集(Quantum Data)上的任务。而 TensorFlow Quantum 正是帮助我们构建这种混合量子 — 经典机器学习模型的利器。接下来,我们会对量子计算的若干基本概念进行简介,然后介绍使用 TensorFlow Quantum 和谷歌的量子计算库 Cirq 构建 PQC、将 PQC 嵌入 Keras 模型、并在量子数据集上训练混合模型的流程。
TensorFlow Quantum (见 TensorFlow DevSummit 视频)
https://www.bilibili.com/video/av95867255/
https://tensorflow.google.cn/quantum
简单教程
https://medium.com/mdr-inc/tensorflow-quantum-basic-tutorial-explained-with-supplementary-2f69011036c0
量子计算基本概念
本节将简述量子计算的一些基本概念,包括量子比特、量子门、量子线路等。
推荐阅读
如果你希望更深入地了解量子力学以及量子计算的基本原理,建议可以从以下两本书入手:
- 吴飚,简明量子力学(简洁明快的量子力学入门教程,即将由北京大学出版社出版,可先行阅读在线版本 http://www.phy.pku.edu.cn/~wubiao/pop_qm_pkupress.pdf)
- Hidary, Jack D. Quantum Computing: An Applied Approach . Cham: Springer International Publishing, 2019. https://doi.org/10.1007/978-3-030-23922-0. (注重代码实操的量子计算教程,GitHub上有配套源码:https://github.com/JackHidary/quantumcomputingbook)
量子比特
同时,我们可以用布洛赫球面(Bloch Sphere)来形象地展示单个量子比特的状态。球面的最顶端为 ,最底端为 ,而从原点到球面上任何一点的单位向量都可以是一个量子比特的状态。
布洛赫球面(Bloch Sphere):其中 Z 轴正负方向的量子态分别为基本态 和 ,X 轴正负方向的量子态分别为 和 ,Y 轴正负方向的量子态分别为 和
值得特别注意的是,尽管量子比特 可能的状态相当之多,但一旦我们对其进行观测,则其状态会立即坍缩 [2] 到 和 这两个基本状态中的一个,其概率分别为 和 。
量子逻辑门
在二进制的经典计算机中,我们有 AND(与)、OR(或)、NOT(非)等逻辑门,对输入的比特状态进行变换并输出。在量子计算机中,我们同样有量子逻辑门(Quantum Logic Gate,或简称“量子门”),对量子状态进行变换并输出。如果我们使用向量化的语言来表述量子状态,则量子逻辑门可以看作是一个对状态向量进行变换的矩阵。
例如,量子非门可以表述为 ,于是当我们将量子非门作用于基本态 时,我们得到 。量子门也可以作用在叠加态,例如 (这说明量子非门没能改变量子态 的状态。事实上,量子非门 相当于在布洛赫球面上将量子态绕 X 轴旋转 180 度。而 就在 X 轴上,所以没有变化)。量子与门和或门 [3] 由于涉及到多个量子位而稍显复杂,但同样可以通过尺寸更大的矩阵实现。
量子线路
当我们将量子比特以及量子逻辑门按顺序标记在一条或多条平行的线条上时,就构成了量子线路(Quantum Circuit,或称量子电路)。例如,对于我们在上一节讨论的,使用量子非门 对基本态 进行变换的过程,我们可以写出量子线路如下:
一个简单的量子线路
接下来,我们考虑将上图中量子线路的量子非门 换为阿达马门 :
阿达马门对应的矩阵表示为 ,于是我们可以计算出变换后的量子态为 。这是一个 和 的叠加态,在观测后会坍缩到基本态,其概率分别为 。也就是说,这个量子线路的观测结果类似于扔硬币。假若观测 20 次,则大约 10 次的结果是 ,10 次的结果是 。
实例:使用Cirq建立简单的量子线路
Cirq (https://cirq.readthedocs.io/) 是谷歌主导的开源量子计算库,可以帮助我们方便地建立量子线路并模拟测量结果(我们在下一节介绍 TensorFlow Quantum 的时候还会用到它)。Cirq 是一个 Python 库,可以使用pip install cirq
进行安装。以下代码实现了上节所建立的两个简单的量子线路,并分别进行了 20 次的模拟测量。
import cirq
q = cirq.LineQubit(0) # 实例化一个量子比特
simulator = cirq.Simulator() # 实例化一个模拟器
X_circuit = cirq.Circuit( # 建立一个包含量子非门和测量的量子线路
cirq.X(q),
cirq.measure(q)
)
print(X_circuit) # 在终端可视化输出量子线路
# 使用模拟器对该量子线路进行20次的模拟测量
result = simulator.run(X_circuit, repetitions=20)
print(result) # 输出模拟测量结果
H_circuit = cirq.Circuit( # 建立一个包含阿达马门和测量的量子线路
cirq.H(q),
cirq.measure(q)
)
print(H_circuit)
result = simulator.run(H_circuit, repetitions=20)
print(result)
结果如下:
0: ───X───M───
0=11111111111111111111
0: ───H───M───
0=00100111001111101100
可见第一个量子线路的测量结果始终为 1,第二个量子态的 20 次测量结果中有 9 次是0,11 次是 1(如果你多运行几次,会发现 0 和 1 出现的概率趋近于 1/2。可见结果符合我们在上节中的分析。
混合量子 — 经典机器学习
本节介绍混合量子 — 经典机器学习的基本概念,以及使用 TensorFlow Quantum 建立此类模型的方法。
在混合量子 — 经典机器学习过程中,我们使用量子数据集训练混合量子 — 经典模型。混合量子 — 经典模型的前半部分是量子模型(即参数化的量子线路)。量子模型接受量子数据集作为输入,对输入使用量子门进行变换,然后通过测量转换为经典数据。测量后的经典数据输入经典模型,并使用常规的损失函数计算模型的损失值。最后,基于损失函数的值计算模型参数的梯度并更新模型参数。这一过程不仅包括经典模型的参数,也包括量子模型的参数。具体流程如下图所示。
经典机器学习(上图)与混合量子 — 经典机器学习(下图)的流程对比
TensorFlow Quantum 即是一个与 TensorFlow Keras 结合紧密的,可快速建立混合量子 — 经典机器学习模型的开源库,可以使用pip install tensorflow-quantum
进行安装。
后文示例均默认使用以下代码导入 TensorFlow、TensorFlow Quantum 和 Cirq:
import tensorflow as tf
import tensorflow_quantum as tfq
import cirq
推荐阅读
Broughton, Michael, Guillaume Verdon, Trevor McCourt, Antonio J. Martinez, Jae Hyeon Yoo, Sergei V. Isakov, Philip Massey, et al. “ TensorFlow Quantum: A Software Framework for Quantum Machine Learning. (http://arxiv.org/abs/2003.02989) ” ArXiv:2003.02989 [Cond-Mat, Physics:Quant-Ph], March 5, 2020. (TensorFlow Quantum 白皮书)
量子数据集与带参数的量子门
以有监督学习为例,经典数据集由经典数据和标签组成。经典数据中的每一项是一个由不同特征组成的向量。我们可以将经典数据集写作 ,其中 。量子数据集同样由数据和标签组成,而数据中的每一项是一个量子态。以前节单量子比特的量子态为例,我们可以将每一项数据写作 。在具体实现上,我们可以通过量子线路来生成量子数据。也就是说,每一项数据 都对应着一个量子线路。例如,我们可以通过以下代码,使用 Cirq 生成一组量子数据:
q = cirq.GridQubit(0, 0)
q_data = []
for i in range(100):
x_i = cirq.Circuit(
cirq.rx(np.random.rand() * np.pi)(q)
)
q_data.append(x_i)
在这一过程中,我们使用了一个带参数的量子门 cirq.rx(angle)(q)
。和之前我们使用的量子门 cirq.X(q)
, cirq.H(q)
不同的是,这个量子门多了一个参数 angle
,表示将量子比特 q
绕布洛赫球面的 x 轴旋转 angle
角度(弧度制)。以上代码生成了 100 项量子数据,每项数据是从基本态
开始绕布洛赫球面的 x 轴随机旋转
弧度所变换而来的量子态。量子数据集在不少量子相关的领域(如化学、材料科学、生物学和药物发现等)都有应用。
当我们要将量子数据集作为 Keras 的输入时,可以使用 TensorFlow Quantum 的 convert_to_tensor
方法,将量子数据集转换为张量:
q_data = tfq.convert_to_tensor(q_data)
dtype
)需要为
tf.dtypes.string
。
当我们在建立量子线路时使用了带参数的量子门,且该参数可以自由调整时,我们就称这样的量子线路为参数化的量子线路。Cirq 支持结合 SymPy 这一 Python 下的符号运算库实现参数化的量子线路,示例如下:
import sympy
theta = sympy.Symbol('theta')
q_model = cirq.Circuit(cirq.rx(theta)(q))
在上面的代码中,我们建立了如下图所示的量子线路。该量子线路可以将任意输入量子态
绕布洛赫球面的x轴逆时针旋转
度,其中
是使用 sympy.Symbol
声明的符号变量(即参数)。
参数化的量子线路示例
将参数化的量子线路嵌入机器学习模型
通过 TensorFlow Quantum,我们可以轻松地将参数化的量子线路以 Keras 层的方式嵌入 Keras 模型。例如对于前节建立的参数化的量子线路 q_model
,我们可以使用 tfq.layers.PQC
将其直接作为一个 Keras 层使用。
q_layer = tfq.layers.PQC(q_model, cirq.Z(q))
expectation_output = q_layer(q_data_input)
tfq.layers.PQC
的第一个参数为使用 Cirq 建立的参数化的量子线路,第二个参数为测量方式,此处使用
cirq.Z(q)
在布洛赫球面的 Z 轴进行测量。
以上代码也可直接写作:
expectation_output = tfq.layers.PQC(q_model, cirq.Z(q))(q_data_input)
实例:对量子数据集进行二分类
在以下代码中,我们首先建立了一个量子数据集,其中一半的数据项为基本态 绕布洛赫球面的 x 轴逆时针旋转 弧度(即 ),另一半则为 弧度(即 )。所有的数据均加入了绕x,y轴方向旋转的,标准差为 的高斯噪声。对于这个量子数据集,如果不加变换而直接测量,则所有数据都会和抛硬币一样等概率随机坍缩到基本态 和 ,从而无法区分。
为了区分这两类数据,我们接下来建立了一个量子模型,这个模型将单比特量子态绕布洛赫球面的 x 轴逆时针旋转 弧度。变换过后量子态数据的测量值送入“全连接层+softmax”的经典机器学习模型,并使用交叉熵作为损失函数。模型训练过程会自动同时调整量子模型中 的值和全连接层的权值,使得整个混合量子-经典机器学习模型的准确度较高。
import cirq
import sympy
import numpy as np
import tensorflow as tf
import tensorflow_quantum as tfq
q = cirq.GridQubit(0, 0)
# 准备量子数据集(q_data, label)
add_noise = lambda x: x + np.random.normal(0, 0.25 * np.pi)
q_data = tfq.convert_to_tensor(
[cirq.Circuit(
cirq.rx(add_noise(0.5 * np.pi))(q),
cirq.ry(add_noise(0))(q)
) for _ in range(100)] +
[cirq.Circuit(
cirq.rx(add_noise(1.5 * np.pi))(q),
cirq.ry(add_noise(0))(q)
) for _ in range(100)]
)
label = np.array([0] * 100 + [1] * 100)
# 建立参数化的量子线路(PQC)
theta = sympy.Symbol('theta')
q_model = cirq.Circuit(cirq.rx(theta)(q))
# 建立量子层和经典全连接层
q_layer = tfq.layers.PQC(q_model, cirq.Z(q))
dense_layer = tf.keras.layers.Dense(2, activation=tf.keras.activations.softmax)
# 使用Keras建立训练流程。量子数据首先通过PQC,然后通过经典的全连接模型
q_data_input = tf.keras.Input(shape=() ,dtype=tf.dtypes.string)
expectation_output = q_layer(q_data_input)
classifier_output = dense_layer(expectation_output)
model = tf.keras.Model(inputs=q_data_input, outputs=classifier_output)
# 编译模型,指定优化器、损失函数和评估指标,并进行训练
model.compile(
optimizer=tf.keras.optimizers.SGD(learning_rate=0.01),
loss=tf.keras.losses.sparse_categorical_crossentropy,
metrics=[tf.keras.metrics.sparse_categorical_accuracy]
)
model.fit(x=q_data, y=label, epochs=200)
# 输出量子层参数(即theta)的训练结果
print(q_layer.get_weights())
输出:
...
200/200 [==============================] - 0s 165us/sample - loss: 0.1586 - sparse_categorical_accuracy: 0.9500
[array([-1.5279944], dtype=float32)]
可见,通过训练,模型在训练集上可以达到95%的准确率, 。而当 时,恰好可以使得两种类型的数据分别接近基本态 和 ,从而达到最易区分的状态。
[1] 此手册的行文时间为公元2020年,如果你来自未来,请理解作者的时代局限性。
[2] “坍缩”一词多用于量子观测的哥本哈根诠释,除此以外还有多世界理论等。此处使用“坍缩”一词仅是方便表述。
[3] 其实更常见的基础二元量子门是“量子选择非门”(CNOT)和“交换门”(SWAP)。
[4] 这种矩阵被称之为“幺正矩阵”或“酉矩阵”。
《简单粗暴 TensorFlow 2 》目录
TensorFlow Quantum: 混合量子 — 经典机器学习 (本文)