diff --git a/README.md b/README.md index 8ffe208..2c0323e 100644 --- a/README.md +++ b/README.md @@ -163,7 +163,33 @@ for(decl:col) { - [C++中如何将string类型转换为int类型?](./basic_content/extent/string_int.md) -### 5.代码运行 +### 5.自己写的工具 + +- [容器快捷输出工具](./tool/output) + +对吴老师的代码进行了修改,[点击此处直通代码](./tool/output/output_container.h) + +输入: + +```cpp +map mp{ + {1, 1}, + {2, 4}, + {3, 9}}; + cout << mp << endl; +``` + +输出结果显示: + +```cpp +{ 1 => 1, 2 => 4, 3 => 9 } +``` + +感谢吴老师的代码,原代码: + +> https://github.com/adah1972/output_container/blob/master/output_container.h + +### 6.代码运行 - **代码环境** @@ -173,7 +199,7 @@ Ubuntu 18.04 CLion gcc/g++ -### 6.关于作者 +### 7.关于作者 个人公众号: diff --git a/tool/output/container.cpp b/tool/output/container.cpp new file mode 100644 index 0000000..bf2cd74 --- /dev/null +++ b/tool/output/container.cpp @@ -0,0 +1,43 @@ +// +// Created by light on 19-12-16. +// + +#include +#include +#include +#include +#include "output_container.h" + + +using namespace std; + +int main() { + map mp{ + {1, 1}, + {2, 4}, + {3, 9}}; + cout << mp << endl; + vector> vv{ + {1, 1}, + {2, 4}, + {3, 9}}; + cout << vv << endl; + + pair p{1, 2}; + cout << p << endl; + + set s{1, 2, 3}; + cout << s << endl; + + vector v{'a', 'b'}; + cout << v << endl; + set vs{"a", "b"}; + cout << vs << endl; + + map mm{ + {1, "23"}, + {2, "234hi"} + }; + cout << mm << endl; + +} diff --git a/tool/output/output_container.h b/tool/output/output_container.h new file mode 100644 index 0000000..952fbf3 --- /dev/null +++ b/tool/output/output_container.h @@ -0,0 +1,150 @@ +// +// Created by light on 20-1-8. +// + +#ifndef MORDEN_C_OUTPUT_CONTAINER_H +#define MORDEN_C_OUTPUT_CONTAINER_H + +#include // std::ostream +#include // std::false_type/true_type/decay_t/is_same_v +#include // std::declval/pair + +// 检测是否是pair +template +struct is_pair : std::false_type { +}; +template +struct is_pair> : std::true_type { +}; +template +inline constexpr bool is_pair_v = is_pair::value; + +// 检测输出函数是否存在 +template +struct has_output_function { + template + static auto output(U *ptr) + -> decltype(std::declval() << *ptr, + std::true_type()); + + template + static std::false_type output(...); + + static constexpr bool value = + decltype(output(nullptr))::value; +}; + +template +inline constexpr bool has_output_function_v = + has_output_function::value; + + +enum CHARS { + ORD, // 其他类型 + CHAR, // char 类型 + STRING // string 类型 +}; + +template +int ischarOrString(T &elem) { + using std::decay_t; + using std::is_same_v; + using element_type = decay_t; + constexpr bool is_char_v = is_same_v; + constexpr bool is_string_v = is_same_v || + is_same_v || + is_same_v; + if (is_char_v) + return CHAR; + else if (is_string_v) + return STRING; + else + return ORD; +} + +template +void output(T &elem, int type, std::ostream &os) { + switch (type) { + case CHAR: + os << '\'' << elem << '\''; + break; + case STRING: + os << '\"' << elem << '\"'; + break; + case ORD: + os << elem; + break; + } +} + +template +auto output_element(std::ostream &os, const T &element, + const Cont &) +-> typename std::enable_if::value, bool>::type { + int ftype = ischarOrString(element.first); + int stype = ischarOrString(element.second); + + output(element.first, ftype, os); + os << " => "; + output(element.second, stype, os); + return true; +} + +template +auto output_element(std::ostream &os, const T &element, + const Cont &) +-> typename std::enable_if::value, bool>::type { + int etype = ischarOrString(element); + output(element, etype, os); + return false; +} + +template +std::ostream &operator<<(std::ostream &os, const std::pair &pr) { + os << '(' << pr.first << ", " << pr.second << ')'; + return os; +} + +//template +//auto output_element(std::ostream& os, const T& element, +// const Cont&, const std::true_type) +//-> decltype(std::declval(), os) +//{ +// os << element.first << " => " << element.second; +// return os; +//} +// +//template +//auto output_element(std::ostream& os, const T& element, +// const Cont&, ...) +//-> decltype(os) +//{ +// os << element; +// return os; +//} + + + +// 针对没有输出函数的容器处理 +template>> +auto operator<<(std::ostream &os, const T &container) +-> decltype(container.begin(), container.end(), os) { + os << "{ "; + if (!container.empty()) { + bool on_first_element = true; + for (auto elem:container) { + if (!on_first_element) { + os << ", "; + } else { + on_first_element = false; + } + output_element(os, elem, container); + } + } + os << " }"; + return os; +} + + +#endif //MORDEN_C_OUTPUT_CONTAINER_H