-
Notifications
You must be signed in to change notification settings - Fork 0
/
BeamOpticsTutorial.html
273 lines (261 loc) · 82.1 KB
/
BeamOpticsTutorial.html
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
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html><head><meta http-equiv="Content-Type" content="text/html; charset=utf-8"><meta http-equiv="X-UA-Compatible" content="IE=edge,IE=9,chrome=1"><meta name="generator" content="MATLAB 2021a"><title>Beam optics tutorial (Section 3.3.1)</title><style type="text/css">.rtcContent { padding: 30px; } .S0 { margin: 2px 10px 9px 4px; padding: 0px; line-height: 21px; min-height: 0px; white-space: pre-wrap; color: rgb(0, 0, 0); font-family: Helvetica, Arial, sans-serif; font-style: normal; font-size: 14px; font-weight: normal; text-align: left; }
.S1 { margin: 15px 10px 5px 4px; padding: 0px; line-height: 28.8px; min-height: 0px; white-space: pre-wrap; color: rgb(213, 80, 0); font-family: Helvetica, Arial, sans-serif; font-style: normal; font-size: 24px; font-weight: normal; text-align: left; }
.CodeBlock { background-color: #F7F7F7; margin: 10px 0 10px 0;}
.S2 { border-left: 1px solid rgb(233, 233, 233); border-right: 1px solid rgb(233, 233, 233); border-top: 1px solid rgb(233, 233, 233); border-bottom: 0px none rgb(0, 0, 0); border-radius: 4px 4px 0px 0px; padding: 6px 45px 0px 13px; line-height: 17.234px; min-height: 18px; white-space: nowrap; color: rgb(0, 0, 0); font-family: Menlo, Monaco, Consolas, "Courier New", monospace; font-size: 14px; }
.S3 { border-left: 1px solid rgb(233, 233, 233); border-right: 1px solid rgb(233, 233, 233); border-top: 0px none rgb(0, 0, 0); border-bottom: 0px none rgb(0, 0, 0); border-radius: 0px; padding: 0px 45px 0px 13px; line-height: 17.234px; min-height: 18px; white-space: nowrap; color: rgb(0, 0, 0); font-family: Menlo, Monaco, Consolas, "Courier New", monospace; font-size: 14px; }
.S4 { border-left: 1px solid rgb(233, 233, 233); border-right: 1px solid rgb(233, 233, 233); border-top: 0px none rgb(0, 0, 0); border-bottom: 1px solid rgb(233, 233, 233); border-radius: 0px 0px 4px 4px; padding: 0px 45px 4px 13px; line-height: 17.234px; min-height: 18px; white-space: nowrap; color: rgb(0, 0, 0); font-family: Menlo, Monaco, Consolas, "Courier New", monospace; font-size: 14px; }
.S5 { margin: 10px 10px 9px 4px; padding: 0px; line-height: 21px; min-height: 0px; white-space: pre-wrap; color: rgb(0, 0, 0); font-family: Helvetica, Arial, sans-serif; font-style: normal; font-size: 14px; font-weight: normal; text-align: left; }
.S6 { border-left: 1px solid rgb(233, 233, 233); border-right: 1px solid rgb(233, 233, 233); border-top: 1px solid rgb(233, 233, 233); border-bottom: 1px solid rgb(233, 233, 233); border-radius: 4px 4px 0px 0px; padding: 6px 45px 4px 13px; line-height: 17.234px; min-height: 18px; white-space: nowrap; color: rgb(0, 0, 0); font-family: Menlo, Monaco, Consolas, "Courier New", monospace; font-size: 14px; }
.S7 { color: rgb(64, 64, 64); padding: 10px 0px 6px 17px; background: rgb(255, 255, 255) none repeat scroll 0% 0% / auto padding-box border-box; font-family: Menlo, Monaco, Consolas, "Courier New", monospace; font-size: 14px; overflow-x: hidden; line-height: 17.234px; }
.variableValue { width: 100% !important; }
.embeddedOutputsMatrixElement,.eoOutputWrapper .matrixElement { min-height: 18px; box-sizing: border-box;}
.embeddedOutputsMatrixElement .matrixElement,.eoOutputWrapper .matrixElement,.rtcDataTipElement .matrixElement { position: relative;}
.matrixElement .variableValue,.rtcDataTipElement .matrixElement .variableValue { white-space: pre; display: inline-block; vertical-align: top; overflow: hidden;}
.embeddedOutputsMatrixElement.inlineElement {}
.embeddedOutputsMatrixElement.inlineElement .topHeaderWrapper { display: none;}
.embeddedOutputsMatrixElement.inlineElement .veTable .body { padding-top: 0 !important; max-height: 100px;}
.inlineElement .matrixElement { max-height: 300px;}
.embeddedOutputsMatrixElement.rightPaneElement {}
.rightPaneElement .matrixElement,.rtcDataTipElement .matrixElement { overflow: hidden; padding-left: 9px;}
.rightPaneElement .matrixElement { margin-bottom: -1px;}
.embeddedOutputsMatrixElement .matrixElement .valueContainer,.eoOutputWrapper .matrixElement .valueContainer,.rtcDataTipElement .matrixElement .valueContainer { white-space: nowrap; margin-bottom: 3px;}
.embeddedOutputsMatrixElement .matrixElement .valueContainer .horizontalEllipsis.hide,.embeddedOutputsMatrixElement .matrixElement .verticalEllipsis.hide,.eoOutputWrapper .matrixElement .valueContainer .horizontalEllipsis.hide,.eoOutputWrapper .matrixElement .verticalEllipsis.hide,.rtcDataTipElement .matrixElement .valueContainer .horizontalEllipsis.hide,.rtcDataTipElement .matrixElement .verticalEllipsis.hide { display: none;}
.embeddedOutputsVariableMatrixElement .matrixElement .valueContainer.hideEllipses .verticalEllipsis, .embeddedOutputsVariableMatrixElement .matrixElement .valueContainer.hideEllipses .horizontalEllipsis { display:none;}
.embeddedOutputsMatrixElement .matrixElement .valueContainer .horizontalEllipsis,.eoOutputWrapper .matrixElement .valueContainer .horizontalEllipsis { margin-bottom: -3px;}
.eoOutputWrapper .embeddedOutputsVariableMatrixElement .matrixElement .valueContainer { cursor: default !important;}
.embeddedOutputsVariableElement { white-space: pre-wrap; word-wrap: break-word; min-height: 18px; max-height: 250px; overflow: auto;}
.variableElement {}
.embeddedOutputsVariableElement.inlineElement {}
.inlineElement .variableElement {}
.embeddedOutputsVariableElement.rightPaneElement { min-height: 16px;}
.rightPaneElement .variableElement { padding-top: 2px; padding-left: 9px;}
.variableNameElement { margin-bottom: 3px; display: inline-block;}
/* * Ellipses as base64 for HTML export. */.matrixElement .horizontalEllipsis,.rtcDataTipElement .matrixElement .horizontalEllipsis { display: inline-block; margin-top: 3px; /* base64 encoded version of images-liveeditor/HEllipsis.png */ width: 30px; height: 12px; background-repeat: no-repeat; background-image: url("");}
.matrixElement .verticalEllipsis,.textElement .verticalEllipsis,.rtcDataTipElement .matrixElement .verticalEllipsis,.rtcDataTipElement .textElement .verticalEllipsis { margin-left: 35px; /* base64 encoded version of images-liveeditor/VEllipsis.png */ width: 12px; height: 30px; background-repeat: no-repeat; background-image: url("");}
.S8 { margin: 10px 0px 20px; padding-left: 0px; font-family: Helvetica, Arial, sans-serif; font-size: 14px; }
.S9 { margin-left: 56px; line-height: 21px; min-height: 0px; text-align: left; white-space: pre-wrap; }
.S10 { border-left: 1px solid rgb(233, 233, 233); border-right: 1px solid rgb(233, 233, 233); border-top: 0px none rgb(0, 0, 0); border-bottom: 1px solid rgb(233, 233, 233); border-radius: 0px; padding: 0px 45px 4px 13px; line-height: 17.234px; min-height: 18px; white-space: nowrap; color: rgb(0, 0, 0); font-family: Menlo, Monaco, Consolas, "Courier New", monospace; font-size: 14px; }
.S11 { border-left: 1px solid rgb(233, 233, 233); border-right: 1px solid rgb(233, 233, 233); border-top: 1px solid rgb(233, 233, 233); border-bottom: 0px none rgb(0, 0, 0); border-radius: 0px; padding: 6px 45px 0px 13px; line-height: 17.234px; min-height: 18px; white-space: nowrap; color: rgb(0, 0, 0); font-family: Menlo, Monaco, Consolas, "Courier New", monospace; font-size: 14px; }
.S12 { border-left: 1px solid rgb(233, 233, 233); border-right: 1px solid rgb(233, 233, 233); border-top: 1px solid rgb(233, 233, 233); border-bottom: 1px solid rgb(233, 233, 233); border-radius: 4px; padding: 6px 45px 4px 13px; line-height: 17.234px; min-height: 18px; white-space: nowrap; color: rgb(0, 0, 0); font-family: Menlo, Monaco, Consolas, "Courier New", monospace; font-size: 14px; }</style></head><body><div class = rtcContent><div class = 'S0'><span>Companion software for "Volker Ziemann, </span><span style=' font-style: italic;'>Hands-on Accelerator physics using MATLAB, CRCPress, 2019</span><span>" (https://www.crcpress.com/9781138589940)</span></div><h1 class = 'S1'><span>Beam optics tutorial (Section 3.3.1)</span></h1><div class = 'S0'><span>Volker Ziemann, 211106, CC-BY-SA-4.0</span></div><div class = 'S0'><span>In this tutorial we'll develop a first beam optics code that allows us to track single particles, or "rays" through a sequence of drift spaces and thin lenses. Let's therefore first clear the MATLAB workspace and define anonymous functions (those with an '@') that return the transfer matrices. Here D(L) will return the 2x2 matrix for a drift space and Q(F) that of a thin lens with focal length F</span></div><div class="CodeBlock"><div class="inlineWrapper"><div class = 'S2'><span style="white-space: pre"><span >clear </span><span style="color: rgb(170, 4, 249);">all</span><span >; close </span><span style="color: rgb(170, 4, 249);">all</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span >D=@(L)[1,L;0,1];</span></span></div></div><div class="inlineWrapper"><div class = 'S4'><span style="white-space: pre"><span >Q=@(F)[1,0;-1/F,1];</span></span></div></div></div><div class = 'S5'><span>Now it is easy to obtain the transfer matrix of a beam line consisting of a sequnece of such elements. For example, a beamline, where the first element is a drift space with a length of 2m, followed by a lens with a focal length of 1 m and a second drift space having a length of 3m.</span></div><div class="CodeBlock"><div class="inlineWrapper outputs"><div class = 'S6'><span style="white-space: pre"><span >A=D(3)*Q(1)*D(2)</span></span></div><div class = 'S7'><div class="inlineElement eoOutputWrapper embeddedOutputsVariableMatrixElement" uid="AF17D69A" data-scroll-top="null" data-scroll-left="null" data-width="1334" data-testid="output_0" style="width: 1364px; white-space: normal; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;"><div class="matrixElement veSpecifier saveLoad" style="white-space: normal; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;"><div class="veVariableName variableNameElement double" style="width: 1334px; white-space: normal; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;"><div class="headerElementClickToInteract" style="white-space: normal; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;"><span style="white-space: normal; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;">A = </span><span class="veVariableValueSummary veMetaSummary" style="white-space: normal; font-style: normal; color: rgb(179, 179, 179); font-size: 12px;">2×2</span></div></div><div class="valueContainer" data-layout="{"columnWidth":44,"totalColumns":"2","totalRows":"2","charsPerColumn":6}" style="white-space: nowrap; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;"><div class="variableValue" style="width: 90px; white-space: pre; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;"> -2 -1
-1 -1
</div><div class="horizontalEllipsis hide" style="white-space: nowrap; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;"></div><div class="verticalEllipsis hide" style="white-space: nowrap; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;"></div></div></div></div></div></div></div><div class = 'S5'><span>I suggest to verify the numerical calculation by multiplying the individual matrices by hand on a peice of paper.</span></div><div class = 'S0'><span>Directly coding the matrix multiplications is inconvenient and we therefore introduce a very simple way to describe sequences of elements as an array, here called </span><span style=' font-family: monospace;'>fodo.</span><span> This array has four columns with the following interpretations</span></div><ul class = 'S8'><li class = 'S9'><span>First column: code, drift=1, quad=2</span></li><li class = 'S9'><span>Second column: repeat code for the element</span></li><li class = 'S9'><span>Third column: length of one segment</span></li><li class = 'S9'><span>Fourth column: strength of one segment, focal length for a thin quadrupole.</span></li></ul><div class = 'S0'><span>The second column with the repeat code is not strictly necessary but will make the graphs easier to interpret, because we can plot many intermediate points, rather than connecting values at the end of elements by a straight line.</span></div><div class = 'S0'><span>This particular sequence starts with an element having a 1 in the first column and is therefore a drift space. It consists of 5 segments, each 0.2 m long. The last column is not used here and contains a zero, The next element is a lens, beacuse the first code is 2 and it is defocusing, because the fourth column contains a negative number for the focal length. The third element is again a driftspace, this time with 10 segments of 0.2 m, followed by a focusing lens and another drift space. Such a sequence of focusing lens, drift space, defocusing lens and another drift space is usually referred to as a FODO (get it?) cell. </span></div><div class="CodeBlock"><div class="inlineWrapper"><div class = 'S2'><span style="white-space: pre"><span >F=2.1;</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span >fodo=[1, 5, 0.2, 0;</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > 2, 1, 0, -F;</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > 1, 10, 0.2, 0;</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > 2, 1, 0, F;</span></span></div></div><div class="inlineWrapper outputs"><div class = 'S10'><span style="white-space: pre"><span > 1, 5, 0.2, 0] </span></span></div><div class = 'S7'><div class="inlineElement eoOutputWrapper embeddedOutputsVariableMatrixElement" uid="C8B1D865" data-scroll-top="null" data-scroll-left="null" data-width="1334" data-testid="output_1" style="width: 1364px; white-space: normal; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;"><div class="matrixElement veSpecifier saveLoad" style="white-space: normal; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;"><div class="veVariableName variableNameElement double" style="width: 1334px; white-space: normal; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;"><div class="headerElementClickToInteract" style="white-space: normal; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;"><span style="white-space: normal; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;">fodo = </span><span class="veVariableValueSummary veMetaSummary" style="white-space: normal; font-style: normal; color: rgb(179, 179, 179); font-size: 12px;">5×4</span></div></div><div class="valueContainer" data-layout="{"columnWidth":72,"totalColumns":"4","totalRows":"5","charsPerColumn":10}" style="white-space: nowrap; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;"><div class="variableValue" style="width: 290px; white-space: pre; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;"> 1.0000 5.0000 0.2000 0
2.0000 1.0000 0 -2.1000
1.0000 10.0000 0.2000 0
2.0000 1.0000 0 2.1000
1.0000 5.0000 0.2000 0
</div><div class="horizontalEllipsis hide" style="white-space: nowrap; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;"></div><div class="verticalEllipsis hide" style="white-space: nowrap; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;"></div></div></div></div></div></div></div><div class = 'S5'><span>Now we build our beamline as a sequence of one or many FODO cells. The latter we achieve by simply concatenating</span><span style=' font-family: monospace;'> fodo</span><span> vertically (note the semicolon!). We will always use</span><span style=' font-family: monospace;'> beamline</span><span> as a reserved name of the beamline that we work on.</span></div><div class="CodeBlock"><div class="inlineWrapper"><div class = 'S2'><span style="white-space: pre"><span style="color: rgb(2, 128, 9);">%beamline=fodo; % one cell</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span style="color: rgb(2, 128, 9);">%beamline=[fodo;fodo] % two cells</span></span></div></div><div class="inlineWrapper outputs"><div class = 'S10'><span style="white-space: pre"><span >beamline=repmat(fodo,10,1) </span><span style="color: rgb(2, 128, 9);">% ten cells</span></span></div><div class = 'S7'><div class="inlineElement eoOutputWrapper embeddedOutputsVariableMatrixElement" uid="5CC5EFC4" data-scroll-top="null" data-scroll-left="null" data-width="1334" data-testid="output_2" style="width: 1364px; white-space: normal; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;"><div class="matrixElement veSpecifier saveLoad" style="white-space: normal; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;"><div class="veVariableName variableNameElement double" style="width: 1334px; white-space: normal; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;"><div class="headerElementClickToInteract" style="white-space: normal; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;"><span style="white-space: normal; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;">beamline = </span><span class="veVariableValueSummary veMetaSummary" style="white-space: normal; font-style: normal; color: rgb(179, 179, 179); font-size: 12px;">50×4</span></div></div><div class="valueContainer" data-layout="{"columnWidth":72,"totalColumns":"4","totalRows":"50","charsPerColumn":10}" style="white-space: nowrap; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;"><div class="variableValue" style="width: 290px; white-space: pre; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;"> 1.0000 5.0000 0.2000 0
2.0000 1.0000 0 -2.1000
1.0000 10.0000 0.2000 0
2.0000 1.0000 0 2.1000
1.0000 5.0000 0.2000 0
1.0000 5.0000 0.2000 0
2.0000 1.0000 0 -2.1000
1.0000 10.0000 0.2000 0
2.0000 1.0000 0 2.1000
1.0000 5.0000 0.2000 0
</div><div class="horizontalEllipsis hide" style="white-space: nowrap; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;"></div><div class="verticalEllipsis" style="white-space: nowrap; font-style: normal; color: rgb(64, 64, 64); font-size: 12px;"></div></div></div></div></div></div></div><div class = 'S5'><span>Before we can assemble the transfer matrices, we have to do a bit of accounting, like counting the number of elements </span><span style=' font-family: monospace;'>nlines</span><span> in the beamline description and the number </span><span style=' font-family: monospace;'>nmat</span><span> of matrices that we will need to store the matrices. This is just given by the sum of the segments in the second column plus one. Next we allocate space for the nmat </span><span texencoding="2\times 2" style="vertical-align:-5px"><img src="" width="34.5" height="18" /></span><span> matrices in the array Racc and initialize the first matrix as the </span><span texencoding="2\times2" style="vertical-align:-5px"><img src="" width="34.5" height="18" /></span><span>unit matrix eye(2). This additional matrix at the beginning accounts for the "plus one" in the definition of</span><span style=' font-family: monospace;'> nmat</span><span>. Finally we allocate the array </span><span style=' font-family: monospace;'>spos</span><span> to store the position of each segment along the beam line.</span></div><div class="CodeBlock"><div class="inlineWrapper outputs"><div class = 'S6'><span style="white-space: pre"><span >nlines=size(beamline,1)</span></span></div><div class = 'S7'><div class='variableElement' style='font-family: Menlo, Monaco, Consolas, "Courier New", monospace; font-size: 12px; '>nlines = 50</div></div></div><div class="inlineWrapper"><div class = 'S11'><span style="white-space: pre"><span >nmat=sum(beamline(:,2))+1;</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span >Racc=zeros(2,2,nmat);</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span >Racc(:,:,1)=eye(2);</span></span></div></div><div class="inlineWrapper"><div class = 'S4'><span style="white-space: pre"><span >spos=zeros(nmat,1);</span></span></div></div></div><div class = 'S5'><span>Now we are ready to loop through the beamline description by looping over the lines, labeled by</span><span style=' font-family: monospace;'> line</span><span>, and then over the segments, labeled by </span><span style=' font-family: monospace;'>seg</span><span>. As we do that we increment the counter</span><span style=' font-family: monospace;'> ic</span><span> that keeps track of which segment we're currently working on. Then we initialize the transfer matrix of the current segment </span><span style=' font-family: monospace;'>Rcurr</span><span> by the unit marix before looking at the first column to decide whether the element is a drift space or a lens and fill </span><span style=' font-family: monospace;'>Rcurr</span><span> with the appropriate transfer matrix. In</span><span style=' font-family: monospace;'> case</span><span> the first column (</span><span style=' font-family: monospace;'>beamline(line,1)</span><span>) is a 1, it is a drift and we use the value from the third column as the length of the drift space. In </span><span style=' font-family: monospace;'>case</span><span> the first column contains a 2, it is a lens and we use the focal length from the fourth column as the argument to the function </span><span style=' font-family: monospace;'>Q(F).</span><span> In case we encounter any other number in the first column, we display a warning. At this point </span><span style=' font-family: monospace;'>Rcurr</span><span> contains the transfer matrix for the current segment and we find </span><span style=' font-weight: bold;'>the transfer matrix from the start of the beamline to the end of the current element</span><span> </span><span style=' font-family: monospace;'>Racc(:,:,ic)</span><span> by left-multiplying Rcurr to </span><span style=' font-family: monospace;'>Racc(:,:,ic-1)</span><span> which already contains the transfer matrix from the start to the end of he previous segment. Finally we calculate the position </span><span style=' font-family: monospace;'>spos(ic)</span><span> of the current segment from that of the previous segment </span><span style=' font-family: monospace;'>spos(ic-1)</span><span> by adding the length of the current segment to it.</span></div><div class="CodeBlock"><div class="inlineWrapper"><div class = 'S2'><span style="white-space: pre"><span >ic=1;</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span style="color: rgb(14, 0, 255);">for </span><span >line=1:nlines</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > </span><span style="color: rgb(14, 0, 255);">for </span><span >seg=1:beamline(line,2)</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > ic=ic+1;</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > Rcurr=eye(2);</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > </span><span style="color: rgb(14, 0, 255);">switch </span><span >beamline(line,1)</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > </span><span style="color: rgb(14, 0, 255);">case </span><span >1</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > Rcurr=D(beamline(line,3));</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > </span><span style="color: rgb(14, 0, 255);">case </span><span >2</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > Rcurr=Q(beamline(line,4));</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > </span><span style="color: rgb(14, 0, 255);">otherwise</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > disp(</span><span style="color: rgb(170, 4, 249);">'unknown element'</span><span >);</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > </span><span style="color: rgb(14, 0, 255);">end</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > Racc(:,:,ic)=Rcurr*Racc(:,:,ic-1);</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > spos(ic)=spos(ic-1)+beamline(line,3); </span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > </span><span style="color: rgb(14, 0, 255);">end</span></span></div></div><div class="inlineWrapper"><div class = 'S4'><span style="white-space: pre"><span style="color: rgb(14, 0, 255);">end</span></span></div></div></div><div class = 'S5'><span>At this point</span><span style=' font-family: monospace;'> Racc</span><span> contains all trasnfer matrices from the start of the beam line to the end of each and every segment while </span><span style=' font-family: monospace;'>spos</span><span> contains the positions of each segment.</span></div><div class = 'S0'><span>Now we can use the transfer matrices to plot the trajectory of a particle with starting coordinates </span><span texencoding="x_0 = 0" style="vertical-align:-6px"><img src="" width="41" height="20" /></span><span> and </span><span texencoding="x_0'=10^{-3}" style="vertical-align:-10px"><img src="" width="61" height="25" /></span><span> rad by mapping the initial (column) vector </span><span style=' font-family: monospace;'>x0 </span><span>with </span><span style=' font-family: monospace;'>Racc</span><span> to the ened of each segment and record both position</span><span style=' font-family: monospace;'> x(1)</span><span> and angle </span><span style=' font-family: monospace;'>x(2)</span><span> in the previously allocated array </span><span style=' font-family: monospace;'>data</span><span>. </span></div><div class="CodeBlock"><div class="inlineWrapper"><div class = 'S2'><span style="white-space: pre"><span >data=zeros(nmat,2); </span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span >x0=[0;1e-3];</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span style="color: rgb(14, 0, 255);">for </span><span >k=1:nmat</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > x=Racc(:,:,k)*x0;</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > data(k,1)=x(1);</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > data(k,2)=x(2);</span></span></div></div><div class="inlineWrapper"><div class = 'S4'><span style="white-space: pre"><span style="color: rgb(14, 0, 255);">end</span></span></div></div></div><div class = 'S5'><span>Now we can plot the trajectory and annotate the axes. </span></div><div class="CodeBlock"><div class="inlineWrapper"><div class = 'S2'><span style="white-space: pre"><span >plot(spos,1000*data(:,1),</span><span style="color: rgb(170, 4, 249);">'k'</span><span >) </span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span >xlabel(</span><span style="color: rgb(170, 4, 249);">'s [m]'</span><span >)</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span >ylabel(</span><span style="color: rgb(170, 4, 249);">'x [mm]'</span><span >) </span></span></div></div><div class="inlineWrapper outputs"><div class = 'S10'><span style="white-space: pre"><span >title(</span><span style="color: rgb(170, 4, 249);">'Trajectory'</span><span >)</span></span></div><div class = 'S7'><div class="inlineElement eoOutputWrapper embeddedOutputsFigure" uid="D4409CDA" data-scroll-top="null" data-scroll-left="null" data-testid="output_4" style="width: 1364px;"><div class="figureElement"><img class="figureImage figureContainingNode" src="" style="width: 560px;"></div></div></div></div></div><div class = 'S0'><span>We observe an oscillation of the particle's position along the beam line. This type of transverse oscillation is usually called </span><span style=' font-style: italic;'>betatron oscillation</span><span>.</span></div><div class = 'S0'><span>Now you can go back to the start of the script and change the focal length F and observe how the oscillation changes. You might also want to increase the length of the beam line by adding more copies of the FODO cells in the definition of beamline by increasing the number of copies in </span><span style=' font-family: monospace;'>repmat()</span><span>. Explore!</span></div><div class = 'S0'><span>Calculating transfer matrices is a task we repeatedly have to do and it is therefore convenient to encapsulate the code that does that in a dedicated function; we call it </span><span style=' font-family: monospace;'>calcmat().</span><span> It receives the </span><span style=' font-family: monospace;'>beamline</span><span> as input and returns </span><span style=' font-family: monospace;'>Racc, spos,nmat</span><span>, and</span><span style=' font-family: monospace;'> nlines</span><span>. Internally it calculates nmat and nlines and then allocates arrays for </span><span style=' font-family: monospace;'>Racc </span><span>and </span><span style=' font-family: monospace;'>spos</span><span> and then loops over all lines and segments, just as we did in the code above. We use such a function in the following way</span></div><div class="CodeBlock"><div class="inlineWrapper"><div class = 'S12'><span style="white-space: pre"><span >[Racc,spos,nmat,nlines]=calcmat(beamline);</span></span></div></div></div><div class = 'S0'><span>and--viola--all transfer matrces and the positions of all segments are immediately available. Here is the definition of the function </span><span style=' font-family: monospace;'>calcmat()</span></div><div class="CodeBlock"><div class="inlineWrapper"><div class = 'S2'><span style="white-space: pre"><span style="color: rgb(14, 0, 255);">function </span><span >[Racc,spos,nmat,nlines]=calcmat(beamline)</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span >nlines=size(beamline,1);</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span >nmat=sum(beamline(:,2))+1;</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span >Racc=zeros(2,2,nmat);</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span >Racc(:,:,1)=eye(2);</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span >spos=zeros(nmat,1);</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span >ic=1;</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span style="color: rgb(14, 0, 255);">for </span><span >line=1:nlines</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > </span><span style="color: rgb(14, 0, 255);">for </span><span >seg=1:beamline(line,2)</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > ic=ic+1;</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > Rcurr=eye(2);</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > </span><span style="color: rgb(14, 0, 255);">switch </span><span >beamline(line,1)</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > </span><span style="color: rgb(14, 0, 255);">case </span><span >1</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > Rcurr=D2(beamline(line,3));</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > </span><span style="color: rgb(14, 0, 255);">case </span><span >2</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > Rcurr=Q2(beamline(line,4));</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > </span><span style="color: rgb(14, 0, 255);">otherwise</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > disp(</span><span style="color: rgb(170, 4, 249);">'unknown element'</span><span >);</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > </span><span style="color: rgb(14, 0, 255);">end</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > Racc(:,:,ic)=Rcurr*Racc(:,:,ic-1);</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > spos(ic)=spos(ic-1)+beamline(line,3); </span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span > </span><span style="color: rgb(14, 0, 255);">end</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span style="color: rgb(14, 0, 255);">end</span></span></div></div><div class="inlineWrapper"><div class = 'S4'><span style="white-space: pre"><span style="color: rgb(14, 0, 255);">end</span></span></div></div></div><div class = 'S5'><span>Inside this function the function D(L) an Q(F) are unknown. We could either copy the two lines with their definitions into calcmat, or define additional functions that provide the same functionality. First for the drift space</span></div><div class="CodeBlock"><div class="inlineWrapper"><div class = 'S2'><span style="white-space: pre"><span style="color: rgb(14, 0, 255);">function </span><span >out=D2(L)</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span >out=[1,L;0,1];</span></span></div></div><div class="inlineWrapper"><div class = 'S4'><span style="white-space: pre"><span style="color: rgb(14, 0, 255);">end</span></span></div></div></div><div class = 'S5'><span>and for the lens</span></div><div class="CodeBlock"><div class="inlineWrapper"><div class = 'S2'><span style="white-space: pre"><span style="color: rgb(14, 0, 255);">function </span><span >out=Q2(F)</span></span></div></div><div class="inlineWrapper"><div class = 'S3'><span style="white-space: pre"><span >out=[1,0;-1/F,1];</span></span></div></div><div class="inlineWrapper"><div class = 'S4'><span style="white-space: pre"><span style="color: rgb(14, 0, 255);">end</span></span></div></div></div><div class = 'S5'><span>Here I have given new names (D2, Q2)to functions for drift and quad, because the names "D" and "Q" are already used for the matrices in the first few lines of this script.</span></div><div class = 'S0'><span>These service functions will make all further examples much more compact and readable, because we never have to copy and paste the enclosed code verbatim into our scripts. This simple tutorial only contains the bare essentials of beam optics calculations. Have a look at the web site where you picked this script up and check out the other more advanced examples. </span></div>
<br>
<!--
##### SOURCE BEGIN #####
%%
% Companion software for "Volker Ziemann, _Hands-on Accelerator physics using
% MATLAB, CRCPress, 2019_" (https://www.crcpress.com/9781138589940)
%% Beam optics tutorial (Section 3.3.1)
% Volker Ziemann, 211106, CC-BY-SA-4.0
%
% In this tutorial we'll develop a first beam optics code that allows us to
% track single particles, or "rays" through a sequence of drift spaces and thin
% lenses. Let's therefore first clear the MATLAB workspace and define anonymous
% functions (those with an '@') that return the transfer matrices. Here D(L) will
% return the 2x2 matrix for a drift space and Q(F) that of a thin lens with focal
% length F
clear all; close all
D=@(L)[1,L;0,1];
Q=@(F)[1,0;-1/F,1];
%%
% Now it is easy to obtain the transfer matrix of a beam line consisting of
% a sequnece of such elements. For example, a beamline, where the first element
% is a drift space with a length of 2m, followed by a lens with a focal length
% of 1 m and a second drift space having a length of 3m.
A=D(3)*Q(1)*D(2)
%%
% I suggest to verify the numerical calculation by multiplying the individual
% matrices by hand on a peice of paper.
%
% Directly coding the matrix multiplications is inconvenient and we therefore
% introduce a very simple way to describe sequences of elements as an array, here
% called |fodo.| This array has four columns with the following interpretations
%%
% * First column: code, drift=1, quad=2
% * Second column: repeat code for the element
% * Third column: length of one segment
% * Fourth column: strength of one segment, focal length for a thin quadrupole.
%%
% The second column with the repeat code is not strictly necessary but will
% make the graphs easier to interpret, because we can plot many intermediate points,
% rather than connecting values at the end of elements by a straight line.
%
% This particular sequence starts with an element having a 1 in the first column
% and is therefore a drift space. It consists of 5 segments, each 0.2 m long.
% The last column is not used here and contains a zero, The next element is a
% lens, beacuse the first code is 2 and it is defocusing, because the fourth column
% contains a negative number for the focal length. The third element is again
% a driftspace, this time with 10 segments of 0.2 m, followed by a focusing lens
% and another drift space. Such a sequence of focusing lens, drift space, defocusing
% lens and another drift space is usually referred to as a FODO (get it?) cell.
F=2.1;
fodo=[1, 5, 0.2, 0;
2, 1, 0, -F;
1, 10, 0.2, 0;
2, 1, 0, F;
1, 5, 0.2, 0]
%%
% Now we build our beamline as a sequence of one or many FODO cells. The latter
% we achieve by simply concatenating |fodo| vertically (note the semicolon!).
% We will always use |beamline| as a reserved name of the beamline that we work
% on.
%beamline=fodo; % one cell
%beamline=[fodo;fodo] % two cells
beamline=repmat(fodo,10,1) % ten cells
%%
% Before we can assemble the transfer matrices, we have to do a bit of accounting,
% like counting the number of elements |nlines| in the beamline description and
% the number |nmat| of matrices that we will need to store the matrices. This
% is just given by the sum of the segments in the second column plus one. Next
% we allocate space for the nmat $2\times 2$ matrices in the array Racc and initialize
% the first matrix as the $2\times2$unit matrix eye(2). This additional matrix
% at the beginning accounts for the "plus one" in the definition of |nmat|. Finally
% we allocate the array |spos| to store the position of each segment along the
% beam line.
nlines=size(beamline,1)
nmat=sum(beamline(:,2))+1;
Racc=zeros(2,2,nmat);
Racc(:,:,1)=eye(2);
spos=zeros(nmat,1);
%%
% Now we are ready to loop through the beamline description by looping over
% the lines, labeled by |line|, and then over the segments, labeled by |seg|.
% As we do that we increment the counter |ic| that keeps track of which segment
% we're currently working on. Then we initialize the transfer matrix of the current
% segment |Rcurr| by the unit marix before looking at the first column to decide
% whether the element is a drift space or a lens and fill |Rcurr| with the appropriate
% transfer matrix. In |case| the first column (|beamline(line,1)|) is a 1, it
% is a drift and we use the value from the third column as the length of the drift
% space. In |case| the first column contains a 2, it is a lens and we use the
% focal length from the fourth column as the argument to the function |Q(F).|
% In case we encounter any other number in the first column, we display a warning.
% At this point |Rcurr| contains the transfer matrix for the current segment and
% we find *the transfer matrix from the start of the beamline to the end of the
% current element* |Racc(:,:,ic)| by left-multiplying Rcurr to |Racc(:,:,ic-1)|
% which already contains the transfer matrix from the start to the end of he previous
% segment. Finally we calculate the position |spos(ic)| of the current segment
% from that of the previous segment |spos(ic-1)| by adding the length of the current
% segment to it.
ic=1;
for line=1:nlines
for seg=1:beamline(line,2)
ic=ic+1;
Rcurr=eye(2);
switch beamline(line,1)
case 1
Rcurr=D(beamline(line,3));
case 2
Rcurr=Q(beamline(line,4));
otherwise
disp('unknown element');
end
Racc(:,:,ic)=Rcurr*Racc(:,:,ic-1);
spos(ic)=spos(ic-1)+beamline(line,3);
end
end
%%
% At this point |Racc| contains all trasnfer matrices from the start of the
% beam line to the end of each and every segment while |spos| contains the positions
% of each segment.
%%
% Now we can use the transfer matrices to plot the trajectory of a particle
% with starting coordinates $x_0 = 0$ and $x_0'=10^{-3}$ rad by mapping the initial
% (column) vector |x0| with |Racc| to the ened of each segment and record both
% position |x(1)| and angle |x(2)| in the previously allocated array |data|.
data=zeros(nmat,2);
x0=[0;1e-3];
for k=1:nmat
x=Racc(:,:,k)*x0;
data(k,1)=x(1);
data(k,2)=x(2);
end
%%
% Now we can plot the trajectory and annotate the axes.
plot(spos,1000*data(:,1),'k')
xlabel('s [m]')
ylabel('x [mm]')
title('Trajectory')
%%
% We observe an oscillation of the particle's position along the beam line.
% This type of transverse oscillation is usually called _betatron oscillation_.
%
% Now you can go back to the start of the script and change the focal length
% F and observe how the oscillation changes. You might also want to increase the
% length of the beam line by adding more copies of the FODO cells in the definition
% of beamline by increasing the number of copies in |repmat()|. Explore!
%
% Calculating transfer matrices is a task we repeatedly have to do and it is
% therefore convenient to encapsulate the code that does that in a dedicated function;
% we call it |calcmat().| It receives the |beamline| as input and returns |Racc,
% spos,nmat|, and |nlines|. Internally it calculates nmat and nlines and then
% allocates arrays for |Racc| and |spos| and then loops over all lines and segments,
% just as we did in the code above. We use such a function in the following way
[Racc,spos,nmat,nlines]=calcmat(beamline);
%%
% andREPLACE_WITH_DASH_DASHviolaREPLACE_WITH_DASH_DASHall transfer matrces and the positions of all segments are immediately
% available. Here is the definition of the function |calcmat()|
function [Racc,spos,nmat,nlines]=calcmat(beamline)
nlines=size(beamline,1);
nmat=sum(beamline(:,2))+1;
Racc=zeros(2,2,nmat);
Racc(:,:,1)=eye(2);
spos=zeros(nmat,1);
ic=1;
for line=1:nlines
for seg=1:beamline(line,2)
ic=ic+1;
Rcurr=eye(2);
switch beamline(line,1)
case 1
Rcurr=D2(beamline(line,3));
case 2
Rcurr=Q2(beamline(line,4));
otherwise
disp('unknown element');
end
Racc(:,:,ic)=Rcurr*Racc(:,:,ic-1);
spos(ic)=spos(ic-1)+beamline(line,3);
end
end
end
%%
% Inside this function the function D(L) an Q(F) are unknown. We could either
% copy the two lines with their definitions into calcmat, or define additional
% functions that provide the same functionality. First for the drift space
function out=D2(L)
out=[1,L;0,1];
end
%%
% and for the lens
function out=Q2(F)
out=[1,0;-1/F,1];
end
%%
% Here I have given new names (D2, Q2)to functions for drift and quad, because
% the names "D" and "Q" are already used for the matrices in the first few lines
% of this script.
%
% These service functions will make all further examples much more compact and
% readable, because we never have to copy and paste the enclosed code verbatim
% into our scripts. This simple tutorial only contains the bare essentials of
% beam optics calculations. Have a look at the web site where you picked this
% script up and check out the other more advanced examples.
##### SOURCE END #####
-->
</div></body></html>