将C ++文件读/写转换为C#文件读/写

时间:2013-05-28 22:59:17

标签: c# c++ porting

所有

我正在用C ++进行一些编程练习,然后尝试将它们翻译成C#和其他语言来刷新。

这个例子来自Ch。 11.2来自Nell Dale和Chip Weems的“使用C ++编程和解决问题”(第5版),第521-522页。这是一个接收此文件的程序(学生姓名,GPA,四个年级和课程成绩) - 让我们称之为rec.in

  

Mary Abbott   3.4 80 90 60 99 B

     鲍勃贝克   4.0 90 90 90 95 A

     莎莉卡特   2.0 70 70 70 65 C

     比尔丹斯克   3.0 65 50 60 70 D

     

JoEllen Zureck   1.9 90 95 80 99 A

并且(或多或少)只是将其复制到另一个文件。

以下是执行此操作的代码:

#include <iostream>
#include <string>
#include <fstream>

using namespace std;

const int MAX_STUDENTS = 150;

enum GradeType{A,B,C,D,F};

struct StudentRec
{
    string stuName;
    float gpa;
    int examScore[4];
    GradeType courseGrade;
};
StudentRec gradeBook[MAX_STUDENTS];

int main()
{
    StudentRec record[MAX_STUDENTS];
    ofstream outFile;
    outFile.open("rec.out");
    ifstream inFile;
    inFile.open("rec.in");
    for (int count = 0; count < MAX_STUDENTS; count++)
    ReadValues(inFile, record[count]);
    for (int count = 0; count <= MAX_STUDENTS; count++)
    PrintValues(outFile, record[count]);
    inFile.close();
    outFile.close();
    return 0;
}

void ReadValues(ifstream& inFile, StudentRec& record)
{
    char letter;
    char throwAway;

    getline(inFile, record.stuName);
    inFile >> record.gpa>>record.examScore[0]
    >> record.examScore[1] >> record.examScore[2]
    >> record.examScore[3] >> letter;

    inFile.get(throwAway);

    switch(letter)
    {
        case 'A' : record.courseGrade = A;
           break;
        case 'B' : record.courseGrade = B;
           break;
        case 'C' : record.courseGrade = C;
           break;
            case 'D' : record.courseGrade = D;
           break;
        case 'F' : record.courseGrade = F;
               break;
   }
}

void PrintValues(ofstream& outFile, StudentRec& record)
{
    outFile << record.stuName << endl;
    outFile << record.gpa << ' ' << record.examScore[0] << ' '
        << record.examScore[1] << ' '
        << record.examScore[2] << ' '
        << record.examScore[3] << ' ';
    switch (record.courseGrade)
    {
        case A : outFile << 'A';
        break;
        case B : outFile << 'B';
        break;
        case C : outFile << 'C';
        break;
        case D : outFile << 'D';
        break;
        case F : outFile << 'F';
        break;
    }

outFile << endl;
}

我开始使用C#的翻译。我以前做过C#工作,但没有安装控制台工作。结果,它有点......凌乱......努力翻译东西。我对在C#中匹配C ++的getline函数感到困惑。当然,有System.IO.StreamReader,但是一次只能读取一行文件。 getline以流格式执行,如上所示。这会使用单词/字符之间的空格/换行符将其分解。如何使用C#的库选择获得相同的结果?我与System.IO.StreamReader解决方案并没有完全结合。

这是我的开始:

using System;
using System.IO;

namespace _112a
{
    class Program
    {
        const int MAX_STUDENTS = 150;
        enum GradeType{A,B,C,D,F};
        struct StudentRec{
            string stuName;
            float gpa;
            GradeType courseGrade;
        };

        StudentRec[] gradeBook = new StudentRec[MAX_STUDENTS];

        static void Main(string[] args)
        {
            StudentRec[] record = new StudentRec[MAX_STUDENTS];
            System.IO.StreamWriter outFile = new System.IO.StreamWriter("rec.out"); 
            System.IO.StreamReader inFile = new System.IO.StreamReader("rec.in");

            for (int count = 0; count < MAX_STUDENTS; count++)
            {
                ReadValues(inFile, record[count]);
            }

            for (int count = 0; count <= MAX_STUDENTS; count++)
            {
                PrintValues(outFile, record[count]);
            }

            inFile.Close();
            outFile.Close();
        }

        static void ReadValues(StreamReader inFile, StudentRec record)
        {
            char letter;
            char throwAway;

            /*can't use getline, what do?*/

            inFile.ReadLine();

            switch (letter)
            {
                case 'A':
                    record.courseGrade = (GradeType)Enum.Parse(typeof(GradeType), "A");
                    break;
                case 'B':
                    record.courseGrade = (GradeType)Enum.Parse(typeof(GradeType), "B");
                    break;
                case 'C':
                    record.courseGrade = (GradeType)Enum.Parse(typeof(GradeType), "C");
                    break;
                case 'D':
                    record.courseGrade = (GradeType)Enum.Parse(typeof(GradeType), "D");
                    break;
                case 'F':
                    record.courseGrade = (GradeType)Enum.Parse(typeof(GradeType), "F");
                    break;
            }


        }

        static void PrintValues(StreamWriter outFile, StudentRec record)
        {
            outFile.Write("{0}\n{1} {2} {3} ", record.stuName, record.examScore[0], record.examScore[1], record.examScore[2], record.examScore[3]);

            switch(record.courseGrade)
            {
                case GradeType.A: 
                    outFile.Write("A\n");
                    break;
                case GradeType.B: 
                    outFile.Write("B\n");
                    break;
                case GradeType.C: 
                    outFile.Write("C\n");
                    break;
                case GradeType.D:
                    outFile.Write("D\n");
                    break;
                case GradeType.F:
                    outFile.Write("F\n");
            }
        }
    }
}

关于您在方法PrintValues中看到的交换机结构,我还有一个问题:在每种情况下,它在C ++文件中指定它只需要record.courseGrade字段并将其与case后面指定的参数。 C#是一种强大的语言,不会让这种情况发生......如果你将char'A'与record.courseGrade进行比较,就会出现类型不匹配错误。或者,将其保留为单引号(并将其保留为record.courseGrade字段)会产生The name 'A' does not exist in the current context.错误,如何最好地解决此问题? - 已解决

使用ReadLine()的困惑源于它的工作方式。在这种情况下,您有StudentRec结构的多个成员被填充,但它们是结构中的独立实体。我担心会做类似的事情:

StudentRec.stuName = inFile.ReadLine();

并且StudentRec.stuName填充整行,其他结构字段中没有任何内容。

此外,我们还有letter被取消分配并用作交换机的问题;这是因为它从未收到ReadLine的值。然而,如果ReadLine确实分裂了,那就是学术解决方案。

2 个答案:

答案 0 :(得分:3)

您想分别使用StreamReader和Writer。您可能正在寻找的方法是ReadLineWriteLine,StreamReader文档底部有一个可以满足您需求的工作示例。您只需要在while循环中更改文件路径和逻辑。文档可以在这里找到;

http://msdn.microsoft.com/en-us/library/system.io.streamreader.aspx

StreamWriter文档在这里,并有一个示例,底部使用读者和作者;

http://msdn.microsoft.com/en-us/library/system.io.streamwriter.aspx

所有方法都具有逻辑名称,并且比c ++对应方法更易于实现。

我认为您的开关导致编译错误,因为案例需要case "A":而不是case A:。如果你把文字放在那里,它将进行字符串比较。它不理解A是什么,除非那是局部变量的名称。

编辑:读取文件的示例代码;

// we need to alloc/init the streams inside for using blocks so they're properly disposed of
List<string> grades = new List<string>();
using (StreamReader inFile = new StreamReader("rec.in"))
{
     string line;
     while ((line = inFile.ReadLine()) != null)
     {
         string[] tokens = line.Split(' '); // split line on whitespace
         if (tokens.Length > 7)
             grades.Add(tokens[7]);
     }
}

using (StreamWriter outFile = new StreamWriter("rec.out"))
{
      foreach (string s in grades)
          outFile.WriteLine(s);
}

我上面使用的技术并不是完全必要的,例如你可以在StreamReaders中嵌套StreamWriter并在同一范围内进行写作。在C#版本中,您将逐行循环遍历文件的内容。每当我们得到一条线时,我们想要将它分割成空格以获得它的子组件。然后我将字母等级添加到列表中。我们不希望输出字符串中有换行符,因为WriteLine会自动执行此操作。在我们读取文件之后,我们遍历列表中的每个字符串并将其写入输出文件。

答案 1 :(得分:1)

对于switch的问题,试试这个:

case GradeType.A: 
    outFile.Write("A\n");
    break;

您使用枚举并且必须将其交给切换案例。

如果您想使用文件中的每一行,其他问题必须将其分配给字符串,然后处理它以通过拆分等方法提取所需的信息。

string line = inFile.ReadLine();
string[] data = line.Split(' ');
record.stuName = string.Format("{0} {1}", data[0], data[1]);
record.gpa = float.Parse(data[2]);
....