Generic wrapper behaving in the same way as wrapped type(泛型包装的行为方式与包装类型相同)
                            本文介绍了泛型包装的行为方式与包装类型相同的处理方法,对大家解决问题具有一定的参考价值,需要的朋友们下面随着小编来一起学习吧!
                        
                        问题描述
#include <utility>
#include <vector>
template <typename Wrapped>
class Wrapper
{
public:
    template <typename... Args>
    Wrapper(Args&&... args)
    : wrapped(std::forward<Args>(args)...)
    {
    }
private:
    Wrapped wrapped;
};
Wrapper<std::vector<int>> intended()
{
    std::vector<int>::allocator_type allocator;
    return { { 1, 2, 3 }, allocator }; // doesn't compile
}
Wrapper<std::vector<int>> unintended()
{
    return 123; // calls 'explicit vector(size_type count)'
}
Wrapper几乎不可见?例如,返回std::vector<int>不允许编译这样的函数:
std::vector<int> get_vector()
{
    return 123; // doesn't compile
}
推荐答案
对于T的默认构造函数,可以通过检查const T&是否可以用{}初始化来判断它是否隐式。
对于一个类型是否可以从另一个类型隐式构造,我们可以结合std::is_constructible和std::is_convertible来检查。
因此您的Wrapper的构造函数可以根据上面的规则有条件地explicit,例如:
#include <utility>
template<typename T, typename... Args>
concept implicitly_constructible = 
  requires { [](const T&) {}({std::declval<Args>()...}); };
template<typename Wrapped>
class Wrapper {
 public:
  template<typename T>
    requires std::is_constructible_v<Wrapped, T>
  explicit(!std::is_convertible_v<T, Wrapped>)
  Wrapper(T&& t)
  : wrapped(std::forward<T>(t)) { }
  template<typename... Args>
    requires (sizeof...(Args) != 1) && 
             std::is_constructible_v<Wrapped, Args...>
  explicit(!implicitly_constructible<Wrapped, Args...>)
  Wrapper(Args&&... args)
  : wrapped(std::forward<Args>(args)...) { }
 private:
  Wrapped wrapped;
};
需要注意的是,{1, 2, 3}不能从Args&&推导出来,所以您还需要明确指定它的类型,比如std::initializer_list<int>{1, 2, 3}。
Demo.
这篇关于泛型包装的行为方式与包装类型相同的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持编程学习网!
				 沃梦达教程
				
			本文标题为:泛型包装的行为方式与包装类型相同
 
				
         
 
            
        基础教程推荐
             猜你喜欢
        
	     - C++结构和函数声明。为什么它不能编译? 2022-11-07
- 这个宏可以转换成函数吗? 2022-01-01
- 我有静态或动态 boost 库吗? 2021-01-01
- 如何通过C程序打开命令提示符Cmd 2022-12-09
- 在 C++ 中计算滚动/移动平均值 2021-01-01
- 静态库、静态链接动态库和动态链接动态库的 .lib 文件里面是什么? 2021-01-01
- 常量变量在标题中不起作用 2021-01-01
- 如何将 std::pair 的排序 std::list 转换为 std::map 2022-01-01
- 如何检查GTK+3.0中的小部件类型? 2022-11-30
- 如何在 C++ 中初始化静态常量成员? 2022-01-01
 
    	 
    	 
    	 
    	 
    	 
    	 
    	 
    	 
						 
						 
						 
						 
						 
				 
				 
				 
				