-
Notifications
You must be signed in to change notification settings - Fork 128
/
Copy pathgrover.cpp
67 lines (52 loc) · 2.15 KB
/
grover.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
// Grover's searching
// Source: ./examples/grover.cpp
#include <cmath>
#include <iostream>
#include <numeric>
#include <tuple>
#include <vector>
#include "qpp/qpp.hpp"
int main() {
using namespace qpp;
idx n = 10; // number of qubits
std::cout << ">> Grover on n = " << n << " qubits\n";
std::vector<idx> dims(n, 2); // local dimensions
std::vector<idx> subsys(n); // ordered subsystems
std::iota(std::begin(subsys), std::end(subsys), 0);
// number of elements in the database
auto N = static_cast<idx>(std::llround(std::pow(2, n)));
std::cout << ">> Database size: " << N << '\n';
// mark an element randomly
idx marked = randidx(0, N - 1);
std::cout << ">> Marked state: " << marked << " -> ";
std::cout << disp(n2multiidx(marked, dims),
IOManipContainerOpts{}.set_sep(" "))
<< '\n';
ket psi = mket(std::vector<idx>(n, 0)); // computational |0>^\otimes n
// apply H^\otimes n, no aliasing
psi = (kronpow(gt.H, n) * psi).eval();
cmat G = 2 * prj(psi) - gt.Id(N); // Diffusion operator
Timer<> t;
// number of queries
auto nqueries = static_cast<idx>(std::ceil(pi / 4 * std::sqrt(N)));
std::cout << ">> We run " << nqueries << " queries\n";
for (idx i = 0; i < nqueries; ++i) {
psi(marked) = -psi(marked); // apply the oracle first, no aliasing
psi = (G * psi).eval(); // then the diffusion operator, no aliasing
}
// we now measure the state in the computational basis, destructively
auto measured = measure_seq(psi, subsys, dims);
std::cout << ">> Probability of the marked state: "
<< prod(std::get<PROB>(measured)) << '\n';
// sample
std::cout << ">> Let's sample...\n";
auto result = std::get<RES>(measured);
if (result == n2multiidx(marked, dims)) {
std::cout << ">> Hooray, we obtained the correct result: ";
} else {
std::cout << ">> Not there yet... we obtained: ";
}
std::cout << multiidx2n(result, dims) << " -> ";
std::cout << disp(result, IOManipContainerOpts{}.set_sep(" ")) << '\n';
std::cout << ">> Run time: " << t.toc() << " seconds\n";
}