Проблемы с пользовательским строковым классом типа C - PullRequest
0 голосов
/ 29 августа 2011

У меня есть домашнее задание, которое мне нужно, чтобы создать собственный класс строки, используя строки типа C. Большинство из них выглядит нормально, но я получаю ошибку во время выполнения, как только я тестирую ее, и она не выполняет все мои тесты, как должно. В частности, у моего оператора + = что-то не так, но я не могу понять, что именно.

Я не могу добавить или изменить ни один из прототипов, и я должен использовать конструкции C ++ поверх типа C, где это возможно.

Спасибо!

#include <iostream>
#include <string>
#include "tstr.h"
using namespace std;

//Default constructor to initialize the string to null
TStr::TStr() {
    strPtr = 0;
    strSize = 0;
}  
//constructor; conversion from the char string
TStr::TStr(const char *str) {
    int i=0;
    while (str[i] != '\0') {
        ++i;
    }
    strSize = i;
    strPtr = new char [i+1];
    for (i=0; i < strSize; ++i) {
        strPtr[i] = str[i];
    }
} 
//Copy constructor
TStr::TStr(const TStr& str) {
    strPtr = new char[str.strSize];
    strcpy(strPtr, str.strPtr);
}
//Destructor
TStr::~TStr() {
    delete[] strPtr;
}

//subscript operators-checks for range
char& TStr::operator [] (int i) {
    assert (i >= 0 && i < strSize);
    return strPtr[i];
}
const char& TStr::operator [] (int i) const {
    assert (i >= 0 && i < strSize);
    return strPtr[i];
}

//overload the concatenation oprerator
TStr TStr::operator += (const TStr& str) {
    char *buffer = new char[strSize + str.strSize + 1];
    strcpy(buffer, strPtr);
    strcat(buffer, str.strPtr);
    delete [] strPtr;
    strPtr = buffer;
    return *this;
}

//overload the assignment operator
const TStr& TStr::operator = (const TStr& str) {
    if (this != &str) {
        delete[] strPtr;
        strPtr = new char[strSize = str.strSize];
        assert(strPtr);
        for (int i=0; i<strSize; ++i) {
            strPtr[i] = str.strPtr[i];
        }
    }
    return *this;
}

//overload two relational operators as member functions
bool TStr::operator == (const TStr& str) const {
    int value = strcmp(strPtr, str.strPtr);
    if (value == 0) {
        return true;
    } else {
        return false;
    }
    /*int counter=0;
    for (int i=0; i < strSize; ++i) {
        if (strPtr[i] == str.strPtr[i]) {
            ++counter;
        }
    }
    if (counter == strSize) {
        return true;
    } else {
        return false;
    }
    return (strPtr == str.strPtr && strSize == str.strSize);*/
}
bool TStr::operator < (const TStr& str) const {
    return (strPtr < str.strPtr && strSize < str.strSize);
}
//the length of the string
int TStr::size() {
    return strSize;
}

//Overload the stream insertion and extraction operators.
ostream& operator << (ostream& out, const TStr& str) {
int size = str.strSize;
for (int i=0; i < size; ++i) {
    out << str[i];
}
return out;
}
istream& operator >> (istream& in, TStr& str) {
return in;
}

//overload two other relational operators as global functions
bool operator != (const TStr& S1, const TStr& S2) {
    return !(S1 == S2);
}
bool operator <= (const TStr& S1, const TStr& S2) {
    return (S1 < S2 || S1 == S2);
}
bool operator > (const TStr& S1, const TStr& S2) {
    return !(S1 < S2);
}
bool operator >= (const TStr& S1, const TStr& S2) {
    return !(S1 < S2 || S1 == S2);
}

//overload the concatenation operator as a global function
TStr operator + (const TStr& str1, const TStr& str2) {
    //return (str1 += str2);
    //return (str1 + str2);
}

Вот что я тестирую:

int main() {
authors();
TStr str1 = "VENI";         //initialize str1 using 
                                //the assignment   operator 
const TStr str2("VEDI");    //initialize str2 using the
                                    //conversion constructor
TStr str3; //initialize str3 to null
TStr str4; //initialize str4 to null

cout << "\nTest 1: str1: " << str1 << " str2 " << str2 
     << " str3 " << str3 << " ###.\n" ;          //Test 1

if (str1 <= str2)                               //Test 2  
    cout << "\nTest 2: " << str1 << " is less "
         << "than " << str2 << endl;            
else                                            
    cout << "\nTest 2: " << str2 << " is less "
         << "than " << str1 << endl;            

str1=" Pride is what we have.";

str3 = str1;                                    //Test 3
cout << "\nTest 3: The new value of str3 = "
     << str3 << endl;                          


str3 += " Vanity is what others have. ";              
cout<<"\nTest 4: The str3: '" << str3<<"'  \nhas ";        //test 4
cout<< countVowels(str3)<<" vowels "<< endl;

/*TStr str5 = str1 + str2 + str3;
cout<<"\nTest 5: The str5: '" << str5<<"' \nhas ";        //test 5
cout<<countVowels(str5)<<" vowels \n";

cout<<"\nTest 6: The str3 again: " << str3 <<endl;    //test 6

cout<<"\n\n Bye, Bye!";*/


return 0;
}

РЕДАКТИРОВАТЬ: Я должен был добавить, что мне нужно, чтобы тесты работали как часть домашней работы.

РЕДАКТИРОВАТЬ 2: Вот функции, которые я изменил до сих пор.

//constructor; conversion from the char string
TStr::TStr(const char *str) {
    int i=0;
    while (str[i] != '\0') {
        ++i;
    }
    strSize = i;
    strPtr = new char [i+1];
    for (i=0; i < strSize; ++i) {
        strPtr[i] = str[i];
        strPtr[i + 1] = '\0'; //<- this
    }
} 
//Copy constructor
TStr::TStr(const TStr& str) {
    strPtr = new char[str.strSize + 1]; //<-this
    strcpy(strPtr, str.strPtr);
}
//overload the concatenation oprerator
TStr TStr::operator += (const TStr& str) {
    char *buffer = new char[strSize + str.strSize + 1]; //<- this
    strcpy(buffer, strPtr);
    strcat(buffer, str.strPtr);
    delete [] strPtr;
    strPtr = buffer;
    return *this;
}

//overload the assignment operator
const TStr& TStr::operator = (const TStr& str) {
    if (this != &str) {
        delete[] strPtr;
        strPtr = new char[strSize = str.strSize + 1]; //<- this
        assert(strPtr);
        for (int i=0; i<strSize; ++i) {
            strPtr[i] = str.strPtr[i];
        }
    }
    return *this;
}

РЕДАКТИРОВАТЬ 3: К сожалению, изменение его ниже ничего не сделал на самом деле. Это может быть оператор =, который вызывает проблему, так как теперь он терпит неудачу сразу после теста 3.

//constructor; conversion from the char string
TStr::TStr(const char *str) {
    int i=0;
    while (str[i] != '\0') {
        ++i;
    }
    strSize = i;
    strPtr = new char [i+1];
    strcpy(strPtr, str);
} 

Ответы [ 5 ]

1 голос
/ 29 августа 2011

Поскольку никто еще не указал на это, ваш strSize остается неопределенным в вашем конструкторе копирования;и в вашем операторе присваивания.

Вам нужно завершить строки только нулем, если они будут взаимодействовать со строками в стиле C (например, char *, strcopy и т. д.).

1 голос
/ 29 августа 2011

Есть несколько мест, в которых вам нужно +1, чтобы зарезервировать окончание \0 для строк. Например, в конструкторе копирования

strPtr = new char[str.strSize];

(это должно быть str.strSize+1) и в operator=(). Это могут быть источником ваших проблем.

НО вы должны внимательно изучить все подобные ошибки и протестировать каждую функцию отдельно (может быть, с хорошим набором тестов, вы знаете, написанным перед самим классом) :)

1 голос
/ 29 августа 2011

Ваш char* конструктор преобразования не завершает NUL содержимое (он резервирует пространство для NUL, но не сохраняет его), что приводит к сбою strcpy и strcat позже.

1 голос
/ 29 августа 2011

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

  1. В конструкторе, конвертируемом из const char *, вы не копируете нулевой терминатор в strPtr. (исправлено)
  2. В вашем конструкторе копирования вы не выделяете достаточно места для нулевого терминатора. (исправлено)
  3. В вашем операторе присваивания вы не выделяете достаточно места для нулевого терминатора. (исправлено ... не совсем, см. # 5)
  4. В вашем операторе присваивания вы не копируете через нулевой терминатор.
  5. Вы добавили новую проблему в свой оператор присваивания, когда зафиксировали распределение. Теперь вы устанавливаете strSize на неправильный размер.
0 голосов
/ 29 августа 2011

operator +=() должен возвращать ссылку - TStr&. Вы возвращаете копию, что, вероятно, не то, что вы хотите.

...