从输入文件到输出文件的算术?

时间:2014-03-26 20:33:05

标签: c++ visual-studio-2012

这是一个两部分问题,我希望自己能够理解。我会根据需要进行编辑!我正在尝试编写一个程序,它将从输入文件中连续进行计算。文本文件如下所示:

int + 25 10
double / 5.5 8.5
...

每个实例都以int,double,float等类型开头,然后是计算类型(加,减等),然后是两个数字。我希望能够连续读取每个并将总和,产品等输出到输出文件中。例如,如果我们使用上面的第一个例子,文件中的输出将是:

int 25 10 = 35

我的代码可以进行以下计算:

void doAddition(ifstream &inFile) {
int num1, num2;
inFile >> num1 >> num2;
cout << num1 << num2 << " = "<< (num1+num2) << '\n'; }

唯一的问题是我不知道如何添加变量的类型(我尝试过使用字符串,但它似乎不起作用),例如“int”或“double”所以我得到:

25 10 = 35

而不是:

int 25 10 = 35

我可能会看到的第二个问题是,当我真正想将其添加到outfile时,我正在使用“cout”在屏幕上显示信息。这里有更多信息:

我用什么来移动到下一行:

void readToNextLine(ifstream& inFile) {
string t1, t2;
inFile >> t1 >> t2; }

我的主要代码:

ifstream inFile;
//ofstream outFile;
char ch;
int num1, num2;

inFile.open("infile.txt");
//outFile.open("outfile.txt");

if (inFile.is_open()){

    inFile >> ch;
    while (!inFile.eof())
    {
        switch (ch) 
        {
            case '+':
                doAddition(inFile);
                break;
...

正如你所看到的,我注释掉了游戏部分,因为我无法让它正常工作。有什么建议?我现在有大约10个窗口和两本C ++书籍,只是试图将它们逻辑地组合在一起!

编辑:我不确定切换是否是最好的方法。我需要程序看到“int”并意识到它是一个单词。如果我使用了4个变量类型,比如int,double,float和long,也许我可以检查每个变量的第一个字母:i,d,f,l然后一旦它知道它可以进入+, - 的类型,等检查。感觉就像从逻辑上做这个,我只是花了更多的时间来使用一系列课程,但我不知道从哪里开始。

2 个答案:

答案 0 :(得分:1)

让我们从您提供的示例开始:

int + 25 10

“类型”和算术运算符的类型分别很简单,std::stringchar

std::ifstream in("infile.txt");
std::string type; char op;

if (in >> type >> op)
{
    // ...
}

对于其他两个值,您还必须将它们提取到字符串中,因为您首先必须找到type的值才能转换它们:

if (in >> type >> op >> a >> b) // a and b are strings

现在使用函数检查type并将ab转换为正确的类型:

void convertTo(std::string const& typeName, std::string const& a, std::string const& b, char op)
{
    if (typeName == "int")
    {
        int a1 = std::stoi(a),
            b2 = std::stoi(b);

         doOperation(op, a1, b2)
    } else if (typeName == "double") {
        double a1 = std::stod(a),
               b2 = std::stod(b);

        doOperation(op, a1, b2);
    } else if (typeName == "float") {
        // std::stof()
    }
}

doOperation()是模板化的,并按照以下方式实现:

template<typename T>
struct F;

template<> struct F<int> { static const std::string value = "int"; };
template<> struct F<double> { static const std::string value = "double"; };
template<> struct F<float> { static const std::string value = "float"; };

template<typename U, std::string name = F<U>::value>
void doOperation(char op, U a, U b)
{
    std::ofstream out("outfile.txt");
    switch (op)
    {
        case '+':
            out << name << " " << op << " " << (a + b);
        case '-':
            out << name << " " << op << " " << (a - b);
        case '/':
           // ...
        // ...
    }
}

答案 1 :(得分:1)

我真的不理解从文件中读取所有这些麻烦。 Stackoverflow和Web上有太多示例。也许是人们没有搜索,或者他们要求的例子与他们的确切代码相匹配。

试试这个:

struct Input_Record
{
    std::string data_type_as_string;
    std::string operation;
    std::string value1_as_string;
    std::string value2_as_string;

    friend std::istream& operator>>(std::istream& inp, Input_Record& r);
};

std::istream& operator>>(std::istream& inp, Input_Record& r)
{
    inp >> r.data_type_as_string;
    inp >> r.operation;
    inp >> r.value1_as_string;
    std::getline(inp, r.value2_as_string);  // Use getline to eat the line ending.
}

// ...
Input_Record r;
while (input_file >> r)
{
  // Do stuff with r
};

如果你真的想要玩得开心,你可以使用父基类和工厂模式来根据输入一般创建对象:

class Binary_Operation // Base class for factory pattern.
{
  public:
    //! Evaluate the object and return the result as a string
    //  e.g. numbers as text
    virtual std::string evaluate(void) const = 0; 
};
class Binary_Integer_Operation : public Binary_Operation
{
  public:
    std::string evaluate(void) const
    {
        // Convert values to integers than perform the operation.
        // Convert result to string using std::istringstream.
    };
};
class Binary_Double_Operation : public Binary_Operation
{
  // See Binary_Integer_Operation above.
};

这允许您执行以下操作:

Binary_Operation * factory_create(const Input_Record& r)
{
  Binary_Operation * p_obj = nullptr;
  if (r.data_type == "int")
  {
    p_obj = new Binary_Integer_Operation;
    // Initialize fields
  }
  if (r.data_type == "double")
  {
    p_obj = new Binary_Double_Operation;
    // Initialize fields;
  }
  return p_obj;
}

您的处理循环如下:

Input_Record r;
while (input_file >> r)
{
  Binary_Operation * p_operation = factory_create(r);
  std::string result = p_operation->evaluate();
  cout << "result = " << result << "\n";
  delete p_operation;
}