data:image/s3,"s3://crabby-images/829bb/829bb8615cede921ab9aeaaeeb44793b3e5ccde9" alt=""
Object Oriented Programming
data:image/s3,"s3://crabby-images/cf538/cf538a25b6c82331b175322c92f9e5cfb5ee661a" alt="image-20230314095605396"
Fundamentals
.h vs .cpp
- .h defines public interface API
- .cpp handle all the dirty details
data:image/s3,"s3://crabby-images/00e13/00e13df3eb8824c181d8544f402b5c787be04325" alt="image-20230314100210163"
why so may extensions?
header file: .h, .hh, .hpp
source file: .cc, .cpp, .cxx, .c++, .C
Depends on the compiler
data:image/s3,"s3://crabby-images/1269d/1269ddc9227edea638a625f6cd28a743c9580c22" alt="image-20230314100748047"
Const
data:image/s3,"s3://crabby-images/c81ce/c81ce9f380a17484173d346b903b9965a4a9fe44" alt="image-20230323093656327"
Why use CONST?
There is no excuse for ignoring the safety mechanisms provided with a product, and there is particularly no excuse for programmers too lazy to write const-correct code.
same as Why don’t we use global variables?
- “Non-const variables can be read or modified by any part of the function“ <**NOT SAFE!!!**>
data:image/s3,"s3://crabby-images/769a4/769a40377793ea6228e06d087b68641665b98c36" alt="image-20230314101927243"
- if we don’t use const, we can’t spot the bug easily in the above code
data:image/s3,"s3://crabby-images/b9ae0/b9ae098bf58f63f52ff4075778ad9293a02ef345" alt="image-20230314102259807"
- if we use const instead of the one above:
data:image/s3,"s3://crabby-images/aeaaf/aeaaf5b6224a655cdb10115433cd45e9caa108b0" alt="image-20230314102420495"
- we could get errors from the compiler(compiler time error):
data:image/s3,"s3://crabby-images/60842/608426ce51ded5c7c7da86a7484a6c9b3fb68cac" alt="image-20230314102522080"
- Besides, const allows us to reason about whether a variable will be changed
data:image/s3,"s3://crabby-images/26fcc/26fccae6c281f369927ae4bb76733981635d6bad" alt="image-20230314102658693"
const and Classes
data:image/s3,"s3://crabby-images/569ee/569eec91112457d761d48cc4dd5bb22bde7a2c65" alt="image-20230314103712508"
- can’t call non-const member function on const object:
data:image/s3,"s3://crabby-images/18cc6/18cc66841e5789f9010c1e545300a83f5fd33948" alt=""
Const Iterators
const vector
::iterator itr acts like int* const itr to make an iterator read-only, define a new const_iterator
data:image/s3,"s3://crabby-images/ed48f/ed48fa44b1975a9bf0356351024b35d02e057016" alt="image-20230314104743484"
data:image/s3,"s3://crabby-images/17fdb/17fdb8abf53083939fdf1c79a71b0c10887e5863" alt="image-20230314105022855"
Operators
Operator Overloading
data:image/s3,"s3://crabby-images/15d0a/15d0a17a79710cd22a27ce600616d789ecdc0bf4" alt="image-20230316092301560"
When dealing with user-defined class:
data:image/s3,"s3://crabby-images/57644/57644943cfcf460c613d000ee5822c41de0da57d" alt="image-20230316092614143"
data:image/s3,"s3://crabby-images/c06f7/c06f74a0daf0577d728d580dfc8c018ac6b12b2c" alt="image-20230316092944550"
1 | // member |
1 | // const member |
So, implement as Member of Non_Member?
- [], (), ->, = must be implemented as member
- <<, implemented as non-member
- +,< if binary op treats both operands equally, implement as non-member
- += if binary op doesn’t treat both operands equally( change lhs)
1 | // non-member function |
POLA(Principle of Least Astonishment)
design operators primarily to mimic conventional usage
if operator is ambiguous , don’t overload that operator
use non-member functions for symmetric operators
data:image/s3,"s3://crabby-images/bf406/bf406f0263986df08be26c6eab767019d0ee21c1" alt="image-20230316102026110"
The second one doesn’t work if overloaded as member function.
- always provide all out of a set of related operators
data:image/s3,"s3://crabby-images/9bb20/9bb2082b9ce33aa95ba845b30f0944aa0f072938" alt="image-20230316102230667"
don’t astonish others: if you implement < , you should also implement > etc.
- always think about const vs non-const for member functions:
data:image/s3,"s3://crabby-images/d3fef/d3fef8942bd86430fa152d3dae972c421922e58b" alt="image-20230316104219230"
Functors (basically lambdas)
data:image/s3,"s3://crabby-images/098bf/098bfda6e846e15f0d0ba60351dd46af92e319fe" alt="image-20230316102703125"
Advanced Multi-threading Support (C++20)
1 | awaiter operator co_await() const noexcept { |
Special Member Functions
1 | StringVector vec7 = vec4;// copy constructor |
when we don’t use default copy constructor?
data:image/s3,"s3://crabby-images/4c409/4c409837b87a50488c113f6e50a6b1fd5322272e" alt="image-20230316112057876"
when we have ownership issues:
- ownership issues: a member is a handle on a resource outside of the class
- pointers
- mutexes
- filestreams
Rule of Three
- if you explicitly define a copy constructor, copy assignment of destructor, you should define all three
when we need to declare a destructor, we have to take care of the ownership issues, the issues happen to other two.
Move Semantics
emplace_back?
- based on the philosophy: if you don’t need any copy, don’t do it
data:image/s3,"s3://crabby-images/ef681/ef6815aa7f0a576b7e224287743b4bb88549f573" alt=""
- for larger objects
All things have its beginning
What happens when running the following codes?
1 | StrVector readNames(size_t size){ |
- you may find it surprising, lots of copy constructor and destructor!
data:image/s3,"s3://crabby-images/2ce69/2ce69e51b1cad5ca6ff84cf6696fb976368e75a2" alt="image-20230319143511666"
so any optimization?
copy elision, c++17, compiler make the copy only once in the caller function(main)
data:image/s3,"s3://crabby-images/7af8f/7af8f3a51d773d3e025fee6dacd0833fe639cdda" alt="image-20230319144541047"
still have any further optimization ?
YES!! The cool thing move can save us more time.
in the following code:
1 | name2 = readNames(54321234); |
- readNames is a temporary value, and will become homeless and will be destroyed, why we need to copy it? There is no need!
- We could move(steal) the resources instead!
lvalue vs. rvalue
- lvalue: has a name, can find address
- rvalue: no name, temporary
1 | v1[1] = 4*i; // v1[1] is a reference to named value "lvalue" |
lvalue reference vs. rvalue reference
1 | auto&& v4 = v1+v2; // save the rvalue, or expands its life time! |
- any changes to v4 will change (v1+v2) temporary object.
1 | auto& ptr3 = &val; // Wrong! can't bind lvalue reference to rvalue |
- but we could bind const lvalue reference to rvalue
1 | const auto& ptr3 = ptr + 5; // correct! const save us a lot! |
why it works?
because you are not changing it anyway, so it is fine.
move constructor and assignment
why rvalue are key to move semantics?
- lvalue is NOT disposable, so you can copy from, but definitely cannot move from.
- rvalue is disposable, so you can move from
1 | // StringVector(StringVector&& other) noexcept; |
- rhs itself is actually lvalue, so we need to std::move
swap
The first very primitive attempt:
1 | template<typename T> |
- Bad! make three copies, sadly.
with move, second attempt with the knowledge of rvalue reference:
1 | template<typename T> |
- Cool! We use no copy with move!
Rule of Five
- If you declare one, you should declare all of them.(plus move constructor and move assignment)
Inheritance
Namespaces
1 | namespace Lecture{ |
Motivation
interface
1 | class Drink{ |
- cannot be instantiated
Abstract Classes
cannot be instantiated
has at least one pure virtual function
Non-virtual Destructors
1 | class Base{ |
Templetes vs. Derived Classes
When to use each?
static vs. dynamic polymorphism
- Templetes: at complile time generate real codes
- Derived Classes: at run time
data:image/s3,"s3://crabby-images/45537/4553759e9bb494d6f21cf972c024365497b59961" alt="image-20230323112910581"
RAII
How many code paths are there ?
data:image/s3,"s3://crabby-images/ead4a/ead4a423496e13ffe6445f4d051acb6bdd719846" alt="image-20230325110400783"
data:image/s3,"s3://crabby-images/87a3b/87a3b920cd6589dbbd20dec8bad6d0ad2f98c291" alt="image-20230325110927161"
Can you guarantee there is no memory leek in the code?
data:image/s3,"s3://crabby-images/81fe3/81fe334a900d91df2bf826022924c98fe5fd887b" alt="image-20230325111156256"
- For normal code paths, this code is fine.
- But if there is an exception, we will never reach the
delete e
.
How to enforce exception safety?
data:image/s3,"s3://crabby-images/8897e/8897e8c4434d6b85599695df307177945e05c617" alt="image-20230325112707980"
RAII: Resource Acquisition Is Initialization
(Scope Based Memory Management) (Constructor Acquires, Destructor Releases)(Pointer to Implementation)
What is RAII?
- all resources should be acquired in the constructor
- all resources should be released in the destructor
an example: ifstream satisfies RAII
data:image/s3,"s3://crabby-images/e848a/e848a037f7846c23837bfab78184cb0d883f14d1" alt="image-20230325113657964"
- you should not call close
another example
data:image/s3,"s3://crabby-images/a070e/a070e70fe378be95246920eeef6e170788671185" alt="image-20230325114215587"
fix:
data:image/s3,"s3://crabby-images/b1a34/b1a347bbba1c9e7a1bf686bb37632ddeaf7e9600" alt="image-20230325114229502"
data:image/s3,"s3://crabby-images/c84cc/c84cc1a9ca283710542cebc25441e3cd88b0976d" alt="image-20230325114333457"
besides:
data:image/s3,"s3://crabby-images/19a3d/19a3dd0e0a8b0a9c2db8628c2dfa970e8a26fa39" alt="image-20230325114736038"
Is automatic memory management a good or bad thing?
data:image/s3,"s3://crabby-images/e93db/e93dbb64e714ef4dac483eeb58b625eb9e737f10" alt="image-20230325115402637"
Unique Pointer
- uniquely owns its resource and deletes it when the object is destroyed.
- can’t be copied.
data:image/s3,"s3://crabby-images/536af/536afec314cefb53c09b3659610954406d9ddf7b" alt="image-20230325115455131"
data:image/s3,"s3://crabby-images/3f5bc/3f5bcab4a957c519c4b93578e33d7a2ab9dca65d" alt="image-20230325115919877"
data:image/s3,"s3://crabby-images/cf60f/cf60fde2b9589075a348d1bf6b5e08e90a215e85" alt="image-20230325120030213"
Shared pointer
- Resources can be stored by any number of shared_ptrs.
- Delete when none of them point to it
data:image/s3,"s3://crabby-images/ee25a/ee25a9984227514043a02693b0c8c648ab94b254" alt="image-20230325120328899"
data:image/s3,"s3://crabby-images/ef2db/ef2dbe9d8b65b8e5e03079cec48eb6b24ba0deff" alt="image-20230329101204208"
- Post title:CS106L笔记:OOP
- Post author:sixwalter
- Create time:2023-08-05 11:14:26
- Post link:https://coelien.github.io/2023/08/05/course-learning/CS-106L/OOP/
- Copyright Notice:All articles in this blog are licensed under BY-NC-SA unless stating additionally.