IPB
ЛогинПароль:

> Внимание!

1. Пользуйтесь тегами кода. - [code] ... [/code]
2. Точно указывайте язык, название и версию компилятора (интерпретатора).
3. Название темы должно быть информативным. В описании темы указываем язык!!!

> Ошибка в коде программы(С++), Сортировка строк методом пузырька
V.k.l.chr.by
сообщение 7.05.2008 23:42
Сообщение #1


Новичок
*

Группа: Пользователи
Сообщений: 29
Пол: Мужской

Репутация: -  0  +


Будьте добры,уважаемые форумчане можете исправить ошибку в данном коде программе.Уж очень срочно надо.Буду очень признателен.А вот и само условие и код программы:
Исходные данные находятся в текстовом файле. Разделитель – конец строки. Программа запрашивает имя входного и результирующего (отсортиро-ванного) файла. Использовать динамическое распределение памяти.И надо сортировать строки методом пузырька.Тут как бы всё правилььно выполняет,только вот пишет ошибку .. и как мне подсказывали что-то с указателями.Или после кода программы"void main(void)..."


#include<stdio.h>
#include<windows.h>
#include<string.h>
#include<iostream.h>

#define maxline 1000

void sort(char* ptr[],int n_lines)
{
int i,j;
char *tmp;
for(i=0;i<n_lines-1;i++)

for(j=0;j<n_lines-1;j++)

if(strcmp(ptr[j],ptr[j+1])>0)
{
tmp=ptr[j];
ptr[j]=ptr[j+1];
ptr[j+1]=tmp;
}


}
void write_lines(char* ptr[],int n_lines)
{
int i;
char *str = new char[];
for(i=0;i<n_lines;i++)
{
CharToOem(ptr[i],str);
cout<<str<<endl;

}
}

void write_in_file(char* ptr[],int n_lines, FILE* fp)
{
int i;
char *str = new char[];
for(i=0;i<n_lines;i++)
{
fputs(ptr[i],fp);
fputs("\n",fp);


}
}

void main(void)
{

char s[] = "Введите путь к файлу : ";
CharToOem(s,s);
char r[] = "Введите путь к отсортированному файлу : ";
CharToOem(r,r);
char a[] = "Ошибка при открытии файла\n";
CharToOem(a,a);
char ns[] = "До сортировки :";
CharToOem(ns,ns);
char ps[] = "После сортировки :";
CharToOem(ps,ps);


FILE *in;
char *str = new char[];
char *name = new char[];
int counter=0;
char buf[maxline];
char *ptr[maxline];
cout<<s;
cin>>name;

if((in=fopen(name,"rt"))==NULL)
{
cout<<a;
return;
}

FILE *fp;
char *nam = new char[];
cout<<r;
cin>>nam;
fp=fopen(nam,"wt");

for(counter=0;(!feof(in))&&counter<maxline;counter++)
{
fgets(buf,maxline,in);
if(buf[strlen(buf)-1]=='\n')
{
buf[strlen(buf)-1]='\0';
}
ptr[counter] = new char[strlen(buf)+1];
strcpy(ptr[counter],buf);
buf[0]='\0';

}

fclose(in);
cout<<ns<<endl;
write_lines(ptr,counter);
cout<<endl<<endl<<ps<<endl;
sort(ptr,counter);
write_lines(ptr,counter);
write_in_file(ptr,counter,fp);
fclose(fp);

}
P.S.Прошу прощение,за повтор темы.Хотел удалить ,но не нашёл.

Сообщение отредактировано: volvo - 27.11.2009 12:22
 Оффлайн  Профиль  PM 
 К началу страницы 
+ Ответить 
 
 Ответить  Открыть новую тему 
Ответов
V.k.l.chr.by
сообщение 15.05.2008 22:03
Сообщение #2


Новичок
*

Группа: Пользователи
Сообщений: 29
Пол: Мужской

Репутация: -  0  +


Вообщем всё понятно,но возникли небольшие вопросики.
1-Почему,в коде программы нельзя было просто написать до N,а не n_lines?
2-И у Вас в комментарии написано,что "до n_lines - 1"Причём тут единица.
3-И если не трудно,можно ещё пару слов об "os"ЧТо это такое?Я как понял,это назваине потока,благодаря чему будет копироваться троки в файл.Я так понял?

-
// для всех номеров строк от 0 до n_lines - 1 (в С++ индексация всегда начинается с 0)
for(int i = 0; i < n_lines; i++) {


Я так подумал вышлю остальной код программы(если не трудно прокментировать его),чтобы после сего,по частям разбарал бы и если что вопрсоызадавал бы.Думаю так быстрее будет,чем по частям высылать и так же уже нужно скоро уже относить программу.
p/S/Я ещё раз хочу выразить благодарность за помощь студенту.
-
void sort(char* ptr[], int n_lines) {
for(int i = 0; i < n_lines; i++) {
for(int j = n_lines - 1; j > i; j--) {
if(strcmp(ptr[j - 1], ptr[j]) > 0) {
char *tmp = ptr[j-1];
ptr[j-1] = ptr[j];
ptr[j] = tmp;
}
}
}
}


int main()
{

char s_in_name[] = "Введите путь к файлу : ";
CharToOem(s_in_name, s_in_name);

char s_out_name[] = "Введите путь к отсортированному файлу : ";
CharToOem(s_out_name, s_out_name);

char s_err_open[] = "Ошибка при открытии файла\n";
CharToOem(s_err_open, s_err_open);

char s_before[] = "До сортировки :";
CharToOem(s_before, s_before);

char s_after[] = "После сортировки :";
CharToOem(s_after, s_after);

char in_name[64], out_name[64];

const int maxLines = 1000;
char *ptr[maxLines];


cout << s_in_name;
cin >> in_name;

ifstream in(in_name, ios::in);
if(!in) {
cout << s_err_open << endl;
return -1;
}

cout << s_out_name;
cin >> out_name;

ofstream out(out_name, ios::out);
if(!out) {
cout << s_err_open << endl;
return -1;
}


const int buf_size = 1024;

char buffer[buf_size];
int current = -1;
while(in.getline(buffer, buf_size)) {
ptr[++current] = new char[strlen(buffer) + 1];
strcpy(ptr[current], buffer);
}

cout << s_before << endl;
write_lines(cout, ptr, current + 1);
sort(ptr, current + 1);
cout << s_after << endl;
write_lines(cout, ptr, current + 1);

write_lines(out, ptr, current + 1);
for(int i = 0; i <= current; i++) {
delete ptr[i];
}

return 0;

}
 Оффлайн  Профиль  PM 
 К началу страницы 
+ Ответить 
volvo
сообщение 15.05.2008 23:19
Сообщение #3


Гость






Цитата(V.k.l.chr.by @ 15.05.2008 22:03) *
1-Почему,в коде программы нельзя было просто написать до N,а не n_lines?
Ну, если тебе удобнее до N, то пиши до N, я привык давать "говорящие" имена переменным...
Цитата(V.k.l.chr.by @ 15.05.2008 22:03) *
в комментарии написано,что "до n_lines - 1"Причём тут единица.
При том, что если число строк это n_lines, то значит, их индексы - от 0 до (n_lines-1)
Цитата(V.k.l.chr.by @ 15.05.2008 22:03) *
можно ещё пару слов об "os"ЧТо это такое?
Это поток вывода, os (сокращение от output stream)... Понимаешь в чем дело... Когда запускается программа на С++, создаются несколько потоков. Один (cout, стандартный вывод на экран) - для вывода данных, другой (cin, стандартный ввод с клавиатуры) - для ввода данных в программу. Ну, есть еще cerr/clog, но это пока не важно... Так вот, cout имеет тип ostream. От этого же типа унаследован и файловый поток (ofstream), то есть, в свою процедуру я могу передать (по правилу совместимости типов) не только сам cout, но и любого его наследника, и если я передам файловый поток, то все то, что должно было бы напечататься на экране, будет выведено в файл... Чем я и воспользовался... Когда мне нужно - вывожу данные на монитор, а когда нужно - в файл...

Теперь что касается остальной программы:

Это - самая обычная сортировка "пузырьком" массива ptr, содержащего n_lines строк. Алгоритм этот найдешь на любом сайте, да хоть у нас в FAQ-е по Паскалю: Методы сортировок

void sort(char* ptr[], int n_lines) {
for(int i = 0; i < n_lines; i++) {
for(int j = n_lines - 1; j > i; j--) {
if(strcmp(ptr[j - 1], ptr[j]) > 0) {
char *tmp = ptr[j-1];
ptr[j-1] = ptr[j];
ptr[j] = tmp;
}
}
}
}




// а это - основная программа
int main()
{

// Сначала задаем все строковые константы и приводим их к виду,
// в котором они нормально отображаются на экране...

char s_in_name[] = "Введите путь к файлу : ";
CharToOem(s_in_name, s_in_name);

char s_out_name[] = "Введите путь к отсортированному файлу : ";
CharToOem(s_out_name, s_out_name);

char s_err_open[] = "Ошибка при открытии файла\n";
CharToOem(s_err_open, s_err_open);

char s_before[] = "До сортировки :";
CharToOem(s_before, s_before);

char s_after[] = "После сортировки :";
CharToOem(s_after, s_after);

// место для имен входного/выходного файлов
char in_name[64], out_name[64];

// максимальное число строк, которые будут обрабатываться.
// Если в файле будет больше строк, чем указано здесь - то массив,
// описанный чуть ниже, переполнится и получишь Access Violation - ошибку доступа
const int maxLines = 1000;

// Вот, собственно, массив для хранения указателей на строки
char *ptr[maxLines];

// запрашиваем у пользователя и вводим имя входного файла
cout << s_in_name;
cin >> in_name;

// открываем файловый поток in для чтения
ifstream in(in_name, ios::in);
if(!in) { // поток не открылся, выходим с ошибкой
cout << s_err_open << endl;
return -1;
}

// запрашиваем и принимаем от пользователя имя выходного файла
cout << s_out_name;
cin >> out_name;

// опять же открываем файловый поток, но теперь уже на запись
ofstream out(out_name, ios::out);
if(!out) { // не открылся - ошибка
cout << s_err_open << endl;
return -1;
}
// максимальное количество символов, читаемое из файла за один раз
const int buf_size = 1024;
// ну, и буфер для временного хранения прочитанной из файла строки
char buffer[buf_size];

// здесь будет номер строки, прочитанной из файла на каждой итерации цикла
// для начала (-1), потом поймешь почему...
int current = -1;

// попытка прочесть до buf_size символов (на самом деле может быть прочитано
// и меньше, поскольку чтение завершится как только встретится символ "\n")
// из файла in в buffer...
// пока эти попытки успешны (т.е., строка читается)
while(in.getline(buffer, buf_size)) {
// увеличиваем номер текущей строки (теперь понятно, почему изначально был (-1)?
// Потому что начинается индексация с 0), и для текущего номера выделяется память,
// достаточная для хранения прочитанной из файла строки...
ptr[++current] = new char[strlen(buffer) + 1];

// как только память выделена - копируем строку из временного буфера в дин. память
strcpy(ptr[current], buffer);
}

// Ну, а дальше - все просто: выводим все прочитанные строки в cout
// (т.е., на монитор), как именно, я уже объяснял
cout << s_before << endl;
write_lines(cout, ptr, current + 1);

// сортируем массив строк (поскольку current содержит индекс последней
// прочитанной строки начиная с 0, а в процедуру сортировки нужно передать
// КОЛИЧЕСТВО строк, то передаем current + 1)
sort(ptr, current + 1);

// и уже отсортированные строки выводим сначала на монитор (cout)...
cout << s_after << endl;
write_lines(cout, ptr, current + 1);

// ... а потом - в файловый поток, открытый нами для записи
write_lines(out, ptr, current + 1);

// Все, строки сохранены, можно дин. память освобождать
for(int i = 0; i <= current; i++) {
delete ptr[i];
}

return 0;

}
По-моему несложно...
 К началу страницы 
+ Ответить 

Сообщений в этой теме
V.k.l.chr.by   Ошибка в коде программы(С++)   7.05.2008 23:42
volvo   Можно уточнить? зачем понадобилось перемешивать чи...   7.05.2008 23:57
V.k.l.chr.by   Вообщем я просто брал нечто похожую программу с ла...   8.05.2008 0:08
volvo   Неправильно... Разбираться надо, ЧТОБЫ программа з...   8.05.2008 1:27
V.k.l.chr.by   Просто в условии дано ,чтобы было динамичесткое ра...   8.05.2008 1:30
V.k.l.chr.by   Проверил ещё раз,всё ровно русские символы не сорт...   9.05.2008 12:46
volvo   Ну, так возьми и чуть-чуть подкорректируй write_li...   9.05.2008 15:08
V.k.l.chr.by   Ух..выручили Вы меня всё тепреь работает!...   9.05.2008 18:26
volvo   Задавай, конечно... Для этого форум и существует -...   9.05.2008 19:33
V.k.l.chr.by   И так мне уже в начале не понятно первые строчки.А...   10.05.2008 23:40
volvo   // функция write_lines описывается как void - не ...   11.05.2008 0:04
V.k.l.chr.by   Вообщем всё понятно,но возникли небольшие вопросик...   15.05.2008 22:03
volvo   1-Почему,в коде программы нельзя было просто напис...   15.05.2008 23:19
V.k.l.chr.by   Спасибо.Но и в правду как бы всё понятно. Так же х...   16.05.2008 0:42
volvo   Дело не в диапазоне... Дело в том, что память в пр...   16.05.2008 1:49


 Ответить  Открыть новую тему 
1 чел. читают эту тему (гостей: 1, скрытых пользователей: 0)
Пользователей: 0

 



- Текстовая версия 21.06.2025 16:13
Хостинг предоставлен компанией "Веб Сервис Центр" при поддержке компании "ДокЛаб"