c – 为什么编译器找不到这个运算符<< overload?

c – 为什么编译器找不到这个运算符<< overload?,第1张

概述我正在尝试编写运算符的重载<<对于将存储在 boost::variant中的标准库容器的特定实例化.这是一个说明问题的小例子: #include <iostream>#include <vector>std::ostream & operator<<( std::ostream & os, const std::vector< int > & ) { os << "Streaming o 我正在尝试编写运算符的重载<<对于将存储在 boost::variant中的标准库容器的特定实例化.这是一个说明问题的小例子:

#include <iostream>#include <vector>std::ostream & operator<<( std::ostream & os,const std::vector< int > & ) {  os << "Streaming out std::vector< int >";  return os;}std::ostream & operator<<( std::ostream & os,const std::vector< double > & ) {  os << "Streaming out std::vector< double >";  return os;}#include <boost/variant.hpp>typedef boost::variant< std::vector< int >,std::vector< double > > MyVariant;int main( int argc,char * argv[] ) {  std::cout << MyVariant();  return 0;}

Clang的第一个错误是

boost/variant/detail/variant_io.hpp:64:14: error: invalID operands to binary Expression ('std::basic_ostream<char>' and 'const std::vector<int,std::allocator<int>>')        out_ << operand;        ~~~~ ^  ~~~~~~~

我意识到#include< boost / variant.hpp>在一个奇怪的地方.我很确定问题与模板中的两阶段名称查找有关,所以我移动了#include以尝试从clang documentation on lookup实现修复#1.修复#2来自该文档并不是一个好选择,因为我相信添加我的重载运算符<<到std命名空间会导致未定义的行为. 不应该在#include允许编译器找到定义之前定义我的运算符<< s?这个技术似乎适用于以下示例,改编自同一个clang页面.

#include <iostream>namespace ns {  struct Data {};}std::ostream& operator<<(std::ostream& out,const ns::Data & data) {  return out << "Some data";}namespace ns2 {  template<typename T>  voID Dump( std::ostream & out,const T & value) {    out << value;  }}int main( int argc,char * argv[] ) {  ns2::Dump( std::cout,ns::Data() );}
解决方法 在模板实例化期间,仅在阶段II查找期间找到取决于模板类型的功能模板.第二阶段查找不考虑在使用点可见的名称,而只考虑基于参数依赖查找找到的名称.由于std :: ostream和std :: vector< int>的唯一关联命名空间是namespace std它不会查找在全局命名空间中定义的输出运算符.当然,不允许将这些运算符添加到命名空间std中,这是一个真正的问题:您只能为容器定义这些运算符,至少包含一个用户定义类型!在这种限制的可能方法是添加一个自定义分配器,它只是从std :: allocator< T>中派生出来的.但它位于合适的用户定义命名空间中:然后您可以在此命名空间中定义输出运算符.这种方法的缺点是std :: vector< T> (即,没有allocator参数)几乎是词汇类型.

移动声明并没有帮助:第二阶段名称查找实际上并不依赖于声明的顺序,除了声明必须在实例化之前.唯一正确的解决方法是在阶段II查找中寻找名称空间中的运算符,这几乎意味着要打印的类型必须涉及用户定义的类型.

总结

以上是内存溢出为你收集整理的c – 为什么编译器找不到这个运算符<< overload?全部内容,希望文章能够帮你解决c – 为什么编译器找不到这个运算符<< overload?所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/langs/1215409.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-06-05
下一篇 2022-06-05

发表评论

登录后才能评论

评论列表(0条)

保存