How can I use non-default delimiters when reading a text file with std::fstream?

前端 未结 2 1631
没有蜡笔的小新
没有蜡笔的小新 2020-11-30 05:01

In my C++ code, I want to read from a text file (*.txt) and tokenize every entry. More specifically, I want to be able to read individual words from a file, such as \"format

相关标签:
2条回答
  • 2020-11-30 05:32

    You can use

    istream::getline(char* buffer, steamsize maxchars, char delim)
    

    although this only supports a single delimiter. To further split the lines on your different delimiters, you could use

    char* strtok(char* inString, const char* delims)  
    

    which takes multiple delimeters. When you use strtok you only need to pass it the address of your buffer the first time - after that just pass in a null and it will give you the next token from the last one it gave you, returning a null pointer when there are no more.

    EDIT: A specific implementation would be something like

    char buffer[120]; //this size is dependent on what you expect the file to contain
    while (!myIstream.eofbit) //I may have forgotten the exact syntax of the end bit
    {
        myIstream.getline(buffer, 120); //using default delimiter of \n
        char* tokBuffer;
        tokBuffer = strtok(buffer, "'- ");
        while (tokBuffer != null) {
            cout << "token is: " << tokBuffer << "\n";
            tokBuffer = strtok(null, "'- "); //I don't need to pass in the buffer again because it remembers the first time I called it
        }
    }
    
    0 讨论(0)
  • 2020-11-30 05:34

    An istream treats "white space" as delimiters. It uses a locale to tell it what characters are white space. A locale, in turn, includes a ctype facet that classifies character types. Such a facet could look something like this:

    #include <locale>
    #include <iostream>
    #include <algorithm>
    #include <iterator>
    #include <vector>
    #include <sstream>
    
    class my_ctype : public
    std::ctype<char>
    {
        mask my_table[table_size];
    public:
        my_ctype(size_t refs = 0)  
            : std::ctype<char>(&my_table[0], false, refs)
        {
            std::copy_n(classic_table(), table_size, my_table);
            my_table['-'] = (mask)space;
            my_table['\''] = (mask)space;
        }
    };
    

    And a little test program to show it works:

    int main() {
        std::istringstream input("This is some input from McDonald's and Burger-King.");
        std::locale x(std::locale::classic(), new my_ctype);
        input.imbue(x);
    
        std::copy(std::istream_iterator<std::string>(input),
            std::istream_iterator<std::string>(),
            std::ostream_iterator<std::string>(std::cout, "\n"));
    
        return 0;
    }
    

    Result:

    This
    is
    some
    input
    from
    McDonald
    s
    and
    Burger
    King.
    

    istream_iterator<string> uses >> to read the individual strings from the stream, so if you use them directly, you should get the same results. The parts you need to include are creating the locale and using imbue to make the stream use that locale.

    0 讨论(0)
提交回复
热议问题