-
Notifications
You must be signed in to change notification settings - Fork 1.1k
/
Copy pathintro.ipynb
2938 lines (2938 loc) · 389 KB
/
intro.ipynb
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
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
{
"cells": [
{
"cell_type": "code",
"execution_count": null,
"metadata": {
"cellView": "form",
"id": "906e07f6e562"
},
"outputs": [],
"source": [
"# @title Copyright 2022 The Cirq Developers\n",
"# Licensed under the Apache License, Version 2.0 (the \"License\");\n",
"# you may not use this file except in compliance with the License.\n",
"# You may obtain a copy of the License at\n",
"#\n",
"# https://www.apache.org/licenses/LICENSE-2.0\n",
"#\n",
"# Unless required by applicable law or agreed to in writing, software\n",
"# distributed under the License is distributed on an \"AS IS\" BASIS,\n",
"# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n",
"# See the License for the specific language governing permissions and\n",
"# limitations under the License."
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "19599098c1f9"
},
"source": [
"# Introduction to Cirq"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "8bd3406cf99e"
},
"source": [
"<table class=\"tfo-notebook-buttons\" align=\"left\">\n",
" <td>\n",
" <a target=\"_blank\" href=\"https://quantumai.google/cirq/start/intro\"><img src=\"https://quantumai.google/site-assets/images/buttons/quantumai_logo_1x.png\" />View on QuantumAI</a>\n",
" </td>\n",
" <td>\n",
" <a target=\"_blank\" href=\"https://colab.research.google.com/github/quantumlib/Cirq/blob/master/docs/start/intro.ipynb\"><img src=\"https://quantumai.google/site-assets/images/buttons/colab_logo_1x.png\" />Run in Google Colab</a>\n",
" </td>\n",
" <td>\n",
" <a target=\"_blank\" href=\"https://github.com/quantumlib/Cirq/blob/master/docs/start/intro.ipynb\"><img src=\"https://quantumai.google/site-assets/images/buttons/github_logo_1x.png\" />View source on GitHub</a>\n",
" </td>\n",
" <td>\n",
" <a href=\"https://storage.googleapis.com/tensorflow_docs/Cirq/docs/start/intro.ipynb\"><img src=\"https://quantumai.google/site-assets/images/buttons/download_icon_1x.png\" />Download notebook</a>\n",
" </td>\n",
"</table>"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "8m9ye4AS6dE4"
},
"source": [
"[Cirq](https://github.com/quantumlib/cirq) is a framework for writing quantum algorithms for noisy intermediate scale quantum (NISQ) devices. Roughly speaking, NISQ devices are those with O(100) qubits that can enact O(1000) gates. Because the resources for NISQ devices are so constrained, we believe that a framework for writing programs on these devices needs to be aware of all of the architectural properties of the device on which the algorithm is written. This is in contrast to other frameworks where there is a clean separation between the abstract model being used and the details of the device. \n",
"\n",
"In this tutorial we will walk through the basics of writing quantum algorithms in Cirq. Our final goal will be to write a variational ansatz for use in an optimization algorithm."
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "cc948e49cecb"
},
"source": [
"# Installing Cirq"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "rPgPbry6-mF3"
},
"source": [
"To use Cirq one first needs to install Cirq. Installation instructions are available at [https://quantumai.google/cirq/start/install](https://quantumai.google/cirq/start/install). For the purpose of this tutorial, we run `pip install cirq` as shown in the following code cell to install the latest release of Cirq. \n",
"\n",
"> Different notebook execution systems exist, but for most part they have \"run\" button on a cell which you can click, or \"shift + enter\" is often the shortcut to run the cell. \n",
"\n",
"\n",
"Note: this notebook relies on unreleased Cirq features. If you want to try these features, make sure you install cirq via `pip install cirq --pre`.\n"
]
},
{
"cell_type": "code",
"execution_count": null,
"metadata": {
"id": "RlJBDvNgC00H"
},
"outputs": [],
"source": [
"try:\n",
" import cirq\n",
"except ImportError:\n",
" print(\"installing cirq...\")\n",
" !pip install --quiet cirq --pre\n",
" print(\"installed cirq.\")\n",
" import cirq\n",
"\n",
"import matplotlib.pyplot as plt\n",
"import numpy as np"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "GPjUqrK8DJTq"
},
"source": [
"Let's check that Cirq has been successfully installed by importing Cirq and printing out a diagram of Google's Sycamore device.\n",
"\n",
"\n",
""
]
},
{
"cell_type": "code",
"execution_count": null,
"metadata": {
"id": "FTrmLyq4C2gf"
},
"outputs": [],
"source": [
"\"\"\"Test successful installation by printing out the Sycamore device.\"\"\"\n",
"import cirq_google\n",
"\n",
"print(cirq_google.Sycamore)"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "09zRgohCMiBs"
},
"source": [
"This cell should run successfully, and the output should in fact be the grid of qubits for the Sycamore device. If so, the install worked!\n",
"\n",
"> Be aware that Cirq is still alpha software, meaning **breaking changes can happen at any time**. If you don't want your project to suddenly go from working to not working when we a new version is released, you should depend on a *specific version* of Cirq and periodically bump that version to the latest one. For example, you can run `pip install cirq==x.y.z` to install version `x.y.z` of Cirq."
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "3340594dd8c1"
},
"source": [
"# Qubits, Operations, Moments and Circuits"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "8A7a3jcql1l5"
},
"source": [
"In Cirq, circuits are represented by a `Circuit` object. Conceptually:\n",
"\n",
"- A `Circuit` is a collection of `Moment`s. \n",
"- A `Moment` is a collection of `Operation`s that all act during the same abstract time slice. \n",
"- An `Operation` is a an effect that operates on a specific subset of Qubits. \n",
" - The most common type of `Operation` is a `Gate` applied to several qubits (a \"`GateOperation`\"). \n",
"- The `Qubit`s of a circuit are implicitly defined by the operations - you can't allocate qubits to a Circuit\n",
" "
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "03b7d753ecd5"
},
"source": [
"<img width=\"50%\" src=\"\"/>"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "ca7d9d7b3953"
},
"source": [
"These ideas are illustrated by the above diagram, where time goes from left to right, horizontal lines are different qubits, operations acting on qubits are the boxes (sometimes spanning multiple qubits) and a moment is a group of operations that \"happen at the same time\". "
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "9768d38c9151"
},
"source": [
"## Create a `Circuit`"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "VFwmWPf7D057"
},
"source": [
"A typical way to create a `Circuit` is shown below."
]
},
{
"cell_type": "code",
"execution_count": null,
"metadata": {
"id": "pE88WsFeDGfs"
},
"outputs": [],
"source": [
"\"\"\"Creating a circuit.\"\"\"\n",
"# Define three qubits.\n",
"a = cirq.NamedQubit(\"a\")\n",
"b = cirq.NamedQubit(\"b\")\n",
"c = cirq.NamedQubit(\"c\")\n",
"\n",
"# Define a list of operations.\n",
"ops = [cirq.H(a), cirq.H(b), cirq.CNOT(b, c), cirq.H(b)]\n",
"\n",
"# Create a circuit from the list of operations.\n",
"circuit = cirq.Circuit(ops)\n",
"print(\"Circuit:\\n\")\n",
"print(circuit)"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "j0qHxsoyIrcj"
},
"source": [
"## Exercise: Create a Circuit\n",
"\n",
"Write a program to create the following circuit.\n",
"\n",
"```\n",
" ┌──┐\n",
"0: ───H─────@────────\n",
" │\n",
"1: ───H────@┼────H───\n",
" ││\n",
"2: ────────X┼────────\n",
" │\n",
"3: ─────────X────────\n",
" └──┘\n",
"```\n",
"\n",
"Note that the circuit has a total of 3 different moments."
]
},
{
"cell_type": "code",
"execution_count": null,
"metadata": {
"cellView": "form",
"id": "6a5TEN5bKAPz"
},
"outputs": [],
"source": [
"# @title Attempt the solution here\n",
"\n",
"# Define 4 qubits.\n",
"\n",
"# Define a list of operations.\n",
"\n",
"# Create a circuit from the list of operations."
]
},
{
"cell_type": "code",
"execution_count": null,
"metadata": {
"cellView": "form",
"id": "Q52e1pX_JIdi"
},
"outputs": [],
"source": [
"# @title Expand to view the solution\n",
"\"\"\"Creating a circuit.\"\"\"\n",
"# Define four line qubits. NamedQubits would also work, however this demonstrates a more succint syntax.\n",
"q = cirq.LineQubit.range(4)\n",
"\n",
"# Define a list of operations.\n",
"ops = [cirq.H(q[0]), cirq.H(q[1]), cirq.CNOT(q[1], q[2]), cirq.CNOT(q[0], q[3]), cirq.H(q[1])]\n",
"\n",
"# Create a circuit from the list of operations.\n",
"print(\"Circuit:\\n\")\n",
"print(cirq.Circuit(ops))"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "-06jQwEdI4DJ"
},
"source": [
"## Unpacking the circuit\n",
"We can unpack this a bit and see all of the components for the circuit.\n",
"\n",
"The first thing we do is pick some qubits to use. There are many different types of qubits in Cirq, and you can define your own by inheriting from the `cirq.Qid` class. There's nothing inherently special or magical about these quantum id types such as `cirq.NamedQubit`. They simply identify what you wish to operate on, which is relevant when you are targeting a specific device. For example, if we were creating a circuit for the Sycamore device and wanted to refer to the qubit in the left-most position, we would use `cirq.GridQubit(5, 0)`. (See the first diagram of the Sycamore device we printed out.) For simplicity, in the previous cell we defined `cirq.NamedQubit`s which are simply qubits that can be identified by a name.\n",
"\n",
"Next, we encounter the object `cirq.H` which is a Hadamard gate with unitary\n",
"\n",
"$$\n",
"H = {1 \\over \\sqrt{2}} \\left[ \\begin{array}[cc] & 1 & 1 \\\\ 1 & -1 \\end{array}\\right] .\n",
"$$\n",
"\n",
"In Cirq, `cirq.H` is an instance of the `cirq.HPowGate` class, which itself is a subclass of `Gate` (along with other classes). We can use Cirq to see the unitary matrix of `Gate` objects as follows."
]
},
{
"cell_type": "code",
"execution_count": null,
"metadata": {
"id": "YKfg575v1DQB"
},
"outputs": [],
"source": [
"\"\"\"Get the unitary of a gate, here the Hadamard gate.\"\"\"\n",
"cirq.unitary(cirq.H)"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "hJMAciW21KEg"
},
"source": [
"We see that this agrees with the unitary for the Hadamard gate above.\n",
"\n",
"`Gate` objects have the ability to be applied \"on\" one or more qubits. There are two ways to do this for gates, either using the `on` method or by directly calling the gate on the qubits as if the gate were a function and the qubits were arguments. For example to apply the `H` on qubit `a` we can say `cirq.H.on(a)` or `cirq.H(a)`.\n",
"\n",
"The result of those expressions is typically a `GateOperation` object, which is a type of `Operation`.\n",
"\n",
"> **Note**: In Cirq, there is a strong distinction between `Operation`s and `Gate`s. An `Operation` is associated with specific qubits and can be put in `Circuit`s. A `Gate` has unspecified qubits, and will produce an operation when acting on qubits.\n",
"\n",
"Once you have a collection of operations, you can construct a `Circuit` by passing the operations into the constructor for a `Circuit`:\n",
"\n",
"```\n",
"ops = [list of operations]\n",
"circuit = cirq.Circuit(ops)\n",
"```\n",
"\n",
"The last thing we did in the example code was use the (surprisingly useful) ability to print the circuit as a text diagram.\n",
"\n",
"The diagram is visually helpful, but it doesn't really get into the internal details of how the `Circuit` is represented. As mentioned, a `Circuit` is made up of a sequence of `Moment` objects, and each `Moment` object is a list of non-overlapping `Operation`s. To see this internal structure, we can iterate over the `Moment`s in the `Circuit` and print them out."
]
},
{
"cell_type": "code",
"execution_count": null,
"metadata": {
"id": "hH-y4JiEMv25"
},
"outputs": [],
"source": [
"\"\"\"Print out the moments in a circuit.\"\"\"\n",
"print(\"Circuit:\\n\")\n",
"print(circuit)\n",
"\n",
"# Inspecting individual moments.\n",
"print(\"\\nMoments in the circuit:\\n\")\n",
"for i, moment in enumerate(circuit):\n",
" print(f'Moment {i}: \\n{moment}')"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "pm5iC7MNQY6-"
},
"source": [
"We see that this circuit consists of three moments. For even more on the underlying structure of a circuit, we can print the circuit's `repr`. This returns a more detailed (and usually less readable) expression."
]
},
{
"cell_type": "code",
"execution_count": null,
"metadata": {
"id": "2Y6zG_peQG1y"
},
"outputs": [],
"source": [
"\"\"\"Print the repr of a circuit.\"\"\"\n",
"print(repr(circuit))"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "zyVbU8yfW_qi"
},
"source": [
"Although it is less readable, the usefulness of printing the `repr` is that it includes *all* the gory details which can be useful when debugging. The `repr` is also a valid python expression that evaluates to the circuit.\n",
"For example, if we notice that a circuit generated in some complicated way triggers a bug in a simulator, copy-pasting the generated circuit's `repr` into a test, and then working from there, is a simple way to decouple the reproduction of the bug from the circuit generation code."
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "0bb8611c3865"
},
"source": [
"## More ways to create `Circuit`s"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "uaDb6B_jPgrb"
},
"source": [
"Above we created a `Circuit` by passing in a list of operations to its constructor. In Cirq, there are many ways to construct and modify circuits, and each of these is useful in different contexts. Here are a few examples:\n",
"\n",
"\n",
"1. `Circuit(...)`: This is the simplest way to make a circuit. Give this method some operations, and out pops a circuit.\n",
"2. `append`: `Circuit`s are mutable. You can start with an empty `circuit = cirq.Circuit()` and simply `circuit.append(operations)` to add on more and more operations .\n",
"3. `insert`: Instead of appending, you can insert before a particular moment location (labeled by an integer index).\n",
"\n",
"One interesting, and extremely convenient, fact about `Circuit(...)`, `append`, and `insert` is that they \"auto flatten\" whatever you give them.\n",
"You *can* give them a list of operations, but you can also give them\n",
"\n",
"- a list *of lists* of operations,\n",
"- a generator function that sometimes yields tuples of operations and other times yields individual operations,\n",
"- or just a single operation (without a list around it).\n",
"\n",
"If it can recursively iterated into individual operations, these three methods will take it.\n",
"\n",
"> The above idea uses a concept we call an `OP_TREE` in Cirq. An `OP_TREE` is not a class, but a contract. The basic idea is that if the input can be iteratively flattened into a list of operations, then the input is an `OP_TREE`.\n",
"\n",
"The main place where auto-flattening is useful is when you are building a circuit's operations using generators. \n",
"\n",
"> Recall that, in Python, functions that have a `yield` statement are *generators*. Generators are functions that act as *iterators*. \n",
"\n",
"In this context, auto-flattening means that generators producing operations for a circuit can simply `yield` sub-generators (instead of iterating over them and yielding their items). We show an example of this below."
]
},
{
"cell_type": "code",
"execution_count": null,
"metadata": {
"id": "QFoV-eOE1tGN"
},
"outputs": [],
"source": [
"\"\"\"Creating a circuit from generator functions.\"\"\"\n",
"\n",
"\n",
"def xor_swap(a, b):\n",
" \"\"\"Swaps two qubits with three CNOTs.\"\"\"\n",
" yield cirq.CNOT(a, b) # |a> |b> --> |a> |a ^ b>\n",
" yield cirq.CNOT(b, a) # |a> |a ^ b> --> |a ^ a ^ b> | a ^ b> = |b>|a^b>\n",
" yield cirq.CNOT(a, b) # |b> |a ^ b> --> |b>|a ^ b ^ b> = |b> |a>\n",
"\n",
"\n",
"cirq.Circuit(xor_swap(a, b))"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "Z76Kcs27EcY5"
},
"source": [
"## Exercise: Create a circuit to left rotate 5 qubits. \n",
"\n",
"Now that we've learnt how to build a circuit to swap the state of two qubits, use this to build a circuit which left rotates 5 qubits i.e. on applying the circuit on 5 qubits (0 - 4), the state of qubits should change as follows:\n",
"``` \n",
"0 --> 4 \n",
"4 --> 3\n",
"3 --> 2\n",
"2 --> 1\n",
"1 --> 0\n",
"```"
]
},
{
"cell_type": "code",
"execution_count": null,
"metadata": {
"cellView": "form",
"id": "LbIZIMINEzD9"
},
"outputs": [],
"source": [
"# @title Attempt the solution here"
]
},
{
"cell_type": "code",
"execution_count": null,
"metadata": {
"cellView": "form",
"id": "5oqmyccsE1kK"
},
"outputs": [],
"source": [
"# @title Expand to view the solution\n",
"def left_rotate(qubits):\n",
" \"\"\"Rotates qubits to the left.\"\"\"\n",
" for i in range(len(qubits) - 1):\n",
" a, b = qubits[i : i + 2]\n",
" yield xor_swap(a, b)\n",
"\n",
"\n",
"# Get five qubits on a line.\n",
"line = cirq.LineQubit.range(5)\n",
"\n",
"# Create a circuit which rotates the qubits to the left.\n",
"# uncomment the next line to compare with your circuit\n",
"# print(cirq.Circuit(left_rotate(line)))"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "ae159315c56d"
},
"source": [
"One can see how this method of creating circuits is quite powerful. \n",
"\n",
"> Note that `cirq.SWAP` is a pre-defined gate in Cirq. We used three `cirq.CNOT`s instead of `cirq.SWAP` in the above example to demonstrate auto-flattening with generators."
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "60d8516a19b2"
},
"source": [
"## Insert strategies"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "p9LUxAU41wWs"
},
"source": [
"You may have noticed that there is a hole in what we've explained so far. We have been passing a one-dimensional sequence of operations, but the output is a two-dimensional circuit (a list-of-lists-of-operations). There is a degree of freedom that hasn't been account for. Specifically, how does Cirq choose the moment that each operation will be placed within?\n",
"\n",
"The answer is the concept of a `cirq.InsertStrategy`. An `InsertStrategy` defines how `Operation`s are placed in a `Circuit` when requested to be inserted at a given location. Here a `location` is identified by the index of the `Moment` in the `Circuit` that operations should be placed before. \n",
"\n",
"> *Note*: In the case of `Circuit.append` this means inserting at the index `len(circuit)` which is one more than the largest moment index and so represents the end of the circuit.\n",
"\n",
"There are currently four insertion strategies in Cirq:\n",
"\n",
"1. `InsertStrategy.EARLIEST` (the default),\n",
"2. `InsertStrategy.NEW`,\n",
"3. `InsertStrategy.INLINE`,\n",
"4. `InsertStrategy.NEW_THEN_INLINE`.\n",
"\n",
"The strategy `InsertStrategy.EARLIEST` is defined as follows:\n",
"\n",
"> `InsertStrategy.EARLIEST`: Scans backward from the insert\n",
"> location until a moment with operations touching qubits affected by the\n",
"> operation to insert is found. The operation is added into the moment just\n",
"> after that location.\n",
"\n",
"For example, if we first create an `Operation` in a single moment,\n",
"and then use `InsertStrategy.EARLIEST` the `Operation` can slide back to this\n",
"first `Moment` if there is space."
]
},
{
"cell_type": "code",
"execution_count": null,
"metadata": {
"id": "wNek1WjpX4MR"
},
"outputs": [],
"source": [
"\"\"\"Appending operations with InsertStrategy.EARLIEST.\"\"\"\n",
"# Create an empty circuit.\n",
"circuit = cirq.Circuit()\n",
"\n",
"# Append an operation.\n",
"# Note: InsertStrategy.EARLIEST is used by default if not otherwise specified.\n",
"circuit.append([cirq.CZ(a, b)])\n",
"\n",
"# Append more operations.\n",
"# Note: InsertStrategy.EARLIEST is used by default if not otherwise specified.\n",
"circuit.append([cirq.H(a), cirq.H(b), cirq.H(c)])\n",
"\n",
"# Display the circuit.\n",
"print(\"Circuit:\\n\")\n",
"print(circuit)"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "4d93a69cfcb8"
},
"source": [
"After creating the first moment with a `CZ` gate, the second\n",
"append uses the `InsertStrategy.EARLIEST` strategy. The\n",
"`H` on ``a`` and ``b`` cannot slide back, while the `H` on ``c`` can and so ends up in the first `Moment`."
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "TcHeZM6qXvbS"
},
"source": [
"While `InsertStrategy.EARLIEST` is the default strategy, the second most important strategy is `InsertStrategy.NEW_THEN_INLINE`, defined as follows:\n",
"\n",
"> `InsertStrategy.NEW_THEN_INLINE`: For the first operation, add it to a new \n",
"> `Moment` the insertion point. Attempts to add the operation after the first \n",
"> operation to insert into the moment just before the desired insert location. \n",
"> But, if there's already an existing operation affecting any of the qubits\n",
"> touched by the operation to insert, a new moment is created instead and this \n",
"> `Moment` is the one that is subsequently used for insertions.\n",
"\n",
"To see an example of this strategy, we create a circuit with the same operations but inserting them with a different strategy."
]
},
{
"cell_type": "code",
"execution_count": null,
"metadata": {
"id": "qWVDhLxFYuRp"
},
"outputs": [],
"source": [
"\"\"\"Appending operations with InsertStrategy.NEW_THEN_INLINE.\"\"\"\n",
"# Create an empty circuit.\n",
"circuit = cirq.Circuit()\n",
"\n",
"# Append an operation.\n",
"circuit.append([cirq.CZ(a, b)], strategy=cirq.InsertStrategy.NEW_THEN_INLINE)\n",
"\n",
"# Append more operations.\n",
"circuit.append([cirq.H(a), cirq.H(b), cirq.H(c)], strategy=cirq.InsertStrategy.NEW_THEN_INLINE)\n",
"\n",
"# Display the circuit.\n",
"print(\"Circuit:\\n\")\n",
"print(circuit)"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "69a53a1f5de2"
},
"source": [
"In contrast to the previous codeblock using `InsertStrategy.EARLIEST`, we see that the three `cirq.H` gates appended after the `cirq.CZ` gate appear in the same moment when we use `InsertStrategy.NEW_THEN_INLINE`."
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "y9conKPAPn26"
},
"source": [
"## Exercise: Create the given circuit using least number of appends\n",
"\n",
"Now that you've learned about `InsertStrategy`s, here is an exercise to validate your understanding. Create, **using the least number of appends**, the following circuit:\n",
"\n",
"\n",
"\n",
"```\n",
"a: ───@───H───────────H───H───\n",
" │\n",
"b: ───@───────H───@───H───────\n",
" │\n",
"c: ───H───────────@───────────\n",
"```\n",
"\n",
"Here imagine that you want exactly the moments indicated by the spacing of the circuit so that there are six moments in this circuit."
]
},
{
"cell_type": "code",
"execution_count": null,
"metadata": {
"cellView": "form",
"id": "-HXXD801OFGF"
},
"outputs": [],
"source": [
"# @title Attempt the solution here"
]
},
{
"cell_type": "code",
"execution_count": null,
"metadata": {
"cellView": "form",
"id": "jP4VkPeHcjJT"
},
"outputs": [],
"source": [
"# @title Expand to view the solution\n",
"# Define three qubits.\n",
"a = cirq.NamedQubit('a')\n",
"b = cirq.NamedQubit('b')\n",
"c = cirq.NamedQubit('c')\n",
"\n",
"# Get an empty circuit.\n",
"circuit = cirq.Circuit()\n",
"\n",
"# Append these gates using cirq.InsertStrategy.EARLIEST (the default strategy).\n",
"circuit.append([cirq.CZ(a, b), cirq.H(c), cirq.H(a)])\n",
"\n",
"# Append these gates using cirq.InsertStrategy.NEW_THEN_INLINE.\n",
"circuit.append(\n",
" [cirq.H(b), cirq.CZ(b, c), cirq.H(b), cirq.H(a), cirq.H(a)],\n",
" strategy=cirq.InsertStrategy.NEW_THEN_INLINE,\n",
")\n",
"\n",
"# Display the circuit.\n",
"print(\"Circuit:\\n\")\n",
"print(circuit)"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "X15yPl_KQ20Z"
},
"source": [
"# Simulations of a Circuit\n",
"\n",
"Now that we know how to construct `Circuit`s in Cirq, let's see how to execute them on a simulator. First we create a simple circuit to simulate in the following cell."
]
},
{
"cell_type": "code",
"execution_count": null,
"metadata": {
"id": "V6tZk3qGqBoH"
},
"outputs": [],
"source": [
"\"\"\"Get a circuit to simulate.\"\"\"\n",
"\n",
"\n",
"def basic_circuit(measure=True):\n",
" \"\"\"Returns a simple circuit with some one- and two-qubit gates,\n",
" as well as (optionally) measurements.\n",
" \"\"\"\n",
" # Gates we will use in the circuit.\n",
" sqrt_x = cirq.X**0.5\n",
" cz = cirq.CZ\n",
"\n",
" # Yield the operations.\n",
" yield sqrt_x(a), sqrt_x(b)\n",
" yield cz(a, b)\n",
" yield sqrt_x(a), sqrt_x(b)\n",
" if measure:\n",
" yield cirq.measure(a, b)\n",
"\n",
"\n",
"# Create a circuit including measurements.\n",
"circuit = cirq.Circuit(basic_circuit())\n",
"print(circuit)"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "WpywVOeDqi4Q"
},
"source": [
"The main simulator in Cirq is the `cirq.Simulator`. The general pattern of simulation is to instantiate this simulator, then pass in a circuit to either the `run` or `simulate` methods (more on this below)."
]
},
{
"cell_type": "code",
"execution_count": null,
"metadata": {
"id": "KmGuMjvGw_Ef"
},
"outputs": [],
"source": [
"\"\"\"Example of simulating a circuit in Cirq.\"\"\"\n",
"# Get a simulator.\n",
"simulator = cirq.Simulator()\n",
"\n",
"# Pass the circuit to the simulator.run method.\n",
"result = simulator.run(circuit, repetitions=1)\n",
"print(\"Measurement results:\")\n",
"print(result)"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "aHugx9T0z047"
},
"source": [
"Running this multiple times should result in different measurement results, since the circuit produces a superposition over all computational basis states."
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "416e9c012263"
},
"source": [
"Above we used the `run` method of the `simulator`. In Cirq, `run` methods mimic the actual hardware in that they don't give one access to unphysical objects like the wavefunction. The `repetitions` argument is how many times to sample from the circuit.\n",
"\n",
"\n",
"\n",
"If one wants to get the wavefunction, the `simulate` methods can be used as shown below."
]
},
{
"cell_type": "code",
"execution_count": null,
"metadata": {
"id": "Apj7WiFZ0WFm"
},
"outputs": [],
"source": [
"\"\"\"Simulating a circuit with the `simulate` method.\"\"\"\n",
"# Get a circuit without measurements.\n",
"circuit = cirq.Circuit(basic_circuit(measure=False))\n",
"\n",
"# Simulate the circuit.\n",
"result = simulator.simulate(circuit, qubit_order=[a, b])\n",
"\n",
"# Print the final state vector (wavefunction).\n",
"print(\"State vector:\")\n",
"print(np.around(result.final_state_vector, 3))\n",
"\n",
"# Print the state vector in Dirac notation.\n",
"print(\"\\nDirac notation:\")\n",
"print(result.dirac_notation())"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "t18-sIJc0cvf"
},
"source": [
"Notice that we passed a `qubit_order` into the `simulate` method. This order helps define the order of the kronecker (tensor) product used in the resulting `final_state_vector`. \n",
"\n",
"> *Note*: The `qubit_order` argument is optional. When it is omitted, qubits are sorted ascending according to the ordering methods defined by their Python class (for example `cirq.NamedQubit` sorts lexicographically by name).\n",
"If there are multiple types of qubits in one circuit, the name of the type is used as a tie breaker.\n",
"\n",
"The simplest `qubit_order` value you can provide is a list of the qubits in the desired order. Any qubits from the circuit that are not in the list will be ordered using the default `__str__` ordering, but come after qubits that are in the list. \n",
"\n",
"> **Note**: Be aware that all qubits in the list are included in the simulation, even if they are not operated on by the circuit.\n",
"\n",
"The mapping from the order of the qubits to the order of the amplitudes in the wave function can be tricky to understand. Basically, it is the same as the ordering used by `numpy.kron`.\n",
"\n",
"> If the state vector is the array \n",
">> (0.1, 0.2, 0.3, 0.4),\n",
"\n",
"> then this is \n",
">> 0.1|00⟩ + 0.2|01⟩ + 0.3|10⟩ + 0.4|11⟩ \n",
"\n",
"> in Dirac notation. If \n",
">> qubit order = [a, b]\n",
"\n",
"> then |00> means qubit a is in 0 and qubit b is in 0, |01> means \n",
"> qubit a is 0 and qubit b is 1, etc.\n",
"\n",
"Another way to think about the qubit-to-amplitude ordering is as \"for loop ordering\":\n",
"\n",
"```\n",
"for a in [0, 1]:\n",
" for b in [0, 1]:\n",
" print(a, b)\n",
"```\n",
"\n",
"The first index (the outermost loop) is the slowest to vary."
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "YLpiz0aN1Jd6"
},
"source": [
"## Repetitions and histograms\n",
"\n",
"As mentioned, the simulator `run` methods also take an option for repeating the circuit, namely, the `repetitions` argument. If the measurements in the circuit are terminal and all other operations are unitary, this simulator is optimized to not recompute the state vector before sampling from the circuit."
]
},
{
"cell_type": "code",
"execution_count": null,
"metadata": {
"id": "QxkmBlo21lrQ"
},
"outputs": [],
"source": [
"\"\"\"Simulate a circuit using 1000 repetitions.\"\"\"\n",
"# Get a circuit with terminal measurements to simulate.\n",
"circuit = cirq.Circuit(basic_circuit())\n",
"\n",
"# Sample from the circuit 1000 times.\n",
"result = simulator.run(circuit, repetitions=1000)\n",
"\n",
"# Get a histogram of measurement results.\n",
"print(result.histogram(key=\"a,b\"))\n",
"\n",
"# Plot a state histogram of the result.\n",
"cirq.plot_state_histogram(result)"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "bD0zX0zP2HxQ"
},
"source": [
"Here we have also demonstrated the use of the `histogram` method on the `result` which sums over all the different results for all of the different repetitions.\n",
"\n",
"The `histogram` method can also be given a `fold_func` argument, in order to group measurement results under some key before counting them up.\n",
"For example, we can group by whether or not the two measurement results agreed:"
]
},
{
"cell_type": "code",
"execution_count": null,
"metadata": {
"id": "rPqVUsD9snYf"
},
"outputs": [],
"source": [
"print(\n",
" result.histogram(\n",
" key=\"a,b\", fold_func=lambda bits: \"agree\" if bits[0] == bits[1] else \"disagree\"\n",
" )\n",
")"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "qFsytBIbOVD8"
},
"source": [
"# The Deutsch-Jozsa Algorithm\n",
"\n",
"The very first indication that quantum computers could be more powerful than classical computers was provided by David Deutsch in his 1985 paper\n",
"\n",
"> David Deutsch, \"[Quantum Theory, the Church-Turing Principle and the Universal Quantum Computer](https://www.daviddeutsch.org.uk/wp-content/deutsch85.pdf)\" *Proc. R. Soc. Lond.* A **400** 97–117. http://doi.org/10.1098/rspa.1985.0070\n",
"\n",
"This algorithm was extended by Deutsch and Richard Jozsa to a more convincing algorithmic seperation and what is now called the Deutsch-Jozsa algorithm. In this section we will show how to write circuits for the Deutsch algorithm and then as an exercise in using Cirq for algorithms for a small version of the Deutsch-Jozsa algorithm.\n",
"\n",
"Let's begin with the Deutsch algorithm. In Deutsch's algorithm you are given access to a box which computes a one bit boolean function. That is it is a box which takes in a bit and outputs a bit. If we want to be a mathematician or theoretical computer scientist we write the function $f$ as $f: \\{0, 1\\} \\rightarrow \\{0, 1\\}$. There are exactly four such boolean functions which we can write out in a table\n",
"\n",
"| $x$ | $f_0$ | $f_1$ | $f_x$ | $f_{\\bar{x}}$ |\n",
"| --- | --- | --- | --- | --- |\n",
"| 0 | 0 | 1 | 0 | 1\n",
"| 1 | 0 | 1 | 1 | 0\n",
"\n",
"The first two of these are *constant* functions, $f_0$ and $f_1$. That is they always output a constant value (independent of the input). The other two $f_x$ and $f_\\bar{x}$ are *balanced*. Over their inputs $0$ and $1$, they have an equal number of $0$s and $1$s in their truth table. \n",
"\n",
"We can now state Deutsch's problem:\n",
"\n",
"> Given access to a one bit input one bit output boolean function, determine by querying the function whether the function is *balanced* or *constant*.\n",
"\n",
"It shouldn't take you much to convince yourself that in order to solve this problem classically you need to call the function on both possible input values. The easiest way to see this is just to consider what happens if you query the function on one particular input and notice that, for either input, learning the value of the function does not separate the constant from balanced functions. In summary:\n",
"\n",
"*Classically one must query the binary function twice to distinguish the constant function from the balanced function.*"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "UAec5ZBuSWYU"
},
"source": [
"Now lets turn to the quantum approach to this problem. There is one bit of book keeping we need to take care of. Above we have described a classical function on bits that is not reversible. That is, knowing the values of the output does not allow us to determine uniquely the value of the input. In order to run this on a quantum computer, however we need to make this computation reversible. A trick for taking a classical non-reversible function and making it \"quantum happy\" is to compute the value in an extra register and store the input. Suppose we have an $n$ bit input $x$ and we are computing a (potentially non-reverisble) boolean function $f(x)$. Then we can implement this via a Unitary $U_f$ that acts like on $n + 1$ qubits\n",
"\n",
"$$\n",
"U_f |x\\rangle |y\\rangle = |x\\rangle | y \\oplus f(x)\\rangle .\n",
"$$\n",
"\n",
"Here $\\oplus$ is addition modulo $2$ (XOR) and we have identified how $U_f$ acts by its action on all computational basis states $|x\\rangle$ ($n$ input qubits) and $|y\\rangle$ ($1$ output qubit). To see that this is reversible one can note that applying the transformation twice returns the state to its original form.\n",
"\n",
"Let's see how to implement these functions in Cirq.\n",
"\n",
"$f_0$ enacts the transform\n",
"$$\n",
"\\begin{eqnarray}\n",
"|00\\rangle &\\rightarrow& |00\\rangle \\\\\n",
"|01\\rangle &\\rightarrow& |01\\rangle \\\\\n",
"|10\\rangle &\\rightarrow& |10\\rangle \\\\\n",
"|11\\rangle &\\rightarrow& |11\\rangle \\\\\n",
"\\end{eqnarray}\n",
"$$\n",
"Well this is just the identity transform, i.e. an empty circuit.\n",
"\n",
"$f_1$ enacts the transform\n",
"$$\n",