C ++: добавление к векторной строке

Я пишу программу для свиней на латыни; читать ввод от пользователей (имя и фамилия), сделайте ввод строчными буквами и измените имя в зависимости от того, что было в имени. Если первая буква (как имени, так и фамилии) была гласной, мы должны добавить путь до ее конца.

Если первая буква была согласной, мы должны были взять первую букву, переместить ее в конец строки и добавить в конец строки «у».

Мой код выдает ошибки при попытке добавить текст в конец строки. Он говорит, что не может преобразовать строку в символ, и я не совсем уверен, что это значит. В нем также говорится, что я не могу использовать выходной операнд ‹< для строк, хотя я использовал его раньше.

Ошибки возникают с strcpy и окончательным кодом, в котором я выводю имена.

37: ошибка: невозможно преобразовать 'std::vector<std::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::allocator<std::basic_string<char, std::char_traits<char>, std::allocator<char> > > >' в 'char*' для аргумента '1' в 'char* strcpy(char*, const char*)'

47: ошибка: невозможно преобразовать 'std::vector<std::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::allocator<std::basic_string<char, std::char_traits<char>, std::allocator<char> > > >' в 'char*' для аргумента '1' в 'char* strcpy(char*, const char*)'

54: ошибка: нет совпадений для 'operator<<' в 'std::cout << first'

Мне просто нужна помощь, чтобы исправить ошибки и посмотреть, где я ошибся. Полный код прилагается.

#include <iostream>
#include <vector>
#include <cstring>
#include <string>
using namespace std;
int main()
{
    int q, s;
    char shea[] = "way";
    char gavin_stop_looking_at_ponies[] = "ay";
    vector <string> first;
    vector <string> last;
    cout << "Please enter your first name." << endl;
    for (int i = 0; i < first.size(); i++)
    {
        getline (cin, first[i]);
        string nfirst = first[i];
        while (nfirst[q])
        {
            nfirst[q] = tolower(nfirst[q]);
        }
        first[i] = nfirst;
    }
    cout << "Please enter your last name." << endl;
    for (int j = 0; j < last.size(); j++)
    {
        getline (cin, last[j]);
        string nlast = last[j];
        while (nlast[s])
        {
            nlast[s] = tolower(nlast[s]);
        }
        last[j] = nlast;
        }
    if ( (first[0] == "a") ||( first [0] == "e") || (first [0] == "i") || (first [0] == "o")     || (first [0] == "u"))
    {
        strcpy (first, "way");
    }
    else
    {
        first[first.size()] = first[0] + "ay";
    }
    
    if ( (last[0] == "a") ||( last [0] == "e") || (last [0] == "i") || (last [0] == "o") || (last [0] == "u"))
    {
        strcpy (last, "way");
    }
    else
    {
        last[last.size()] = last[0] + "ay";
    }
    cout << first << last << endl;
    return 0;
}

person Shea Hunter Belsky    schedule 04.04.2013    source источник
comment
Было бы неплохо знать, где возникают ошибки, и иметь точные сообщения об ошибках.   -  person chris    schedule 04.04.2013
comment
Я пытался сказать, когда добавлял текст к строке. Строки, содержащие strcpy.   -  person Shea Hunter Belsky    schedule 04.04.2013
comment
Не используйте strcpy с C ++ std :: string. Посмотрите на функциональность, доступную в std :: string.   -  person Pete Fordham    schedule 04.04.2013
comment
Ну, первая и последняя - векторные ‹string› объекты. Я не верю, что вы можете просто так передать их cout. Вы должны перебирать их и передавать каждую отдельную строку. См. Это сообщение: stackoverflow.com/questions/10750057/   -  person confused_at_times    schedule 04.04.2013
comment
first [0] .c_str () для преобразования вашей String в const char *. Опять же, вы вызываете strcpy с первым параметром, являющимся вектором объекта строк, а не строкой.   -  person confused_at_times    schedule 04.04.2013
comment
Вы не инициализируете q, и я думаю, вы не понимаете, что такое строка ... строка i не один символ, а его последовательность.   -  person johannes    schedule 04.04.2013
comment
Возможно, вы захотите сделать шаг назад и прочитать хорошую книгу по C ++ . Ваш код содержит бесконечные while циклы, чтение неинициализированных переменных, ошибки выхода индекса за пределы допустимого диапазона и путаницу в том, как использовать std::vector.   -  person Blastfurnace    schedule 04.04.2013


Ответы (4)


Я аннотировал ваш код с объяснением некоторых проблем и предложениями по их решению. Если вы чего-то не понимаете, прокомментируйте, и я постараюсь уточнить.

#include <iostream>

// You don't need 'vector' for this.
#include <vector>

// You won’t often need the C string header in C++.
#include <cstring>

#include <string>
using namespace std;
int main()
{
    // These variables are unused.
    int q, s;
    char shea[] = "way";
    char gavin_stop_looking_at_ponies[] = "ay";

    // 'first' and 'last' are both names, not collections
    // of names.
    string first;
    string last;
    vector <string> first;
    vector <string> last;

    // 'endl' is unnecessary here; it outputs a newline and
    // flushes the stream, but standard output is usually
    // line-buffered, meaning that newline flushes the
    // stream regardless.
    cout << "Please enter your first name.\n"
    cout << "Please enter your first name." << endl;

    // If you just want to get one name, 'getline' is perfect.
    getline(cin, first);

    // This loop would run 0 times because 'first' is an
    // empty vector.
    for (int i = 0; i < first.size(); i++)
    {
        getline (cin, first[i]);
        string nfirst = first[i];
        while (nfirst[q])
        {
            nfirst[q] = tolower(nfirst[q]);
        }
        first[i] = nfirst;
    }

    // To make a string lowercase, use 'tolower' on each character.
    // Here's one way to do it:
    for (string::size_type i = 0; i < first.size(); ++i)
        first[i] = tolower(first[i]);

    // Here's another, with C++11 enabled:
    for (auto& c : first)
        c = tolower(c);

    cout << "Please enter your last name.\n";
    cout << "Please enter your last name." << endl;

    // Same thing.
    getline(cin, last);
    for (int j = 0; j < last.size(); j++)
    {
        getline (cin, last[j]);
        string nlast = last[j];
        while (nlast[s])
        {
            nlast[s] = tolower(nlast[s]);
        }
        last[j] = nlast;
    }

    // Now 'first' is a string, and 'first[0]' is a 'char'.
    // "a" is a string literal; 'a' is a character literal.
    // You can compare each character individually:
    if (first[0] == 'a' || first[0] == 'e' || first[0] == 'i' || first[0] == 'o' || first[0] == 'u')

    // Or you can say "if the character was found in this
    // set of vowels".
    if (string("aeiou").find(first[0]) != string::npos)

    if ( (first[0] == "a") ||( first [0] == "e") || (first [0] == "i") || (first [0] == "o")     || (first [0] == "u"))
    {
        // This would try to copy "way" into 'first':
        // formerly a vector of string objects, now just a
        // string object. 'strcpy' wants a character buffer,
        // and will overwrite characters in that buffer—
        // probably not what you want:
        //
        // "aaron" => "wayon"
        // 
        strcpy (first, "way");

        // Instead, just append "way":
        first += "way";
    }
    else
    {
        // This says "take the first first character of the
        // string, add the value of that character to a
        // pointer to a buffer containing "ay", then try to
        // copy the resulting pointer past the end of the
        // string. Again, not quite what you intended!
        first[first.size()] = first[0] + "ay";

        // Think of it instead like this: take everything
        // after the first character, add a string consisting
        // of the first character back onto the end, then add
        // "ay" after that.
        first = first.substr(1) + string(1, first[0]) + "ay";
    }

    // Duplicated code! You could move the above logic into
    // a function to avoid this duplication. Then you only
    // have to work on it in one place. :)
    if ( (last[0] == "a") ||( last [0] == "e") || (last [0] == "i") || (last [0] == "o") || (last [0] == "u"))
    {
        strcpy (last, "way");
    }
    else
    {
        last[last.size()] = last[0] + "ay";
    }

    // I need a space between my first and last names!
    cout << first << ' ' << last << '\n';
    cout << first << last << endl;
    return 0;
}
person Jon Purdy    schedule 04.04.2013

first - это vector<string>, а не string. vector не поддерживает << с cout. Если вы хотите вывести каждый std::string в std::vector, попробуйте перебирать std::vector и выводить их по одному символу за раз.

Точно так же first - это vector<string>, который не может быть неявно преобразован в char*. strcpy работает с необработанными блоками char данных, от которых first очень далеко. strcpy предназначен для операций уровня C со строковыми буферами (и даже в этом случае опасно использовать).

first[first.size()] = first[0] + "ay"; - это неопределенное поведение, поскольку вы обращаетесь к элементу «один за последним» из first, что является недопустимой памятью. Если вы хотите вставить что-нибудь в заднюю часть first, попробуйте first.push_back( first[0] + "ay" );.

Вполне вероятно, что вас смущает разница между std::string, char и char*. Это совершенно разные вещи. std::string - это управляемый буфер char. char - это одиночное 8-битное значение, часто используемое для хранения литералов, таких как 'a' (который имеет тип char). char* - это указатель на один char, часто используемый как указатель на начало неуправляемого буфера, плотно упакованного char.

Ваш std::vector<std::string> - это управляемый буфер из управляемых char буферов. Каждый элемент в vector - это полный буфер с неизвестным количеством char, а не один char.

person Yakk - Adam Nevraumont    schedule 04.04.2013

Я согласен с некоторыми из предыдущих респондентов, но я не хочу вдаваться в подробный обзор кода, потому что он будет большим и может упустить суть. Я думаю, что возникшая у вас проблема является признаком фундаментального непонимания правильного использования стандартной библиотеки C ++. Чаще всего вы смешиваете строковые реализации стандартной библиотеки C со «строковым» объектом C ++. В C есть набор функций, которые принимают указатели на символы и работают с ними как со строками, а в C ++ есть строковый объект, который инкапсулирует строку символов и предоставляет набор операций над ней. В некоторых случаях строка C ++ может использоваться там, где обычно существует C char *, но в большинстве случаев это невозможно.

Даже в C ++ STL (стандартная библиотека шаблонов) вы немного запутались. Судя по вашему коду, похоже, что вы хотите получить имя и фамилию пользователя, то есть две отдельные строки. Однако вы объявляете 2 коллекции (в данном случае векторы) строк, а не только две отдельные строки. Из этого вытекает множество других проблем в вашей программе.

Думаю, вам может понравиться информация, найденная здесь: http://en.cppreference.com/w/

person alpartis    schedule 08.04.2013

Я быстро написал код для этого, просто чтобы немного поиграться с ним. Вот что я придумал - я думаю, что это довольно хорошо упрощено, понятно и легко для понимания:

#include <iostream>
#include <string>
#include <algorithm>
using namespace std;

/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
/* simple pig-latin program ... converts a given "first name" and "last name"*/
/* from the user to pig latin translations.  Obviously, the so-called names  */
/* are really just any 2 random words, but that's the theme anyway.          */
/*                                                                           */
/*                                                                           */
/*                                                                           */
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */


string xlat_to_piglatin(const string source)
{
    string result;

    //
    // translation: if word begins with a vowel, simply append "way" to
    //  it.  Otherwise, move the initial consonant from the beginning to
    //  the end, and append "ay" to it.
    //
    if (source.find_first_of("aeiou") == 0) {
        result = source;
        result.append("way");
    } else {
        result = source.substr(1, source.size());
        result.append(source.substr(0, 1));
        result.append("ay");
    }
    return result;
}

int main()
{
    string  first;
    string  last;

    //
    // get the first and last names from the user on standard in ...
    //
    cout << "Please enter your first name:" << endl;
    cin >> first;
    cout << "Please enter your last name:" << endl;
    cin >> last;

    //
    // convert strings to lower case ...
    //
    transform(first.begin(), first.end(), first.begin(), ptr_fun<int, int>(tolower));
    transform(last.begin(), last.end(), last.begin(), ptr_fun<int, int>(tolower));

    cout << xlat_to_piglatin(first) << " " << xlat_to_piglatin(last) << endl;
    return 0;
}
person alpartis    schedule 09.04.2013