layout | title |
---|---|
presentation |
Week 3, session 1: advanced string formatting, structs & classes |
class: title
5CCYB041
By now, you should be able to: .columns[ .col[
- use the terminal and navigate around the filesystem
- write, compile and run a small C++ program
- use
for
loops andif
statements - use standard data types, including
std::string
andstd::vector
] .col[ - write and use your own functions
- read and write basic information to & from file
- handle command-line arguments and options
- split your code across multiple
cpp
and header files ] ]
Ideally, you should also have a basic understanding of: .columns[ .col[
- error handling using C++ exceptions
- (const) references
- variable scope ] .col[
- the
inline
keyword - namespaces
- the compile & link process ] ]
If you struggle with any of these aspects, please ask for help!
We continue working on our DNA shotgun sequencing project
You can find the most up to date version in the project's solution/
folder
.explain-bottom[ Make sure your code is up to date now! ]
name: format
We've already seen how to concatenate strings to form more complex strings, and
how to convert numeric values to strings using std::to_string()
- but this isn't always very convenient or easy to follow
--
The C++20 standard introduces a new function to help with string formatting:
std::format()
--
Its use is best illustrated with an example. Instead of writing:
debug::log ("read " + std::to_string (fragments.size()) + " fragments");
--
we can write:
#include <format>
...
debug::log (std::format ("read {} fragments", fragments.size()));
The std::format()
template function has the following (highly simplified) declaration:
namespace std {
string format (format_string fmt, ArgType1 arg1, ArgType2 arg2, ...);
}
The std::format()
template function has the following (highly simplified) declaration:
*namespace std {
string format (format_string fmt, ArgType1 arg1, ArgType2 arg2, ...);
*}
- it is declared within the
std
namespace
The std::format()
template function has the following (highly simplified) declaration:
namespace std {
`string` format (format_string fmt, ArgType1 arg1, ArgType2 arg2, ...);
}
- it is declared within the
std
namespace - it returns a
std::string
The std::format()
template function has the following (highly simplified) declaration:
namespace std {
string format (`format_string fmt`, ArgType1 arg1, ArgType2 arg2, ...);
}
- it is declared within the
std
namespace - it returns a
std::string
- the first argument is the format string, of type
std::format_string
- it contains the text for the output string, with braces
{}
where substitutions are to be inserted
- it contains the text for the output string, with braces
The std::format()
template function has the following (highly simplified) declaration:
namespace std {
string format (format_string fmt, `ArgType1 arg1`, ArgType2 arg2, ...);
}
- it is declared within the
std
namespace - it returns a
std::string
- the first argument is the format string, of type
std::format_string
- it contains the text for the output string, with braces
{}
where substitutions are to be inserted
- it contains the text for the output string, with braces
- each subsequent argument is a variable to be converted to text and inserted
into the format string instead of the matching
{}
For example:
std::string name = "Joe";
std::string colour = "orange";
std::cout << std::format ("My name is {}, my favorite colour is {}\n", name, colour);
would produce:
My name is Joe, my favorite colour is orange
But the arguments to be substituted don't need to be strings:
int num_iter = 101;
double func_value = 0.023859;
std::cout << std::format ("after {} iterations, function value = {}\n",
num_iter, func_value);
would produce:
after 101 iterations, function value = 0.023859
For numeric arguments, the conversion to text can be carefully controlled:
int num_iter = 101;
double func_value = 0.023859;
std::cout << std::format ("after {} iterations, function value = `{:.3f}`\n",
num_iter, func_value);
would produce the second argument to 3 decimal places:
after 101 iterations, function value = 0.024
There are many more formatting options – too many to cover in this course!
For details, please refer to the relevant documentation
.explain-bottom[
Exercise: use the std::format()
function where relevant in your own code
]
name: struct
class: section
Looking at our project, we would like to add a find_biggest_overlap()
function to:
- identify the fragment that has the biggest overlap with the current sequence
- remove it from the list of candidate fragments
- and return the size of the overlap
We need to return two pieces of information from that function!
--
One approach to this problem relies on references:
- one of the arguments to our function is a reference to an existing variable, and the function will assign the correct value to that variable before returning:
int find_biggest_overlap (const std::string& sequence, Fragments& fragments,
`int& index`)
- The
index
variable is passed by non-const reference, allowing the function to assign a value to it that will also update the original variable. - the function would then be free to use the return value to provide the size of the corresponding overlap
We would then be able to use this function as follows:
int index_of_fragment;
int overlap_size = find_biggest_overlap (sequence, fragments, index_of_fragment);
Since index_of_fragment
is passed by reference, the function can update its value
- we can now rely on both
overlap_size
andindex_of_fragment
being set correctly.
--
However, this is a cumbersome approach
- we need to declare a variable before invoking the function
- the intent is not immediately clear
--
A better solution would be to return a single variable of a type capable of holding multiple values
- for example, we could return a
std::vector<int>
here - but what if the two values to be returned were of a different type?
A cleaner solution would be to declare our own compound data type, composed of the two variables we need.
This can be done using structures
- structures are an old concept: they predate C++ and were already present in C
--
Structures allow us to define a new compound data type, composed of other data types, grouped together into a single entity.
- each member variable is named, and can therefore be assigned a clear interpretation
- the struct can then be treated as any other variable, passed to & from function calls, etc.
--
This is best illustrated with an example
Structures are declared using the struct
keyword, followed by the list of
members (along with their types) enclosed in braces:
struct Overlap {
int size;
int fragment;
};
--
This can then be used as a regular data type in our function declaration:
`Overlap` find_biggest_overlap (const std::string& sequence, Fragments& fragments);
--
We can use our function as follows, and access the member variables using dot-notation:
`auto overlap` = find_biggest_overlap (sequence, fragments);
std::cerr << std::format ("overlap of size {} at index {}\n",
`overlap.size`, `overlap.fragment`);
In the implementation of our function (the function definition), we need to
return a variable of type Overlap
. We can do that like this:
...
Overlap retval;
retval.size = biggest_overlap;
retval.fragment = fragment_with_biggest_overlap;
return retval;
}
--
... but a much cleaner solution is to use aggregate initialisation in the return
statement:
...
return { biggest_overlap, fragment_with_biggest_overlap };
}
--
Since the return type is known, the compiler knows that this needs to
initialise an Overlap
struct.
- each member of the struct is then initialised with the matching variable in the brace-delimited list
Declaring our own custom struct
allows us to return multiple pieces of
information as a single variable
- this is a cleaner way to solve our problem
--
.explain-bottom[
Exercise: add the find_biggest_overlap()
function to your own code
]
name: structured_binding
class: section
The C++17 standard introduced the ability to use structured binding
- this allows us to declare and initialise multiple variables from the individual members of a larger composite variable
--
This is best illustrated by example, using our find_biggest_overlap()
function:
`auto overlap` = find_biggest_overlap (sequence, fragments);
std::cerr << std::format ("overlap of size {} at index {}\n",
`overlap.size`, `overlap.fragment`);
--
Using structured binding, we can declare independent variables for each
member of the Overlap
structure returned:
auto [ overlap, index ] = find_biggest_overlap (sequence, fragments);
std::cerr << std::format ("overlap of size {} at index {}\n", `overlap`, `index`);
--
.explain-bottom[ Exercise: modify your own code to use structured binding as shown here ]
class: info
The C++ STL provides a convenience template class that can be used as a
lightweight replacement for a dedicated struct
in a variety
of contexts: std::tuple
One use for std::tuple
is for structured binding:
#include <tuple>
std::tuple<int,int> find_biggest_overlap (const std::string& sequence,
Fragments& fragments)
{
...
return { biggest_overlap, fragment_with_biggest_overlap };
}
...
auto [ overlap, index ] = find_biggest_overlap (sequence, fragments);
class: info
While appealing in its simplicity, std::tuple
does not provide any indication
about the interpretation or meaning of each member
- in our case, all we know is that we get 2
int
s - which one represents the overlap size, and which the index?
- We have to rely on additional documentation to work this out
- this could be described in a comment, or some other external document
--
On the other hand, using a struct
provides named members, whose interpretation is
immediately much clearer.
⇒ Better to use a struct
with named members rather than a std::tuple
where possible
--
The std::tuple
class is very useful in other contexts, especially when
working with generic functions and containers
- a generic function or container will not know ahead of time what it is to contain
std::tuple
provides a simple container for an arbitrary number of variables of arbitrary typesstd::tuple
is therefore used in many places throughout the C++ STL
Structured binding offers another way to return multiple values from a function
- this can sometimes be more convenient or more legible
--
There are many other uses for structured binding
- many of these are too advanced for this course
- indeed, arguably some of the most useful were only introduced in C++23!
- if interested, have a look online for examples
name: string_view
class: section
Let's take a look at our code so far, and try to reason about where most of the time is spent
--
The bulk of our algorithm consists of a series of nested loops:
• loop while there are fragments left to merge
• loop over all candidate fragments
• loop over all overlap sizes
• extract the corresponding substrings and compare
• merge the fragment with the largest overlap
• remove it from the list of candidate fragments
--
When looking for opportunities to improve performance, we should focus our attention on the innermost loop
- this is the portion of code that will be executed most often
--
⇒ what is going on in the innermost loop?
int compute_overlap (const std::string& sequence, const std::string& fragment)
{
...
for (int overlap = fragment.size(); overlap > 0; --overlap) {
const auto seq_start = sequence.substr(0, overlap);
const auto frag_end = fragment.substr(fragment.size()-overlap);
if (seq_start == frag_end) {
largest_overlap = overlap;
break;
}
}
...
-- We actually have three loops, although we did not write them up ourselves!
int compute_overlap (const std::string& sequence, const std::string& fragment)
{
...
for (int overlap = fragment.size(); overlap > 0; --overlap) {
const auto seq_start = `sequence.substr(0, overlap)`;
const auto frag_end = `fragment.substr(fragment.size()-overlap)`;
if (seq_start == frag_end) {
largest_overlap = overlap;
break;
}
}
...
We actually have three loops, although we did not write them up ourselves!
- one for extracting each substring
int compute_overlap (const std::string& sequence, const std::string& fragment)
{
...
for (int overlap = fragment.size(); overlap > 0; --overlap) {
const auto seq_start = sequence.substr(0, overlap);
const auto frag_end = fragment.substr(fragment.size()-overlap);
if (`seq_start == frag_end`) {
largest_overlap = overlap;
break;
}
}
...
We actually have three loops, although we did not write them up ourselves!
- one for extracting each substring
- one for the comparison itself
int compute_overlap (const std::string& sequence, const std::string& fragment)
{
...
for (int overlap = fragment.size(); overlap > 0; --overlap) {
`const auto seq_start` = sequence.substr(0, overlap);
`const auto frag_end` = fragment.substr(fragment.size()-overlap);
if (seq_start == frag_end) {
largest_overlap = overlap;
break;
}
}
...
We actually have three loops, although we did not write them up ourselves!
- one for extracting each substring
- one for the comparison itself
We also have some implicit memory allocation & deallocation for each substring
- we will cover this much later in the course
int compute_overlap (const std::string& sequence, const std::string& fragment)
{
...
for (int overlap = fragment.size(); overlap > 0; --overlap) {
const auto seq_start = sequence.substr(0, overlap);
const auto frag_end = fragment.substr(fragment.size()-overlap);
if (seq_start == frag_end) {
largest_overlap = overlap;
break;
}
}
...
- there is not much that can be done about the comparison itself
- thankfully, it will tend to terminate early (as soon as any character doesn't match) --
- but can we avoid copying each substring into a whole new
std::string
to perform the comparison?
int compute_overlap (const std::string& sequence, const std::string& fragment)
{
...
for (int overlap = fragment.size(); overlap > 0; --overlap) {
bool match = true;
for (int n = 0; n < overlap; ++n) {
if (sequence[n] != fragment[fragment.size()-overlap+n]) {
match = false;
break;
}
}
if (match) {
largest_overlap = overlap;
break;
}
}
...
We can perform the comparison manually character-by-character, ensuring we compare the characters at the correct positions
int compute_overlap (const std::string& sequence, const std::string& fragment)
{
...
for (int overlap = fragment.size(); overlap > 0; --overlap) {
const auto seq_start = sequence.substr(0, overlap);
const auto frag_end = fragment.substr(fragment.size()-overlap);
if (seq_start == frag_end) {
largest_overlap = overlap;
break;
}
}
...
... or we can use the std::string_view
class to achieve the same effect!
#include <string_view>
...
int compute_overlap (const std::string& sequence, const std::string& fragment)
{
...
for (int overlap = fragment.size(); overlap > 0; --overlap) {
const auto seq_start = `std::string_view (sequence)`.substr(0, overlap);
const auto frag_end = `std::string_view (fragment)`.substr(fragment.size()-overlap);
if (seq_start == frag_end) {
largest_overlap = overlap;
break;
}
}
...
... or we can use the std::string_view
class to achieve the same effect!
How does this work?
Each std::string
holds information consisting of:
- the size of the string
- a handle to the memory location corresponding to the start of the text
- this is actually a pointer – which we are trying to avoid as much as possible on this course!
↓
(*#&\:;*'¬|\^$(&*^$%("()!`this is the text that our string contains`@~@:~#*(£';~"£*(£|#@
-- Memory is essentially a very long list of 8-bit bytes
- most of these are going to contain data in machine representation (binary)
- none of that will make sense as text
- the text for our string will occupy some contiguous section of memory
- the
std::string
class only needs to know where it starts and how long it is
- the
--
The std::string_view
class holds exactly the same information
The std::string_view
class provides a non-owning (and non-modifiable) view
of a string
Notably, it provides its own substr()
method, which also returns a
std::string_view
- with a different start and size for the substring
- no copying is performed!
--
Original std::string
:
- start at position 1000, size 41 bytes
↓
(*#&\:;*'¬|\^$(&*^$%("()!`this is the text that our string contains`@~@:~#*(£';~"£*(£|#@
The std::string_view
class provides a non-owning (and non-modifiable) view
of a string
Notably, it provides its own substr()
method, which also returns a
std::string_view
- with a different start and size for the substring
- no copying is performed!
Original std::string
:
- start at position 1000, size 41 bytes
↓
(*#&\:;*'¬|\^$(&*^$%("()!this is the text that `our string` contains@~@:~#*(£';~"£*(£|#@
↑
std::string_view
for "our string"
:
- start at position 1022, size 10 bytes
The main difference between std::string
and std::string_view
is one of
ownership
--
Each std::string
'owns' the memory where its text is
stored, and is responsible for its management
- it needs to allocate the memory when some text needs to be stored
- it needs to free or deallocate that memory when no longer in use, so that the memory can be used for other purposes
--
Each std::string_view
only provides a view into some existing text in
memory
- it does not own that memory, and is not responsible for managing it
--
Consequently, std::string_view
instances are very cheap to create and destroy
- no memory management!
- no copying!
--
But we need to be very careful with the lifetime of that memory location...
When extracting a substring using the original std::string
, we got a new
std::string
- this new
std::string
contained a fully self-contained copy of the corresponding text - its lifetime was not tied to the lifetime of the original
std::string
--
But when we extract a substring using the new std::string_view
, we
get a non-owning view
- the text in that view is completely dependent on the original
std::string
- its lifetime is intrisincally tied to that of the original
std::string
--
There is nothing to prevent the use of a std::string_view
that is no longer
valid!
- no mechanism exists in C++ to prevent this
- if the original
std::string
no longer exists, or points to new location, we will get undefined behaviour- the code may crash, we could get nonsense back, etc.
-- It is up to the developer (you!) to ensure the lifetime of the text exceeds the lifetime of the view
- this is a very common source of bugs and security vulnerabilities
std::string text ("this is the text that our string contains");
std::cout << text << "\n";
auto view = std::string_view (text).substr (22, 10);
std::cout << view << "\n";
text.clear(); // ⇐ original string is no longer valid!
std::cout << view << "\n";
--
If you try this, you may find that the code works just fine
- this is because the memory where the text was stored hasn't been modified
- that memory location has been marked as unused and is available for other uses
- ... but we haven't needed to store anything else there – yet!
--
This is why the result of the last line of code is undefined behaviour
- it may work as expected sometimes, or it may crash other times
- it may depend on what compiler is used, or what settings were used for the compiler, etc.
std::string text ("this is the text that our string contains");
std::cout << text << "\n";
auto view = std::string_view (text).substr (22, 10);
std::cout << view << "\n";
text = "a longer piece of text that won't fit in the original memory location");
// ⇑ forces re-allocation: original string is no longer valid!
std::cout << view << "\n";
--
Assigning a different string to our variable can also force a re-allocation
- particularly if the new string is longer than the original
- there is no guarantee that the new string will be allocated at the same location!
std::string_view get_substr ()
{
std::string original = read_from_file();
return std::string_view (original);
}
// ⇓ original string has gone out of scope and no longer exists!
std::cout << get_substr() << "\n";
--
Returning a non-owning view or reference to a function scope, local variable is a common mistake!
- this applies to any non-owning reference or similar construct
// a vector of 16 integers:
std::vector<int> v = (16, 0);
// we take an iterator to the first element:
auto iter = v.begin();
// we take a reference to the second element:
int& ref (v[1]);
// we append another element to the vector:
v.push_back (1);
// ⇑ this may cause a re-allocation of the contents of the vector
// to a different memory location!
This issue is much broader than just std::string_view
- it affects any constructs that refers to data owned or managed by some other construct
- it can affect references and STL iterators too!
These are examples of a more general class of problems, generally referred to as dangling references
- you will also come across 'dangling pointers', which is essentially the same problem
This applies to references, iterators, pointers and any non-owning construct
--
It is not always obvious when such references become invalidated
- the original owner has been destroyed or gone out of scope
- the original owner has been cleared of its contents
- the original owner has moved its contents to a different location
--
For these reasons, use non-owning constructs with caution!
- in general, they are safe as function arguments
- the owning object lives in a broader scope with a longer lifetime, and will not change while the function runs
- they are not safe as return values
- especially when referring to function scope variables!
- they are not safe when there is a possibility of modification of the original container
`#include <string_view>`
...
int compute_overlap (const std::string& sequence, const std::string& fragment)
{
...
for (int overlap = fragment.size(); overlap > 0; --overlap) {
const auto seq_start = `std::string_view (sequence)`.substr(0, overlap);
const auto frag_end = `std::string_view (fragment)`.substr(fragment.size()-overlap);
if (seq_start == frag_end) {
largest_overlap = overlap;
break;
}
}
...
--
In our case, we can see that:
- the lifetime of both
std::string_view
s is shorter than the originalsequence
&fragments
- there is no scope for our code to modify either
sequence
orfragment
while our views are live- indeed, we enforced this by providing these to
compute_overlap()
asconst
references!
- indeed, we enforced this by providing these to
We can measure the impact of this simple change using the time
command:
Before the change:
$ time ./shotgun ../data/fragments-1.txt out
initial sequence has size 1000
final sequence has length 20108
`real 0m0.359s`
user 0m0.359s
sys 0m0.000s
We can measure the impact of this simple change using the time
command:
After the change:
$ time ./shotgun ../data/fragments-1.txt out
initial sequence has size 1000
final sequence has length 20108
`real 0m0.041s`
user 0m0.037s
sys 0m0.004s
--
This is almost 10× faster!
name: classes
class: section
Classes can be thought of an extension of structures
- indeed, in C++,
struct
are also classes!
--
Classes are user-defined data types that can be used to group data, but also:
- allow the class to provide member functions to interact with the data
- provide access specifiers to limit access to some or all member variables
--
Classes are central to Object-Oriented Programming
- a class is essentially a blueprint for objects of that type
- a class is used to represent a broadly independent aspect of our program
- an instance of a class is also referred to as an object
--
We have already used a number of standard classes:
std::string
,std::vector
,std::string_view
, ...
Methods or member functions are functions that are accessed via an existing instance of a class using the [dot operator]((https://www.geeksforgeeks.org/dot-operator-in-cpp/)
--
You have already been using methods throughout the course so far:
s.size();
v.size();
v.push_back();
v.insert();
- ...
--
These are a feature of C++ classes
- you can define your own methods for your own classes
Member variables or functions can be declared as public or private
--
When public, the corresponding variable or function can be used from outside the class
When private, the variable or function can only be used within another member function of the same class
--
The ability to protect members in this way supports encapsulation and abstraction
- private data can only be modified using public methods:
- the author of the class can then ensure the consistency of the internal state of their class (encapsulation)
- users of the class only need to understand the abstract interface provided via the public methods (abstraction)
--
Encapsulation and abstraction are fundamental features of Object-Oriented Programming
- we will cover them in more detail later in the course, when they will make more sense
In C++, there is actually very little practical difference between struct
and class
- there is however a big conceptual difference!
--
As far as the compiler is concerned, struct
& class
are essentially the
same thing
- the only actual difference between the two is that unless otherwise
specified:
- members of a
struct
are public by default - members of a
class
are private by default
- members of a
--
Nonetheless, you are encouraged to reserve the use of struct
for small, lightweight containers with public data members only
- for example, as a way of grouping variables into a single entity that can be returned from a function
- by using a
struct
, you are expressing the notion that thestruct
itself is unimportant: it's the data members that matters - do not use a
struct
for anything that should provide an abstract interface, and/or where maintaining consistency between member variables is important
--
⇒ In general, prefer to define a class
We already use plenty of classes in our project:
std::string
std::vector
std::vector<std:string>
(aliased toFragments
)
These already provide all the functionality we need for our program to function
--
But what if we need to use our DNA shotgun sequencing algorithm as part of a broader project?
- it would be better to encapsulate the algorithm into a distinct, discrete module of some form
--
⇒ let's use a class to represent our algorithm!
layout: true
Let's set up a class called ShotgunSequencer
to encapsulate our algorithm:
class ShotgunSequencer {
};
`class` ShotgunSequencer {
};
- a class is declared using the keyword
class
- this is similar to declaring a
struct
- this is similar to declaring a
class `ShotgunSequencer` {
};
- a class is declared using the keyword
class
- this is similar to declaring a
struct
- this is similar to declaring a
- we provide a suitable name for the class
- this is the name of the type – not an instance
- there are many conventions for naming – on this course, we recommend using PascalCase
- it is important to chose a name that clearly expresses what kind of object this class represents
class ShotgunSequencer `{`
`};`
- a class is declared using the keyword
class
- this is similar to declaring a
struct
- this is similar to declaring a
- we provide a suitable name for the class
- this is the name of the type – not an instance
- there are many conventions for naming – on this course, we recommend using PascalCase
- it is important to chose a name that clearly expresses what kind of object this class represents
- the contents of the class are then declared between braces
- don't forget the final semicolon!
layout: true
Now let's add some data members to our class:
class ShotgunSequencer {
private:
const int m_minimum_overlap = 10;
std::string m_sequence;
Fragments m_fragments;
};
class ShotgunSequencer {
`private:`
const int m_minimum_overlap = 10;
std::string m_sequence;
Fragments m_fragments;
};
- we are going to declare our member variables as private
- this is done using the
private
keyword, followed by a colon (:
) - all subsequent declarations will be private
- this is done using the
class ShotgunSequencer {
private:
* const int m_minimum_overlap = 10;
* std::string m_sequence;
* Fragments m_fragments;
};
- we are going to declare our member variables as private
- this is done using the
private
keyword, followed by a colon (:
) - all subsequent declarations will be private
- this is done using the
- we can now declare our member variables, in exactly the same way as we did
with
struct
- there are many naming conventions – for member variables, we
recommend
snake_case
with them_
prefix
- there are many naming conventions – for member variables, we
recommend
class ShotgunSequencer {
private:
* const int m_minimum_overlap = 10;
std::string m_sequence;
Fragments m_fragments;
};
- we are going to declare our member variables as private
- this is done using the
private
keyword, followed by a colon (:
) - all subsequent declarations will be private
- this is done using the
- we can now declare our member variables, in exactly the same way as we did
with
struct
- there are many naming conventions – for member variables, we
recommend
snake_case
with them_
prefix
- there are many naming conventions – for member variables, we
recommend
- note that member variables can be default-initialised as shown
- we need to initialise
m_minimum_overlap
since we have declared itconst
– we won't be able to modify it later! - note: this type of in-class member initialisation was introduced in C++11
- we need to initialise
layout: true
We now need to add methods to allow users of our class to interact with it:
class ShotgunSequencer {
public:
void init (const Fragments& fragments);
bool iterate ();
void check_remaining_fragments ();
private:
const int m_minimum_overlap = 10;
std::string m_sequence;
Fragments m_fragments;
};
class ShotgunSequencer {
`public`:
void init (const Fragments& fragments);
bool iterate ();
void check_remaining_fragments ();
private:
const int m_minimum_overlap = 10;
std::string m_sequence;
Fragments m_fragments;
};
- this time, our methods will need to be public
- this is done using the
public
keyword, in much the same way as withprivate
- this is done using the
class ShotgunSequencer {
public:
* void init (const Fragments& fragments);
* bool iterate ();
* void check_remaining_fragments ();
private:
const int m_minimum_overlap = 10;
std::string m_sequence;
Fragments m_fragments;
};
- this time, our methods will need to be public
- this is done using the
public
keyword, in much the same way as withprivate
- this is done using the
- we can now add our method declarations
- the names of these methods should mirror the actions performed in the algorithm
- these look like regular function declarations –
but are declared within the scope of our
ShotgunSequencer
class
class ShotgunSequencer {
public:
* void init (const Fragments& fragments);
bool iterate ();
void check_remaining_fragments ();
private:
const int m_minimum_overlap = 10;
std::string m_sequence;
Fragments m_fragments;
};
.init()
is used to provide the list of fragments to initialise the algorithm- it does not need to return anything (return type is therefore
void
)
- it does not need to return anything (return type is therefore
class ShotgunSequencer {
public:
void init (const Fragments& fragments);
* bool iterate ();
void check_remaining_fragments ();
private:
const int m_minimum_overlap = 10;
std::string m_sequence;
Fragments m_fragments;
};
.iterate()
performs a single iteration of the algorithm- it will identify the fragment with the largest overlap, and if found, merge it
- we return a
bool
to indicate the status of the iteration:
⇒ iffalse
, no fragment was found, and the algorithm should stop
class ShotgunSequencer {
public:
void init (const Fragments& fragments);
bool iterate ();
* void check_remaining_fragments ();
private:
const int m_minimum_overlap = 10;
std::string m_sequence;
Fragments m_fragments;
};
.check_remaining_fragments()
performs the final check- the remaining fragments should all already be contained within the estimated sequence
- we could have decided to return
bool
to indicate the status of the check – this is a design decision! - ... but if any fragments remain, we consider this to be unexpected, but not fatal ⇒ we issue a warning
- there is therefore to need for a return value – the return type is
also
void
class ShotgunSequencer {
public:
void init (const Fragments& fragments);
* bool iterate ();
* void check_remaining_fragments ();
private:
const int m_minimum_overlap = 10;
std::string m_sequence;
Fragments m_fragments;
};
- note that we don't need to provide any arguments to these methods
- this is because the class members will all be available within the scope of these methods
- they will already have full access to the private
m_minimum_overlap
,m_sequence
andm_fragments
variables!
layout: true
How do we use our class elsewhere in our code? In shotgun.cpp
:
*#include "shotgun_sequencer.h"
...
auto fragments = load_fragments (args[1]);
* ShotgunSequencer solver;
* solver.init (fragments);
* while (solver.iterate());
* solver.check_remaining_fragments();
std::cerr << "final sequence has length " << solver.sequence().size() << "\n";
write_sequence (args[2], solver.sequence());
}
*#include "shotgun_sequencer.h"
...
auto fragments = load_fragments (args[1]);
ShotgunSequencer solver;
solver.init (fragments);
while (solver.iterate());
solver.check_remaining_fragments();
std::cerr << "final sequence has length " << solver.sequence().size() << "\n";
write_sequence (args[2], solver.sequence());
}
- we need to
#include
our new header to ensure the declarations are accessible in this file
#include "shotgun_sequencer.h"
...
auto fragments = load_fragments (args[1]);
* ShotgunSequencer solver;
solver.init (fragments);
while (solver.iterate());
solver.check_remaining_fragments();
std::cerr << "final sequence has length " << solver.sequence().size() << "\n";
write_sequence (args[2], solver.sequence());
}
- we need to
#include
our new header to ensure the declarations are accessible in this file - at the apppropriate point, we can create an instance of our new
ShotgunSequencer
class
#include "shotgun_sequencer.h"
...
auto fragments = load_fragments (args[1]);
ShotgunSequencer solver;
* solver.init (fragments);
while (solver.iterate());
solver.check_remaining_fragments();
std::cerr << "final sequence has length " << solver.sequence().size() << "\n";
write_sequence (args[2], solver.sequence());
}
- we need to
#include
our new header to ensure the declarations are accessible in this file - at the apppropriate point, we can create an instance of our new
ShotgunSequencer
class - we use the
.init()
method to supply the list of fragments and initialise the algorithm
#include "shotgun_sequencer.h"
...
auto fragments = load_fragments (args[1]);
ShotgunSequencer solver;
solver.init (fragments);
* while (solver.iterate());
solver.check_remaining_fragments();
std::cerr << "final sequence has length " << solver.sequence().size() << "\n";
write_sequence (args[2], solver.sequence());
}
- we can now iterate through the algorithm
- the simplest approach is to use a
while
loop here: we keep going whileiterate()
returnstrue
- as everything is done within the
.iterate()
method, we can leave the loop empty
- the simplest approach is to use a
#include "shotgun_sequencer.h"
...
auto fragments = load_fragments (args[1]);
ShotgunSequencer solver;
solver.init (fragments);
while (solver.iterate());
* solver.check_remaining_fragments();
std::cerr << "final sequence has length " << solver.sequence().size() << "\n";
write_sequence (args[2], solver.sequence());
}
Finally, we can perform the final check to ensure all the remaining fragments are indeed already contained in the final sequence
layout: true
We have declared our methods, but we have not defined them!
Let's create a shotgun_sequencer.cpp
file to match the corresponding
header:
#include <iostream>
#include <algorithm>
#include <format>
#include "fragments.h"
#include "overlap.h"
#include "shotgun_sequencer.h"
#include "debug.h"
void ShotgunSequencer::init (const Fragments& fragments)
{ ... }
bool ShotgunSequencer::iterate ()
{ ... }
void ShotgunSequencer::check_remaining_fragments ()
{ ... }
#include <iostream>
#include <algorithm>
#include <format>
#include "fragments.h"
#include "overlap.h"
*#include "shotgun_sequencer.h"
#include "debug.h"
void ShotgunSequencer::init (const Fragments& fragments)
{ ... }
bool ShotgunSequencer::iterate ()
{ ... }
void ShotgunSequencer::check_remaining_fragments ()
{ ... }
.explain-bottom[
As before, we need to #include
all the necessary headers that declare the
functionality we are going to use – including our new header!
]
#include <iostream>
#include <algorithm>
#include <format>
#include "fragments.h"
#include "overlap.h"
#include "shotgun_sequencer.h"
#include "debug.h"
*void ShotgunSequencer::init (const Fragments& fragments)
{ ... }
*bool ShotgunSequencer::iterate ()
{ ... }
*void ShotgunSequencer::check_remaining_fragments ()
{ ... }
.explain-top[ We can now provide the definitions for our methods. As before, we need to start each definition by replicating the declaration, so that the compiler can match it with the original declaration in the header
But there are some clear differences! ]
#include <iostream>
#include <algorithm>
#include <format>
#include "fragments.h"
#include "overlap.h"
#include "shotgun_sequencer.h"
#include "debug.h"
void `ShotgunSequencer::`init (const Fragments& fragments)
{ ... }
bool `ShotgunSequencer::`iterate ()
{ ... }
void `ShotgunSequencer::`check_remaining_fragments ()
{ ... }
.explain-top[ The name of each method is now prefixed with the class name and the scope resolution operator
This is because these definitions are now *outside* the scope of the class declaration (outside the braces within which we declared our member variables and functions).
This is how we can refer to member functions of a class. This essentially means: the `init()` method that was declared within the scope of the `ShotgunSequencer` class ]
#include <iostream>
#include <algorithm>
#include <format>
#include "fragments.h"
#include "overlap.h"
#include "shotgun_sequencer.h"
#include "debug.h"
*void init (const Fragments& fragments)
{ ... }
*bool iterate ()
{ ... }
*void check_remaining_fragments ()
{ ... }
.explain-top[
If we tried to define our methods without this scope resolution, the compiler
would (rightly) assume that we are defining completely different, global functions,
that are entirely independent of our ShotgunSequencer
class!
For example, we would end up with:
- an unexpected
iterate()
function- potentially with compiler errors as we try to access member variables
- no definition for our
ShotgunSequencer::iterate()
method- leading to linker errors at a later stage in the build process (unresolved symbol) ]
#include <iostream>
#include <algorithm>
#include <format>
#include "fragments.h"
#include "overlap.h"
#include "shotgun_sequencer.h"
#include "debug.h"
void ShotgunSequencer::init (const Fragments& fragments)
{ `...` }
bool ShotgunSequencer::iterate ()
{ `...` }
void ShotgunSequencer::check_remaining_fragments ()
{ `...` }
.explain-top[ Let's now focus on what will go in the body of our functions ]
layout: true
void ShotgunSequencer::init (const Fragments& fragments)
{
m_fragments = fragments;
if (debug::verbose)
fragment_statistics (m_fragments);
m_sequence = extract_longest_fragment (m_fragments);
}
void ShotgunSequencer::init (const Fragments& fragments)
{
`m_fragments` = fragments;
if (debug::verbose)
fragment_statistics (`m_fragments`);
`m_sequence` = extract_longest_fragment (`m_fragments`);
}
Note that we can access the members of our class directly within the body of our method
- technically, these are the members of the current instance of our class
- each instance will have its own independent version of these variables
void ShotgunSequencer::init (const Fragments& fragments)
{
* m_fragments = fragments;
if (debug::verbose)
fragment_statistics (m_fragments);
m_sequence = extract_longest_fragment (m_fragments);
}
Note that we can access the members of our class directly within the body of our method
- technically, these are the members of the current instance of our class
- each instance will have its own independent version of these variables
We start by copying the list of fragments over from the argument provided
(fragments
) into the corresponding member variable (m_fragments
)
- note how using a clear naming strategy for class members helps to avoid confusion!
void ShotgunSequencer::init (const Fragments& fragments)
{
m_fragments = fragments;
* if (debug::verbose)
* fragment_statistics (m_fragments);
* m_sequence = extract_longest_fragment (m_fragments);
}
Note that we can access the members of our class directly within the body of our method
- technically, these are the members of the current instance of our class
- each instance will have its own independent version of these variables
We start by copying the list of fragments over from the argument provided
(fragments
) into the corresponding member variable (m_fragments
)
- note how using a clear naming strategy for class members helps to avoid confusion!
The rest of the function mirrors what was done in shotgun.cpp
previously
bool ShotgunSequencer::iterate ()
{
debug::log ("---------------------------------------------------");
debug::log (std::format ("{} fragments left", m_fragments.size()));
auto [ overlap, index ] = find_biggest_overlap (m_sequence, m_fragments);
if (index < 0)
return false;
if (std::abs (overlap) < m_minimum_overlap)
return false;
debug::log (
std::format ("fragment with biggest overlap is at index {}, overlap = {}",
index, overlap));
merge (m_sequence, m_fragments[index], overlap);
m_fragments.erase (m_fragments.begin() + index);
return true;
}
bool ShotgunSequencer::iterate ()
{
debug::log ("---------------------------------------------------");
debug::log (std::format ("{} fragments left", m_fragments.size()));
auto [ overlap, index ] = find_biggest_overlap (m_sequence, m_fragments);
if (index < 0)
return false;
if (std::abs (overlap) < m_minimum_overlap)
return false;
debug::log (
std::format ("fragment with biggest overlap is at index {}, overlap = {}",
index, overlap));
merge (m_sequence, m_fragments[index], overlap);
m_fragments.erase (m_fragments.begin() + index);
return true;
}
.explain-bottom[
This mirrors almost exactly what was previously performed in shotgun.cpp
– this time using the member variables (m_minimum_overlap
,
m_fragments
, m_sequence
)
]
bool ShotgunSequencer::iterate ()
{
debug::log ("---------------------------------------------------");
debug::log (std::format ("{} fragments left", m_fragments.size()));
auto [ overlap, index ] = find_biggest_overlap (m_sequence, m_fragments);
if (index < 0)
`return false`;
if (std::abs (overlap) < m_minimum_overlap)
`return false`;
debug::log (
std::format ("fragment with biggest overlap is at index {}, overlap = {}",
index, overlap));
merge (m_sequence, m_fragments[index], overlap);
m_fragments.erase (m_fragments.begin() + index);
`return true`;
}
.explain-top[
The main difference is that we now return
to indicate success or failure.
]
void ShotgunSequencer::check_remaining_fragments ()
{
debug::log (std::format (
"{} fragments remaining unmatched"
m_fragments.size()));
int num_unmatched = 0;
for (auto& frag : m_fragments) {
if (m_sequence.find (frag) == std::string::npos)
++num_unmatched;
}
if (num_unmatched)
std::cerr << "WARNING: " << num_unmatched << " fragments remain unmatched!\n";
else
debug::log ("all remaining fragments matched OK");
}
Likewise, the code in ShotgunSequencer::check_remaining_fragments()
works exactly as it
did previously in shotgun.cpp
layout: false name: getset
There is one final piece required for us to be able to use our
ShotgunSequencer
class:
- a way to retrieve the resulting sequence
--
For this, we can use a getter method
class ShotgunSequencer {
public:
...
* const std::string& sequence () const { return m_sequence; }
private:
...
};
--
Let's unpack what is going on here...
class ShotgunSequencer {
public:
...
`const std::string& sequence () const` { return m_sequence; }
private:
...
};
This is the declaration of our method
class ShotgunSequencer {
public:
...
const std::string& `sequence` () const { return m_sequence; }
private:
...
};
We have given our getter method a simple name: sequence()
- note that many style guides would recommend a name such as
get_sequence()
orgetSequence()
- use whichever coding standards are in use on whichever project you may be contributing to!
class ShotgunSequencer {
public:
...
`const std::string&` sequence () const { return m_sequence; }
private:
...
};
Our getter returns a const reference to our member variable
- this is a common construct: returning a full-blown copy could rapidly become prohibitive
- returning a
const
reference guarantees our private variable remains read-only- it cannot be modified from outside the code
class ShotgunSequencer {
public:
...
const std::string& sequence `()` const { return m_sequence; }
private:
...
};
Note that our getter method does not take any arguments
- we can simply invoke it as
solver.sequence()
- this is often the case with getters: they only need to return the corresponding value
name: const_method
class ShotgunSequencer {
public:
...
const std::string& sequence () `const` { return m_sequence; }
private:
...
};
The const
keyword has a special meaning when placed at the end of our method
declaration, after the argument list:
- it states that this method cannot modify any of the class members
- calling this method is therefore guaranteed to leave the class itself completely unmodified
- the compiler is responsible for enforcing this
class ShotgunSequencer {
public:
...
const std::string& sequence () const { `return m_sequence;` }
private:
...
};
In this case, we have decided to insert the method definition right in the class declaration
- this differs from our previous methods, which were defined separately in the
corresponding
.cpp
file
--
This declares the member function implicitly as inline
- remember:
inline
means the definition is allowed to appear across multiple translation units - this makes sense for small functions such as getters & setters
- it provides opportunities for the compiler to optimise away the function call
- it can simply substitute the body of the function where it might otherwise have called the function
- there is now no need to supply the corresponding function definition in a
separate
.cpp
file
class ShotgunSequencer {
public:
* void init (const Fragments& fragments);
...
const std::string& sequence () const { return m_sequence; }
private:
...
};
Setters perform the opposite action from getters
- they allow users to set class parameters
- they typically do not need to return anything, so usually have a
void
return type - since they modify class members, they cannot be declared
const
--
Our init()
method is in many ways a setter method:
- it sets the (initial) list of fragments
- we could have called this method
set_fragments()
or similar- here, we have chosen to call it
init()
since setting the fragment list implicitly (re-)initialises the algorithm
- here, we have chosen to call it
Getters and setters are an important tool to implement encapsulation
-
the getter can ensure the member variable cannot be modified directly from outside the class
-
the setter can perform any additional actions that may be required when modifying member variables
- for example, our
init()
method (if viewed as a setter method) needs to re-initialise the whole algorithm, including resetting the current estimate of the sequence to the longest fragment in the list - simply setting the list of fragments without reinitialising the algorithm would leave the class in an inconsistent state – breaking encapsulation
- for example, our
class: section name: exercises
Have a go at implementing the changes necessary to create the
ShotgunSequencer
class.
Move the functionality previously in shotgun.cpp
(currently within the run()
function) into dedicated methods.