0

在 C++ 中调用模板函数时,我无法理解错误的原因。有问题的函数是 rapidjson 的一部分,定义如下:

template <unsigned parseFlags, typename InputStream, typename Handler>
    ParseResult Parse(InputStream& is, Handler& handler) 

我的调用代码是:

#include <iostream>
#include <sstream>
#include <fstream>
#include "rapidjson/rapidjson.h"
#include "rapidjson/reader.h"

class IStreamWrapper {
public:
  typedef char Ch;
  IStreamWrapper(std::istream& is) : is_(is) {
  }
  Ch Peek() const { // 1
    int c = is_.peek();
    return c == std::char_traits<char>::eof() ? '\0' : (Ch)c;
  }
  Ch Take() { // 2
    int c = is_.get();
    return c == std::char_traits<char>::eof() ? '\0' : (Ch)c;
  }
  size_t Tell() const { return (size_t)is_.tellg(); } // 3
  Ch* PutBegin() { assert(false); return 0; }
  void Put(Ch) { assert(false); }
  void Flush() { assert(false); }
  size_t PutEnd(Ch*) { assert(false); return 0; }
private:
  IStreamWrapper(const IStreamWrapper&);
  IStreamWrapper& operator=(const IStreamWrapper&);
  std::istream& is_;
};

struct EmptyHandler {
  bool Null() { return true; }
  bool Bool(bool b) { return true; }
  bool Int(int i) { return true; }
  bool Uint(unsigned u) { return true; }
  bool Int64(int64_t i) { return true; }
  bool Uint64(uint64_t u) { return true; }
  bool Double(double d) { return true; }
  bool String(const char* str, rapidjson::SizeType length, bool copy) { return true; }
  bool StartObject() { return true; }
  bool Key(const char* str, rapidjson::SizeType length, bool copy) { return true; }
  bool EndObject(rapidjson::SizeType memberCount) { return true; }
  bool StartArray() { return true; }
  bool EndArray(rapidjson::SizeType elementCount) { return true; }
};

int main(int argc, char*argv[]){
  std::ifstream input("example.json");
  IStreamWrapper is(std::istream(input));
  EmptyHandler handler;
  rapidjson::Reader reader;
  reader.Parse<rapidjson::kParseDefaultFlags, IStreamWrapper, EmptyHandler>(is, handler);
  return 0;
}

但是 MS Visual Studio 2013 向我显示了错误:

error C2664: 'rapidjson::ParseResult rapidjson::GenericReader<rapidjson::UTF8<char>,rapidjson::UTF8<char>,rapidjson::CrtAllocator>::Parse<0,IStreamWrapper,EmptyHandler>(InputStream &,Handler &)' : cannot convert argument 1 from 'IStreamWrapper (__cdecl *)(std::istream)' to 'IStreamWrapper &'

据我了解,用于实例化的类型与传递给模板函数的类型完全相同。这段代码只是为了演示这个问题——在我的原始代码中,istream 不是来自文件——所以为来自 rapidjson 的文件建议一个不同的读取类并不能解决我的问题。我也想了解,为什么这里会发生这个错误。

有任何想法吗?

4

1 回答 1

2
IStreamWrapper is(std::istream(input));

相当于

IStreamWrapper is(std::istream input);

即,它声明了一个函数,该函数调用了一个名为is的类型的参数并返回一个.std::istreaminputIStreamWrapper

即使解释为变量声明(不是),std::istream(input)也没有任何意义。它将制作的子对象的istream临时副本,在过程中进行切片- 除了它不会编译,因为无法复制流。inputistreaminput

此外,由于IStreamWrapper存储了引用,因此使用将立即销毁的临时流也不是一个好主意。

于 2015-02-27T08:33:37.633 回答