从 'Type (__cdecl *)(std::istream)' 转换为 'Type &'
Conversion from 'Type (__cdecl *)(std::istream)' to 'Type &'
我无法理解在 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() ? '[=13=]' : (Ch)c;
}
Ch Take() { // 2
int c = is_.get();
return c == std::char_traits<char>::eof() ? '[=13=]' : (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 的文件进行不同的读取 class 不会解决我的问题。我也想了解为什么会出现此错误。
有什么想法吗?
IStreamWrapper is(std::istream(input));
等同于
IStreamWrapper is(std::istream input);
即,它声明了一个名为 is
的函数,该函数接受名为 input
的类型 std::istream
的参数并返回一个 IStreamWrapper
.
即使解释为变量声明(不是),std::istream(input)
也毫无意义。它将制作 input
的 istream
子对象的 istream
临时副本,并在此过程中切片 input
- 除了它不会编译,因为无法复制流。
此外,由于 IStreamWrapper
存储了一个引用,因此使用将立即被销毁的临时流也不是一个好主意。
我无法理解在 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() ? '[=13=]' : (Ch)c;
}
Ch Take() { // 2
int c = is_.get();
return c == std::char_traits<char>::eof() ? '[=13=]' : (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 的文件进行不同的读取 class 不会解决我的问题。我也想了解为什么会出现此错误。
有什么想法吗?
IStreamWrapper is(std::istream(input));
等同于
IStreamWrapper is(std::istream input);
即,它声明了一个名为 is
的函数,该函数接受名为 input
的类型 std::istream
的参数并返回一个 IStreamWrapper
.
即使解释为变量声明(不是),std::istream(input)
也毫无意义。它将制作 input
的 istream
子对象的 istream
临时副本,并在此过程中切片 input
- 除了它不会编译,因为无法复制流。
此外,由于 IStreamWrapper
存储了一个引用,因此使用将立即被销毁的临时流也不是一个好主意。