293 / 14 / 3
Регистрация: 21.12.2010
Сообщений: 293
1

Удалить комментарии из файла. Помогите найти ошибку.

15.03.2011, 14:43. Показов 4036. Ответов 10

Студворк — интернет-сервис помощи студентам
программа должна распознать комменитарии вида /* */ и // из входного файла, и записать в выходной файл программу без комментариев...
1 комментарии не могут быть вложенными
2 коментарий не может являться частью строковой константы
3 строковые и символьные константы могут включать в себя \' и \"
Остальные требования не буду, писать, так как я их реализовала

В общем у меня проблема с 3 пунктом...Он зацикливается сдесь и не переходит в состояние Normal
В чем проблема? помогите найти ошибку?

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
//---------------------------------------------------------------------------
 
#include <vcl.h>
#pragma hdrstop
 
#include <stdio.h>
#include <tchar.h>
#include <iostream.h>
#include <conio.h>
 
using namespace std;
 
//---------------------------------------------------------------------------
 
#pragma argsused
 
typedef enum States { Normal, Slash, Comment,EndComment, EndFile,StrState,OneComment /* перечисление всех
                                                  состояний автомата */ } States;
int main(int argc, char ** argv)//
{
FILE * fi, * fo;         /* входной и выходной файл */
States State = Normal;   /* текущее состояние */
int/*space*/c;                   /* считанный символ (только один текущий!) */
  /* все, никакие переменные больше не нужны, этих вполне достаточно */
 
  if (argc != 3)
  {
    /* по заданию, программа должна получать имена входного и выходного
       файлов при помощи параметров командной строки, поэтому здесь можно
       вывести сообщение об ошибке, инструкцию и т.п. А можно обойтись и
       без этой проверки вообще, следующих двух условных конструкций будет
       достаточно */
    return 3;   /* это зачем? почему 3? подумайте. */
  }
  fi = fopen(argv[1], "rb");
  if (!fi)
  {
    fprintf(stderr, "Input file \"%s/*\" open error.\n", argv[1]);
    return 1;
  }
  fo = fopen(argv[2], "wb");
  if (!fo)
  {
    fclose(fi);
    fprintf(stderr, "Output file \"%s\" open error.\n", argv[2]);
    return 2;
  }
 
  while ((c=fgetc(fi)) != EOF)  /* считываем символ и проверяем, не конец ли файла? */
   {
    switch (State)   /* если нет, то обрабатываем в зависимости от текущего состояния */
    {
      case Normal:
            State=Normal;
            if (c == '/')         /* если встретили слэш, */
                 State = Slash;     /* то перешли в соответствующее состояние */
            else if (c == '"')
                 {State=StrState;
                 fputc(c,fo);}
                 else fputc(c,fo);
            break;
 
        case Slash:          /* если предыдущим был слэш, то ... */
            if (c == '*')
                State=Comment;
            else if (c == '/') 
                State=OneComment;
                else              
                 
            {
                State=Normal;
                fputc('/',fo);
                fputc(c,fo);
            }
             break;
 
        case Comment:
            if ( c=='*')
                State=EndComment;
            else State=Comment;
            break;
 
        case EndComment:
            State=Comment;
            if (c=='/')
                State=Normal;
            if (c=='*')
                State=EndComment;
            break;
 
        case StrState:
            fputc(c,fo);
            if  (c == '"') State = Normal;
 
            if ( (c == '\'') || (c == '\"')) State=StrState;
            
                  
            
            
        break;
            
        case OneComment:
                 if ( (c == '\r') || (c=='\n'))
                 {
                 State=Normal;
                 fputc(c,fo);}
                 else State = OneComment;
        break;
        
        }
 
 
 
 
            }
   if (State=Slash)   fputc(c,fo);
 
 
  fclose(fi);    /* не забывайте закрывать файлы! */
  fclose(fo);    /* особенно выходной, открытый для записи */
  return 0;      /* а это зачем? */
}

вот пример входного файла
Вложения
Тип файла: txt lab1_in.txt (2.4 Кб, 62 просмотров)
0
Лучшие ответы (1)
Programming
Эксперт
94731 / 64177 / 26122
Регистрация: 12.04.2006
Сообщений: 116,782
15.03.2011, 14:43
Ответы с готовыми решениями:

Удалить значение из базы.Не работает форма.Помогите найти ошибку?
&lt;form id=&quot;myform&quot; &gt;&lt;select id=&quot;mymenu&quot; &gt;&lt;option value=&quot;Алексей СИБВТ&quot;&gt;Алексей...

Удалить комментарии из файла
Здравствуйте, написал программу, которая удаляет комментарии в файле #include &lt;iostream&gt;...

Необходимо удалить комментарии из html файла
Задание: Написать скрипт, который будет удалять все комментарии из html файла. Я написал код, но...

Удалить из текстового файла комментарии вида {comments}
uses crt; var f,g:text; p1,p2:byte; nf,ng,s,st:string; begin clrscr; write('Имя файла...

10
293 / 14 / 3
Регистрация: 21.12.2010
Сообщений: 293
15.03.2011, 14:45  [ТС] 2
я подозреваю что проблема в 91-95 строках...
0
Формучанин
364 / 294 / 42
Регистрация: 02.11.2010
Сообщений: 1,242
15.03.2011, 20:43 3
если тебе встречается слеш (\) то нужно записать в выходной файл два символа подряд(если state позволяет)
а вот эти две строки:
C
1
2
 if  (c == '"') State = Normal;//Является ли c двойной кавычкой?
if ( (c == '\'') || (c == '\"')) State=StrState;//является ли c одинарной или двойной кавычкой?
логически эквивалентны:
C
1
if ( (c == '\'') || (c == '\"')) State=StrState;
state никогда не примет значение Normal.

чтобы проверять на слеш нужно использовать такую конструкцию:
C
1
if (c=='\\') ...
Добавлено через 10 минут
корректно будет так:
C
1
2
3
if  (c == '"') State = Normal;
 
                        if ( (c == '\'') || (c == '\"')) State=StrState;
заменить на:

C
1
2
3
4
5
6
7
if (c=='"') State= Normal; else
if  (c == '\\') 
{
if ((c=fgetc(fi)) == EOF) {/*Дописать сюда код завершения цикла, достигнут конец файла*/}
fputc(c,fo);
continue;
}
Добавлено через 6 минут
C
1
2
3
4
5
case Comment:
                        if ( c=='*')
                                State=EndComment;
                        else State=Comment;
                        break;
если после комментария /* встретится чтото типа a*=2, то программа посчитает это за конец комментария.
Нужно сравнивать два символа: */

Добавлено через 4 минуты
C
1
2
3
4
5
6
7
8
9
 case Normal:
                        State=Normal;
                        if (c == '/')         /* если встретили слэш, */
                                 State = Slash;     /* то перешли в соответствующее состояние */
                        else if (c == '"')
                                 {State=StrState;
                                 fputc(c,fo);}
                                 else fputc(c,fo);
                        break;
можно сократить до:
C
1
2
3
4
5
6
7
8
9
10
 case Normal:
                        
                        if (c == '/')         /* если встретили слэш, */
                                 State = Slash;     /* то перешли в соответствующее состояние */
                        else {
                                 fputc(c,fo);
                                 if (c == '"') State=StrState;
                              }
                                 
                          break;
Добавлено через 2 минуты
C
1
return 0;      /* а это зачем? */
А это затем чтобы вызывающее приложение знало код, который вернула твоя программа. 0=успех.

Добавлено через 3 минуты
C
1
  if (State=Slash)   fputc(c,fo);
это условие выполнятся будет либо всегда либо никогда, в зависимости от численного значения Slash.
Правильно будет так:
C
1
  if (State==Slash)   fputc(c,fo);
Добавлено через 2 минуты
107 строка:
C
1
     else State = OneComment;
не несёт смысловой нагрузки, и её можно вобще удалить
1
4866 / 3288 / 468
Регистрация: 10.12.2008
Сообщений: 10,570
16.03.2011, 04:57 4
Цитата Сообщение от nxnx
логически эквивалентны:
а где State = Normal ?

C
1
2
3
if ((c=fgetc(fi)) == EOF) {/*Дописать сюда код завершения цикла, достигнут конец файла*/}
fputc(c,fo);
continue;
чего ? дописать EOF в файл, зачем ?
0
4866 / 3288 / 468
Регистрация: 10.12.2008
Сообщений: 10,570
16.03.2011, 06:02 5
Лучший ответ Сообщение было отмечено как решение

Решение

C
1
2
/* удаляет многострочные и однострочные комментарии из файла
   комментарии в двойных кавычках не рассматривает */
Вложения
Тип файла: zip sing_multi_comments.zip (1.2 Кб, 270 просмотров)
4
Формучанин
364 / 294 / 42
Регистрация: 02.11.2010
Сообщений: 1,242
16.03.2011, 11:06 6
Цитата Сообщение от accept Посмотреть сообщение
а где State = Normal ?

C
1
2
3
if ((c=fgetc(fi)) == EOF) {/*Дописать сюда код завершения цикла, достигнут конец файла*/}
fputc(c,fo);
continue;
чего ? дописать EOF в файл, зачем ?
State = Normal в той конструкции не имеет смысла.
Проверяется не достигнут ли конец. Если достигнут-выходим. если нет, пишем дальше.
0
293 / 14 / 3
Регистрация: 21.12.2010
Сообщений: 293
16.03.2011, 21:23  [ТС] 7
Цитата Сообщение от nxnx Посмотреть сообщение
чтобы проверять на слеш нужно использовать такую конструкцию:
Код C1 if (c=='\\') ...
спасибо, вот именно в этом и была проблема
0
4866 / 3288 / 468
Регистрация: 10.12.2008
Сообщений: 10,570
17.03.2011, 03:48 8
Цитата Сообщение от nxnx
Проверяется не достигнут ли конец. Если достигнут-выходим. если нет, пишем дальше.
а, ну это так выглядит, if на одной строке записан

записал бы в одну строку и это
C
1
2
fputc(c,fo);
continue;
а потом бы вообще
C
1
if ((c = fgetc(fi)) == EOF) { код; код; }; fputc(c, fo); continue;
как удобно наблюдать за одной строкой
0
293 / 14 / 3
Регистрация: 21.12.2010
Сообщений: 293
23.03.2011, 19:17  [ТС] 9
Вот коректный код, может кому пригодится.

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
//---------------------------------------------------------------------------
 
#include <vcl.h>
#pragma hdrstop
 
#include <stdio.h>
#include <tchar.h>
#include <iostream.h>
#include <conio.h>
 
using namespace std;
 
//---------------------------------------------------------------------------
 
#pragma argsused
 
typedef enum States { Normal, Slash, Comment,EndComment, EndFile,StrState,SimState,OneComment /* перечисление всех
                                                  состояний автомата */ } States;
int main(int argc, char ** argv)//
{
FILE * fi, * fo;         /* входной и выходной файл */
States State = Normal;   /* текущее состояние */
int/*space*/c;                   /* считанный символ (только один текущий!) */
  /* все, никакие переменные больше не нужны, этих вполне достаточно */
 
  if (argc != 3)
  {
    /* по заданию, программа должна получать имена входного и выходного
       файлов при помощи параметров командной строки, поэтому здесь можно
       вывести сообщение об ошибке, инструкцию и т.п. А можно обойтись и
       без этой проверки вообще, следующих двух условных конструкций будет
       достаточно */
    return 3;   /* это зачем? почему 3? подумайте. */
  }
  fi = fopen(argv[1], "rb");
  if (!fi)
  {
    fprintf(stderr, "Input file \"%s/*\" open error.\n", argv[1]);
    return 1;
  }
  fo = fopen(argv[2], "wb");
  if (!fo)
  {
    fclose(fi);
    fprintf(stderr, "Output file \"%s\" open error.\n", argv[2]);
    return 2;
  }
 
  while ((c=fgetc(fi)) != EOF)  /* считываем символ и проверяем, не конец ли файла? */
   {
    switch (State)   /* если нет, то обрабатываем в зависимости от текущего состояния */
    {
      case Normal:
 
                        if (c == '/')         /* если встретили слэш, */
                                 State = Slash;     /* то перешли в соответствующее состояние */
                        else {
                                 fputc(c,fo);
                                 if (c == '"') State=StrState;
                                 if (c == '\'') State=SimState;
                              }
 
                        break;
 
        case Slash:          /* если предыдущим был слэш, то ... */
            if (c == '*')
                State=Comment;
            else if (c == '/') 
                State=OneComment;
                else              
                 
            {
                State=Normal;
                fputc('/',fo);
                fputc(c,fo);
            }
             break;
 
        case Comment:
            if ( c=='*')
                State=EndComment;
            //else State=Comment;
            break;
 
        case EndComment:
            State=Comment;
            if (c=='/')
                {State=Normal; fputc(' ',fo);}
            if (c=='*')
                State=EndComment;
            break;
 
        case StrState:
            fputc(c,fo);
            if (c=='"') State= Normal; else
            if (c == '\\')
            {
            if ((c=fgetc(fi)) == EOF) break;
            fputc(c,fo);
            }
        break;
 
        case SimState:
            fputc(c,fo);
            if (c=='\'') State= Normal; else
            if (c == '\\')
            {
            if ((c=fgetc(fi)) == EOF) break;
            fputc(c,fo);
            }
        break;
 
        case OneComment:
                 if ( (c == '\r') || (c=='\n'))
                 {
                 State=Normal;
                 fputc(c,fo);}
        break;
        
        }
   }      if (State==Slash)   fputc(c,fo);
 
  fclose(fi);    /* не забывайте закрывать файлы! */
  fclose(fo);    /* особенно выходной, открытый для записи */
  return 0;      /* а это зачем? */
}
0
0 / 0 / 0
Регистрация: 09.04.2011
Сообщений: 5
12.04.2011, 21:20 10
а как её запустить?Я скопировал текст программы в .cpp файл и запустил.Но не работает.Можете подсказать,что я не правильно сделал?Может,надо было создать проект?Какой именно?Пользуюсь VS 6.0
0
Формучанин
364 / 294 / 42
Регистрация: 02.11.2010
Сообщений: 1,242
12.04.2011, 22:57 11
Цитата Сообщение от flexin777 Посмотреть сообщение
а как её запустить?Я скопировал текст программы в .cpp файл и запустил.Но не работает.Можете подсказать,что я не правильно сделал?Может,надо было создать проект?Какой именно?Пользуюсь VS 6.0
имена файлов нужно указать в параметрах запуска.
что вывела программа и с каким кодом завершилась?
0
IT_Exp
Эксперт
87844 / 49110 / 22898
Регистрация: 17.06.2006
Сообщений: 92,604
12.04.2011, 22:57
Помогаю со студенческими работами здесь

Удалить комментарии из файла, содержащего программный код Си
удалить коментарии с файла, которое имеет С-програму

Удалить все комментарии и теги из HTML файла
Есть задание: Написать сценарий для sed и awk для удаления тегов и комментариев HTML. Теги и...

Удалить из файла все комментарии вида {комментарий}
Дан текстовый файл, содержащий строки длиной до 255 символов. Файл является программой на языке...

Найти | Удалить все дочерние комментарии
Доброго времени суток. У меня как всегда проблема... На сайте можно комментировать все, что...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
11
Ответ Создать тему
Опции темы

КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2023, CyberForum.ru