author | Christian Urban <christian.urban@kcl.ac.uk> |
Mon, 08 Nov 2021 01:52:56 +0000 | |
changeset 406 | ad24f50c484d |
parent 399 | b17a98b0c52f |
child 411 | f55742af79ef |
permissions | -rw-r--r-- |
276 | 1 |
% !TEX program = xelatex |
6 | 2 |
\documentclass{article} |
11
417869f65585
updated
Christian Urban <christian dot urban at kcl dot ac dot uk>
parents:
9
diff
changeset
|
3 |
\usepackage{../style} |
195 | 4 |
\usepackage{disclaimer} |
199 | 5 |
\usepackage{../langs} |
6 | 6 |
|
335 | 7 |
|
8 |
||
6 | 9 |
\begin{document} |
10 |
||
399 | 11 |
\section*{Main Part 1 (Scala, 6 Marks)} |
199 | 12 |
|
399 | 13 |
\IMPORTANT{This part is about Scala. It is due on \cwSIXa{} at 5pm and worth 6\%.} |
195 | 14 |
|
127 | 15 |
\noindent |
195 | 16 |
Also note that the running time of each part will be restricted to a |
199 | 17 |
maximum of 30 seconds on my laptop. |
192 | 18 |
|
196 | 19 |
\DISCLAIMER{} |
6 | 20 |
|
201 | 21 |
\subsection*{Reference Implementation} |
6 | 22 |
|
199 | 23 |
Like the C++ assignments, the Scala assignments will work like this: you |
24 |
push your files to GitHub and receive (after sometimes a long delay) some |
|
25 |
automated feedback. In the end we take a snapshot of the submitted files and |
|
266 | 26 |
apply an automated marking script to them.\medskip |
199 | 27 |
|
266 | 28 |
\noindent |
306 | 29 |
In addition, the Scala coursework comes with a reference implementation |
30 |
in form of \texttt{jar}-files. This allows you to run any test cases |
|
199 | 31 |
on your own computer. For example you can call Scala on the command |
335 | 32 |
line with the option \texttt{-cp drumb.jar} and then query any |
199 | 33 |
function from the template file. Say you want to find out what |
345 | 34 |
the function \code{get_january_data} |
35 |
produces: for this you just need to prefix them with the object name |
|
396 | 36 |
\texttt{M1} and call them with some arguments: |
199 | 37 |
|
38 |
\begin{lstlisting}[language={},numbers=none,basicstyle=\ttfamily\small] |
|
345 | 39 |
$ scala -cp drumb.jar |
199 | 40 |
|
396 | 41 |
scala> M1.get_january_data("FB", 2014) |
345 | 42 |
val res2: List[String] = List(2014-01-02,54.709999,....) |
199 | 43 |
\end{lstlisting}%$ |
44 |
||
201 | 45 |
\subsection*{Hints} |
46 |
||
47 |
\noindent |
|
396 | 48 |
\textbf{For Main Part 1:} useful string functions: |
266 | 49 |
\texttt{.startsWith(...)} for checking whether a string has a given |
50 |
prefix, \texttt{\_ ++ \_} for concatenating two strings; useful option |
|
51 |
functions: \texttt{.flatten} flattens a list of options such that it |
|
52 |
filters way all \texttt{None}'s, \texttt{Try(...).getOrElse ...} runs |
|
53 |
some code that might raise an exception---if yes, then a default value |
|
54 |
can be given; useful list functions: \texttt{.head} for obtaining the |
|
55 |
first element in a non-empty list, \texttt{.length} for the length of |
|
56 |
a list; \texttt{.filter(...)} for filtering out elements in a list; |
|
57 |
\texttt{.getLines.toList} for obtaining a list of lines from a file; |
|
58 |
\texttt{.split(",").toList} for splitting strings according to a |
|
59 |
comma.\bigskip |
|
201 | 60 |
|
396 | 61 |
\noindent\alert |
266 | 62 |
\textbf{Note!} Fortunately Scala supports operator overloading. But |
63 |
make sure you understand the difference between \texttt{100 / 3} and |
|
201 | 64 |
\texttt{100.0 / 3}! |
65 |
||
66 |
\newpage |
|
399 | 67 |
\subsection*{Main Part 1 (6 Marks, file drumb.scala)} |
199 | 68 |
|
69 |
A purely fictional character named Mr T.~Drumb inherited in 1978 |
|
70 |
approximately 200 Million Dollar from his father. Mr Drumb prides |
|
71 |
himself to be a brilliant business man because nowadays it is |
|
72 |
estimated he is 3 Billion Dollar worth (one is not sure, of course, |
|
73 |
because Mr Drumb refuses to make his tax records public). |
|
74 |
||
75 |
Since the question about Mr Drumb's business acumen remains open, |
|
76 |
let's do a quick back-of-the-envelope calculation in Scala whether his |
|
77 |
claim has any merit. Let's suppose we are given \$100 in 1978 and we |
|
78 |
follow a really dumb investment strategy, namely: |
|
79 |
||
80 |
\begin{itemize} |
|
81 |
\item We blindly choose a portfolio of stocks, say some Blue-Chip stocks |
|
82 |
or some Real Estate stocks. |
|
83 |
\item If some of the stocks in our portfolio are traded in January of |
|
84 |
a year, we invest our money in equal amounts in each of these |
|
85 |
stocks. For example if we have \$100 and there are four stocks that |
|
86 |
are traded in our portfolio, we buy \$25 worth of stocks |
|
306 | 87 |
from each. (Be careful to also test cases where you trade with 3 stocks.) |
199 | 88 |
\item Next year in January, we look at how our stocks did, liquidate |
89 |
everything, and re-invest our (hopefully) increased money in again |
|
90 |
the stocks from our portfolio (there might be more stocks available, |
|
91 |
if companies from our portfolio got listed in that year, or less if |
|
92 |
some companies went bust or were de-listed). |
|
276 | 93 |
\item We do this for 41 years until January 2019 and check what would |
199 | 94 |
have become out of our \$100. |
95 |
\end{itemize} |
|
96 |
||
97 |
\noindent |
|
266 | 98 |
Until Yahoo was bought by Altaba a few years ago, historical stock market |
199 | 99 |
data for such back-of-the-envelope calculations was freely available |
201 | 100 |
online. Unfortunately nowadays this kind of data is more difficult to |
199 | 101 |
obtain, unless you are prepared to pay extortionate prices or be |
306 | 102 |
severely rate-limited. Therefore this part comes with a number |
199 | 103 |
of files containing CSV-lists with the historical stock prices for the |
104 |
companies in our portfolios. Use these files for the following |
|
396 | 105 |
tasks.\medskip |
106 |
||
107 |
\noindent\alert |
|
108 |
\textbf{Note:} Do not hardcode the path to the CSV-files. The testing |
|
109 |
framework will assume that these files are in the same directory as the |
|
110 |
drumb.scala file. |
|
111 |
\bigskip |
|
199 | 112 |
|
201 | 113 |
\newpage |
199 | 114 |
\noindent |
115 |
\textbf{Tasks} |
|
116 |
||
117 |
\begin{itemize} |
|
118 |
\item[(1)] Write a function \texttt{get\_january\_data} that takes a |
|
119 |
stock symbol and a year as arguments. The function reads the |
|
120 |
corresponding CSV-file and returns the list of strings that start |
|
121 |
with the given year (each line in the CSV-list is of the form |
|
399 | 122 |
\texttt{someyear-01-someday,someprice}).\hfill[0.5 Marks] |
199 | 123 |
|
124 |
\item[(2)] Write a function \texttt{get\_first\_price} that takes |
|
125 |
again a stock symbol and a year as arguments. It should return the |
|
126 |
first January price for the stock symbol in the given year. For this |
|
127 |
it uses the list of strings generated by |
|
128 |
\texttt{get\_january\_data}. A problem is that normally a stock |
|
129 |
exchange is not open on 1st of January, but depending on the day of |
|
130 |
the week on a later day (maybe 3rd or 4th). The easiest way to solve |
|
131 |
this problem is to obtain the whole January data for a stock symbol |
|
132 |
and then select the earliest, or first, entry in this list. The |
|
133 |
stock price of this entry should be converted into a double. Such a |
|
134 |
price might not exist, in case the company does not exist in the given |
|
135 |
year. For example, if you query for Google in January of 1980, then |
|
136 |
clearly Google did not exist yet. Therefore you are asked to |
|
137 |
return a trade price with type \texttt{Option[Double]}\ldots\texttt{None} |
|
138 |
will be the value for when no price exists; \texttt{Some} if there is a |
|
139 |
price.\hfill[1 Mark] |
|
140 |
||
141 |
\item[(3)] Write a function \texttt{get\_prices} that takes a |
|
142 |
portfolio (a list of stock symbols), a years range and gets all the |
|
143 |
first trading prices for each year in the range. You should organise |
|
144 |
this as a list of lists of \texttt{Option[Double]}'s. The inner |
|
145 |
lists are for all stock symbols from the portfolio and the outer |
|
146 |
list for the years. For example for Google and Apple in years 2010 |
|
147 |
(first line), 2011 (second line) and 2012 (third line) you obtain: |
|
148 |
||
149 |
\begin{verbatim} |
|
314 | 150 |
List(List(Some(312.204773), Some(26.782711)), |
315 | 151 |
List(Some(301.0466), Some(41.244694)), |
314 | 152 |
List(Some(331.462585), Some(51.464207)))) |
201 | 153 |
\end{verbatim}\hfill[1 Mark] |
266 | 154 |
|
155 |
||
156 |
%\end{itemize} |
|
199 | 157 |
|
266 | 158 |
%\subsection*{Advanced Part 3 (4 Marks, continue in file drumb.scala)} |
159 |
% |
|
160 |
%\noindent |
|
161 |
%\textbf{Tasks} |
|
199 | 162 |
|
266 | 163 |
%\begin{itemize} |
199 | 164 |
|
165 |
\item[(4)] Write a function that calculates the \emph{change factor} (delta) |
|
166 |
for how a stock price has changed from one year to the next. This is |
|
167 |
only well-defined, if the corresponding company has been traded in both |
|
168 |
years. In this case you can calculate |
|
169 |
||
170 |
\[ |
|
171 |
\frac{price_{new} - price_{old}}{price_{old}} |
|
172 |
\] |
|
173 |
||
174 |
If the change factor is defined, you should return it |
|
175 |
as \texttt{Some(change\_factor)}; if not, you should return |
|
176 |
\texttt{None}.\mbox{}\hfill\mbox{[1 Mark]} |
|
177 |
||
178 |
\item[(5)] Write a function that calculates all change factors |
|
266 | 179 |
(deltas) for the prices we obtained in Task (2). For the running |
199 | 180 |
example of Google and Apple for the years 2010 to 2012 you should |
181 |
obtain 4 change factors: |
|
182 |
||
266 | 183 |
\begin{verbatim} |
184 |
List(List(Some(-0.03573991804411003), Some(0.539974575389325)), |
|
185 |
List(Some(0.10103414222249969), Some(0.24777764141006836))) |
|
199 | 186 |
\end{verbatim} |
187 |
||
188 |
That means Google did a bit badly in 2010, while Apple did very well. |
|
189 |
Both did OK in 2011. Make sure you handle the cases where a company is |
|
190 |
not listed in a year. In such cases the change factor should be \texttt{None} |
|
266 | 191 |
(recall Task~(4)). |
199 | 192 |
\mbox{}\hfill\mbox{[1 Mark]} |
193 |
||
194 |
\item[(6)] Write a function that calculates the ``yield'', or |
|
195 |
balance, for one year for our portfolio. This function takes the |
|
196 |
change factors, the starting balance and the year as arguments. If |
|
197 |
no company from our portfolio existed in that year, the balance is |
|
198 |
unchanged. Otherwise we invest in each existing company an equal |
|
199 |
amount of our balance. Using the change factors computed under Task |
|
266 | 200 |
(2), calculate the new balance. Say we had \$100 in 2010, we would have |
199 | 201 |
received in our running example involving Google and Apple: |
202 |
||
203 |
\begin{verbatim} |
|
266 | 204 |
$50 * -0.03573991804411003 + $50 * 0.539974575389325 |
205 |
= $25.21173286726075 |
|
199 | 206 |
\end{verbatim} |
207 |
||
208 |
as profit for that year, and our new balance for 2011 is \$125 when |
|
209 |
converted to a \texttt{Long}.\mbox{}\hfill\mbox{[1 Mark]} |
|
210 |
||
211 |
\item[(7)] Write a function that calculates the overall balance |
|
212 |
for a range of years where each year the yearly profit is compounded to |
|
213 |
the new balances and then re-invested into our portfolio. |
|
214 |
For this use the function and results generated under (6).\\ |
|
399 | 215 |
\mbox{}\hfill\mbox{[0.5 Marks]} |
199 | 216 |
\end{itemize}\medskip |
217 |
||
218 |
||
219 |
||
220 |
\noindent |
|
221 |
\textbf{Test Data:} File \texttt{drumb.scala} contains two portfolios |
|
222 |
collected from the S\&P 500, one for blue-chip companies, including |
|
223 |
Facebook, Amazon and Baidu; and another for listed real-estate |
|
224 |
companies, whose names I have never heard of. Following the dumb |
|
266 | 225 |
investment strategy from 1978 until 2019 would have turned a starting |
226 |
balance of \$100 into roughly \$39,162 for real estate and a whopping |
|
227 |
\$462,199 for blue chips. Note when comparing these results with your |
|
199 | 228 |
own calculations: there might be some small rounding errors, which |
229 |
when compounded lead to moderately different values.\bigskip |
|
230 |
||
231 |
||
232 |
\noindent |
|
233 |
\textbf{Moral:} Reflecting on our assumptions, we are over-estimating |
|
234 |
our yield in many ways: first, who can know in 1978 about what will |
|
235 |
turn out to be a blue chip company. Also, since the portfolios are |
|
236 |
chosen from the current S\&P 500, they do not include the myriad |
|
237 |
of companies that went bust or were de-listed over the years. |
|
238 |
So where does this leave our fictional character Mr T.~Drumb? Well, given |
|
239 |
his inheritance, a really dumb investment strategy would have done |
|
396 | 240 |
equally well, if not much better. And one would assume this guy is |
241 |
by now locked up in prison and the key thrown away, but alas he |
|
242 |
is still around annoying commonsense people.\medskip |
|
199 | 243 |
|
244 |
\end{document} |
|
245 |