无法分配或复制 iostream 对象?
iostream 和其他流类实际上不是类,而是 typedef,对吧?
问题是,我尝试在初始化列表中初始化一个 istream 对象,但不幸的是我收到了一个错误,代码如下:
class A
{
public:
A(istream &is=cin): ais(is)
{}
private:
istream ais;
};
无法使用 g++ 编译,错误:
synthesized method ‘std::basic_istream<char, std::char_traits<char> >::basic_istream(const std::basic_istream<char, std::char_traits<char> >&)’ first required here
我搜索了,发现, iostream 无法分配或复制
。但为什么我无法在初始化列表中初始化它呢?
因为我认为,初始化列表将调用对象的构造函数/复制构造函数,对吧?
iostream
and other stream class are not actually class, but typedef
s, right?
Here is the problem, I tried to initialize a istream
object in initialization list, but unfortunately I got an error, code goes bellow:
class A
{
public:
A(istream &is=cin): ais(is)
{}
private:
istream ais;
};
Can't compile with g++, error:
synthesized method ‘std::basic_istream<char, std::char_traits<char> >::basic_istream(const std::basic_istream<char, std::char_traits<char> >&)’ first required here
I searched SO, found that, iostream cannot be assigned or copy
. But why cannot I initialize it in the initialization list?
Cuz I think, the initialization list will invoke the object's constructor/copy-constructor, right?
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(6)
您的代码尝试将一个
istream
(传递给构造函数的一个)转换为两个istream
(传递给构造函数的一个和ais
) >。istream
对象代表实际的流本身。只有一个流,无法以某种方式将其变成两个流。甚至还不清楚这意味着什么。如果流上有一些数据,那么首先读取的流是否会获取它?还是他们都明白了?如果是这样,谁或什么复制了它?
istream
就像一个文件本身。如果您不亲自将数据从一个文件复制到另一个文件,则无法将一个文件变成两个文件。但是,您可以根据需要拥有任意数量的指向同一istream
的引用或指针。解决您问题的方法可能是让ais
成为参考。Your code attempts to turn one
istream
, the one passed to the constructor, into twoistream
s, the one that was passed to the constructor andais
. Anistream
object represents that actual stream itself. There is only one stream and there is no way to somehow turn it into two streams.It's not even clear what that would mean. If there's some data on the stream, does whichever stream reads first get it? Or do they both get it? If so, who or what duplicates it?
An
istream
is like a file itself. You cannot turn one file into two files without copying the data from one to the other yourself. You can, however, have as many references or pointers to the sameistream
as you want. The solution to your problem is probably to makeais
a reference.您将成员变量声明为:
意味着将在初始值设定项列表中调用复制构造函数。这不是特定于初始化列表的:您不能在任何地方执行此操作。
您可以将成员变量更改为引用:
但这意味着您必须确保
ais
引用的istream
在A< 的生命周期内有效/代码>。
You have the member variable declared as:
meaning the copy constructor will be invoked in the initializer list. This is not specific to the initializer list: you can't do this anywhere.
You could change the member variable to be a reference:
but that means you have to ensure that the
istream
to whichais
refers is valid for the lifetime ofA
.是的,但这不相关。
因为它既不能被复制,也不能被赋值。 Ctor init list 并不神奇。要从
is
创建ais
,您需要制作副本,但不能。Yes, but that's not relevant.
Because it can be neither copied nor assigned. Ctor init list is not magical. To create
ais
fromis
you need to make a copy, and you can't.流对象表示数据流的句柄。抄袭是没有意义的。另外,仅复制流的基本部分并不是正确的做法:这将切掉有趣的部分(尽管您可以通过使用流缓冲区来获取实际有趣的部分)。
如果类包含流,则您要在类中初始化的是对该流的引用:
当然,这意味着只要您正在使用流,某些外部就需要保持该流处于活动状态。如果这不是您想要或需要的,您可以让您的类保存例如
std::ifstream
并采用std::string
作为构造函数参数:如果字符串非空则打开相应的文件。否则,您需要将流的rdbuf()
设置为std::cin
中的:The stream object represent a handle to streams of data. Copying that doesn't make sense. Also, this wouldn't be the right thing to copy only tge base part of a stream: this would be slicing off the interesting bit (although you could get hold of thr actually interesting portion by using the stream buffer).
What you want to initialize in your class if it holds a stream is a reference to the stream:
Of course, this means that some external needs to keep the stream alive as long as you are using it. If this isn't what you want or need you can have your class hold e.g. an
std::ifstream
and take astd::string
as constructor argument: if the string is non-empty you'd open the corresponding file. Otherwise you'd set your stream'srdbuf()
to the one fromstd::cin
:除了您已经得到的答案之外,您还可以使用传入的流的 streambuf 初始化
ais
(根据您的定义)(当然,同样的警告与引用一样:您必须确保 Streambuf 保持有效,直到对象被破坏)。使用底层 Streambuf 进行初始化可能是最接近复制流的方法:虽然两者仍然从同一文件读取并使用相同的缓冲(以及 Streambuf 可能执行的其他处理),但所有格式化和错误处理标志都是分开的。
要使用streambuf初始化
ais
,您可以编写:In addition to the answers you already got, you can also initialize
ais
(as of your definition) with the streambuf of the stream you passed in (of course, the same caveat as with the reference applies: You must ensure that the streambuf stays valid until the object is destructed).Initializing with the underlying streambuf is probably the closest to copying a stream you can do: While both still read from the same file and use the same buffering (and other processing the streambuf may do), all the formatting and error handling flags are separate.
To initialize
ais
with the streambuf, you write:请注意,该成员是引用类型。正如您所写的,您正在制作一个副本来初始化该成员。
Notice the member is a reference type. As you've written it, you're making a copy to initialize the member.