2

我正在用 C++ 开发一个具有大量文件 io 操作的程序。我在公共标头中定义了一个静态 ofstream,以便在项目中的任何地方都可以访问它。代码结构如下:所有公共变量都定义在com.h中,test.h和test.cpp是针对一个叫OPClass的类,main.cpp承载主程序

COM.H:

#ifndef __CLCOM__
#define __CLCOM__
#include <sstream>
#include <fstream>
#include <iostream>

using namespace std;

static ofstream out;
static stringstream ss;

#endif

测试.H:

#ifndef __CL__
#define __CL__
#include <iostream>
#include <fstream>
#include "com.h"

using namespace std;

class OPClass
{
  public:
   void run(void);
   void show(ostream &o) const;
};
#endif

测试.CPP:

#include "com.h"
#include "test.h"

void OPClass::run(void)
{
  out << "Here is run()" << endl;
  show(out);
}

void OPClass::show(ostream &o) const
{
  o << "hello!" << endl;
}

主.CPP:

#include "com.h"
#include "test.h"

void runmain(void)
{
  OPClass op;
  out.open("output.txt", ios::out | ios::trunc);
  out << endl << "State changed!" << endl;
  op.run();
  if (out.is_open()) out.close();
}

int main(int argc, char* argv[])
{
  runmain();
  return  0;
}

可以看到,静态的 ofstream 被命名为 out,将在主程序和类中调用。我正在使用 mingw32 并且在编译或运行时没有看到任何问题。但似乎只有 runmain() 中的信息会被写入输出文件。写入类中该文件的任何其他消息永远不会出现在输出文件中。为什么会这样,我如何编写一个通用文件流,以便项目中的任何地方都可以访问它?谢谢。

4

2 回答 2

3

每个编译单元都有自己的ssout. 因此,main.cpp 与 test.cpp 看到的实例不同。

你真的不需要在这里静态。为了解决这个问题,而不是在头文件中声明变量及其分配,您只需使用extern关键字对它们进行原型设计。

#ifndef __CLCOM__
#define __CLCOM__
#include <sstream>
#include <fstream>
#include <iostream>

// Note: don't put "using" statements in headers
// use fully qualified names instead 
extern std::ofstream out;
extern std::stringstream ss;

#endif

您实际将声明放在哪里取决于您,只需确保它仅在一个地方即可。这可能是一个 com.cpp 文件,或者如果这适合您的项目,您可以将其粘贴在 main.cpp 中。

std::ofstream out;
std::stringstream ss;

不是说像这样的全局变量是个好主意,无论如何......

于 2012-04-21T04:47:12.523 回答
1

先发制人的声明:您应该接受@HostileFork 的回答。

就像附录一样,显示正在发生的事情的一种简单方法是out在您尝试使用它时打印出地址。

如果您添加以下几个语句:

void OPClass::run(void)
{
  cout << "Address of 'out' = " << &out << endl;
  out << "Here is run()" << endl;
  show(out);
}

和:

void runmain(void)
{
  cout << "Address of 'out' = " << &out << endl; 
  OPClass op;
  out.open("output.txt", ios::out | ios::trunc);
  out << endl << "State changed!" << endl;
  op.run();
  if (out.is_open()) out.close();
}

您会注意到两个打印语句用于out显示两个不同的地址。这应该告诉您,您实际上将两个outcreated 实例作为两个不同的变量。您的方法OPClass正在尝试写入完全不同的输出流。它与您static在全球范围内使用的方式有关;它的行为不像你想象的那样。在全局上下文中,声明某些static内容会将其绑定到它所在文件的本地范围。

于 2012-04-21T04:54:09.847 回答