Langkau ke kandungan utama

Kompilasi kuantum hampiran untuk litar evolusi masa

Anggaran penggunaan: Lima minit pada pemproses Eagle (NOTA: Ini adalah anggaran sahaja. Masa jalan sebenar anda mungkin berbeza.)

Latar belakang

Tutorial ini menunjukkan cara melaksanakan Kompilasi Kuantum Hampiran menggunakan rangkaian tensor (AQC-Tensor) dengan Qiskit untuk meningkatkan prestasi litar kuantum. Kita menerapkan AQC-Tensor dalam konteks evolusi masa yang di-Trotter-kan untuk mengurangkan kedalaman litar sambil mengekalkan ketepatan simulasi, mengikuti rangka kerja Qiskit untuk penyediaan keadaan dan pengoptimuman. Kamu akan belajar cara membuat litar ansatz berkedalaman rendah daripada litar Trotter awal, mengoptimumkannya dengan rangkaian tensor, dan menyediakannya untuk pelaksanaan perkakasan kuantum.

Objektif utama adalah untuk mensimulasikan evolusi masa bagi model Hamiltonian dengan kedalaman litar yang dikurangkan. Ini dicapai menggunakan addon Qiskit AQC-Tensor, qiskit-addon-aqc-tensor, yang memanfaatkan rangkaian tensor, khususnya keadaan produk matriks (MPS), untuk memampatkan dan mengoptimumkan litar awal. Melalui pelarasan berulang, litar ansatz yang dimampatkan mengekalkan kesetiaan kepada litar asal sambil kekal boleh dilaksanakan pada perkakasan kuantum jangka terdekat. Maklumat lanjut boleh didapati dalam docs yang berkaitan dengan contoh mudah untuk memulakan.

Kompilasi Kuantum Hampiran amat berguna dalam simulasi kuantum yang melebihi masa koherensi perkakasan, kerana ia membolehkan simulasi yang rumit dilakukan dengan lebih cekap. Tutorial ini membimbing kamu melalui persediaan aliran kerja AQC-Tensor dalam Qiskit, merangkumi permulaan Hamiltonian, penjanaan litar Trotter, dan transpilasi litar yang dioptimumkan untuk peranti sasaran.

Keperluan

Sebelum memulakan tutorial ini, pastikan kamu telah memasang yang berikut:

  • Qiskit SDK v1.0 atau lebih baru, dengan sokongan visualisasi
  • Qiskit Runtime v0.22 atau lebih baru (pip install qiskit-ibm-runtime)
  • Addon Qiskit AQC-Tensor (pip install 'qiskit-addon-aqc-tensor[aer,quimb-jax]')

Persediaan

# Added by doQumentation — required packages for this notebook
!pip install -q matplotlib numpy qiskit qiskit-addon-aqc-tensor qiskit-addon-utils qiskit-ibm-runtime quimb rustworkx scipy
import numpy as np
import quimb.tensor
import datetime
import matplotlib.pyplot as plt

from scipy.optimize import OptimizeResult, minimize

from qiskit.quantum_info import SparsePauliOp, Pauli
from qiskit.transpiler import CouplingMap
from qiskit.transpiler.preset_passmanagers import generate_preset_pass_manager
from qiskit import QuantumCircuit
from qiskit.synthesis import SuzukiTrotter

from qiskit_addon_utils.problem_generators import (
generate_time_evolution_circuit,
)
from qiskit_addon_aqc_tensor.ansatz_generation import (
generate_ansatz_from_circuit,
)
from qiskit_addon_aqc_tensor.objective import MaximizeStateFidelity
from qiskit_addon_aqc_tensor.simulation.quimb import QuimbSimulator
from qiskit_addon_aqc_tensor.simulation import tensornetwork_from_circuit
from qiskit_addon_aqc_tensor.simulation import compute_overlap

from qiskit_ibm_runtime import QiskitRuntimeService
from qiskit_ibm_runtime import EstimatorV2 as Estimator

from rustworkx.visualization import graphviz_draw

Bahagian I. Contoh skala kecil

Bahagian pertama tutorial ini menggunakan contoh skala kecil dengan 10 tapak untuk menggambarkan proses pemetaan masalah simulasi kuantum kepada litar kuantum yang boleh dilaksanakan. Di sini, kita akan meneroka dinamik model XXZ 10-tapak, membolehkan kita membina dan mengoptimumkan litar kuantum yang mudah diurus sebelum berskala kepada sistem yang lebih besar.

Model XXZ dikaji secara meluas dalam fizik untuk mengkaji interaksi spin dan sifat magnetik. Kita menyediakan Hamiltonian dengan syarat sempadan terbuka dan interaksi antara tapak jiran yang bergantung kepada tapak di sepanjang rantai.

Model Hamiltonian dan boleh cerap

Hamiltonian untuk model XXZ 10-tapak kita ditakrifkan sebagai:

H^XXZ=i=1L1Ji,(i+1)(XiX(i+1)+YiY(i+1)+2ZiZ(i+1)),\hat{\mathcal{H}}_{XXZ} = \sum_{i=1}^{L-1} J_{i,(i+1)}\left(X_i X_{(i+1)}+Y_i Y_{(i+1)}+ 2\cdot Z_i Z_{(i+1)} \right) \, ,

di mana Ji,(i+1)J_{i,(i+1)} adalah pekali rawak yang sepadan dengan tepi (i,i+1)(i, i+1), dan L=10L=10 adalah bilangan tapak.

Dengan mensimulasikan evolusi sistem ini dengan kedalaman litar yang dikurangkan, kita dapat memperoleh pandangan tentang penggunaan AQC-Tensor untuk memampatkan dan mengoptimumkan litar.

Sediakan Hamiltonian dan boleh cerap

Sebelum kita memetakan masalah kita, kita perlu menyediakan peta gandingan, Hamiltonian, dan boleh cerap untuk model XXZ 10-tapak.

# L is the number of sites, also the length of the 1D spin chain
L = 10

# Generate the coupling map
edge_list = [(i - 1, i) for i in range(1, L)]
# Generate an edge-coloring so we can make hw-efficient circuits
even_edges = edge_list[::2]
odd_edges = edge_list[1::2]
coupling_map = CouplingMap(edge_list)

# Generate random coefficients for our XXZ Hamiltonian
np.random.seed(0)
Js = np.random.rand(L - 1) + 0.5 * np.ones(L - 1)
hamiltonian = SparsePauliOp(Pauli("I" * L))
for i, edge in enumerate(even_edges + odd_edges):
hamiltonian += SparsePauliOp.from_sparse_list(
[
("XX", (edge), Js[i] / 2),
("YY", (edge), Js[i] / 2),
("ZZ", (edge), Js[i]),
],
num_qubits=L,
)

# Generate a ZZ observable between the two middle qubits
observable = SparsePauliOp.from_sparse_list(
[("ZZ", (L // 2 - 1, L // 2), 1.0)], num_qubits=L
)

print("Hamiltonian:", hamiltonian)
print("Observable:", observable)
graphviz_draw(coupling_map.graph, method="circo")
Hamiltonian: SparsePauliOp(['IIIIIIIIII', 'IIIIIIIIXX', 'IIIIIIIIYY', 'IIIIIIIIZZ', 'IIIIIIXXII', 'IIIIIIYYII', 'IIIIIIZZII', 'IIIIXXIIII', 'IIIIYYIIII', 'IIIIZZIIII', 'IIXXIIIIII', 'IIYYIIIIII', 'IIZZIIIIII', 'XXIIIIIIII', 'YYIIIIIIII', 'ZZIIIIIIII', 'IIIIIIIXXI', 'IIIIIIIYYI', 'IIIIIIIZZI', 'IIIIIXXIII', 'IIIIIYYIII', 'IIIIIZZIII', 'IIIXXIIIII', 'IIIYYIIIII', 'IIIZZIIIII', 'IXXIIIIIII', 'IYYIIIIIII', 'IZZIIIIIII'],
coeffs=[1. +0.j, 0.52440675+0.j, 0.52440675+0.j, 1.0488135 +0.j,
0.60759468+0.j, 0.60759468+0.j, 1.21518937+0.j, 0.55138169+0.j,
0.55138169+0.j, 1.10276338+0.j, 0.52244159+0.j, 0.52244159+0.j,
1.04488318+0.j, 0.4618274 +0.j, 0.4618274 +0.j, 0.9236548 +0.j,
0.57294706+0.j, 0.57294706+0.j, 1.14589411+0.j, 0.46879361+0.j,
0.46879361+0.j, 0.93758721+0.j, 0.6958865 +0.j, 0.6958865 +0.j,
1.391773 +0.j, 0.73183138+0.j, 0.73183138+0.j, 1.46366276+0.j])
Observable: SparsePauliOp(['IIIIZZIIII'],
coeffs=[1.+0.j])

Output of the previous code cell

Dengan Hamiltonian yang telah ditakrifkan, kita boleh meneruskan untuk membina keadaan awal.

# Generate an initial state
initial_state = QuantumCircuit(L)
for i in range(L):
if i % 2:
initial_state.x(i)

Langkah 1: Petakan input klasik kepada masalah kuantum

Kini setelah kita membina Hamiltonian, yang mentakrifkan interaksi spin-spin dan medan magnet luaran yang mencirikan sistem, kita mengikuti tiga langkah utama dalam aliran kerja AQC-Tensor:

  1. Jana litar AQC yang dioptimumkan: Menggunakan Trotterisasi, kita menghampiri evolusi awal, yang kemudiannya dimampatkan untuk mengurangkan kedalaman litar.
  2. Buat litar evolusi masa selebihnya: Tangkap evolusi untuk masa yang tinggal selepas segmen awal.
  3. Gabungkan litar-litar: Gabungkan litar AQC yang dioptimumkan dengan litar evolusi selebihnya menjadi litar evolusi masa lengkap yang sedia untuk dilaksanakan.

Pendekatan ini mencipta ansatz berkedalaman rendah untuk evolusi sasaran, menyokong simulasi yang cekap dalam kekangan perkakasan kuantum jangka terdekat.

Tentukan bahagian evolusi masa untuk disimulasikan secara klasik

Matlamat kita adalah untuk mensimulasikan evolusi masa model Hamiltonian yang ditakrifkan sebelum ini menggunakan evolusi Trotter. Untuk menjadikan proses ini cekap bagi perkakasan kuantum, kita membahagikan evolusi kepada dua segmen:

  • Segmen Awal: Bahagian evolusi awal ini, dari ti=0.0t_i = 0.0 hingga tf=0.2t_f = 0.2, boleh disimulasikan dengan MPS dan boleh "dikompil" secara cekap menggunakan AQC-Tensor. Dengan menggunakan addon Qiskit AQC-Tensor, kita menjana litar yang dimampatkan untuk segmen ini, yang dirujuk sebagai aqc_target_circuit. Oleh kerana segmen ini akan disimulasikan pada simulator rangkaian tensor, kita boleh menggunakan bilangan lapisan Trotter yang lebih tinggi tanpa memberi kesan yang ketara kepada sumber perkakasan. Kita menetapkan aqc_target_num_trotter_steps = 32 untuk segmen ini.

  • Segmen Seterusnya: Bahagian evolusi yang tinggal ini, dari t=0.2t = 0.2 hingga t=0.4t = 0.4, akan dilaksanakan pada perkakasan kuantum, yang dirujuk sebagai subsequent_circuit. Memandangkan had perkakasan, kita bertujuan untuk menggunakan lapisan Trotter sesedikit mungkin untuk mengekalkan kedalaman litar yang boleh diurus. Untuk segmen ini, kita menggunakan subsequent_num_trotter_steps = 3.

Pilih masa pemisahan

Kita memilih t=0.2t = 0.2 sebagai masa pemisahan untuk mengimbangi kemampuan simulasi klasik dengan kebolehlaksanaan perkakasan. Pada awal evolusi, keterjeratan dalam model XXZ kekal cukup rendah untuk kaedah klasik seperti MPS untuk menghampiri dengan tepat.

Ketika memilih masa pemisahan, panduan yang baik adalah memilih titik di mana keterjeratan masih boleh diurus secara klasik tetapi menangkap cukup evolusi untuk menyederhanakan bahagian yang dilaksanakan pada perkakasan. Percubaan dan kesilapan mungkin diperlukan untuk mencari keseimbangan terbaik bagi Hamiltonian yang berbeza.

# Generate the AQC target circuit (initial segment)
aqc_evolution_time = 0.2
aqc_target_num_trotter_steps = 32

aqc_target_circuit = initial_state.copy()
aqc_target_circuit.compose(
generate_time_evolution_circuit(
hamiltonian,
synthesis=SuzukiTrotter(reps=aqc_target_num_trotter_steps),
time=aqc_evolution_time,
),
inplace=True,
)
# Generate the subsequent circuit
subsequent_num_trotter_steps = 3
subsequent_evolution_time = 0.2

subsequent_circuit = generate_time_evolution_circuit(
hamiltonian,
synthesis=SuzukiTrotter(reps=subsequent_num_trotter_steps),
time=subsequent_evolution_time,
)
subsequent_circuit.draw("mpl", fold=-1)

Output of the previous code cell

Untuk membolehkan perbandingan yang bermakna, kita akan menjana dua litar tambahan:

  • Litar perbandingan AQC: Litar ini berevolusi sehingga aqc_evolution_time tetapi menggunakan tempoh langkah Trotter yang sama seperti subsequent_circuit. Ia berfungsi sebagai perbandingan kepada aqc_target_circuit, menunjukkan evolusi yang akan kita perhatikan tanpa menggunakan bilangan langkah Trotter yang ditingkatkan. Kita akan merujuk litar ini sebagai aqc_comparison_circuit.

  • Litar rujukan: Litar ini digunakan sebagai garis dasar untuk mendapatkan keputusan yang tepat. Ia mensimulasikan evolusi penuh menggunakan rangkaian tensor untuk mengira hasil yang tepat, memberikan rujukan untuk menilai keberkesanan AQC-Tensor. Kita akan merujuk litar ini sebagai reference_circuit.

# Generate the AQC comparison circuit
aqc_comparison_num_trotter_steps = int(
subsequent_num_trotter_steps
/ subsequent_evolution_time
* aqc_evolution_time
)
print(
"Number of Trotter steps for comparison:",
aqc_comparison_num_trotter_steps,
)

aqc_comparison_circuit = generate_time_evolution_circuit(
hamiltonian,
synthesis=SuzukiTrotter(reps=aqc_comparison_num_trotter_steps),
time=aqc_evolution_time,
)
Number of Trotter steps for comparison: 3
# Generate the reference circuit
evolution_time = 0.4
reps = 200

reference_circuit = initial_state.copy()
reference_circuit.compose(
generate_time_evolution_circuit(
hamiltonian,
synthesis=SuzukiTrotter(reps=reps),
time=evolution_time,
),
inplace=True,
)

Jana ansatz dan parameter awal daripada litar Trotter dengan langkah yang lebih sedikit

Kini setelah kita membina empat litar kita, mari kita teruskan dengan aliran kerja AQC-Tensor. Pertama, kita membina litar yang "baik" yang mempunyai masa evolusi yang sama dengan litar sasaran, tetapi dengan langkah Trotter yang lebih sedikit (dan dengan itu lapisan yang lebih sedikit).

Kemudian kita menghantar litar yang "baik" ini ke fungsi generate_ansatz_from_circuit AQC-Tensor. Fungsi ini menganalisis ketersambungan dua-qubit litar dan mengembalikan dua perkara:

  1. Litar ansatz berparameter umum dengan ketersambungan dua-qubit yang sama seperti litar input.
  2. Parameter yang, apabila dimasukkan ke dalam ansatz, menghasilkan litar input (yang baik).

Tidak lama lagi kita akan mengambil parameter ini dan menyesuaikannya secara berulang untuk membawa litar ansatz sedekat mungkin dengan MPS sasaran.

aqc_ansatz_num_trotter_steps = 1

aqc_good_circuit = initial_state.copy()
aqc_good_circuit.compose(
generate_time_evolution_circuit(
hamiltonian,
synthesis=SuzukiTrotter(reps=aqc_ansatz_num_trotter_steps),
time=aqc_evolution_time,
),
inplace=True,
)

aqc_ansatz, aqc_initial_parameters = generate_ansatz_from_circuit(
aqc_good_circuit
)
aqc_ansatz.draw("mpl", fold=-1)

Output of the previous code cell

print(f"AQC Comparison circuit: depth {aqc_comparison_circuit.depth()}")
print(f"Target circuit: depth {aqc_target_circuit.depth()}")
print(
f"Ansatz circuit: depth {aqc_ansatz.depth()}, with {len(aqc_initial_parameters)} parameters"
)
AQC Comparison circuit: depth 36
Target circuit: depth 385
Ansatz circuit: depth 7, with 156 parameters

Pilih tetapan untuk simulasi rangkaian tensor

Di sini, kita menggunakan simulator litar keadaan produk matriks Quimb, bersama-sama dengan jax untuk menyediakan kecerunan.

simulator_settings = QuimbSimulator(
quimb.tensor.CircuitMPS, autodiff_backend="jax"
)

Seterusnya, kita membina representasi MPS bagi keadaan sasaran yang akan dihampiri menggunakan AQC-Tensor. Representasi ini membolehkan pengendalian keterjeratan yang cekap, memberikan penerangan padat keadaan kuantum untuk pengoptimuman lanjut.

aqc_target_mps = tensornetwork_from_circuit(
aqc_target_circuit, simulator_settings
)
print("Target MPS maximum bond dimension:", aqc_target_mps.psi.max_bond())

# Obtains the reference MPS, where we can obtain the exact expectation value by examining the `local_expectation``
reference_mps = tensornetwork_from_circuit(
reference_circuit, simulator_settings
)
reference_expval = reference_mps.local_expectation(
quimb.pauli("Z") & quimb.pauli("Z"), (L // 2 - 1, L // 2)
).real.item()
print("Reference MPS maximum bond dimension:", reference_mps.psi.max_bond())
Target MPS maximum bond dimension: 5
Reference MPS maximum bond dimension: 7

Perhatikan bahawa, dengan memilih bilangan langkah Trotter yang lebih besar untuk keadaan sasaran, kita secara berkesan telah mengurangkan ralat Trotter berbanding litar awal. Kita boleh menilai kesetiaan (ψ1ψ22|\langle \psi_1 | \psi_2 \rangle|^2) antara keadaan yang disediakan oleh litar awal dan keadaan sasaran untuk mengukur perbezaan ini.

good_mps = tensornetwork_from_circuit(aqc_good_circuit, simulator_settings)
starting_fidelity = abs(compute_overlap(good_mps, aqc_target_mps)) ** 2
print("Starting fidelity:", starting_fidelity)
Starting fidelity: 0.9982464959067222

Optimumkan parameter ansatz menggunakan pengiraan MPS

Dalam langkah ini, kita mengoptimumkan parameter ansatz dengan meminimumkan fungsi kos mudah, MaximizeStateFidelity, menggunakan pengoptimum L-BFGS daripada SciPy. Kita memilih kriteria berhenti untuk kesetiaan yang memastikan ia melebihi kesetiaan litar awal tanpa AQC-Tensor. Setelah ambang ini dicapai, litar yang dimampatkan akan menunjukkan kedua-dua ralat Trotter yang lebih rendah dan kedalaman yang dikurangkan berbanding litar asal. Dengan menggunakan masa CPU tambahan, pengoptimuman selanjutnya boleh terus meningkatkan kesetiaan.

# Setting values for the optimization
aqc_stopping_fidelity = 1
aqc_max_iterations = 500

stopping_point = 1.0 - aqc_stopping_fidelity
objective = MaximizeStateFidelity(
aqc_target_mps, aqc_ansatz, simulator_settings
)

def callback(intermediate_result: OptimizeResult):
fidelity = 1 - intermediate_result.fun
print(
f"{datetime.datetime.now()} Intermediate result: Fidelity {fidelity:.8f}"
)
if intermediate_result.fun < stopping_point:
# Good enough for now
raise StopIteration

result = minimize(
objective,
aqc_initial_parameters,
method="L-BFGS-B",
jac=True,
options={"maxiter": aqc_max_iterations},
callback=callback,
)
if (
result.status
not in (
0,
1,
99,
)
): # 0 => success; 1 => max iterations reached; 99 => early termination via StopIteration
raise RuntimeError(
f"Optimization failed: {result.message} (status={result.status})"
)

print(f"Done after {result.nit} iterations.")
aqc_final_parameters = result.x
2025-04-14 11:46:52.174235 Intermediate result: Fidelity 0.99795851
2025-04-14 11:46:52.218249 Intermediate result: Fidelity 0.99822826
2025-04-14 11:46:52.280924 Intermediate result: Fidelity 0.99829675
2025-04-14 11:46:52.356214 Intermediate result: Fidelity 0.99832474
2025-04-14 11:46:52.411609 Intermediate result: Fidelity 0.99836131
2025-04-14 11:46:52.453747 Intermediate result: Fidelity 0.99839954
2025-04-14 11:46:52.496184 Intermediate result: Fidelity 0.99846517
2025-04-14 11:46:52.542046 Intermediate result: Fidelity 0.99865029
2025-04-14 11:46:52.583679 Intermediate result: Fidelity 0.99872332
2025-04-14 11:46:52.628732 Intermediate result: Fidelity 0.99892359
2025-04-14 11:46:52.690386 Intermediate result: Fidelity 0.99900640
2025-04-14 11:46:52.759398 Intermediate result: Fidelity 0.99907169
2025-04-14 11:46:52.819496 Intermediate result: Fidelity 0.99911423
2025-04-14 11:46:52.884505 Intermediate result: Fidelity 0.99918716
2025-04-14 11:46:52.947919 Intermediate result: Fidelity 0.99921278
2025-04-14 11:46:53.012808 Intermediate result: Fidelity 0.99924853
2025-04-14 11:46:53.083626 Intermediate result: Fidelity 0.99928797
2025-04-14 11:46:53.153235 Intermediate result: Fidelity 0.99933028
2025-04-14 11:46:53.221371 Intermediate result: Fidelity 0.99935757
2025-04-14 11:46:53.286211 Intermediate result: Fidelity 0.99938140
2025-04-14 11:46:53.352391 Intermediate result: Fidelity 0.99940964
2025-04-14 11:46:53.420472 Intermediate result: Fidelity 0.99944051
2025-04-14 11:46:53.486279 Intermediate result: Fidelity 0.99946828
2025-04-14 11:46:53.552338 Intermediate result: Fidelity 0.99948723
2025-04-14 11:46:53.618688 Intermediate result: Fidelity 0.99951011
2025-04-14 11:46:53.690878 Intermediate result: Fidelity 0.99954718
2025-04-14 11:46:53.762725 Intermediate result: Fidelity 0.99956267
2025-04-14 11:46:53.829784 Intermediate result: Fidelity 0.99958949
2025-04-14 11:46:53.897477 Intermediate result: Fidelity 0.99960498
2025-04-14 11:46:53.954633 Intermediate result: Fidelity 0.99961308
2025-04-14 11:46:54.010125 Intermediate result: Fidelity 0.99962894
2025-04-14 11:46:54.064717 Intermediate result: Fidelity 0.99964121
2025-04-14 11:46:54.118892 Intermediate result: Fidelity 0.99964348
2025-04-14 11:46:54.183236 Intermediate result: Fidelity 0.99964860
2025-04-14 11:46:54.245521 Intermediate result: Fidelity 0.99965695
2025-04-14 11:46:54.305792 Intermediate result: Fidelity 0.99966398
2025-04-14 11:46:54.355819 Intermediate result: Fidelity 0.99967816
2025-04-14 11:46:54.409580 Intermediate result: Fidelity 0.99968293
2025-04-14 11:46:54.457979 Intermediate result: Fidelity 0.99968936
2025-04-14 11:46:54.505891 Intermediate result: Fidelity 0.99969223
2025-04-14 11:46:54.551084 Intermediate result: Fidelity 0.99970009
2025-04-14 11:46:54.601817 Intermediate result: Fidelity 0.99970724
2025-04-14 11:46:54.650097 Intermediate result: Fidelity 0.99970987
2025-04-14 11:46:54.714727 Intermediate result: Fidelity 0.99971237
2025-04-14 11:46:54.780052 Intermediate result: Fidelity 0.99971916
2025-04-14 11:46:54.871994 Intermediate result: Fidelity 0.99971940
2025-04-14 11:46:54.958244 Intermediate result: Fidelity 0.99972465
2025-04-14 11:46:55.011057 Intermediate result: Fidelity 0.99972763
2025-04-14 11:46:55.175339 Intermediate result: Fidelity 0.99972894
2025-04-14 11:46:56.688912 Intermediate result: Fidelity 0.99972894
Done after 50 iterations.
parameters = [float(param) for param in aqc_final_parameters]
print("Final parameters:", parameters)
Final parameters: [-7.853983035039254, 1.5707966468427772, 1.5707962768868613, -1.570798010835122, 1.570794480409574, 1.5707972214146968, -1.570796593027083, 1.5707968206822998, -1.5707959018046258, -1.5707991700969144, 1.5707965852600927, 4.712386891737442, -7.853980840717957, 1.5707967508132654, 1.5707943162503217, -1.5707955382023582, 1.5707958007156742, 1.570796096113293, -1.5707928509846847, 1.5707971042943747, -1.570797909276557, -1.5707941020637393, 1.5707980179540793, 4.712389823219363, -1.5707928752386107, 1.5707996426312891, -1.5707975640471001, -1.570794132802984, 1.5707944361599957, 4.712390747060803, 0.1048818190315936, 0.06686710468840577, -0.0668645844756557, -3.1415923537135466, 1.2374931269696063, 6.323169390432535e-07, 3.53229204771738e-08, 2.1091105688681484, 6.283186439944202, 0.12152258846156239, 0.07961752617254866, -0.07961775088604585, -1.6564278051174865e-06, 2.0771163596472384, 3.141592651630471, -6.283185775192653, 1.7691609006726954, 3.1415922910116216, 0.19837572065074083, 0.11114901449078964, -0.11115124544944892, -3.141591983034976, 0.8570788408766729, 4.201601390404146e-07, -3.141593736550978, 0.34652010942396333, 6.283186232785291, 0.13606356527241956, 0.03891676349289617, -0.03891524189533726, -1.5707965732853424, 1.5707968967088564, -0.3086133992238162, 1.5707957152428194, 1.5707968398959653, -0.32062737993080026, 0.11027416939993417, 0.0726167290795046, -0.07262020423334464, -2.3729431959735024e-06, 1.8204437429254703, 9.299060301196612e-07, -3.141592899563451, 2.103269568939461, 3.1415937539734626, 0.11536891854817125, 0.09099022308254198, -0.09098864958606581, -3.1415913307373127, 2.078429034357281, -1.509777998069368e-06, -3.1415922600663255, 1.5189162645358172, -3.1415878461323583, 0.09999070991480716, 0.04352011445148391, -0.04351849541849812, -1.570797642506462, 1.570795238023824, 0.8903442644396505, 1.5707962698006606, 1.5707946765132268, 0.9098791754570567, 0.10448284343424026, 0.07317037684936827, -0.07316718173961152, -3.141592682240966, 2.1665363080039612, -7.450882112394189e-07, -5.771181304929921e-07, 2.615334999517103, -3.1415914971653898, 0.1890887078648001, 0.13578163074571992, -0.13578078143610256, 7.156734195912883e-07, 1.7915385305413096, -5.188866034727312e-07, 1.2827742939197711e-06, 1.2348316581417487, 6.28318357406372, 0.08061187643781703, 0.03820789039271876, -0.03820731868804904, 1.5707964027727628, 1.570798734462218, 4.387336153720882, -1.570795722044763, 1.570798457375325, 4.450361734163248, 0.092360147257953, 0.06047700345049011, -0.06048592856713045, -3.141591214829027, 2.6593289993286047, -2.366937342261038e-07, 8.112162974032695e-08, 1.8907014631413432, 8.355881261853104e-07, 0.23303641819370874, 0.14331998953606456, -0.1433194488304741, -3.141591621822901, 0.7455776479558791, 3.1415914520163586, -3.1415933560496105, 0.7603938554148255, -1.6230983177616282e-06, 0.07186349688535713, 0.03197144517771341, -0.031971177878588546, -4.712389048748508, 1.5707948403165752, 1.2773619319829186, -1.5707990802172127, 1.5707957676951863, 1.289083769394045, 0.13644999397718796, 0.032761460443590046, -0.032762060585195645, -1.5707977610073176, 1.5707964181578042, -3.4826435600366983, -4.712389691708343, 1.570794277502252, 2.799088046133275]

Pada ketika ini, kita hanya perlu mencari parameter akhir untuk litar ansatz. Kita kemudian boleh menggabungkan litar AQC yang dioptimumkan dengan litar evolusi selebihnya untuk mencipta litar evolusi masa lengkap untuk pelaksanaan pada perkakasan kuantum.

aqc_final_circuit = aqc_ansatz.assign_parameters(aqc_final_parameters)
aqc_final_circuit.compose(subsequent_circuit, inplace=True)
aqc_final_circuit.draw("mpl", fold=-1)

Output of the previous code cell

Kita juga perlu menggabungkan aqc_comparison_circuit kita dengan litar evolusi selebihnya. Litar ini akan digunakan untuk membandingkan prestasi litar yang dioptimumkan oleh AQC-Tensor dengan litar asal.

aqc_comparison_circuit.compose(subsequent_circuit, inplace=True)
aqc_comparison_circuit.draw("mpl", fold=-1)

Output of the previous code cell

Langkah 2: Optimumkan masalah untuk pelaksanaan perkakasan kuantum

Pilih perkakasan. Di sini kita akan guna mana-mana peranti IBM Quantum® yang tersedia dengan sekurang-kurangnya 127 qubit.

service = QiskitRuntimeService()
backend = service.least_busy(min_num_qubits=127)
print(backend)

Kita transpile PUB (Circuit dan observable) supaya sepadan dengan ISA (Instruction Set Architecture) backend. Dengan menetapkan optimization_level=3, Transpiler mengoptimumkan Circuit supaya muat dalam rantaian satu dimensi qubit, mengurangkan hingar yang mempengaruhi kesetiaan Circuit. Setelah Circuit diubah ke format yang serasi dengan backend, kita guna transformasi yang sepadan pada observable untuk memastikan ia selaras dengan susun atur qubit yang telah diubah suai.

pass_manager = generate_preset_pass_manager(
backend=backend, optimization_level=3
)
isa_circuit = pass_manager.run(aqc_final_circuit)
isa_observable = observable.apply_layout(isa_circuit.layout)
print("Observable info:", isa_observable)
print("Circuit depth:", isa_circuit.depth())
isa_circuit.draw("mpl", fold=-1, idle_wires=False)
Observable info: SparsePauliOp(['IIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIZZ'],
coeffs=[1.+0.j])
Circuit depth: 111

Output of the previous code cell

Lakukan transpilasi untuk Circuit perbandingan.

isa_comparison_circuit = pass_manager.run(aqc_comparison_circuit)
isa_comparison_observable = observable.apply_layout(
isa_comparison_circuit.layout
)
print("Observable info:", isa_comparison_observable)
print("Circuit depth:", isa_comparison_circuit.depth())
isa_comparison_circuit.draw("mpl", fold=-1, idle_wires=False)
Observable info: SparsePauliOp(['IIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIZZ'],
coeffs=[1.+0.j])
Circuit depth: 158

Output of the previous code cell

Langkah 3: Laksanakan menggunakan primitif Qiskit

Dalam langkah ini, kita laksanakan Circuit yang telah ditranspile pada perkakasan kuantum (atau backend simulasi). Menggunakan kelas EstimatorV2 daripada qiskit_ibm_runtime, kita sediakan estimator untuk menjalankan Circuit dan mengukur observable yang ditentukan. Hasil kerja memberikan jangkaan hasil bagi observable, memberi kita gambaran tentang prestasi Circuit pada perkakasan sasaran.

estimator = Estimator(backend)
job = estimator.run([(isa_circuit, isa_observable)])
print("Job ID:", job.job_id())
job.result()
Job ID: czyhqdxd8drg008hx0yg
PrimitiveResult([PubResult(data=DataBin(evs=np.ndarray(<shape=(), dtype=float64>), stds=np.ndarray(<shape=(), dtype=float64>), ensemble_standard_error=np.ndarray(<shape=(), dtype=float64>)), metadata={'shots': 4096, 'target_precision': 0.015625, 'circuit_metadata': {}, 'resilience': {}, 'num_randomizations': 32})], metadata={'dynamical_decoupling': {'enable': False, 'sequence_type': 'XX', 'extra_slack_distribution': 'middle', 'scheduling_method': 'alap'}, 'twirling': {'enable_gates': False, 'enable_measure': True, 'num_randomizations': 'auto', 'shots_per_randomization': 'auto', 'interleave_randomizations': True, 'strategy': 'active-accum'}, 'resilience': {'measure_mitigation': True, 'zne_mitigation': False, 'pec_mitigation': False}, 'version': 2})

Lakukan pelaksanaan untuk Circuit perbandingan.

job_comparison = estimator.run([(isa_comparison_circuit, isa_observable)])
print("Job Comparison ID:", job.job_id())
job_comparison.result()
Job Comparison ID: czyhqdxd8drg008hx0yg
PrimitiveResult([PubResult(data=DataBin(evs=np.ndarray(<shape=(), dtype=float64>), stds=np.ndarray(<shape=(), dtype=float64>), ensemble_standard_error=np.ndarray(<shape=(), dtype=float64>)), metadata={'shots': 4096, 'target_precision': 0.015625, 'circuit_metadata': {}, 'resilience': {}, 'num_randomizations': 32})], metadata={'dynamical_decoupling': {'enable': False, 'sequence_type': 'XX', 'extra_slack_distribution': 'middle', 'scheduling_method': 'alap'}, 'twirling': {'enable_gates': False, 'enable_measure': True, 'num_randomizations': 'auto', 'shots_per_randomization': 'auto', 'interleave_randomizations': True, 'strategy': 'active-accum'}, 'resilience': {'measure_mitigation': True, 'zne_mitigation': False, 'pec_mitigation': False}, 'version': 2})

Langkah 4: Proses pasca dan kembalikan hasil dalam format klasik yang dikehendaki

Dalam kes ini, pembinaan semula tidak diperlukan. Kita boleh terus periksa hasil dengan mengakses nilai jangkaan daripada output pelaksanaan.

# AQC results
hw_results = job.result()
hw_results_dicts = [pub_result.data.__dict__ for pub_result in hw_results]
hw_expvals = [
pub_result_data["evs"].tolist() for pub_result_data in hw_results_dicts
]
aqc_expval = hw_expvals[0]

# AQC comparison results
hw_comparison_results = job_comparison.result()
hw_comparison_results_dicts = [
pub_result.data.__dict__ for pub_result in hw_comparison_results
]
hw_comparison_expvals = [
pub_result_data["evs"].tolist()
for pub_result_data in hw_comparison_results_dicts
]
aqc_compare_expval = hw_comparison_expvals[0]

print(f"Exact: \t{reference_expval:.4f}")
print(
f"AQC: \t{aqc_expval:.4f}, |∆| = {np.abs(reference_expval- aqc_expval):.4f}"
)
print(
f"AQC Comparison:\t{aqc_compare_expval:.4f}, |∆| = {np.abs(reference_expval- aqc_compare_expval):.4f}"
)
Exact:         	-0.5252
AQC: -0.4903, |∆| = 0.0349
AQC Comparison: 0.5424, |∆| = 1.0676

Carta bar untuk membandingkan hasil AQC, perbandingan, dan Circuit tepat.

plt.style.use("seaborn-v0_8")

labels = ["AQC Result", "AQC Comparison Result"]
values = [abs(aqc_expval), abs(aqc_compare_expval)]

plt.figure(figsize=(10, 6))
bars = plt.bar(labels, values, color=["tab:blue", "tab:purple"])
plt.axhline(
y=abs(reference_expval), color="red", linestyle="--", label="Exact Result"
)
plt.xlabel("Results")
plt.ylabel("Absolute Expected Value")
plt.title("AQC Result vs AQC Comparison Result (Absolute Values)")
plt.legend()
for bar in bars:
y_val = bar.get_height()
plt.text(
bar.get_x() + bar.get_width() / 2.0,
y_val,
round(y_val, 2),
va="bottom",
)
plt.show()

Output of the previous code cell

Bahagian II: tingkatkan skala

Bahagian kedua tutorial ini membina contoh sebelumnya dengan meningkatkan skala kepada sistem yang lebih besar dengan 50 tapak, menunjukkan cara memetakan masalah simulasi kuantum yang lebih kompleks kepada Circuit kuantum yang boleh dilaksanakan. Di sini, kita meneroka dinamik model XXZ 50-tapak, yang membolehkan kita membina dan mengoptimumkan Circuit kuantum yang besar dan mencerminkan saiz sistem yang lebih realistik.

Hamiltonian untuk model XXZ 50-tapak kita ditakrifkan sebagai:

H^XXZ=i=1L1Ji,(i+1)(XiX(i+1)+YiY(i+1)+2ZiZ(i+1)),\hat{\mathcal{H}}_{XXZ} = \sum_{i=1}^{L-1} J_{i,(i+1)}\left(X_i X_{(i+1)}+Y_i Y_{(i+1)}+ 2\cdot Z_i Z_{(i+1)} \right) \, ,

di mana Ji,(i+1)J_{i,(i+1)} ialah pekali rawak yang sepadan dengan tepi (i,i+1)(i, i+1), dan L=50L=50 ialah bilangan tapak. Takrifkan peta gandingan dan tepi untuk Hamiltonian.

L = 50  # L = length of our 1D spin chain

# Generate the edge list for this spin-chain
edge_list = [(i - 1, i) for i in range(1, L)]
# Generate an edge-coloring so we can make hw-efficient circuits
even_edges = edge_list[::2]
odd_edges = edge_list[1::2]

# Instantiate a CouplingMap object
coupling_map = CouplingMap(edge_list)

# Generate random coefficients for our XXZ Hamiltonian
np.random.seed(0)
Js = np.random.rand(L - 1) + 0.5 * np.ones(L - 1)

hamiltonian = SparsePauliOp(Pauli("I" * L))
for i, edge in enumerate(even_edges + odd_edges):
hamiltonian += SparsePauliOp.from_sparse_list(
[
("XX", (edge), Js[i] / 2),
("YY", (edge), Js[i] / 2),
("ZZ", (edge), Js[i]),
],
num_qubits=L,
)

observable = SparsePauliOp.from_sparse_list(
[("ZZ", (L // 2 - 1, L // 2), 1.0)], num_qubits=L
)

# Generate an initial state
L = hamiltonian.num_qubits
initial_state = QuantumCircuit(L)
for i in range(L):
if i % 2:
initial_state.x(i)

Langkah 1: Petakan input klasik kepada masalah kuantum

Untuk masalah yang lebih besar ini, kita mulakan dengan membina Hamiltonian untuk model XXZ 50-tapak, menentukan interaksi spin-spin dan medan magnet luaran merentasi semua tapak. Selepas itu, kita ikuti tiga langkah utama:

  1. Jana Circuit AQC yang dioptimumkan: Gunakan Trotterisasi untuk menghampiri evolusi awal, kemudian mampat segmen ini bagi mengurangkan kedalaman Circuit.
  2. Cipta Circuit evolusi masa yang tinggal: Tangkap evolusi masa yang tinggal selepas segmen awal.
  3. Gabungkan Circuit-circuit: Satukan Circuit AQC yang dioptimumkan dengan Circuit evolusi yang tinggal untuk mencipta Circuit evolusi masa yang lengkap dan bersedia untuk dilaksanakan. Jana Circuit sasaran AQC (segmen awal).
aqc_evolution_time = 0.2
aqc_target_num_trotter_steps = 32

aqc_target_circuit = initial_state.copy()
aqc_target_circuit.compose(
generate_time_evolution_circuit(
hamiltonian,
synthesis=SuzukiTrotter(reps=aqc_target_num_trotter_steps),
time=aqc_evolution_time,
),
inplace=True,
)

Jana Circuit seterusnya (segmen yang tinggal).

subsequent_num_trotter_steps = 3
subsequent_evolution_time = 0.2

subsequent_circuit = generate_time_evolution_circuit(
hamiltonian,
synthesis=SuzukiTrotter(reps=subsequent_num_trotter_steps),
time=subsequent_evolution_time,
)

Jana Circuit perbandingan AQC (segmen awal, tetapi dengan bilangan langkah Trotter yang sama seperti Circuit seterusnya).

# Generate the AQC comparison circuit
aqc_comparison_num_trotter_steps = int(
subsequent_num_trotter_steps
/ subsequent_evolution_time
* aqc_evolution_time
)
print(
"Number of Trotter steps for comparison:",
aqc_comparison_num_trotter_steps,
)

aqc_comparison_circuit = generate_time_evolution_circuit(
hamiltonian,
synthesis=SuzukiTrotter(reps=aqc_comparison_num_trotter_steps),
time=aqc_evolution_time,
)
Number of Trotter steps for comparison: 3

Jana Circuit rujukan.

evolution_time = 0.4
reps = 200

reference_circuit = initial_state.copy()
reference_circuit.compose(
generate_time_evolution_circuit(
hamiltonian,
synthesis=SuzukiTrotter(reps=reps),
time=evolution_time,
),
inplace=True,
)

Jana ansatz dan parameter awal daripada Circuit Trotter dengan bilangan langkah yang lebih sedikit.

aqc_ansatz_num_trotter_steps = 1

aqc_good_circuit = initial_state.copy()
aqc_good_circuit.compose(
generate_time_evolution_circuit(
hamiltonian,
synthesis=SuzukiTrotter(reps=aqc_ansatz_num_trotter_steps),
time=aqc_evolution_time,
),
inplace=True,
)

aqc_ansatz, aqc_initial_parameters = generate_ansatz_from_circuit(
aqc_good_circuit
)
print(f"AQC Comparison circuit: depth {aqc_comparison_circuit.depth()}")
print(f"Target circuit: depth {aqc_target_circuit.depth()}")
print(
f"Ansatz circuit: depth {aqc_ansatz.depth()}, with {len(aqc_initial_parameters)} parameters"
)
AQC Comparison circuit: depth 36
Target circuit: depth 385
Ansatz circuit: depth 7, with 816 parameters

Tetapkan seting untuk simulasi rangkaian tensor, kemudian bina perwakilan keadaan hasil darab matriks (matrix product state) untuk sasaran optimasi. Seterusnya, nilai kesetiaan antara Circuit awal dan keadaan sasaran bagi mengukur perbezaan ralat Trotter.

simulator_settings = QuimbSimulator(
quimb.tensor.CircuitMPS, autodiff_backend="jax"
)

# Build the matrix-product representation of the state to be approximated by AQC
aqc_target_mps = tensornetwork_from_circuit(
aqc_target_circuit, simulator_settings
)
print("Target MPS maximum bond dimension:", aqc_target_mps.psi.max_bond())

# Obtains the reference MPS, where we can obtain the exact expectation value by examining the `local_expectation``
reference_mps = tensornetwork_from_circuit(
reference_circuit, simulator_settings
)
reference_expval = reference_mps.local_expectation(
quimb.pauli("Z") & quimb.pauli("Z"), (L // 2 - 1, L // 2)
).real.item()

# Compute the starting fidelity
good_mps = tensornetwork_from_circuit(aqc_good_circuit, simulator_settings)
starting_fidelity = abs(compute_overlap(good_mps, aqc_target_mps)) ** 2
print("Starting fidelity:", starting_fidelity)
Target MPS maximum bond dimension: 5
Starting fidelity: 0.9926466919924161

Untuk mengoptimumkan parameter ansatz, kita minimumkan fungsi kos MaximizeStateFidelity menggunakan pengoptimum L-BFGS daripada SciPy, dengan kriteria penghentian ditetapkan untuk melepasi kesetiaan Circuit awal tanpa AQC-Tensor. Ini memastikan Circuit yang dimampatkan mempunyai ralat Trotter yang lebih rendah sekaligus kedalaman yang lebih kecil.

# Setting values for the optimization
aqc_stopping_fidelity = 1
aqc_max_iterations = 500

stopping_point = 1.0 - aqc_stopping_fidelity
objective = MaximizeStateFidelity(
aqc_target_mps, aqc_ansatz, simulator_settings
)

def callback(intermediate_result: OptimizeResult):
fidelity = 1 - intermediate_result.fun
print(
f"{datetime.datetime.now()} Intermediate result: Fidelity {fidelity:.8f}"
)
if intermediate_result.fun < stopping_point:
# Good enough for now
raise StopIteration

result = minimize(
objective,
aqc_initial_parameters,
method="L-BFGS-B",
jac=True,
options={"maxiter": aqc_max_iterations},
callback=callback,
)
if (
result.status
not in (
0,
1,
99,
)
): # 0 => success; 1 => max iterations reached; 99 => early termination via StopIteration
raise RuntimeError(
f"Optimization failed: {result.message} (status={result.status})"
)

print(f"Done after {result.nit} iterations.")
aqc_final_parameters = result.x
2025-04-14 11:48:28.705807 Intermediate result: Fidelity 0.99795851
2025-04-14 11:48:28.743265 Intermediate result: Fidelity 0.99822826
2025-04-14 11:48:28.776629 Intermediate result: Fidelity 0.99829675
2025-04-14 11:48:28.816153 Intermediate result: Fidelity 0.99832474
2025-04-14 11:48:28.856437 Intermediate result: Fidelity 0.99836131
2025-04-14 11:48:28.896432 Intermediate result: Fidelity 0.99839954
2025-04-14 11:48:28.936670 Intermediate result: Fidelity 0.99846517
2025-04-14 11:48:28.982069 Intermediate result: Fidelity 0.99865029
2025-04-14 11:48:29.026130 Intermediate result: Fidelity 0.99872332
2025-04-14 11:48:29.067426 Intermediate result: Fidelity 0.99892359
2025-04-14 11:48:29.110742 Intermediate result: Fidelity 0.99900640
2025-04-14 11:48:29.161362 Intermediate result: Fidelity 0.99907169
2025-04-14 11:48:29.207933 Intermediate result: Fidelity 0.99911423
2025-04-14 11:48:29.266772 Intermediate result: Fidelity 0.99918716
2025-04-14 11:48:29.331727 Intermediate result: Fidelity 0.99921278
2025-04-14 11:48:29.401694 Intermediate result: Fidelity 0.99924853
2025-04-14 11:48:29.467980 Intermediate result: Fidelity 0.99928797
2025-04-14 11:48:29.533281 Intermediate result: Fidelity 0.99933028
2025-04-14 11:48:29.600833 Intermediate result: Fidelity 0.99935757
2025-04-14 11:48:29.670816 Intermediate result: Fidelity 0.99938140
2025-04-14 11:48:29.736928 Intermediate result: Fidelity 0.99940964
2025-04-14 11:48:29.802931 Intermediate result: Fidelity 0.99944051
2025-04-14 11:48:29.869177 Intermediate result: Fidelity 0.99946828
2025-04-14 11:48:29.940156 Intermediate result: Fidelity 0.99948723
2025-04-14 11:48:30.005751 Intermediate result: Fidelity 0.99951011
2025-04-14 11:48:30.070853 Intermediate result: Fidelity 0.99954718
2025-04-14 11:48:30.139171 Intermediate result: Fidelity 0.99956267
2025-04-14 11:48:30.210506 Intermediate result: Fidelity 0.99958949
2025-04-14 11:48:30.279647 Intermediate result: Fidelity 0.99960498
2025-04-14 11:48:30.348016 Intermediate result: Fidelity 0.99961308
2025-04-14 11:48:30.414311 Intermediate result: Fidelity 0.99962894
2025-04-14 11:48:30.488910 Intermediate result: Fidelity 0.99964121
2025-04-14 11:48:30.561298 Intermediate result: Fidelity 0.99964348
2025-04-14 11:48:30.632214 Intermediate result: Fidelity 0.99964860
2025-04-14 11:48:30.705703 Intermediate result: Fidelity 0.99965695
2025-04-14 11:48:30.775679 Intermediate result: Fidelity 0.99966398
2025-04-14 11:48:30.842629 Intermediate result: Fidelity 0.99967816
2025-04-14 11:48:30.912357 Intermediate result: Fidelity 0.99968293
2025-04-14 11:48:30.979420 Intermediate result: Fidelity 0.99968936
2025-04-14 11:48:31.049196 Intermediate result: Fidelity 0.99969223
2025-04-14 11:48:31.125391 Intermediate result: Fidelity 0.99970009
2025-04-14 11:48:31.201256 Intermediate result: Fidelity 0.99970724
2025-04-14 11:48:31.272424 Intermediate result: Fidelity 0.99970987
2025-04-14 11:48:31.338907 Intermediate result: Fidelity 0.99971237
2025-04-14 11:48:31.404800 Intermediate result: Fidelity 0.99971916
2025-04-14 11:48:31.475226 Intermediate result: Fidelity 0.99971940
2025-04-14 11:48:31.547746 Intermediate result: Fidelity 0.99972465
2025-04-14 11:48:31.622827 Intermediate result: Fidelity 0.99972763
2025-04-14 11:48:31.819516 Intermediate result: Fidelity 0.99972894
2025-04-14 11:48:33.444538 Intermediate result: Fidelity 0.99972894
Done after 50 iterations.
parameters = [float(param) for param in aqc_final_parameters]

Bina Circuit akhir untuk transpilasi dengan menggabungkan ansatz yang dioptimumkan bersama Circuit evolusi masa yang tinggal.

aqc_final_circuit = aqc_ansatz.assign_parameters(aqc_final_parameters)
aqc_final_circuit.compose(subsequent_circuit, inplace=True)
aqc_comparison_circuit.compose(subsequent_circuit, inplace=True)

Langkah 2: Optimumkan masalah untuk pelaksanaan perkakasan kuantum

Pilih Backend.

service = QiskitRuntimeService()
backend = service.least_busy(min_num_qubits=127)
print(backend)

Transpilkan Circuit yang telah siap pada perkakasan sasaran, menyediakannya untuk pelaksanaan. Circuit ISA yang terhasil kemudiannya boleh dihantar untuk dilaksanakan pada Backend.

pass_manager = generate_preset_pass_manager(
backend=backend, optimization_level=3
)
isa_circuit = pass_manager.run(aqc_final_circuit)
isa_observable = observable.apply_layout(isa_circuit.layout)
print("Observable info:", isa_observable)
print("Circuit depth:", isa_circuit.depth())
isa_circuit.draw("mpl", fold=-1, idle_wires=False)
Observable info: SparsePauliOp(['IIIIIIIIIIIIIIIIIIIIIIIIIIIIIIZZIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIII'],
coeffs=[1.+0.j])
Circuit depth: 122

Output of the previous code cell

isa_comparison_circuit = pass_manager.run(aqc_comparison_circuit)
isa_comparison_observable = observable.apply_layout(
isa_comparison_circuit.layout
)
print("Observable info:", isa_comparison_observable)
print("Circuit depth:", isa_comparison_circuit.depth())
isa_comparison_circuit.draw("mpl", fold=-1, idle_wires=False)
Observable info: SparsePauliOp(['IIIIIIIIIIIIIIIIIIIIIIIIIIIIIIZZIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIII'],
coeffs=[1.+0.j])
Circuit depth: 158

Output of the previous code cell

Langkah 3: Laksanakan menggunakan primitif Qiskit

Dalam langkah ini, kita jalankan Circuit yang telah ditranspilkan pada perkakasan kuantum (atau Backend simulasi) menggunakan EstimatorV2 daripada qiskit_ibm_runtime untuk mengukur observable yang ditentukan. Hasil kerja ini akan memberikan maklumat berharga tentang prestasi Circuit pada perkakasan sasaran.

Untuk contoh berskala lebih besar ini, kita akan meneroka cara menggunakan EstimatorOptions bagi mengurus dan mengawal parameter eksperimen perkakasan dengan lebih baik. Walaupun seting ini adalah pilihan, ia berguna untuk menjejak parameter eksperimen dan memperhalusi pilihan pelaksanaan demi hasil yang optimum.

Untuk senarai lengkap pilihan pelaksanaan yang tersedia, rujuk dokumentasi qiskit-ibm-runtime.

twirling_options = {
"enable_gates": True,
"enable_measure": True,
"num_randomizations": 300,
"shots_per_randomization": 100,
"strategy": "active",
}

zne_options = {
"amplifier": "gate_folding",
"noise_factors": [1, 2, 3],
"extrapolated_noise_factors": list(np.linspace(0, 3, 31)),
"extrapolator": ["exponential", "linear", "fallback"],
}

meas_learning_options = {
"num_randomizations": 512,
"shots_per_randomization": 512,
}

resilience_options = {
"measure_mitigation": True,
"zne_mitigation": True,
"zne": zne_options,
"measure_noise_learning": meas_learning_options,
}

estimator_options = {
"resilience": resilience_options,
"twirling": twirling_options,
}

estimator = Estimator(backend, options=estimator_options)
job = estimator.run([(isa_circuit, isa_observable)])
print("Job ID:", job.job_id())
job.result()
Job ID: czyjx6crxz8g008f63r0
PrimitiveResult([PubResult(data=DataBin(evs=np.ndarray(<shape=(), dtype=float64>), stds=np.ndarray(<shape=(), dtype=float64>), evs_noise_factors=np.ndarray(<shape=(3,), dtype=float64>), stds_noise_factors=np.ndarray(<shape=(3,), dtype=float64>), ensemble_stds_noise_factors=np.ndarray(<shape=(3,), dtype=float64>), evs_extrapolated=np.ndarray(<shape=(3, 31), dtype=float64>), stds_extrapolated=np.ndarray(<shape=(3, 31), dtype=float64>)), metadata={'shots': 30000, 'target_precision': 0.005773502691896258, 'circuit_metadata': {}, 'resilience': {'zne': {'extrapolator': 'exponential'}}, 'num_randomizations': 300})], metadata={'dynamical_decoupling': {'enable': False, 'sequence_type': 'XX', 'extra_slack_distribution': 'middle', 'scheduling_method': 'alap'}, 'twirling': {'enable_gates': True, 'enable_measure': True, 'num_randomizations': 300, 'shots_per_randomization': 100, 'interleave_randomizations': True, 'strategy': 'active'}, 'resilience': {'measure_mitigation': True, 'zne_mitigation': True, 'pec_mitigation': False, 'zne': {'noise_factors': [1, 2, 3], 'extrapolator': ['exponential', 'linear', 'fallback'], 'extrapolated_noise_factors': [0, 0.1, 0.2, 0.30000000000000004, 0.4, 0.5, 0.6000000000000001, 0.7000000000000001, 0.8, 0.9, 1, 1.1, 1.2000000000000002, 1.3, 1.4000000000000001, 1.5, 1.6, 1.7000000000000002, 1.8, 1.9000000000000001, 2, 2.1, 2.2, 2.3000000000000003, 2.4000000000000004, 2.5, 2.6, 2.7, 2.8000000000000003, 2.9000000000000004, 3]}}, 'version': 2})
job_comparison = estimator.run([(isa_comparison_circuit, isa_observable)])
print("Job Comparison ID:", job.job_id())
job_comparison.result()
Job Comparison ID: czyjx6crxz8g008f63r0
PrimitiveResult([PubResult(data=DataBin(evs=np.ndarray(<shape=(), dtype=float64>), stds=np.ndarray(<shape=(), dtype=float64>), evs_noise_factors=np.ndarray(<shape=(3,), dtype=float64>), stds_noise_factors=np.ndarray(<shape=(3,), dtype=float64>), ensemble_stds_noise_factors=np.ndarray(<shape=(3,), dtype=float64>), evs_extrapolated=np.ndarray(<shape=(3, 31), dtype=float64>), stds_extrapolated=np.ndarray(<shape=(3, 31), dtype=float64>)), metadata={'shots': 30000, 'target_precision': 0.005773502691896258, 'circuit_metadata': {}, 'resilience': {'zne': {'extrapolator': 'exponential'}}, 'num_randomizations': 300})], metadata={'dynamical_decoupling': {'enable': False, 'sequence_type': 'XX', 'extra_slack_distribution': 'middle', 'scheduling_method': 'alap'}, 'twirling': {'enable_gates': True, 'enable_measure': True, 'num_randomizations': 300, 'shots_per_randomization': 100, 'interleave_randomizations': True, 'strategy': 'active'}, 'resilience': {'measure_mitigation': True, 'zne_mitigation': True, 'pec_mitigation': False, 'zne': {'noise_factors': [1, 2, 3], 'extrapolator': ['exponential', 'linear', 'fallback'], 'extrapolated_noise_factors': [0, 0.1, 0.2, 0.30000000000000004, 0.4, 0.5, 0.6000000000000001, 0.7000000000000001, 0.8, 0.9, 1, 1.1, 1.2000000000000002, 1.3, 1.4000000000000001, 1.5, 1.6, 1.7000000000000002, 1.8, 1.9000000000000001, 2, 2.1, 2.2, 2.3000000000000003, 2.4000000000000004, 2.5, 2.6, 2.7, 2.8000000000000003, 2.9000000000000004, 3]}}, 'version': 2})

Langkah 4: Proses selepas pelaksanaan dan kembalikan hasil dalam format klasik yang dikehendaki

Di sini, tiada pembinaan semula yang diperlukan seperti sebelumnya; kita boleh terus mengakses nilai jangkaan daripada output pelaksanaan untuk memeriksa hasilnya.

# AQC results
hw_results = job.result()
hw_results_dicts = [pub_result.data.__dict__ for pub_result in hw_results]
hw_expvals = [
pub_result_data["evs"].tolist() for pub_result_data in hw_results_dicts
]
aqc_expval = hw_expvals[0]

# AQC comparison results
hw_comparison_results = job_comparison.result()
hw_comparison_results_dicts = [
pub_result.data.__dict__ for pub_result in hw_comparison_results
]
hw_comparison_expvals = [
pub_result_data["evs"].tolist()
for pub_result_data in hw_comparison_results_dicts
]
aqc_compare_expval = hw_comparison_expvals[0]

print(f"Exact: \t{reference_expval:.4f}")
print(
f"AQC: \t{aqc_expval:.4f}, |∆| = {np.abs(reference_expval- aqc_expval):.4f}"
)
print(
f"AQC Comparison:\t{aqc_compare_expval:.4f}, |∆| = {np.abs(reference_expval- aqc_compare_expval):.4f}"
)
Exact:         	-0.5888
AQC: -0.4809, |∆| = 0.1078
AQC Comparison: 1.1764, |∆| = 1.7652

Plot keputusan AQC, perbandingan, dan Circuit tepat untuk model XXZ 50-tapak.

labels = ["AQC Result", "AQC Comparison Result"]
values = [abs(aqc_expval), abs(aqc_compare_expval)]

plt.figure(figsize=(10, 6))
bars = plt.bar(labels, values, color=["tab:blue", "tab:purple"])
plt.axhline(
y=abs(reference_expval), color="red", linestyle="--", label="Exact Result"
)
plt.xlabel("Results")
plt.ylabel("Absolute Expected Value")
plt.title("AQC Result vs AQC Comparison Result (Absolute Values)")
plt.legend()
for bar in bars:
y_val = bar.get_height()
plt.text(
bar.get_x() + bar.get_width() / 2.0,
y_val,
round(y_val, 2),
va="bottom",
)

plt.show()

Output of the previous code cell

Kesimpulan

Tutorial ini menunjukkan cara menggunakan Approximate Quantum Compilation dengan rangkaian tensor (AQC-Tensor) untuk memampatkan dan mengoptimumkan Circuit bagi simulasi dinamik kuantum pada skala besar. Dengan menggunakan model Heisenberg kecil dan besar, kita menerapkan AQC-Tensor untuk mengurangkan kedalaman Circuit yang diperlukan bagi evolusi masa Trotterisasi. Dengan menjana ansatz berparameter daripada Circuit Trotter yang dipermudahkan dan mengoptimumkannya menggunakan teknik matrix product state (MPS), kita berjaya menghasilkan penghampiran evolusi sasaran berkedalaman rendah yang tepat dan cekap.

Aliran kerja di sini menonjolkan kelebihan utama AQC-Tensor untuk menskalakan simulasi kuantum:

  • Pemampatan Circuit yang ketara: AQC-Tensor mengurangkan kedalaman Circuit yang diperlukan untuk evolusi masa yang kompleks, meningkatkan kebolehlaksanaannya pada peranti semasa.
  • Optimasi yang cekap: Pendekatan MPS menyediakan rangka kerja yang kukuh untuk optimasi parameter, mengimbangi kesetiaan dengan kecekapan pengiraan.
  • Bersedia untuk perkakasan: Transpilasi Circuit akhir yang dioptimumkan memastikan ia memenuhi kekangan perkakasan kuantum sasaran.

Apabila peranti kuantum yang lebih besar dan algoritma yang lebih canggih muncul, teknik seperti AQC-Tensor akan menjadi penting untuk menjalankan simulasi kuantum yang kompleks pada perkakasan jangka hampir, menunjukkan kemajuan yang menjanjikan dalam menguruskan kedalaman dan kesetiaan untuk aplikasi kuantum yang berskala.

Kaji selidik tutorial

Sila ambil kaji selidik ringkas ini untuk memberikan maklum balas tentang tutorial ini. Pandangan anda akan membantu kami memperbaiki kandungan dan pengalaman pengguna kami.

Link to survey

Note: This survey is provided by IBM Quantum and relates to the original English content. To give feedback on doQumentation's website, translations, or code execution, please open a GitHub issue.

Source: IBM Quantum docs — updated 13 Feb 2026
English version on doQumentation — updated 7 Mei 2026
This translation based on the English version of 9 Apr 2026