data:image/s3,"s3://crabby-images/829bb/829bb8615cede921ab9aeaaeeb44793b3e5ccde9" alt=""
Template Metaprogramming
Motivating example
1 |
|
how do you implement your own distance function?
1 | template<typename It> |
- the above code is O(1), which is efficient.
- but the above implementation only works for random access iterator
another optimization!
1 | template<typename It> |
- but the second way is O(N), for any kind of iterators, which is unexpected
How does STL achieve this?
- You have to extract type information from iterator of what it is.
- need to determine if it is random access iterator
- run the piece of code based on the type of iterator:
data:image/s3,"s3://crabby-images/26145/261458aa35784707ab5cd0435807dd2c9a3006bb" alt="image-20230329105106863"
Computations on Types
Comparision of values vs. type computations : A Preview
data:image/s3,"s3://crabby-images/5ea42/5ea424a73cad284a4adae9f9cbea9e4ba3160900" alt="image-20230329105536999"
data:image/s3,"s3://crabby-images/6742e/6742e2e66c574cff39d2e4baa92c73659b5cdf82" alt="image-20230329105611948"
data:image/s3,"s3://crabby-images/c5f51/c5f516fefa22ee9023204e488c0ee84b8cebaf39" alt="image-20230329105640787"
data:image/s3,"s3://crabby-images/26ed0/26ed056aea6aa089d9f18f99dff3d937a3ec740c" alt="image-20230329105723424"
data:image/s3,"s3://crabby-images/bb5b1/bb5b1d402fad3b3bad09acdc94de10502ee4dfdf" alt="image-20230329105942716"
data:image/s3,"s3://crabby-images/4c013/4c013af645aa2924bdbc8fa8dcff5018008ad6d9" alt="image-20230329110203735"
Meta-functions
template types
- use them for template classes
data:image/s3,"s3://crabby-images/4b691/4b691bbd14a3a4d31caf9b2011d69778bf7f8edc" alt="image-20230329110744559"
template values
data:image/s3,"s3://crabby-images/2d91c/2d91cc67229a2fe88ad37bd9d6574cf5a7758b56" alt="image-20230329111146933"
meta-function
abstractly:
- It is a function that operates on some types/values(“parameters”)
- and outputs some types/values(“return values”)
concretely:
- It is a struct that has public member types/fields(output) which depend on what the template types/values(input) are instantiated with.
data:image/s3,"s3://crabby-images/5fc94/5fc94537a6800dd4560afb26bd221c203ba01164" alt="image-20230329111659110"
- a simple example:
data:image/s3,"s3://crabby-images/d4e97/d4e97474c8997efbaa5c17eb57494b023e78e094" alt="image-20230329111946116"
data:image/s3,"s3://crabby-images/99601/9960182d57f6ed239fdd7be912562a2017996a58" alt="image-20230329112718402"
1 | //input type, output value |
Template Deduction
Template Rules
- Template Specialization: you can have a “generic template”, as well as “specialized” templates for particular types
Fully specialize:
data:image/s3,"s3://crabby-images/66ee7/66ee7fb9ff4e8acb1739a1fb77689d1818347508" alt="image-20230329124148296"
why specialization?: using a bit array to store booleans is much more efficient.
partially specialize:
data:image/s3,"s3://crabby-images/3cb94/3cb947417e51a59a8c5b9e78d3c50bc642dae8d9" alt="image-20230329124600043"
Some examples using Template Deduction
let’s build up a collection of predicate meta-functions
data:image/s3,"s3://crabby-images/9eaee/9eaee5d653bf600a5dd32674f82d31a98ccfcf3e" alt="image-20230329125007552"
why does the second one return true?
- the compiler tries to first match the specialized template
- the second one got matched, so return true
data:image/s3,"s3://crabby-images/7808d/7808d44b6a4c251542b6604a38254dee9bd4173b" alt="image-20230329125420606"
- kind of interesting, you can figure out the type info, because of template deduction rules
data:image/s3,"s3://crabby-images/298b6/298b6e6f91182f50e752eae88cf0752fe8e075ab" alt="image-20230329125712112"
- of course, same technique can be used to return a type
Actually it is a “hack”, because we are using the compiler’s template matching rules to implement an if/else statement for types(we can’t directly compare them)
Let finish our myDistance implement:
1 | using category = typename std::iterator_traits<It>::iterator_category; |
need to remove the offending code when the if statement knows that part won’t be run;
pre-C++17: std::enable_if
C++17: if constexpr
1 | using category = typename std::iterator_traits<It>::iterator_category; |
what is constexpr?
you can calculate this expression at compile time.(stronger form of const)
what is if constexpr?
calculate boolean at compile-time , Replace the entire if/else with the code that will be actually be run
- Post title:CS106L笔记:Template Metaprogramming
- Post author:sixwalter
- Create time:2023-08-05 11:14:26
- Post link:https://coelien.github.io/2023/08/05/course-learning/CS-106L/Template_Metaprogramming/
- Copyright Notice:All articles in this blog are licensed under BY-NC-SA unless stating additionally.