forked from microsoft/QuantumKatas
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Tasks.qs
245 lines (210 loc) · 12.8 KB
/
Tasks.qs
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
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT license.
namespace Quantum.Kata.GroversAlgorithm {
open Microsoft.Quantum.Intrinsic;
open Microsoft.Quantum.Canon;
open Microsoft.Quantum.Diagnostics;
open Microsoft.Quantum.Convert;
open Microsoft.Quantum.Math;
//////////////////////////////////////////////////////////////////
// Welcome!
//////////////////////////////////////////////////////////////////
// The "Solving SAT problem with Grover's algorithm" quantum kata is a series of exercises designed
// to get you comfortable with using Grover's algorithm to solve realistic problems
// using boolean satisfiability problem (SAT) as an example.
// It covers the following topics:
// - writing oracles implementing boolean expressions and SAT instances,
// - using Grover's algorithm to solve problems with unknown number of solutions.
// Each task is wrapped in one operation preceded by the description of the task.
// Each task (except tasks in which you have to write a test) has a unit test associated with it,
// which initially fails. Your goal is to fill in the blank (marked with // ... comment)
// with some Q# code to make the failing test pass.
// Within each section, tasks are given in approximate order of increasing difficulty;
// harder ones are marked with asterisks.
//////////////////////////////////////////////////////////////////
// Part I. Oracles for SAT problems
//////////////////////////////////////////////////////////////////
// The most interesting part of learning Grover's algorithm is solving realistic problems.
// This means using oracles which express an actual problem and not simply hard-code a known solution.
// In this section we'll learn how to express boolean satisfiability problems as quantum oracles.
// Task 1.1. The AND oracle: f(x) = x₀ ∧ x₁
// Inputs:
// 1) 2 qubits in an arbitrary state |x⟩ (input/query register)
// 2) a qubit in an arbitrary state |y⟩ (target qubit)
// Goal: Transform state |x, y⟩ into state |x, y ⊕ f(x)⟩ (⊕ is addition modulo 2),
// i.e., flip the target state if all qubits of the query register are in the |1⟩ state,
// and leave it unchanged otherwise.
// Leave the query register in the same state it started in.
// Stretch goal: Can you implement the oracle so that it would work
// for queryRegister containing an arbitrary number of qubits?
operation Oracle_And (queryRegister : Qubit[], target : Qubit) : Unit is Adj {
// ...
}
// Task 1.2. The OR oracle: f(x) = x₀ ∨ x₁
// Inputs:
// 1) 2 qubits in an arbitrary state |x⟩ (input/query register)
// 2) a qubit in an arbitrary state |y⟩ (target qubit)
// Goal: Transform state |x, y⟩ into state |x, y ⊕ f(x)⟩ (⊕ is addition modulo 2),
// i.e., flip the target state if at least one qubit of the query register is in the |1⟩ state,
// and leave it unchanged otherwise.
// Leave the query register in the same state it started in.
// Stretch goal: Can you implement the oracle so that it would work
// for queryRegister containing an arbitrary number of qubits?
operation Oracle_Or (queryRegister : Qubit[], target : Qubit) : Unit is Adj {
// ...
}
// Task 1.3. The XOR oracle: f(x) = x₀ ⊕ x₁
// Inputs:
// 1) 2 qubits in an arbitrary state |x⟩ (input/query register)
// 2) a qubit in an arbitrary state |y⟩ (target qubit)
// Goal: Transform state |x, y⟩ into state |x, y ⊕ f(x)⟩ (⊕ is addition modulo 2),
// i.e., flip the target state if the qubits of the query register are in different states,
// and leave it unchanged otherwise.
// Leave the query register in the same state it started in.
// Stretch goal: Can you implement the oracle so that it would work
// for queryRegister containing an arbitrary number of qubits?
operation Oracle_Xor (queryRegister : Qubit[], target : Qubit) : Unit is Adj {
// ...
}
// Task 1.4. Alternating bits oracle: f(x) = (x₀ ⊕ x₁) ∧ (x₁ ⊕ x₂) ∧ ... ∧ (xₙ₋₂ ⊕ xₙ₋₁)
// Inputs:
// 1) N qubits in an arbitrary state |x⟩ (input/query register)
// 2) a qubit in an arbitrary state |y⟩ (target qubit)
// Goal: Transform state |x, y⟩ into state |x, y ⊕ f(x)⟩ (⊕ is addition modulo 2).
// Leave the query register in the same state it started in.
//
// Note that this oracle marks two states similar to the state explored in task 1.2 of GroversAlgorithm kata:
// |10101...⟩ and |01010...⟩
// It is possible (and quite straightforward) to implement this oracle based on this observation;
// however, for the purposes of learning to write oracles to solve SAT problems we recommend using the representation above.
operation Oracle_AlternatingBits (queryRegister : Qubit[], target : Qubit) : Unit is Adj {
// ...
}
// Task 1.5. Evaluate one clause of a SAT formula
//
// For general SAT problems, f(x) is represented as a conjunction (an AND operation) of several clauses on N variables,
// and each clause is a disjunction (an OR operation) of one or several variables or negated variables:
// clause(x) = ∨ₖ yₖ, where yₖ = either xⱼ or ¬xⱼ for some j in {0, ..., N-1}
//
// For example, one of the possible clauses on two variables is
// clause(x) = x₀ ∨ ¬x₁
//
// Inputs:
// 1) N qubits in an arbitrary state |x⟩ (input/query register)
// 2) a qubit in an arbitrary state |y⟩ (target qubit)
// 3) a 1-dimensional array of tuples "clause" which describes one clause of a SAT problem instance clause(x).
//
// "clause" is an array of one or more tuples, each of them describing one component of the clause.
// Each tuple is an (Int, Bool) pair:
// - the first element is the index j of the variable xⱼ,
// - the second element is true if the variable is included as itself (xⱼ) and false if it is included as a negation (¬xⱼ)
//
// Example:
// The clause clause(x) = x₀ ∨ ¬x₁ can be represented as [(0, true), (1, false)].
operation Oracle_SATClause (queryRegister : Qubit[],
target : Qubit,
clause : (Int, Bool)[]) : Unit is Adj {
// ...
}
// Task 1.6. General SAT problem oracle
//
// For SAT problems, f(x) is represented as a conjunction (an AND operation) of M clauses on N variables,
// and each clause is a disjunction (an OR operation) of one or several variables or negated variables:
// f(x) = ∧ᵢ (∨ₖ yᵢₖ), where yᵢₖ = either xⱼ or ¬xⱼ for some j in {0, ..., N-1}
//
// Inputs:
// 1) N qubits in an arbitrary state |x⟩ (input/query register)
// 2) a qubit in an arbitrary state |y⟩ (target qubit)
// 3) a 2-dimensional array of tuples "problem" which describes the SAT problem instance f(x).
//
// i-th element of "problem" describes the i-th clause of f(x);
// it is an array of one or more tuples, each of them describing one component of the clause.
// Each tuple is an (Int, Bool) pair:
// - the first element is the index j of the variable xⱼ,
// - the second element is true if the variable is included as itself (xⱼ) and false if it is included as a negation (¬xⱼ)
//
// Example:
// A more general case of the OR oracle for 3 variables f(x) = (x₀ ∨ x₁ ∨ x₂) can be represented as [[(0, true), (1, true), (2, true)]].
//
// Goal: Transform state |x, y⟩ into state |x, y ⊕ f(x)⟩ (⊕ is addition modulo 2).
// Leave the query register in the same state it started in.
operation Oracle_SAT (queryRegister : Qubit[],
target : Qubit,
problem : (Int, Bool)[][]) : Unit is Adj {
// ...
}
//////////////////////////////////////////////////////////////////
// Part II. Oracles for exactly-1 3-SAT problem
//////////////////////////////////////////////////////////////////
// The exactly-1 3-SAT problem (also known as "one-in-three 3-SAT") is a variant of a general 3-SAT problem.
// It has a structure similar to a 3-SAT problem, but each clause must have exactly one true literal
// (while in a normal 3-SAT problem each clause must have at least one true literal).
// Task 2.1. "Exactly one |1⟩" oracle
// Inputs:
// 1) 3 qubits in an arbitrary state |x⟩ (input/query register)
// 2) a qubit in an arbitrary state |y⟩ (target qubit)
//
// Goal: Transform the state |x, y⟩ into the state |x, y ⊕ f(x)⟩ (⊕ is addition modulo 2),
// where f(x) = 1 if exactly one bit of x is in the |1⟩ state, and 0 otherwise.
// Leave the query register in the same state it started in.
// Stretch goal: Can you implement the oracle so that it would work
// for queryRegister containing an arbitrary number of qubits?
operation Oracle_Exactly1One (queryRegister : Qubit[], target : Qubit) : Unit is Adj {
// ...
}
// Task 2.2. "Exactly-1 3-SAT" oracle
// Inputs:
// 1) N qubits in an arbitrary state |x⟩ (input/query register)
// 2) a qubit in an arbitrary state |y⟩ (target qubit)
// 3) a 2-dimensional array of tuples "problem" which describes the SAT problem instance f(x).
// "problem" describes the problem instance in the same format as in task 1.6;
// each clause of the formula is guaranteed to have exactly 3 terms.
//
// Goal: Transform state |x, y⟩ into state |x, y ⊕ f(x)⟩ (⊕ is addition modulo 2).
// Leave the query register in the same state it started in.
//
// Example:
// An instance of the problem f(x) = (x₀ ∨ x₁ ∨ x₂) can be represented as [[(0, true), (1, true), (2, true)]],
// and its solutions will be (true, false, false), (false, true, false) and (false, false, true),
// but none of the variable assignments in which more than one variable is true,
// which are solutions for the general SAT problem.
operation Oracle_Exactly1_3SAT (queryRegister : Qubit[],
target : Qubit,
problem : (Int, Bool)[][]) : Unit is Adj {
// Hint: can you reuse parts of the code in section 1?
// ...
}
//////////////////////////////////////////////////////////////////
// Part III. Using Grover's algorithm for problems with multiple solutions
//////////////////////////////////////////////////////////////////
// Task 3.1. Using Grover's algorithm
// Goal: Implement Grover's algorithm and use it to find solutions to SAT instances from parts I and II.
// This task is not covered by a test and allows you to experiment with running the algorithm.
//
// If you want to learn Grover's algorithm itself, try doing the GroversAlgorithm kata first.
operation T31_E2E_GroversAlgorithm_Test () : Unit {
// Hint: Experiment with SAT instances with different number of solutions and the number of algorithm iterations
// to see how the probability of the algorithm finding the correct answer changes depending on these two factors.
// For example,
// - the AND oracle from task 1.1 has exactly one solution,
// - the alternating bits oracle from task 1.4 has exactly two solutions,
// - the OR oracle from task 1.2 for 2 qubits has exactly 3 solutions, and so on.
// ...
}
// Task 3.2. Universal implementation of Grover's algorithm
// Inputs:
// 1) the number of qubits N,
// 2) a marking oracle which implements a boolean expression, similar to the oracles from part I.
// Output:
// An array of N boolean values which satisfy the expression implemented by the oracle
// (i.e., any basis state marked by the oracle).
//
// Note that the similar task in the GroversAlgorithm kata required you to implement Grover's algorithm
// in a way that would be robust to accidental failures, but you knew the optimal number of iterations
// (the number that minimized the probability of such failure).
// In this task you also need to make your implementation robust to not knowing the optimal number of iterations.
operation UniversalGroversAlgorithm (N : Int, oracle : ((Qubit[], Qubit) => Unit is Adj)) : Bool[] {
// ...
return new Bool[N];
}
}