Форум программистов, компьютерный форум CyberForum.ru

Конкатенация строк - C++

Восстановить пароль Регистрация
 
Рейтинг: Рейтинг темы: голосов - 9, средняя оценка - 4.67
bgm313
12 / 12 / 2
Регистрация: 27.07.2012
Сообщений: 208
27.08.2012, 22:41     Конкатенация строк #1
Частью задания является реализовать функцию strcat. Возникает какая-то ошибка (см.фото).
1.Вы могли бы подсказать в чём ошибка?
2.Не по теме: как можно улучшить функцию read?

Вот код:

C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
//read
char *Read(void){
    size_t size = 32;   //первоначальный размер строки
    size_t i = 0;       //индекс элемента строки
    int ch;             
    
    //выделим память для строки
    char *result = (char *)malloc(size);
    if(!result){
        cout << "Ошибка выделения памяти.\n";
        system("pause");
        exit(1);
    }
 
    //очистим стандартный поток ввода
    fflush(stdin);
    
    //считываем строку
    for(;;){
        ch = getchar(); 
        
        //если встретился пробел, то выходим из цикла
        if(isspace(ch)) break;
    
        //если длина строки превышает максимальный размер, то вернём нулевой указатель
        if(i == MAX_SIZE_STR - 1){
            cout << "Максимальная длина строки: " << MAX_SIZE_STR - 1 << '\n';
            return NULL;
        }
        //перевыделим память для, если это потребуется
        else if(i == size - 2){
            size *= 2;
            result = (char *)realloc(result,size);
            if(!result){
                cout << "Ошибка выделения памяти.\n";
                system("pause");
                exit(1);
            }
        }
        
        //запишем 1 символ из потока в строку
        result[i++] = ch;
    }
 
    //поставим завершающий нуль
    result[i] = '\0';
    
    if(ch!='\n') {
        cout << "Cтрока не должна содержать пробелов.\n";
        return NULL;
    }else if(Length(result) == 0) {
        cout << "Длина строку должна быть больше нуля.\n";
        return NULL;
    }
 
    return result;
}
//my strcat
char * Concat(const char *str,const char *str2){
    int i = 0;
    int n = Length(str);
    char *result;
 
    result = (char *)malloc(n + Length(str2) + 1);
    if(!result){
        cout << "Ошибка выделения памяти.\n";
        system("pause");
        exit(1);
    }
    
    while(result[i] = str[i]) i++;
    i = 0;
    while(result[n++] = str2[i++]);
    
    result[n] = 0;
    return result;
 
//main
void main(void){
    char ch;
    char *str, *str2, *result;
 
    setlocale(LC_ALL,"Rus");
 
    for(;;){
        system("cls");
 
        //выбор операции над строками
        ch = SelectOp();
        putchar('\n');
 
        if(ch == '0') break;
        
        do{
            cout << "Введите строку 1: ";
            str = Read();
        }while(!str);
        
        switch(ch){
case'2':
            do{
                cout << "Введите строку 2: ";
                str2 = Read();
            }while(!str2);
            
            putchar('\n');
 
            result = Concat(str,str2);
            
            cout << "строка 1: " << str << '\n';
            cout << "строка 2: " << str2 << '\n';
            cout << "сумма строк Concat: " << result << '\n';
            cout << "сумма строк strcat: " << strcat(str,str2) << '\n'; 
            
            free(str);
            free(str2);
            free(result);
            break;
            }
}
Миниатюры
Конкатенация строк  
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
27.08.2012, 22:41     Конкатенация строк
Посмотрите здесь:

C++ Конкатенация строк
C++ Конкатенация строк
Конкатенация строк C++
Конкатенация строк C++
конкатенация строк C++
После регистрации реклама в сообщениях будет скрыта и будут доступны все возможности форума.
bgm313
12 / 12 / 2
Регистрация: 27.07.2012
Сообщений: 208
27.08.2012, 23:03  [ТС]     Конкатенация строк #2
Ошибка происходит при вводе строк максимальной длины.
Максимальная длина строки определена как глобальная константа:
[CPP]const int MAX_SIZE_STR = 101;
[CPP]

Добавлено через 4 минуты
Ошибка происходит почему-то при вызове free(result).

Добавлено через 13 минут
Если убрать в main free(result), то никаких ошибок не возникает. НО тогда происходит утечка. Почему free(result) вызывает ошибку?
Jupiter
Каратель
Эксперт C++
6542 / 3962 / 226
Регистрация: 26.03.2010
Сообщений: 9,273
Записей в блоге: 1
Завершенные тесты: 2
27.08.2012, 23:03     Конкатенация строк #3
Цитата Сообщение от bgm313 Посмотреть сообщение
2.Не по теме: как можно улучшить функцию read?
не смешивать Си и С++(это касается кода в целом)

Цитата Сообщение от bgm313 Посмотреть сообщение
fflush(stdin);
это непереносимо и работает только под виндовс

Цитата Сообщение от bgm313 Посмотреть сообщение
exit(1);
exit в С++ не используют так как не будут вызваны деструкторы

Цитата Сообщение от bgm313 Посмотреть сообщение
(char *)malloc(size);
чем обосновано использование malloc ?
bgm313
12 / 12 / 2
Регистрация: 27.07.2012
Сообщений: 208
27.08.2012, 23:17  [ТС]     Конкатенация строк #4
malloc обосновано тем, что в программе требуется использовать функцию realloс. Да, конечно, можно было бы перегрузить оператор new, но в программе не используются классы.

Добавлено через 1 минуту
1.Что использовать вместо fflush(stdin)?
2.Почему всё таки выскакивает ошибка про кучу?

Добавлено через 1 минуту
Если надо могут скинуть весь код.
OhMyGodSoLong
~ Эврика! ~
 Аватар для OhMyGodSoLong
1234 / 983 / 42
Регистрация: 24.07.2012
Сообщений: 2,002
27.08.2012, 23:23     Конкатенация строк #5
> Почему всё таки выскакивает ошибка про кучу?
Потому что вы гадите память, так как у вас где-то переполнение буфера. Скорее всего, из-за неправильно работающей Length() и ошибок на единицу. Помните, что нулевой байт в конце тоже учитывается при выделении памяти, но strlen() возвращает длину строки без него.

Далее, strcat или concat? strcat() дописывает вторую строку к первой, а не создаёт третью, где лежит конкатенация первых двух.

И наконец, возвращать указатель на объект, созданный в функции, — это в общем случае нехорошо. Потому что можно забыть вызвать free(). Кто память выделил, тот её и освобождает.
bgm313
12 / 12 / 2
Регистрация: 27.07.2012
Сообщений: 208
27.08.2012, 23:36  [ТС]     Конкатенация строк #6
0.Чем можно заменить fflush(stdin);

1.Требуется, чтобы функция concat записывала в 3 объект результат сложения.
Можно было бы поступить одним из след.способов:
a.Засунуть всё в глобальную переменную.
b.передать 3 параметр в функцию
с.сделать так, как сделал я
Какой из этих вариантов наиболее приемлемый? Может быть есть ещё какие-либо варианты?

2.По поводу ошибок в функциии Length. Я тоже так думал и выделил память для результирующей строки
размером 2*max_size + length(str) + length(str2); И всё равно выскакивает ошибка. Почему?

Вот скидываю полный код:

C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
  #include <iostream>
using namespace std;
 
//максимальный размер строки
const int MAX_SIZE_STR = 101;
 
//сравнение строк 
int Compare(const char *str, const char *str2);
 
//длина строки
int Length(const char *str);
 
//копия строки str2
void Copy(char *str, const char *str2);
 
//конкатенация строк
char * Concat(const char *str,const char *str2);
 
//считает количество подстрок
int CalcSub(char *str, char *substr);
 
//считывает строку
char *  Read(void);
 
//выбор операции над строками
char SelectOp(void);
 
void main(void){
    char ch;
    char *str, *str2, *result;
 
    setlocale(LC_ALL,"Rus");
 
    for(;;){
        system("cls");
 
        //выбор операции над строками
        ch = SelectOp();
        putchar('\n');
 
        if(ch == '0') break;
        
        do{
            cout << "Введите строку 1: ";
            str = Read();
        }while(!str);
        
        switch(ch){
        case'1':
            do{
                cout << "Введите строку 2: ";
                str2 = Read();
            }while(!str);
            
            putchar('\n');
            
            cout << "Результат сравнения Compare: " << Compare(str,str2) << '\n';   
            cout << "Результат сравнения strcmp: " << strcmp(str,str2) << '\n'; 
            
            free(str);
            free(str2);
            break;
        case'2':
            do{
                cout << "Введите строку 2: ";
                str2 = Read();
            }while(!str2);
            
            putchar('\n');
 
            result = Concat(str,str2);
            
            cout << "строка 1: " << str << '\n';
            cout << "строка 2: " << str2 << '\n';
            cout << "сумма строк Concat: " << result << '\n';
            cout << "сумма строк strcat: " << strcat(str,str2) << '\n'; 
            
            free(result);
            free(str);
            free(str2);
            
            break;
        case'3':
            cout << "длина строки Length: " << Length(str) << '\n';
            cout << "длина строки strlen: " << strlen(str) << '\n';
            free(str);
            break;
        case'4':
            do{
                cout << "Введите подстроку: ";
                str2 = Read();
            }while(!str);
 
            putchar('\n');
 
            cout << "строка: " << str << '\n';
            cout << "подстрока: " << str2 << '\n';
            cout << "количество подстрок: " << CalcSub(str,str2) << '\n';
            
            free(str);
            free(str2);
            break;
        }
        
        putchar('\n');
        system("pause");
    }
}
 
char SelectOp(void){
    char ch;
    
    cout << "Меню:\n"
         << " 1.Сравнение строк.\n"
         << " 2.Конкатенация строк.\n"
         << " 3.Длина строки.\n"
         << " 4.Количество подстрок.\n";
    do{
        cout << "Сделайте выбор(0-выход): ";
        fflush(stdin);
    }while(scanf("%c",&ch)!=1 || cin.rdbuf()->in_avail()!=1 ||
        ch < '0' || ch > '4');
    return ch;
}
 
int Compare(const char *str, const char *str2){
    while(*str == *str2){
        if(!*str) return 0;
        str++; str2++;
    }
    if((*str - *str2) < 0) return -1;
    return 1;
}
 
int Length(const char *str){
    int len = 0;
    
    while(str[len]) len++;
    return len;
}
 
char * Concat(const char *str,const char *str2){
    int i = 0;
    int n = strlen(str);
    char *result = (char *)malloc(1000);
    
    if(!result){
        cout << "Ошибка выделения памяти.\n";
        system("pause");
        exit(1);
    }
    
    while(result[i] = str[i]) i++;
    i = 0;
    while(result[n++] = str2[i++]);
    
    result[n] = 0;
    return result;
}
 
int CalcSub(char *str, char *substr){
    int result = 0;
 
    while(strstr(str,substr)){
        result++;
        str+=Length(substr);
    }
    return result;
}
 
char *Read(void){
    size_t size = 32;   //первоначальный размер строки
    size_t i = 0;       //индекс элемента строки
    int ch;             
    
    //выделим память для строки
    char *result = (char *)malloc(size);
    if(!result){
        cout << "Ошибка выделения памяти.\n";
        system("pause");
        exit(1);
    }
 
    //очистим стандартный поток ввода
    fflush(stdin);
    
    //считываем строку
    for(;;){
        ch = getchar(); 
        
        //если встретился пробел, то выходим из цикла
        if(isspace(ch)) break;
    
        //если длина строки превышает максимальный размер, то вернём нулевой указатель
        if(i == MAX_SIZE_STR - 1){
            cout << "Максимальная длина строки: " << MAX_SIZE_STR - 1 << '\n';
            return NULL;
        }
        //перевыделим память для, если это потребуется
        else if(i == size - 2){
            size *= 2;
            result = (char *)realloc(result,size);
            if(!result){
                cout << "Ошибка выделения памяти.\n";
                system("pause");
                exit(1);
            }
        }
        
        //запишем 1 символ из потока в строку
        result[i++] = ch;
    }
 
    //поставим завершающий нуль
    result[i] = '\0';
    
    if(ch!='\n') {
        cout << "Cтрока не должна содержать пробелов.\n";
        return NULL;
    }else if(Length(result) == 0) {
        cout << "Длина строку должна быть больше нуля.\n";
        return NULL;
    }
 
    return result;
}
Добавлено через 5 минут
Хочу заметить в функции concat намеренно выделяется память размером 1000 б. - это в 5 раз больше чем может содержать результат сложения строк максимально допустимой длины.
Yandex
Объявления
27.08.2012, 23:36     Конкатенация строк
Ответ Создать тему
Опции темы

Текущее время: 13:51. Часовой пояс GMT +3.
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2016, vBulletin Solutions, Inc.
Рейтинг@Mail.ru