Como enviar um circuito com o Qiskit para o Azure Quantum
Saiba como enviar um circuito quântico Qiskit usando o azure-quantum
Python pacote. Você pode enviar circuitos Qiskit para o Azure Quantum usando um bloco de azure-quantum
Python anotações do Azure Quantum, que tem um pacote interno, ou de sua máquina local. Você também pode testar seus circuitos usando o simulador local esparso.
Para obter mais informações, consulte Circuitos quânticos.
Pré-requisitos
Para obter detalhes da instalação, consulte Configurar a extensão QDK.
Um espaço de trabalho do Azure Quantum em sua assinatura do Azure. Para criar um espaço de trabalho, consulte Criar um espaço de trabalho do Azure Quantum.
Um Python ambiente com Python e Pip instalado.
VS Code com o Azure Quantum Development Kit e Pythonextensões Jupyter instaladas.
O pacote Azure Quantum
azure-quantum
Python com a tag [qiskit].python -m pip install --upgrade azure-quantum[qiskit] qsharp ipykernel
Importante
Certifique-se de ter a versão mais recente do Qiskit. Para obter mais informações, consulte Python azure-quantum.
Nota
Se o kernel Python do Jupyter
ipykernel
não for detetado, o VS Code solicitará que você o instale.
Criar um novo Jupyter Notebook
- No VS Code, selecione View > Command palette e selecione Create: New Jupyter Notebook.
- No canto superior direito, o VS Code detetará e exibirá a versão e Python o ambiente virtual Python que foi selecionado para o notebook. Se você tiver vários Python ambientes, talvez seja necessário selecionar um kernel usando o seletor de kernel no canto superior direito. Se nenhum ambiente foi detetado, consulte Jupyter Notebooks no VS Code para obter informações de configuração.
Carregue as importações necessárias
Na primeira célula do seu bloco de notas, execute o seguinte código para carregar as importações necessárias:
import azure.quantum
from azure.quantum import Workspace
from qiskit import QuantumCircuit
from qiskit.visualization import plot_histogram
from azure.quantum.qiskit import AzureQuantumProvider
Conectar-se ao serviço Azure Quantum
Para se conectar ao serviço Azure Quantum, você precisa da ID do recurso e do local do seu espaço de trabalho do Azure Quantum.
Inicie sessão na sua conta do Azure, https://portal.azure.com,
Selecione seu espaço de trabalho do Azure Quantum e navegue até Visão geral.
Copie os parâmetros nos campos.
Adicione uma nova célula em seu bloco de anotações e use as informações da sua conta para criar Workspace
e AzureQuantumProvider
objetos para se conectar ao seu espaço de trabalho do Azure Quantum.
workspace = Workspace(
resource_id = "", # Add the resourceID of your workspace
location = "" # Add the location of your workspace (for example "westus")
)
provider = AzureQuantumProvider(workspace)
Listar todos os back-ends
Agora você pode imprimir todos os back-ends de computação quântica disponíveis em seu espaço de trabalho:
print("This workspace's targets:")
for backend in provider.backends():
print("- " + backend.name())
This workspace's targets:
- ionq.qpu
- ionq.qpu.aria-1
- ionq.simulator
- microsoft.estimator
- quantinuum.hqs-lt-s1
- quantinuum.hqs-lt-s1-apival
- quantinuum.hqs-lt-s2
- quantinuum.hqs-lt-s2-apival
- quantinuum.hqs-lt-s1-sim
- quantinuum.hqs-lt-s2-sim
- quantinuum.qpu.h1-1
- quantinuum.sim.h1-1sc
- quantinuum.sim.h1-1e
- rigetti.sim.qvm
- rigetti.qpu.ankaa-9q-3
- rigetti.qpu.ankaa-3
Execute um circuito simples
Primeiro, crie um circuito Qiskit simples para executar.
# Create a Quantum Circuit acting on the q register
circuit = QuantumCircuit(3, 3)
circuit.name = "Qiskit Sample - 3-qubit GHZ circuit"
circuit.h(0)
circuit.cx(0, 1)
circuit.cx(1, 2)
circuit.measure([0,1,2], [0, 1, 2])
# Print out the circuit
circuit.draw()
┌───┐ ┌─┐
q_0: ┤ H ├──■───────┤M├──────
└───┘┌─┴─┐ └╥┘┌─┐
q_1: ─────┤ X ├──■───╫─┤M├───
└───┘┌─┴─┐ ║ └╥┘┌─┐
q_2: ──────────┤ X ├─╫──╫─┤M├
└───┘ ║ ║ └╥┘
c: 3/════════════════╩══╩══╩═
0 1 2
Selecione um target para executar o programa
Executar no simulador IonQ
Antes de correr em hardware real, vamos testar o circuito no simulador. Use get_backend
para criar um Backend
objeto para se conectar ao back-end do IonQ Simulator:
simulator_backend = provider.get_backend("ionq.simulator")
Os back-ends IonQ suportam portas de um conjunto de portas definido, que são compiladas para serem executadas de forma otimizada no hardware. Se o seu circuito contém portões que não estão nesta lista, você precisa transpilar para o suportado gateset
usando a transpile
função fornecida pelo Qiskit:
from qiskit import transpile
circuit = transpile(circuit, simulator_backend)
A função transpile retorna um novo objeto de circuito onde os portões são decompostos em portões que são suportados no back-end especificado.
Agora você pode executar o programa por meio do serviço Azure Quantum e obter o resultado. A célula a seguir envia um trabalho que executa o circuito com 100 disparos:
job = simulator_backend.run(circuit, shots=8)
job_id = job.id()
print("Job id", job_id)
Job id 00000000-0000-0000-0000-000000000000
Para aguardar até que o trabalho seja concluído e retornar os resultados, execute:
result = job.result()
print(result)
Result(backend_name='ionq.simulator', backend_version='1', qobj_id='Qiskit Sample - 3-qubit GHZ circuit', job_id='00000000-0000-0000-0000-000000000000', success=True, results=[ExperimentResult(shots=8, success=True, meas_level=2, data=ExperimentResultData(counts={'000': 4, '111': 4}, memory=['000', '000', '000', '000', '111', '111', '111', '111'], probabilities={'000': 0.5, '111': 0.5}), header=QobjExperimentHeader(name='Qiskit Sample - 3-qubit GHZ circuit', num_qubits=3, metadata={}), status=JobStatus.DONE, name='Qiskit Sample - 3-qubit GHZ circuit')], date=None, status=None, header=None, error_data=None)
Como o resultado é um objeto nativo do pacote Qiskit, você pode usar o Qiskit result.get_counts
e plot_histogram
visualizar os resultados. Para certificar-se de que todos os rótulos de cadeia de bits possíveis estão representados, adicione-os ao counts
.
counts = {format(n, "03b"): 0 for n in range(8)}
counts.update(result.get_counts(circuit))
print(counts)
plot_histogram(counts)
{'000': 4, '001': 0, '010': 0, '011': 0, '100': 0, '101': 0, '110': 0, '111': 4}
Você também pode usar a get_memory()
função para exibir dados de captura individuais do trabalho
result.get_memory(circuit)
['000', '000', '000', '000', '111', '111', '111', '111']
Nota
No IonQ targets, se você enviar um trabalho com um número ímpar de disparos, os resultados serão arredondados para baixo para o próximo número par. Por exemplo, se você especificar 9 disparos, os resultados exibirão dados para 8 disparos.
Estimar o custo do trabalho
Antes de executar um trabalho na QPU, você deve estimar quanto custa para ser executado.
Para obter os detalhes de preços mais atuais, consulte Preços IonQ ou encontre seu espaço de trabalho e veja as opções de preços na guia "Provedor" do seu espaço de trabalho via: aka.ms/aq/myworkspaces.
Executar em IonQ QPU
Para se conectar ao hardware real (uma
qpu_backend = provider.get_backend("ionq.qpu.aria-1")
Envie o circuito para ser executado no Azure Quantum, obtenha os resultados e execute plot_histogram
para plotar os resultados.
Nota
O tempo necessário para executar um circuito na QPU pode variar dependendo dos tempos de fila atuais.
# Submit the circuit to run on Azure Quantum
job = qpu_backend.run(circuit, shots=100)
job_id = job.id()
print("Job id", job_id)
# Get the job results (this method waits for the Job to complete):
result = job.result()
print(result)
counts = {format(n, "03b"): 0 for n in range(8)}
counts.update(result.get_counts(circuit))
print(counts)
plot_histogram(counts)
Job id 00000000-0000-0000-0000-000000000000
Job Status: job has successfully run
Result(backend_name='ionq.qpu.aria-1', backend_version='1', qobj_id='Qiskit Sample - 3-qubit GHZ circuit', job_id='00000000-0000-0000-0000-000000000000', success=True, results=[ExperimentResult(shots=1024, success=True, meas_level=2, data=ExperimentResultData(counts={'0': 505, '1': 6, '2': 1, '3': 1, '4': 1, '5': 10, '6': 11, '7': 488}, probabilities={'0': 0.4932, '1': 0.0059, '2': 0.001, '3': 0.001, '4': 0.001, '5': 0.0098, '6': 0.0117, '7': 0.4766}), header=QobjExperimentHeader(name='Qiskit Sample - 3-qubit GHZ circuit', num_qubits='3', qiskit='True'))])
{'000': 505, '001': 6, '010': 1, '011': 1, '100': 1, '101': 10, '110': 11, '111': 488}
Importante
Atualmente, não há suporte para o envio de vários circuitos em um único trabalho. Como solução alternativa, você pode chamar o método para enviar cada circuito de forma assíncrona backend.run
e, em seguida, buscar os resultados de cada trabalho. Por exemplo:
jobs = []
for circuit in circuits:
jobs.append(backend.run(circuit, shots=N))
results = []
for job in jobs:
results.append(job.result())
Pré-requisitos
- Uma conta do Azure com uma subscrição ativa. Se não tiver uma conta do Azure, registe-se gratuitamente e inscreva-se numa subscrição pré-paga.
- Um espaço de trabalho do Azure Quantum. Para obter mais informações, consulte Criar um espaço de trabalho do Azure Quantum.
Criar um novo bloco de notas na sua área de trabalho
- Faça logon no portal do Azure e selecione o espaço de trabalho criado na etapa anterior.
- Na folha esquerda, selecione Blocos de Notas.
- Clique em Meus Blocos de Anotações e clique em Adicionar Novo.
- Digite um nome para o arquivo, por exemplo Qiskit.ipynb, e clique em Criar arquivo.
Quando o novo bloco de notas é aberto, cria automaticamente o código para a primeira célula, com base na sua subscrição e nas informações da área de trabalho.
from azure.quantum import Workspace
workspace = Workspace (
resource_id = "", # Add your resource_id
location = "" # Add your workspace location (for example, "westus")
)
Nota
Salvo indicação em contrário, deve executar cada célula em ordem à medida que a cria para evitar quaisquer problemas de compilação.
Clique no ícone triangular "play" à esquerda da célula para executar o código.
Carregue as importações necessárias
Primeiro, você precisará importar alguns módulos adicionais.
Clique em + Código para adicionar uma nova célula e, em seguida, adicione e execute o seguinte código:
from qiskit import QuantumCircuit
from qiskit.visualization import plot_histogram
from azure.quantum.qiskit import AzureQuantumProvider
Conectar-se ao serviço Azure Quantum
Em seguida, crie um AzureQuantumProvider
objeto usando o Workspace
objeto da célula anterior para se conectar ao seu espaço de trabalho do Azure Quantum. Adicione uma nova célula com o seguinte código:
provider = AzureQuantumProvider(workspace)
Definir um circuito simples
Em uma nova célula, crie um circuit
objeto. Este exemplo é um gerador de bits aleatórios quânticos simples. Adicione o seguinte código para definir e exibir o circuito:
# Create a Quantum Circuit acting on the q register
circuit = QuantumCircuit(3, 3)
circuit.name = "Qiskit Sample - 3-qubit GHZ circuit"
circuit.h(0)
circuit.cx(0, 1)
circuit.cx(1, 2)
circuit.measure([0, 1, 2], [0, 1, 2])
# Print out the circuit
circuit.draw()
┌───┐ ┌─┐
q_0: ┤ H ├──■───────┤M├──────
└───┘┌─┴─┐ └╥┘┌─┐
q_1: ─────┤ X ├──■───╫─┤M├───
└───┘┌─┴─┐ ║ └╥┘┌─┐
q_2: ──────────┤ X ├─╫──╫─┤M├
└───┘ ║ ║ └╥┘
c: 3/════════════════╩══╩══╩═
0 1 2
Listar tudo targets
Agora você pode exibir toda a computação targetsquântica, ou back-ends, que estão disponíveis em seu espaço de trabalho. Adicione uma nova célula e execute a seguinte linha:
print("This workspace's targets:")
for backend in provider.backends():
print("- " + backend.name())
This workspace's targets:
- ionq.simulator
- ionq.qpu.aria-1
- microsoft.estimator
- quantinuum.qpu.h1-1
- quantinuum.sim.h1-1sc
- quantinuum.sim.h1-1e
- rigetti.sim.qvm
- rigetti.qpu.ankaa-9q-3
- rigetti.qpu.ankaa-3
Selecione um target para executar o programa
Para verificar seu código antes de executá-lo em hardware quântico real, você pode usar o simulador quântico IonQ, ionq.simulator
.
Adicione uma nova célula e crie um objeto para representar o simulador targetquântico IonQ:
# Get IonQ quantum simulator target:
simulator_backend = provider.get_backend("ionq.simulator")
Executar no simulador IonQ
Para executar o seu circuito no simulador, adicione o seguinte código. Este exemplo usa o run
método do para enviar seu trabalho e, em seguida, monitora o status do target trabalho.
# Submit the circuit to run on Azure Quantum
job = simulator_backend.run(circuit, shots=8)
job_id = job.id()
print("Job id", job_id)
Quando o trabalho for executado com êxito, obtenha os resultados do trabalho e exiba-os:
# Get the job results:
result = job.result()
print(result)
Result(backend_name='ionq.simulator', backend_version='1', qobj_id='Qiskit Sample - 3-qubit GHZ circuit', job_id='00000000-0000-0000-0000-000000000000', success=True, results=[ExperimentResult(shots=8, success=True, meas_level=2, data=ExperimentResultData(counts={'000': 4, '111': 4}, memory=['000', '000', '000', '000', '111', '111', '111', '111'], probabilities={'000': 0.5, '111': 0.5}), header=QobjExperimentHeader(name='Qiskit Sample - 3-qubit GHZ circuit', num_qubits=3, metadata={}), status=JobStatus.DONE, name='Qiskit Sample - 3-qubit GHZ circuit')], date=None, status=None, header=None, error_data=None)
Como esse result
tipo é um objeto nativo do pacote Qiskit, você pode usar o Qiskit result.get_counts
e plot_histogram
visualizar os resultados. Para certificar-se de que todos os rótulos de cadeia de bits possíveis estão representados, adicione-os ao counts
.
counts = {format(n, "03b"): 0 for n in range(8)}
counts.update(result.get_counts(circuit))
print(counts)
plot_histogram(counts)
{'000': 4, '001': 0, '010': 0, '011': 0, '100': 0, '101': 0, '110': 0, '111': 4}
Você também pode usar a get_memory()
função para exibir dados de captura individuais do trabalho
result.get_memory(circuit)
['000', '000', '000', '000', '111', '111', '111', '111']
Nota
No IonQ targets, se você enviar um trabalho com um número ímpar de disparos, os resultados serão arredondados para baixo para o próximo número par. Por exemplo, se você especificar 9 disparos, os resultados exibirão dados para 8 disparos.
Estimar o custo do trabalho
Antes de executar um trabalho em hardware quântico real, ou uma unidade de processamento quântico (QPU), você deve estimar quanto custa executar.
Para obter os detalhes de preços mais atuais, consulte Preços IonQ ou localize seu espaço de trabalho e visualize as opções de preços na folha Provedores do seu espaço de trabalho.
Executar em IonQ QPU
Depois de executar seu trabalho com sucesso no simulador IonQ e estimar o custo QPU, é hora de executar seu circuito no hardware.
Nota
O tempo necessário para executar um circuito na QPU varia dependendo dos tempos de fila atuais. Você pode exibir o tempo médio de fila para um target selecionando a folha Provedores do seu espaço de trabalho.
Use o mesmo run
método e operações que você usou anteriormente com o Validador de API para enviar e monitorar seu trabalho:
# Submit the circuit to run on Azure Quantum
job = qpu_backend.run(circuit, shots=100)
job_id = job.id()
print("Job id", job_id)
Quando o trabalho terminar, obtenha os resultados do trabalho como antes e exiba-os em um gráfico:
result = job.result()
print(result)
counts = {format(n, "03b"): 0 for n in range(8)}
counts.update(result.get_counts(circuit))
print(counts)
plot_histogram(counts)
Job id 910b5ac8-98cd-11ec-b3ba-00155d5528cf
Job Status: job has successfully run
Result(backend_name='ionq.simulator', backend_version='1', qobj_id='Qiskit Sample - 3-qubit GHZ circuit', job_id='Job id 54e8c740-98d9-11ec-b382-00155d957f5d', success=True, results=[ExperimentResult(shots=1024, success=True, meas_level=2, data=ExperimentResultData(counts={'0': 505, '1': 6, '2': 1, '3': 1, '4': 1, '5': 10, '6': 11, '7': 488}, probabilities={'0': 0.4932, '1': 0.0059, '2': 0.001, '3': 0.001, '4': 0.001, '5': 0.0098, '6': 0.0117, '7': 0.4766}), header=QobjExperimentHeader(name='Qiskit Sample - 3-qubit GHZ circuit', num_qubits='3', qiskit='True'))])
{'000': 505, '001': 6, '010': 1, '011': 1, '100': 1, '101': 10, '110': 11, '111': 488}
Importante
Atualmente, não há suporte para o envio de vários circuitos em um único trabalho. Como solução alternativa, você pode chamar o método para enviar cada circuito de forma assíncrona backend.run
e, em seguida, buscar os resultados de cada trabalho. Por exemplo:
jobs = []
for circuit in circuits:
jobs.append(backend.run(circuit, shots=N))
results = []
for job in jobs:
results.append(job.result())
Pré-requisitos
Para obter detalhes da instalação, consulte Configurar a extensão QDK.
Um Python ambiente com Python e Pip instalado.
VS Code com o Azure Quantum Development Kit e Python extensões instaladas.
O pacote Azure Quantum
qsharp
Python com asqiskit
tags ewidget
.python pip install "qsharp[qiskit,widgets]>=1.9"
Execute um circuito básico
No VS Code, abra um novo Python arquivo para definir e executar um circuito básico usando o simulador esparso integrado no pacote Qsharp.
# load the required imports
from qiskit.circuit.random import random_circuit
from qsharp.interop.qiskit import QSharpBackend
# define and display the circuit
circuit = random_circuit(2, 2, measure=True)
print(circuit)
# run the circuit using the built-in sparse simulator
backend = QSharpBackend()
job = backend.run(circuit)
counts = job.result().get_counts()
print(counts)
Para executar o programa, selecione o ícone Executar no canto superior direito e selecione Executar Python arquivo. A saída é exibida em uma nova janela do terminal.
┌─────────────────────────┐┌─┐
q_0: ─■───────────┤0 ├┤M├───
│P(0.79983) │ (XX-YY)(1.9337,1.7385) │└╥┘┌─┐
q_1: ─■───────────┤1 ├─╫─┤M├
└─────────────────────────┘ ║ └╥┘
c: 2/═════════════════════════════════════════╩══╩═
0 1
{'11': 680, '00': 344}
Gerar QIR para o circuito
Usando esse mesmo circuito, você pode gerar QIR que é usado para executar em hardware quântico.
Nota
Ao gerar QIR, todos os registros devem ter sido medidos. Se houver algum registro não utilizado, um erro será gerado. Além disso, tentar gerar QIR quando o perfil está definido como Unrestricted
gera um erro. O perfil Unrestricted
só é válido para simulação. TargetProfile.Base ou TargetProfile.Adaptive_RI
devem ser usados. O target_profile
pode ser substituído na chamada backend.qir(...)
para alternar perfis.
Importar
QSharpError
eTargetProfile
from qsharp import QSharpError, TargetProfile
Modificar a saída para gerar QIR
print(backend.qir(circuit, target_profile=TargetProfile.Adaptive_RI))
Seu código agora deve ter esta aparência:
# load the required imports
from qiskit.circuit.random import random_circuit
from qsharp.interop.qiskit import QSharpBackend
from qsharp import QSharpError, TargetProfile
# define and display the circuit
circuit = random_circuit(2, 2, measure=True)
print(circuit)
# generate QIR for the circuit
print(backend.qir(circuit, target_profile=TargetProfile.Adaptive_RI))
com a saída:
┌────────────┐ ┌─┐
q_0: ┤ Rx(2.7195) ├─■───────────┤M├───
└──┬─────┬───┘ │U1(5.5924) └╥┘┌─┐
q_1: ───┤ Tdg ├─────■────────────╫─┤M├
└─────┘ ║ └╥┘
c: 2/════════════════════════════╩══╩═
0 1
%Result = type opaque
%Qubit = type opaque
define void @ENTRYPOINT__main() #0 {
block_0:
call void @__quantum__qis__rx__body(double 2.7194945105768586, %Qubit* inttoptr (i64 0 to %Qubit*))
call void @__quantum__qis__rz__body(double 2.796204066686262, %Qubit* inttoptr (i64 0 to %Qubit*))
call void @__quantum__qis__t__adj(%Qubit* inttoptr (i64 1 to %Qubit*))
call void @__quantum__qis__cx__body(%Qubit* inttoptr (i64 0 to %Qubit*), %Qubit* inttoptr (i64 1 to %Qubit*))
call void @__quantum__qis__rz__body(double -2.796204066686262, %Qubit* inttoptr (i64 1 to %Qubit*))
call void @__quantum__qis__cx__body(%Qubit* inttoptr (i64 0 to %Qubit*), %Qubit* inttoptr (i64 1 to %Qubit*))
call void @__quantum__qis__rz__body(double 2.796204066686262, %Qubit* inttoptr (i64 1 to %Qubit*))
call void @__quantum__qis__m__body(%Qubit* inttoptr (i64 0 to %Qubit*), %Result* inttoptr (i64 0 to %Result*))
call void @__quantum__qis__m__body(%Qubit* inttoptr (i64 1 to %Qubit*), %Result* inttoptr (i64 1 to %Result*))
call void @__quantum__rt__array_record_output(i64 2, i8* null)
call void @__quantum__rt__result_record_output(%Result* inttoptr (i64 1 to %Result*), i8* null)
call void @__quantum__rt__result_record_output(%Result* inttoptr (i64 0 to %Result*), i8* null)
ret void
}
declare void @__quantum__qis__rx__body(double, %Qubit*)
declare void @__quantum__qis__rz__body(double, %Qubit*)
declare void @__quantum__qis__t__adj(%Qubit*)
declare void @__quantum__qis__cx__body(%Qubit*, %Qubit*)
declare void @__quantum__qis__m__body(%Qubit*, %Result*) #1
declare void @__quantum__rt__array_record_output(i64, i8*)
declare void @__quantum__rt__result_record_output(%Result*, i8*)
attributes #0 = { "entry_point" "output_labeling_schema" "qir_profiles"="adaptive_profile" "required_num_qubits"="2" "required_num_results"="2" }
attributes #1 = { "irreversible" }
; module flags
!llvm.module.flags = !{!0, !1, !2, !3, !4, !5, !6, !7, !8, !9, !10}
!0 = !{i32 1, !"qir_major_version", i32 1}
!1 = !{i32 7, !"qir_minor_version", i32 0}
!2 = !{i32 1, !"dynamic_qubit_management", i1 false}
!3 = !{i32 1, !"dynamic_result_management", i1 false}
!4 = !{i32 1, !"classical_ints", i1 true}
!5 = !{i32 1, !"qubit_resetting", i1 true}
!6 = !{i32 1, !"classical_floats", i1 false}
!7 = !{i32 1, !"backwards_branching", i1 false}
!8 = !{i32 1, !"classical_fixed_points", i1 false}
!9 = !{i32 1, !"user_functions", i1 false}
!10 = !{i32 1, !"multiple_target_branching", i1 false}
Nem todos os programas podem ser executados em todo o hardware. Aqui você pode tentar target o perfil Base
, mas obteremos erros detalhados sobre quais partes do programa não são suportadas.
try:
backend.qir(qc, target_profile=TargetProfile.Base)
except QSharpError as e:
print(e)