Aller au contenu principal

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.

Deutsch-Jozsa algorithm

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 f:ΣnΣf:\Sigma^n \rightarrow \Sigma pour un entier positif arbitraire n.n. Comme pour le problème de Deutsch, la tâche consiste à produire 00 si ff est constante et 11 si ff est équilibrée, ce qui signifie que le nombre de chaînes d'entrée pour lesquelles la fonction prend la valeur 00 est égal au nombre de chaînes d'entrée pour lesquelles la fonction prend la valeur 11.

Remarque : lorsque nn est supérieur à 1,1, il existe des fonctions de la forme f:ΣnΣf:\Sigma^n \rightarrow \Sigma qui ne sont ni constantes ni équilibrées. Par exemple, la fonction f:Σ2Σf:\Sigma^2\rightarrow\Sigma définie par

f(00)=0f(01)=0f(10)=0f(11)=1\begin{aligned} f(00) & = 0 \\ f(01) & = 0 \\ f(10) & = 0 \\ f(11) & = 1 \end{aligned}

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 ff est soit constante soit équilibrée.

Problème de Deutsch-Jozsa

Entrée : une fonction f:{0,1}n{0,1}f:\{0,1\}^n\rightarrow\{0,1\}
Promesse : ff est soit constante soit équilibrée
Sortie : 00 si ff est constante, 11 si ff est é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 nn résultats de mesure vaut 0,0, alors la fonction ff est constante ; sinon, si au moins un des résultats de mesure vaut 1,1, alors la fonction ff 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,

H=(12121212),H = \begin{pmatrix} \frac{1}{\sqrt{2}} & \frac{1}{\sqrt{2}} \\[2mm] \frac{1}{\sqrt{2}} & -\frac{1}{\sqrt{2}} \end{pmatrix},

mais on peut aussi exprimer cette opération à travers son action sur les états de la base standard :

H0=120+121H1=120121.\begin{aligned} H \vert 0\rangle & = \frac{1}{\sqrt{2}} \vert 0 \rangle + \frac{1}{\sqrt{2}} \vert 1 \rangle\\[3mm] H \vert 1\rangle & = \frac{1}{\sqrt{2}} \vert 0 \rangle - \frac{1}{\sqrt{2}} \vert 1 \rangle. \end{aligned}

Ces deux équations peuvent être regroupées en une seule formule,

Ha=120+12(1)a1=12b{0,1}(1)abb,H \vert a \rangle = \frac{1}{\sqrt{2}} \vert 0 \rangle + \frac{1}{\sqrt{2}} (-1)^a \vert 1 \rangle = \frac{1}{\sqrt{2}} \sum_{b\in\{0,1\}} (-1)^{ab} \vert b\rangle,

qui est valide pour les deux choix de aΣ.a\in\Sigma.

Supposons maintenant qu'au lieu d'un seul qubit on dispose de nn qubits, et qu'une opération de Hadamard est appliquée à chacun. L'opération combinée sur les nn qubits est décrite par le produit tensoriel HHH\otimes \cdots \otimes H (nn fois), que l'on note HnH^{\otimes n} 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 nn qubits comme suit :

Hnxn1x1x0=(Hxn1)(Hx0)=(12yn1Σ(1)xn1yn1yn1)(12y0Σ(1)x0y0y0)=12nyn1y0Σn(1)xn1yn1++x0y0yn1y0.\begin{aligned} & H^{\otimes n} \vert x_{n-1} \cdots x_1 x_0 \rangle \\ & \qquad = \bigl(H \vert x_{n-1} \rangle \bigr) \otimes \cdots \otimes \bigl(H \vert x_{0} \rangle \bigr) \\ & \qquad = \Biggl( \frac{1}{\sqrt{2}} \sum_{y_{n-1}\in\Sigma} (-1)^{x_{n-1} y_{n-1}} \vert y_{n-1} \rangle \Biggr) \otimes \cdots \otimes \Biggl( \frac{1}{\sqrt{2}} \sum_{y_{0}\in\Sigma} (-1)^{x_{0} y_{0}} \vert y_{0} \rangle \Biggr) \\ & \qquad = \frac{1}{\sqrt{2^n}} \sum_{y_{n-1}\cdots y_0 \in \Sigma^n} (-1)^{x_{n-1}y_{n-1} + \cdots + x_0 y_0} \vert y_{n-1} \cdots y_0 \rangle. \end{aligned}

Notons ici que l'on écrit les chaînes binaires de longueur nn sous la forme xn1x0x_{n-1}\cdots x_0 et yn1y0,y_{n-1}\cdots y_0, 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 n+1n+1 qubits (y compris le qubit le plus à gauche/bas, traité séparément des autres) est

(H1)(Hn00)=12nxn1x0Σnxn1x0.\bigl( H \vert 1 \rangle \bigr) \bigl( H^{\otimes n} \vert 0 \cdots 0 \rangle \bigr) = \vert - \rangle \otimes \frac{1}{\sqrt{2^n}} \sum_{x_{n-1}\cdots x_0 \in \Sigma^n} \vert x_{n-1} \cdots x_0 \rangle.

Lorsque l'opération UfU_f est appliquée, cet état est transformé en

12nxn1x0Σn(1)f(xn1x0)xn1x0\vert - \rangle \otimes \frac{1}{\sqrt{2^n}} \sum_{x_{n-1}\cdots x_0 \in \Sigma^n} (-1)^{f(x_{n-1}\cdots x_0)} \vert x_{n-1} \cdots x_0 \rangle

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

12nxn1x0Σnyn1y0Σn(1)f(xn1x0)+xn1yn1++x0y0yn1y0.\vert - \rangle \otimes \frac{1}{2^n} \sum_{x_{n-1}\cdots x_0 \in \Sigma^n} \sum_{y_{n-1}\cdots y_0 \in \Sigma^n} (-1)^{f(x_{n-1}\cdots x_0) + x_{n-1}y_{n-1} + \cdots + x_0 y_0} \vert y_{n-1} \cdots y_0 \rangle.

Cette expression semble assez complexe, et sans mieux connaître la fonction f,f, 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 00 — car c'est la probabilité que l'algorithme détermine que ff est constante. Cette probabilité a une formule simple.

12nxn1x0Σn(1)f(xn1x0)2={1if f is constant0if f is balanced\Biggl\vert \frac{1}{2^n} \sum_{x_{n-1}\cdots x_0 \in \Sigma^n} (-1)^{f(x_{n-1}\cdots x_0)} \Biggr\vert^2 = \begin{cases} 1 & \text{if $f$ is constant}\\[1mm] 0 & \text{if $f$ is balanced} \end{cases}

Plus en détail, si ff est constante, alors soit f(xn1x0)=0f(x_{n-1}\cdots x_0) = 0 pour toute chaîne xn1x0,x_{n-1}\cdots x_0, auquel cas la valeur de la somme est 2n,2^n, soit f(xn1x0)=1f(x_{n-1}\cdots x_0) = 1 pour toute chaîne xn1x0,x_{n-1}\cdots x_0, auquel cas la valeur de la somme est 2n.-2^n. En divisant par 2n2^n et en prenant le carré de la valeur absolue, on obtient 1.1.

Si, en revanche, ff est équilibrée, alors ff prend la valeur 00 sur la moitié des chaînes xn1x0x_{n-1}\cdots x_0 et la valeur 11 sur l'autre moitié, donc les termes +1+1 et 1-1 de la somme s'annulent et on obtient la valeur 0.0.

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 : 2n1+12^{n-1} + 1 requêtes sont nécessaires dans le pire des cas. Le raisonnement est le suivant : si un algorithme déterministe interroge ff sur 2n12^{n-1} 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 nn aléatoirement, et qu'on interroge ff sur ces chaînes, il est peu probable d'obtenir la même valeur de fonction pour toutes lorsque ff est équilibrée.

Plus précisément, si on choisit kk chaînes d'entrée x1,,xkΣnx^1,\ldots,x^k \in \Sigma^n uniformément au hasard, qu'on évalue f(x1),,f(xk),f(x^1),\ldots,f(x^k), et qu'on répond 00 si toutes les valeurs de la fonction sont identiques, et 11 sinon, alors on sera toujours correct lorsque ff est constante, et on se trompera dans le cas où ff est équilibrée avec une probabilité de seulement 2k+1.2^{-k + 1}. Si l'on prend k=11,k = 11, par exemple, cet algorithme répondra correctement avec une probabilité supérieure à 99,999{,}9%.

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"))

Output of the previous code cell

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))

Output of the previous code cell

'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 x=xn1x0x = x_{n-1} \cdots x_0 et y=yn1y0y = y_{n-1}\cdots y_0 de longueur n,n, on définit

xy=xn1yn1x0y0.x \cdot y = x_{n-1} y_{n-1} \oplus \cdots \oplus x_0 y_0.

On appellera cette opération le produit scalaire binaire. Une autre façon de le définir est la suivante.

xy={1xn1yn1++x0y0 is odd0xn1yn1++x0y0 is evenx \cdot y = \begin{cases} 1 & x_{{n-1}} y_{n-1} + \cdots + x_0 y_0 \text{ is odd}\\[0.5mm] 0 & x_{{n-1}} y_{n-1} + \cdots + x_0 y_0 \text{ is even} \end{cases}

Remarque : il s'agit d'une opération symétrique, ce qui signifie que le résultat ne change pas si on échange xx et y,y, donc on peut le faire librement lorsque c'est pratique. Il est parfois utile de penser au produit scalaire binaire xyx \cdot y comme à la parité des bits de xx aux positions où la chaîne yy vaut 1,1, ou de manière équivalente, la parité des bits de yy aux positions où la chaîne xx vaut 1.1.

Avec cette notation, on peut maintenant définir le problème de Bernstein-Vazirani.

Problème de Bernstein-Vazirani

Entrée : une fonction f:{0,1}n{0,1}f:\{0,1\}^n\rightarrow\{0,1\}
Promesse : il existe une chaîne binaire s=sn1s0s = s_{n-1} \cdots s_0 telle que f(x)=sxf(x) = s\cdot x pour tout xΣnx\in\Sigma^n
Sortie : la chaîne ss

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 nn portes Hadamard sur les états de la base standard de nn qubits comme suit.

Hnx=12nyΣn(1)xyyH^{\otimes n} \vert x \rangle = \frac{1}{\sqrt{2^n}} \sum_{y\in\Sigma^n} (-1)^{x\cdot y} \vert y\rangle

Comme dans l'analyse de l'algorithme de Deutsch, cela s'explique par le fait que la valeur (1)k(-1)^k pour tout entier kk ne dépend que de la parité de k.k.

En revenant au circuit de Deutsch-Jozsa, après l'exécution de la première couche de portes Hadamard, l'état des n+1n+1 qubits est

12nxΣnx.\vert - \rangle \otimes \frac{1}{\sqrt{2^n}} \sum_{x \in \Sigma^n} \vert x \rangle.

La porte de requête est ensuite appliquée, ce qui (via le phénomène de retour de phase) transforme l'état en

12nxΣn(1)f(x)x.\vert - \rangle \otimes \frac{1}{\sqrt{2^n}} \sum_{x \in \Sigma^n} (-1)^{f(x)} \vert x \rangle.

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

12nxΣnyΣn(1)f(x)+xyy.\vert - \rangle \otimes \frac{1}{2^n} \sum_{x \in \Sigma^n} \sum_{y \in \Sigma^n} (-1)^{f(x) + x \cdot y} \vert y \rangle.

On peut maintenant faire quelques simplifications dans l'exposant de 1-1 à l'intérieur de la somme. Il nous est promis que f(x)=sxf(x) = s\cdot x pour une certaine chaîne s=sn1s0,s = s_{n-1} \cdots s_0, donc on peut exprimer l'état comme

12nxΣnyΣn(1)sx+xyy.\vert - \rangle \otimes \frac{1}{2^n} \sum_{x \in \Sigma^n} \sum_{y \in \Sigma^n} (-1)^{s\cdot x + x \cdot y} \vert y \rangle.

Comme sxs\cdot x et xyx\cdot y 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 1-1 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 :

12nxΣnyΣn(1)(sx)(yx)y.\vert - \rangle \otimes \frac{1}{2^n} \sum_{x \in \Sigma^n} \sum_{y \in \Sigma^n} (-1)^{(s\cdot x) \oplus (y \cdot x)} \vert y \rangle.

(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.

(sx)(yx)=(sy)x(s\cdot x) \oplus (y \cdot x) = (s \oplus y) \cdot x

On peut obtenir cette formule à partir d'une formule similaire pour les bits,

(ac)(bc)=(ab)c,(a c) \oplus (b c) = (a \oplus b) c,

combinée avec un développement du produit scalaire binaire et du ou-exclusif bit à bit :

(sx)(yx)=(sn1xn1)(s0x0)(yn1xn1)(y0x0)=(sn1yn1)xn1(s0y0)x0=(sy)x\begin{aligned} (s\cdot x) \oplus (y \cdot x) & = (s_{n-1} x_{n-1}) \oplus \cdots \oplus (s_{0} x_{0}) \oplus (y_{n-1} x_{n-1}) \oplus \cdots \oplus (y_{0} x_{0}) \\ & = (s_{n-1} \oplus y_{n-1}) x_{n-1} \oplus \cdots \oplus (s_{0} \oplus y_{0}) x_{0} \\ & = (s \oplus y) \cdot x \end{aligned}

Cela nous permet d'exprimer l'état du circuit juste avant les mesures comme suit :

12nxΣnyΣn(1)(sy)xy.\vert - \rangle \otimes \frac{1}{2^n} \sum_{x \in \Sigma^n} \sum_{y \in \Sigma^n} (-1)^{(s\oplus y)\cdot x} \vert y \rangle.

La dernière étape consiste à utiliser encore une autre formule, qui s'applique à toute chaîne binaire z=zn1z0.z = z_{n-1}\cdots z_0.

12nxΣn(1)zx={1if z=0n0if z0n\frac{1}{2^n} \sum_{x \in \Sigma^n} (-1)^{z \cdot x} = \begin{cases} 1 & \text{if $z = 0^n$}\\ 0 & \text{if $z\neq 0^n$} \end{cases}

Ici on utilise une notation simple pour les chaînes que l'on utilisera plusieurs fois encore dans ce cours : 0n0^n est la chaîne tout-zéro de longueur n.n.

Une façon simple de vérifier que cette formule est correcte est de considérer les deux cas séparément. Si z=0n,z = 0^n, alors zx=0z\cdot x = 0 pour toute chaîne xΣn,x\in\Sigma^n, donc la valeur de chaque terme de la somme est 1,1, et on obtient 11 en sommant et en divisant par 2n.2^n. En revanche, si l'un des bits de zz vaut 1,1, alors le produit scalaire binaire zxz\cdot x est égal à 00 pour exactement la moitié des choix possibles de xΣnx\in\Sigma^n et à 11 pour l'autre moitié — car la valeur du produit scalaire binaire zxz\cdot x change (de 00 à 11 ou de 11 à 00) si on inverse un bit de xx à une position où zz vaut 1.1.

Si on applique maintenant cette formule pour simplifier l'état du circuit avant les mesures, on obtient

12nxΣnyΣn(1)(sy)xy=s,\vert - \rangle \otimes \frac{1}{2^n} \sum_{x \in \Sigma^n} \sum_{y \in \Sigma^n} (-1)^{(s\oplus y)\cdot x} \vert y \rangle = \vert - \rangle \otimes \vert s \rangle,

grâce au fait que sy=0ns\oplus y = 0^n si et seulement si y=s.y = s. Ainsi, les mesures révèlent précisément la chaîne ss 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 nn 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 nn bits d'information à découvrir — donc au moins nn 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 nn chaînes ayant un seul 1,1, dans chaque position possible, et 00 pour tous les autres bits, ce qui révèle les bits de ss un par un. Ainsi, l'avantage quantique sur le classique pour ce problème est de 11 requête contre nn 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 ss 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"))

Output of the previous code cell

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 11 contre nn 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.