lab10¶
This is a voluntary lab. Only engage if you enjoy doing the exercises.
The dictionary tasks (encode
, reverse_dic
, and decode
) need a
bit of reading to understand the question.
The std_dev
question is more easily done using numpy
.
encode(code, msg)
¶
Cryptography
Background¶
In this part of the practical, we work with simple cryptographic codes. Imagine Alice wants to send a secret message to Bob and does not want third parties to be able to understand the message. Here, we imagine that Alice and Bob can agree on a particular code before they need to exchange messages. The particular code is called the shared (secret) key.
Alice and Bob will use the same key to (i) encode a message (i.e. change from plain text to something less readable) and later (ii) decode the encoded message back into plain text. This is known as Symmetric-key cryptography, where the same key is used both for encryption (=encoding) and decryption (=decoding).
A code in this lab is a mapping that takes a (input) letter of the alphabet and maps it to a different (output) letter. We can represent such a code in Python through a dictionary where the keys of the dictionary are the input characters and the values the output characters.
For example, let’s consider the trivial code that replaces the letter ‘e’ in the input data with the letter ‘x’ in the output, and simultaneously replaces the letter ‘x’ in the input data with the letter ‘e’ in the output. This could be written as \(\mathrm{e} \mapsto \mathrm{x}\) and \(\mathrm{x} \mapsto \mathrm{e}\).
This can be represented by the dictionary {'x': 'e', 'e': 'x'}
. The
function code1()
provides exactly this dictionary:
In [ ]: mycode = code1()
In [ ]: print(mycode)
{'x': 'e', 'e': 'x'}
We use the convention that if a character is not found in the keys of the dictionary, the character should not be changed by the code.
We now encode the string Hello World
using the mapping described by
mycode
, and obtain the string Hxllo World
.
(If we want to encode information without losing data, we need to make sure that no two keys map to the same value, i.e. the mapping has to be injective. Later, we want to reverse the mapping—to decode a coded message—and will need that the mapping has to be bijective, i.e. there has to be a one-to-one correspondence between input and output sets.)
Exercise (encode
)¶
Write a function encode(code, msg)
which takes the arguments:
code
: this is a dictionary describing a codemsg
: this is a string
The function should apply the mapping to each character of the string as described above and return the encoded output string.
We provide the example codes (code1()
, code2()
, …) below: you
can copy and paste them to help with experimentation and the exercises.
Examples
In [ ]: code = code1()
In [ ]: print(code)
{'x': 'e', 'e': 'x'}
In [ ]: encode(code, "Hello World")
Out[ ]: 'Hxllo World'
In [ ]: encode(code, "Jimi Hendrix")
Out[ ]: 'Jimi Hxndrie'
In [ ]: encode(code, "The name xenon (Xe) is derived from greek for strange.")
Out[ ]: 'Thx namx exnon (Xx) is dxrivxd from grxxk for strangx.'
The sentence “the quick brown fox jumps over the lazy dog” contains all letters of the English alphabet, and might be useful here as an example:
In [ ]: msg = "the quick brown fox jumps over the lazy dog"
In [ ]: encode(code1(), msg)
Out[ ]: 'thx quick brown foe jumps ovxr thx lazy dog'
Let us try some other codes, for example code2()
which maps
\(\mathrm{i} \mapsto \mathrm{s}\) and
\(\mathrm{s}\mapsto \mathrm{g}\) and
\(\mathrm{g}\mapsto \mathrm{i}\):
In [ ]: code2()
Out[ ]: {'g': 'i', 'i': 's', 's': 'g'}
In [ ]: encode(code2(), msg)
Out[ ]: 'the qusck brown fox jumpg over the lazy doi'
Moving on from these education examples, the code provided by function
code3()
makes the encoded message fairly unreadable:
In [22]: print(code3())
{'!': '?', ' ': '$', '#': '.', '$': ' ', '.': '#', '?': '!', 'A': 'B', 'C': 'D',
'B': 'C', 'E': 'F', 'D': 'E', 'G': 'H', 'F': 'G', 'I': 'J', 'H': 'I', 'K': 'L',
'J': 'K', 'M': 'N', 'L': 'M', 'O': 'P', 'N': 'O', 'Q': 'R', 'P': 'Q', 'S': 'T',
'R': 'S', 'U': 'V', 'T': 'U', 'W': 'X', 'V': 'W', 'Y': 'Z', 'X': 'Y', 'Z': 'A',
'a': 'b', 'c': 'd', 'b': 'c', 'e': 'f', 'd': 'e', 'g': 'h', 'f': 'g', 'i': 'j',
'h': 'i', 'k': 'l', 'j': 'k', 'm': 'n', 'l': 'm', 'o': 'p', 'n': 'o', 'q': 'r',
'p': 'q', 's': 't', 'r': 's', 'u': 'v', 't': 'u', 'w': 'x', 'v': 'w', 'y': 'z',
'x': 'y', 'z': 'a'}
In [23]: msg
Out[23]: 'the quick brown fox jumps over the lazy dog'
In [24]: encode(code3(), msg)
Out[24]: 'uif$rvjdl$cspxo$gpy$kvnqt$pwfs$uif$mbaz$eph'
Appendix: example code mappings¶
def code0():
"""A trivial code - no change."""
return {}
def code1():
"""A very simple example (symmetric)."""
return {"e": "x", "x": "e"}
def code2():
"""A simple example i->s, s->g and g->i."""
return {"i": "s", "s": "g", "g": "i"}
def code3():
"""A more complicated code."""
dic = {}
# lower case letters
dic["z"] = "a"
for i in range(ord("a"), ord("z")):
dic[chr(i)] = chr(i + 1)
# upper case letters
dic["Z"] = "A"
for i in range(ord("A"), ord("Z")):
dic[chr(i)] = chr(i + 1)
# space, stop and some other special characters
dic[" "] = "$"
dic["."] = "#"
dic["#"] = "."
dic["$"] = " "
dic["?"] = "!"
dic["!"] = "?"
return dic
def check_code_is_reversible(dic):
"""Given a dictionary used as a code mapping, this function checks
whether the set of keys is the same set of values: if the elements
in the keys are the same unique set as those in the values, then
this mapping is bijective (the set of values is then actually a
permutation of the set of input values) and can be inverted.
If this is not the case, some debug information is printed, and a
ValueError exception raised.
"""
unique_keys = set()
unique_vals = set()
for key, val in dic.items():
unique_keys.add(key)
unique_vals.add(val)
if unique_vals != unique_keys:
print("Code is not reversible:")
print("keys are %s", sorted(unique_keys))
print("values are %s", sorted(unique_vals))
raise ValueError("Code is not reversible - stopping here")
return True
def test_codes():
"""Check that codes defined above are reversible."""
for c in (code0(), code1(), code2(), code3()):
assert check_code_is_reversible(c)
secretmessage \
= "Zpv$ibwf$tvddfttgvmmz$efdpefe$uijt$tfdsfu$nfttbhf#$Dpohsbuvmbujpot?"
reverse_dic(d)
¶
Implement a function reverse_dic(d)
that takes a dictionary d
as
the input argument and returns a dictionary r
. If the dictionary
d
has a key k
and an associated value v
, then the dictionary
r
should have a key v
and a value k
.
(This is only expected to work for dictionaries that have a unique set of values although you do not need to check for this.)
Example:
>>> d = {'key1': 'value1'}
>>> r = reverse_dic(d)
>>> r
{'value1': 'key1'}
>>> reverse_dic({'dog': 10, 'cat': 20})
{10: 'dog', 20: 'cat'}
decode(code, encoded_mesg)
¶
Decoding an encrypted message (decode
)
Write a function decode(code, encoded_msg)
that takes a dictionary
code that contains the mapping dictionary with which the string
encoded_msg
has been encoded. The function decode
should return
the decoded message.
Example:
In [ ]: msg = "the quick brown fox jumps over the lazy dog"
In [ ]: code2()
Out[ ]: {'g': 'i', 'i': 's', 's': 'g'}
In [ ]: x = encode(code2(), msg)
In [ ]: x
Out[ ]: 'the qusck brown fox jumpg over the lazy doi'
In [ ]: decode(code2(), x)
Out[ ]: 'the quick brown fox jumps over the lazy dog'
In [ ]: y = encode(code3(), msg)
In [ ]: y
Out[ ]: 'uif$rvjdl$cspxo$gpy$kvnqt$pwfs$uif$mbaz$eph'
In [ ]: decode(code3(), y)
Out[ ]: 'the quick brown fox jumps over the lazy dog'
Hints: Once you have the function reverse_dic
and encode
defined, you can decode a coded message by using the reversed code
dictionary in the encode function:
In [ ]: msg = "the quick brown fox jumps over the lazy dog"
In [ ]: encoded = encode(code2(), msg)
In [ ]: encoded
Out[ ]: 'the qusck brown fox jumpg over the lazy doi'
In [ ]: code2_reversed = reverse_dic(code2())
In [ ]: encode(code2_reversed, encoded)
Out[ ]: 'the quick brown fox jumps over the lazy dog'
Can you now decode the message:
Zpv$ibwf$tvddfttgvmmz$efdpefe$uijt$tfdsfu$nfttbhf#$Dpohsbuvmbujpot?
which has been encoded with the mapping provided by function code3?
trapez(f, a, b, n)
¶
Composite trapezoidal integration
Background¶
The equation below shows that the integral \(I\) of the function \(f(x)\) from \(x=a\) to \(x=b\) can be approximated by \(A\) through the so-called composite trapezoidal integration rule:
where \(h=\frac{b-a}{n}\), and \(x_i=a+ih\), and \(n\) is the number of subdivisions.
Example:
The figure demonstrates the idea for \(f(x)=x^2\), \(a=0\), \(b=2\), \(n=4\) and thus \(h=0.5\):
The composite trapezoidal rule computes the area under the red line exactly, where the red line is an approximation of the actual function of interest (here \(f(x)=x^2\)) shown by a black dashed line.
Exercise (trapez
):¶
Write a function trapez(f, a, b, n)
which is given
a function
f
which depends on one parameter (i.e.f(x)
) and returns a numbera lower (
a
) and upper (b
) integration limitand the number of subdivisions (
n
) to be used.
The function should use the composite trapezoidal rule to compute A
and return this value.
Examples
In [ ]: def f(x):
...: return x
...:
In [ ]: trapez(f, 0, 1, 1)
Out[ ]: 0.5
In [ ]: trapez(f, 0, 1, 5)
Out[ ]: 0.5
In [ ]: def f2(x):
...: return x * x
...:
In [ ]: trapez(f2, 0, 1, 1)
Out[ ]: 0.5
In [ ]: trapez(f2, 0, 1, 10)
Out[ ]: 0.3350000000000001
In [ ]: trapez(f2, 0, 1, 100)
Out[ ]: 0.33335000000000004
In [ ]: trapez(f2, 0, 1, 1000)
Out[ ]: 0.33333349999999995
finderror(n)
¶
Estimate the numerical integration error
Write a function finderror(n)
which uses the trapez()
function
to find the error of the trapezoidal integral approximation. The
function should compute the integral of \(f(x) = x^2\) with
integration limits \(a = 0\) and \(b = 2\) numerically. The
function should then subtract this numerical result \(A\) from the
exact integral value \(I\) and return the difference. Using anytical
methods (i.e. symbolic integration), we can compute the exact integral
to be \(I=\int\limits_0^2f(x)\mathrm{d}x = \frac{8}{3}\).
Example:
In [ ]: finderror(5)
Out[ ]: -0.05333333333333412
(You should find approximately that the error decays by a factor 4 if
you double n
(because the trapez
method has an error of order
\(h^2\) and \(h\) is proportional to \(1/n\).)
using_quad()
¶
Numerical integration
Write a function using_quad()
that computes and returns the integral
\(\int\limits_a^b x^2 f(x) \mathrm{d}x\) with \(f(x) = x^2\)
from \(a=0\) to \(b=2\) using scipy.integrate.quad
. The
function using_quad
should return the value that the quad()
function returns, i.e. a tuple (y, abserr)
where y
is
quad
’s best approximation of the true integral, and abserr
an
absolute error estimate.
You should find that this method (with the default settings) is (far) more accurate than our trapez function.
std_dev(x)
¶
Compute the standard deviation as a data processing exercise
A function std_dev(x)
which takes a list x
of floating point
numbers, and computes and returns the standard deviation of the sample
of the floating point numbers in the list x
.
For clarity, if the list x
with N
elements is defined as
\(x = [x_1, x_2, x_3, \ldots, x_N]\), then the standard deviation is
given by
where \(\mu\) is the (arithmetic) mean
Examples:
In [ ]: std_dev([1.0, 1.0])
Out[ ]: 0.0
In [ ]: std_dev([1.0, 2.0])
Out[ ]: 0.5
Please submit your file lab10.py
for this assignment.
End of lab10.