L'algorithme de Deutsch-Jozsa
L'algorithme de Deutsch surpasse tous les algorithmes classiques pour un problème de requête, mais l'avantage est assez modeste : une requête contre deux. L'algorithme de Deutsch-Jozsa étend cet avantage — et, en fait, il peut être utilisé pour résoudre deux problèmes de requête différents.
Voici une description par circuit quantique de l'algorithme de Deutsch-Jozsa. Une étape supplémentaire de post-traitement classique, non représentée dans la figure, peut également être nécessaire selon le problème spécifique à résoudre.
Bien sûr, nous n'avons pas encore évoqué les problèmes que cet algorithme résout ; c'est l'objet des deux sections qui suivent.
Le problème de Deutsch-Jozsa
Nous allons commencer par le problème de requête que l'algorithme de Deutsch-Jozsa était initialement conçu pour résoudre, connu sous le nom de problème de Deutsch-Jozsa.
La fonction d'entrée de ce problème prend la forme pour un entier positif arbitraire Comme pour le problème de Deutsch, la tâche consiste à produire si est constante et si est équilibrée, ce qui signifie que le nombre de chaînes d'entrée pour lesquelles la fonction prend la valeur est égal au nombre de chaînes d'entrée pour lesquelles la fonction prend la valeur .
Remarque : lorsque est supérieur à il existe des fonctions de la forme qui ne sont ni constantes ni équilibrées. Par exemple, la fonction définie par
n'appartient à aucune de ces deux catégories. Pour le problème de Deutsch-Jozsa, on ne se préoccupe tout simplement pas de fonctions comme celle-ci — elles sont considérées comme des entrées « sans importance ». Autrement dit, pour ce problème nous avons une promesse que est soit constante soit équilibrée.
L'algorithme de Deutsch-Jozsa, avec sa requête unique, résout ce problème de la façon suivante : si chacun des résultats de mesure vaut alors la fonction est constante ; sinon, si au moins un des résultats de mesure vaut alors la fonction est équilibrée. Une autre façon de le formuler est que le circuit décrit ci-dessus est suivi d'une étape de post-traitement classique dans laquelle le OU des résultats de mesure est calculé pour produire la sortie.
Analyse de l'algorithme
Pour analyser les performances de l'algorithme de Deutsch-Jozsa sur le problème de Deutsch-Jozsa, il est utile de commencer par réfléchir à l'action d'une seule couche de portes Hadamard. Une opération de Hadamard peut être exprimée sous forme de matrice de la manière habituelle,
mais on peut aussi exprimer cette opération à travers son action sur les états de la base standard :
Ces deux équations peuvent être regroupées en une seule formule,
qui est valide pour les deux choix de
Supposons maintenant qu'au lieu d'un seul qubit on dispose de qubits, et qu'une opération de Hadamard est appliquée à chacun. L'opération combinée sur les qubits est décrite par le produit tensoriel ( fois), que l'on note par concision et clarté. En utilisant la formule ci-dessus, puis en développant et en simplifiant, on peut exprimer l'action de cette opération combinée sur les états de la base standard de qubits comme suit :
Notons ici que l'on écrit les chaînes binaires de longueur sous la forme et en suivant la convention d'indexation de Qiskit.
Cette formule nous fournit un outil utile pour analyser le circuit quantique ci-dessus. Après l'exécution de la première couche de portes Hadamard, l'état des qubits (y compris le qubit le plus à gauche/bas, traité séparément des autres) est
Lorsque l'opération est appliquée, cet état est transformé en
par exactement le même phénomène de retour de phase que celui observé dans l'analyse de l'algorithme de Deutsch.
Ensuite, la deuxième couche de portes Hadamard est appliquée, ce qui (d'après la formule ci-dessus) transforme cet état en
Cette expression semble assez complexe, et sans mieux connaître la fonction on ne peut pas conclure grand-chose sur les probabilités d'obtenir différents résultats de mesure.
Heureusement, tout ce dont on a besoin est la probabilité que chacun des résultats de mesure soit — car c'est la probabilité que l'algorithme détermine que est constante. Cette probabilité a une formule simple.
Plus en détail, si est constante, alors soit pour toute chaîne auquel cas la valeur de la somme est soit pour toute chaîne auquel cas la valeur de la somme est En divisant par et en prenant le carré de la valeur absolue, on obtient
Si, en revanche, est équilibrée, alors prend la valeur sur la moitié des chaînes et la valeur sur l'autre moitié, donc les termes et de la somme s'annulent et on obtient la valeur
On conclut que l'algorithme fonctionne correctement tant que la promesse est respectée.
Difficulté classique
L'algorithme de Deutsch-Jozsa fonctionne à chaque fois, donnant toujours la bonne réponse lorsque la promesse est respectée, et nécessite une seule requête. Comment cela se compare-t-il aux algorithmes de requête classiques pour le problème de Deutsch-Jozsa ?
Premièrement, tout algorithme classique déterministe qui résout correctement le problème de Deutsch-Jozsa doit faire un nombre exponentiel de requêtes : requêtes sont nécessaires dans le pire des cas. Le raisonnement est le suivant : si un algorithme déterministe interroge sur chaînes différentes ou moins, et obtient la même valeur de fonction à chaque fois, les deux réponses restent possibles. La fonction pourrait être constante, ou elle pourrait être équilibrée mais par malchance toutes les requêtes renvoient la même valeur de fonction.
Cette deuxième possibilité peut sembler peu probable — mais pour les algorithmes déterministes il n'y a pas d'aléatoire ni d'incertitude, donc ils échoueront systématiquement sur certaines fonctions. On dispose donc d'un avantage significatif des algorithmes quantiques sur les algorithmes classiques à cet égard.
Il y a cependant un bémol : les algorithmes classiques probabilistes peuvent résoudre le problème de Deutsch-Jozsa avec une très haute probabilité en utilisant seulement quelques requêtes. En particulier, si l'on choisit quelques chaînes différentes de longueur aléatoirement, et qu'on interroge sur ces chaînes, il est peu probable d'obtenir la même valeur de fonction pour toutes lorsque est équilibrée.
Plus précisément, si on choisit chaînes d'entrée uniformément au hasard, qu'on évalue et qu'on répond si toutes les valeurs de la fonction sont identiques, et sinon, alors on sera toujours correct lorsque est constante, et on se trompera dans le cas où est équilibrée avec une probabilité de seulement Si l'on prend par exemple, cet algorithme répondra correctement avec une probabilité supérieure à %.
Pour cette raison, l'avantage quantique sur le classique reste assez modeste — mais il n'en représente pas moins un avantage quantifiable qui constitue une amélioration par rapport à l'algorithme de Deutsch.
Deutsch-Jozsa avec Qiskit
# Added by doQumentation — required packages for this notebook
!pip install -q numpy qiskit qiskit-aer
from qiskit import QuantumCircuit
from qiskit_aer import AerSimulator
import numpy as np
Pour implémenter l'algorithme de Deutsch-Jozsa dans Qiskit, on va d'abord définir une fonction dj_query qui génère un circuit quantique implémentant une porte de requête, pour une fonction sélectionnée aléatoirement respectant la promesse du problème de Deutsch-Jozsa.
Avec une probabilité de 50 %, la fonction est constante, et avec 50 % de chance la fonction est équilibrée.
Pour chacune de ces deux possibilités, la fonction est choisie uniformément parmi les fonctions de ce type.
L'argument est le nombre de bits d'entrée de la fonction.
def dj_query(num_qubits):
# Create a circuit implementing for a query gate for a random function
# satisfying the promise for the Deutsch-Jozsa problem.
qc = QuantumCircuit(num_qubits + 1)
if np.random.randint(0, 2):
# Flip output qubit with 50% chance
qc.x(num_qubits)
if np.random.randint(0, 2):
# return constant circuit with 50% chance
return qc
# Choose half the possible input strings
on_states = np.random.choice(
range(2**num_qubits), # numbers to sample from
2**num_qubits // 2, # number of samples
replace=False, # makes sure states are only sampled once
)
def add_cx(qc, bit_string):
for qubit, bit in enumerate(reversed(bit_string)):
if bit == "1":
qc.x(qubit)
return qc
for state in on_states:
qc.barrier() # Barriers are added to help visualize how the functions are created.
qc = add_cx(qc, f"{state:0b}")
qc.mcx(list(range(num_qubits)), num_qubits)
qc = add_cx(qc, f"{state:0b}")
qc.barrier()
return qc
On peut afficher le circuit quantique implémentant la porte de requête en utilisant la méthode draw comme d'habitude.
display(dj_query(3).draw(output="mpl"))
Ensuite, on définit une fonction qui crée le circuit de Deutsch-Jozsa, prenant en argument un circuit quantique implémentant une porte de requête.
def compile_circuit(function: QuantumCircuit):
# Compiles a circuit for use in the Deutsch-Jozsa algorithm.
n = function.num_qubits - 1
qc = QuantumCircuit(n + 1, n)
qc.x(n)
qc.h(range(n + 1))
qc.compose(function, inplace=True)
qc.h(range(n))
qc.measure(range(n), range(n))
return qc
Enfin, on définit une fonction qui exécute le circuit de Deutsch-Jozsa une seule fois.
def dj_algorithm(function: QuantumCircuit):
# Determine if a function is constant or balanced.
qc = compile_circuit(function)
result = AerSimulator().run(qc, shots=1, memory=True).result()
measurements = result.get_memory()
if "1" in measurements[0]:
return "balanced"
return "constant"
On peut tester notre implémentation en choisissant une fonction aléatoirement, en affichant le circuit quantique implémentant la porte de requête pour cette fonction, puis en exécutant l'algorithme de Deutsch-Jozsa sur cette fonction.
f = dj_query(3)
display(f.draw("mpl"))
display(dj_algorithm(f))

'balanced'
Le problème de Bernstein-Vazirani
Ensuite, nous allons aborder un problème connu sous le nom de problème de Bernstein-Vazirani. On l'appelle aussi le problème d'échantillonnage de Fourier, bien qu'il existe des formulations plus générales de ce problème qui portent également ce nom.
Commençons par introduire quelques notations. Pour deux chaînes binaires quelconques et de longueur on définit
On appellera cette opération le produit scalaire binaire. Une autre façon de le définir est la suivante.
Remarque : il s'agit d'une opération symétrique, ce qui signifie que le résultat ne change pas si on échange et donc on peut le faire librement lorsque c'est pratique. Il est parfois utile de penser au produit scalaire binaire comme à la parité des bits de aux positions où la chaîne vaut ou de manière équivalente, la parité des bits de aux positions où la chaîne vaut
Avec cette notation, on peut maintenant définir le problème de Bernstein-Vazirani.
On n'a pas besoin d'un nouvel algorithme quantique pour ce problème ; l'algorithme de Deutsch-Jozsa le résout. Par souci de clarté, appelons circuit de Deutsch-Jozsa le circuit quantique ci-dessus, qui n'inclut pas l'étape de post-traitement classique consistant à calculer le OU.
Analyse de l'algorithme
Pour analyser comment le circuit de Deutsch-Jozsa fonctionne pour une fonction respectant la promesse du problème de Bernstein-Vazirani, commençons par une observation rapide. En utilisant le produit scalaire binaire, on peut décrire alternativement l'action de portes Hadamard sur les états de la base standard de qubits comme suit.
Comme dans l'analyse de l'algorithme de Deutsch, cela s'explique par le fait que la valeur pour tout entier ne dépend que de la parité de
En revenant au circuit de Deutsch-Jozsa, après l'exécution de la première couche de portes Hadamard, l'état des qubits est
La porte de requête est ensuite appliquée, ce qui (via le phénomène de retour de phase) transforme l'état en
En utilisant notre formule pour l'action d'une couche de portes Hadamard, on voit que la deuxième couche de portes Hadamard transforme alors cet état en
On peut maintenant faire quelques simplifications dans l'exposant de à l'intérieur de la somme. Il nous est promis que pour une certaine chaîne donc on peut exprimer l'état comme
Comme et sont des valeurs binaires, on peut remplacer l'addition par le ou-exclusif — à nouveau parce que la seule chose qui compte pour un entier dans l'exposant de est de savoir s'il est pair ou impair. En exploitant la symétrie du produit scalaire binaire, on obtient cette expression pour l'état :
(Des parenthèses ont été ajoutées pour la clarté, bien qu'elles ne soient pas strictement nécessaires car il est conventionnel de traiter le produit scalaire binaire comme ayant une priorité plus élevée que le ou-exclusif.)
À ce stade, on va utiliser la formule suivante.
On peut obtenir cette formule à partir d'une formule similaire pour les bits,
combinée avec un développement du produit scalaire binaire et du ou-exclusif bit à bit :
Cela nous permet d'exprimer l'état du circuit juste avant les mesures comme suit :
La dernière étape consiste à utiliser encore une autre formule, qui s'applique à toute chaîne binaire
Ici on utilise une notation simple pour les chaînes que l'on utilisera plusieurs fois encore dans ce cours : est la chaîne tout-zéro de longueur
Une façon simple de vérifier que cette formule est correcte est de considérer les deux cas séparément. Si alors pour toute chaîne donc la valeur de chaque terme de la somme est et on obtient en sommant et en divisant par En revanche, si l'un des bits de vaut alors le produit scalaire binaire est égal à pour exactement la moitié des choix possibles de et à pour l'autre moitié — car la valeur du produit scalaire binaire change (de à ou de à ) si on inverse un bit de à une position où vaut
Si on applique maintenant cette formule pour simplifier l'état du circuit avant les mesures, on obtient
grâce au fait que si et seulement si Ainsi, les mesures révèlent précisément la chaîne que l'on cherche.
Difficulté classique
Alors que le circuit de Deutsch-Jozsa résout le problème de Bernstein-Vazirani avec une seule requête, tout algorithme de requête classique doit faire au moins requêtes pour résoudre ce problème.
On peut le déduire via un argument dit théorique de l'information, qui est très simple dans ce cas. Chaque requête classique révèle un seul bit d'information sur la solution, et il y a bits d'information à découvrir — donc au moins requêtes sont nécessaires.
Il est en fait possible de résoudre le problème de Bernstein-Vazirani classiquement en interrogeant la fonction sur chacune des chaînes ayant un seul dans chaque position possible, et pour tous les autres bits, ce qui révèle les bits de un par un. Ainsi, l'avantage quantique sur le classique pour ce problème est de requête contre requêtes.
Bernstein-Vazirani avec Qiskit
On a déjà implémenté le circuit de Deutsch-Jozsa ci-dessus, et on va l'utiliser ici pour résoudre le problème de Bernstein-Vazirani. On va d'abord définir une fonction qui implémente une porte de requête pour le problème de Bernstein-Vazirani étant donné une chaîne binaire quelconque.
def bv_query(s):
# Create a quantum circuit implementing a query gate for the
# Bernstein-Vazirani problem.
qc = QuantumCircuit(len(s) + 1)
for index, bit in enumerate(reversed(s)):
if bit == "1":
qc.cx(index, len(s))
return qc
display(bv_query("1011").draw(output="mpl"))
On peut maintenant créer une fonction qui exécute le circuit de Deutsch-Jozsa sur la fonction, en utilisant la fonction compile_circuit définie précédemment.
def bv_algorithm(function: QuantumCircuit):
qc = compile_circuit(function)
result = AerSimulator().run(qc, shots=1, memory=True).result()
return result.get_memory()[0]
display(bv_algorithm(bv_query("1011")))
'1011'
Remarque sur la nomenclature
Dans le contexte du problème de Bernstein-Vazirani, il est courant que l'algorithme de Deutsch-Jozsa soit désigné sous le nom d'« algorithme de Bernstein-Vazirani ». C'est quelque peu trompeur, car l'algorithme est l'algorithme de Deutsch-Jozsa, comme Bernstein et Vazirani l'ont clairement indiqué dans leurs travaux.
Ce que Bernstein et Vazirani ont fait après avoir montré que l'algorithme de Deutsch-Jozsa résout le problème de Bernstein-Vazirani (tel qu'il est énoncé ci-dessus) a été de définir un problème beaucoup plus complexe, connu sous le nom de problème d'échantillonnage de Fourier récursif. Il s'agit d'un problème très artificiel où les solutions à différentes instances du problème débloquent effectivement de nouveaux niveaux du problème organisés en une structure arborescente. Le problème de Bernstein-Vazirani n'est essentiellement que le cas de base de ce problème plus complexe.
Le problème d'échantillonnage de Fourier récursif a été le premier exemple connu d'un problème de requête où les algorithmes quantiques ont un avantage dit super-polynomial sur les algorithmes probabilistes, surpassant ainsi l'avantage quantique sur le classique offert par l'algorithme de Deutsch-Jozsa. Intuitivement, la version récursive du problème amplifie l'avantage de contre des algorithmes quantiques à quelque chose de bien plus grand.
L'aspect le plus difficile de l'analyse mathématique établissant cet avantage est de montrer que les algorithmes de requ ête classiques ne peuvent pas résoudre le problème sans faire beaucoup de requêtes. C'est assez typique ; pour de nombreux problèmes, il peut être très difficile d'exclure des approches classiques créatives qui les résolvent efficacement.
Le problème de Simon, et l'algorithme correspondant décrit dans la section suivante, fournit un exemple bien plus simple d'un avantage super-polynomial (et, en fait, exponentiel) des algorithmes quantiques sur les algorithmes classiques, et c'est pourquoi le problème d'échantillonnage de Fourier récursif est moins souvent discuté. Il représente néanmoins un problème computationnel intéressant en lui-même.