我可以限制一个 C++ 函数接受特定类型的任意嵌套向量吗?

huangapple go评论91阅读模式
英文:

Can I constrain a C++ function to accept an arbitrarily nested vector of a specific type?

问题

Sure, here's the translated code portion:

  1. 我想要能够限制一个C++函数,使其只能接受以下参数:
  2. std::vector<T>
  3. std::vector<std::vector<T>>
  4. std::vector<std::vector<std::vector<T>>>
  5. ...

并且,可以接受任意嵌套的std::vector,但类型必须是特定的类型T。

我希望以下代码可以编译通过:

  1. template<typename T>
  2. requires IsVectorInt<T>
  3. auto foo(T t)
  4. {
  5. }
  6. int main()
  7. {
  8. std::vector<int> intvec {1, 2, 3};
  9. std::vector<std::vector<int>> vecofvec {{1, 2}, {1, 3}};
  10. foo(intvec);
  11. foo(vecofvec);
  12. }

到目前为止,我已经使用概念(concepts)成功地限制了一个函数,使其只能接受以下两种参数之一:

std::vector&lt;T&gt;,其中T可能是另一个std::vector<U>

或者:

std::vector&lt;int&gt;,但这不接受任何嵌套的向量。

我的代码如下所示:

  1. template&lt;class, template&lt;class...&gt; class&gt;
  2. inline constexpr bool is_specialization = false;
  3. template&lt;template&lt;class...&gt; class T, class... Args&gt;
  4. inline constexpr bool is_specialization&lt;T&lt;Args...&gt;, T&gt; = true;
  5. template&lt;class T&gt;
  6. concept IsVector = is_specialization&lt;T, std::vector&gt;;
  7. template&lt;class T&gt;
  8. concept IsVectorInt = IsVector&lt;T&gt; &amp;&amp; std::is_same_v&lt;int, typename T::value_type&gt;;
  9. template&lt;typename T&gt;
  10. requires IsVector&lt;T&gt;
  11. auto foo(T t)
  12. {
  13. }
  14. int main()
  15. {
  16. std::vector&lt;int&gt; intvec {1, 2, 3};
  17. std::vector&lt;std::string&gt; vecofstr {&quot;bar&quot;, &quot;baz&quot;};
  18. std::vector&lt;std::vector&lt;int&gt;&gt; vecofvec {{1, 2}, {1, 3}};
  19. foo(intvec);
  20. foo(vecofvec);
  21. foo(vecofstr); // 我希望这一行无法编译通过
  22. }
  1. <details>
  2. <summary>英文:</summary>
  3. I would like to be able to constrain a C++ function such that the only arguments it can accept are:

std::vector<T>
std::vector<std::vector<T>>
std::vector<std::vector<std::vector<T>>>
...

  1. and so on, accepting an arbitrarily nested std::vector, but with a specific type T
  2. I would like the following to compile:

template<typename T>
requires IsVectorInt<T>
auto foo(T t)
{

}

int main()
{
std::vector<int> intvec {1, 2, 3};
std::vector<std::vector<int>> vecofvec {{1, 2}, {1, 3}};
foo(intvec);
foo(vecofvec);
}

  1. So far I have managed to constrain a function using concepts to only accept either:
  2. `std::vector&lt;T&gt;` where T could itself be another std::vector\&lt;U\&gt;
  3. or:
  4. `std::vector&lt;int&gt;`, this however, does not accept any nested vectors.
  5. My code is shown below:

template<class, template<class...> class>
inline constexpr bool is_specialization = false;

template<template<class...> class T, class... Args>
inline constexpr bool is_specialization<T<Args...>, T> = true;

template<class T>
concept IsVector = is_specialization<T, std::vector>;

template<class T>
concept IsVectorInt = IsVector<T> && std::is_same_v<int, typename T::value_type>;

template<typename T>
requires IsVector<T>
auto foo(T t)
{

}

int main()
{
std::vector<int> intvec {1, 2, 3};
std::vector<std::string> vecofstr {"bar", "baz"};
std::vector<std::vector<int>> vecofvec {{1, 2}, {1, 3}};
foo(intvec);
foo(vecofvec);
foo(vecofstr); // I would like this line to fail to compile

}

  1. </details>
  2. # 答案1
  3. **得分**: 4
  4. 如果您的目标是检测算术类型的向量或嵌套向量,这里提供了一个使用递归实现的特性的提案:
  5. ```c++
  6. #include <iostream>
  7. #include <type_traits>
  8. #include <vector>
  9. template <typename T>
  10. struct NestedVector : public std::false_type {};
  11. template <typename T>
  12. struct NestedVector<std::vector<T>>
  13. : public std::conditional_t<std::is_arithmetic<T>::value ||
  14. NestedVector<T>::value,
  15. std::true_type, std::false_type> {};
  16. // C++<20版本
  17. template <typename T, std::enable_if_t<NestedVector<T>::value, bool> = true>
  18. void foo(T t) {}
  19. // C++20版本
  20. // template <typename T>
  21. // requires NestedVector<T>::value
  22. // void foo(T t) {}
  23. int main() {
  24. std::vector<int> intvec{1, 2, 3};
  25. std::vector<std::string> vecofstr{"bar", "baz"};
  26. std::vector<std::vector<int>> vecofvec{{1, 2}, {1, 3}};
  27. foo(intvec);
  28. foo(vecofvec);
  29. // foo(vecofstr); // 不适用
  30. }

Live Demo

英文:

If your goal is to detect vector or nested vector of arithmetic types, here is a proposal with trait implemented recursively:

  1. #include &lt;iostream&gt;
  2. #include &lt;type_traits&gt;
  3. #include &lt;vector&gt;
  4. template &lt;typename T&gt;
  5. struct NestedVector : public std::false_type {};
  6. template &lt;typename T&gt;
  7. struct NestedVector&lt;std::vector&lt;T&gt;&gt;
  8. : public std::conditional_t&lt;std::is_arithmetic&lt;T&gt;::value ||
  9. NestedVector&lt;T&gt;::value,
  10. std::true_type, std::false_type&gt; {};
  11. // C++&lt;20 version
  12. template &lt;typename T, std::enable_if_t&lt;NestedVector&lt;T&gt;::value, bool&gt; = true&gt;
  13. void foo(T t) {}
  14. // C++20 version
  15. // template &lt;typename T&gt;
  16. // requires NestedVector&lt;T&gt;::value
  17. // void foo(T t) {}
  18. int main() {
  19. std::vector&lt;int&gt; intvec{1, 2, 3};
  20. std::vector&lt;std::string&gt; vecofstr{&quot;bar&quot;, &quot;baz&quot;};
  21. std::vector&lt;std::vector&lt;int&gt;&gt; vecofvec{{1, 2}, {1, 3}};
  22. foo(intvec);
  23. foo(vecofvec);
  24. // foo(vecofstr); // KO
  25. }

(NB: I'm using C++<20 so I removed the requires but I think that it may work juste the same)
Live Demo

答案2

得分: 2

你可以通过创建一个特性来简化并使你的示例更易读,以找到向量的最内层类型,然后像下面所示使用static_assert

  1. // 用于获取最内层类型的特性
  2. template <typename T>
  3. struct inner_type
  4. {
  5. using type = T;
  6. };
  7. // 特化
  8. template <typename T>
  9. struct inner_type<std::vector<T>>
  10. {
  11. using type = typename inner_type<T>::type;
  12. };
  13. // 为了方便使用的别名
  14. template<typename T>
  15. using inner_type_t = typename inner_type<T>::type;
  16. template<typename T>
  17. auto foo(std::vector<T> t)
  18. {
  19. static_assert(std::is_arithmetic_v<inner_type_t<T>>);
  20. }
  21. int main()
  22. {
  23. std::vector<int> intvec {1, 2, 3};
  24. std::vector<std::vector<int>> vecofvec {{1, 2}, {1, 3}};
  25. foo(intvec); // 正常工作
  26. foo(vecofvec); // 正常工作
  27. std::vector<std::string> vecofstr{"bar", "baz"};
  28. // foo(vecofstr); // 预期失败
  29. }

运行示例


使用require(或enable_if_t)而不是static_assert非常简单,如下所示:

  1. template<typename T>
  2. auto foo(std::vector<T> t) requires(std::is_arithmetic_v<inner_type_t<T>>)
  3. {
  4. }

示例

英文:

You can simplify and make your example more readable by creating a trait to find the inner most type of the vector and then using a static_assert as shown below:

  1. //trait to get the innermost type
  2. template &lt;typename T&gt;
  3. struct inner_type
  4. {
  5. using type = T;
  6. };
  7. //specialization
  8. template &lt;typename T&gt;
  9. struct inner_type&lt;std::vector&lt;T&gt;&gt;
  10. {
  11. using type = typename inner_type&lt;T&gt;::type;
  12. };
  13. //alias for convenience
  14. template&lt;typename T&gt;
  15. using inner_type_t = typename inner_type&lt;T&gt;::type;
  16. template&lt;typename T&gt;
  17. auto foo(std::vector&lt;T&gt; t)
  18. {
  19. static_assert(std::is_arithmetic_v&lt;inner_type_t&lt;T&gt;&gt;);
  20. }
  21. int main()
  22. {
  23. std::vector&lt;int&gt; intvec {1, 2, 3};
  24. std::vector&lt;std::vector&lt;int&gt;&gt; vecofvec {{1, 2}, {1, 3}};
  25. foo(intvec); //works
  26. foo(vecofvec); //works
  27. std::vector&lt;std::string&gt; vecofstr{&quot;bar&quot;, &quot;baz&quot;};
  28. //foo(vecofstr); //fails as expected
  29. }

Working demo


It is trivial to to use require(or enable_if_t) instead of static_assert as shown below:

  1. template&lt;typename T&gt;
  2. auto foo(std::vector&lt;T&gt; t) requires(std::is_arithmetic_v&lt;inner_type_t&lt;T&gt;&gt;)
  3. {
  4. }

Demo

huangapple
  • 本文由 发表于 2023年5月25日 17:25:37
  • 转载请务必保留本文链接:https://go.coder-hub.com/76330744.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定