sscanf () пропускающий разделитель и ширина - PullRequest
0 голосов
/ 05 ноября 2018

Я пишу ассемблер для 6502 и пытаюсь прочитать инструкции и данные кода операции из файла, который я подготовил. Использование sscanf для хранения данных, и это работало только частично ...

Файл:

ADC,Im,69,2    
ADC,ZP,65,2
ADC,ZPx,75,2
ADC,Ab,6D,3
ADC,Abx,7D,3
ADC,Aby,79,3
...

Вот только часть кода, связанная с проблемой. fgets отлично работает. Проблемная строка прокомментирована ниже. Будет загружать больше, если это необходимо.

Код:

  FILE *fp = ...
  char bf[15];
  char name[3];
  char mode[3];
  char op[2];
  int bytes;

  while (fgets(bf,15,fp)) {
    //below is the problem line
    sscanf(bf, "%3[^,],%3[^,],%2[^,],%d", name, mode, op, &bytes);
  }
  printf("%s,%s,%s,%d\n", name, mode, op, bytes);

Выход:

ADCIm,Im,69,2
ADCZP,ZP,65,2
ADCZPx75,ZPx75,75,2
ADCAb,Ab,6D,3
ADCAbx7D,Abx7D,7D,3
...

Ожидается, что будет (так же, как формат файла):

ADC,Im,69,2
ADC,ZP,65,2
ADC,ZPx75,75,2
ADC,Ab,6D,3
ADC,Abx7D,7D,3
...

Кажется, что все операции и байты работают нормально, но что-то не так с переменными name и mode, хотя я содержал в аргументе ширину и разделитель.

1 Ответ

0 голосов
/ 05 ноября 2018

На самом деле, это не sscanf, что виноват. У вас неопределенное поведение из-за переполнения буфера - printf ничего не знает о том факте, что ваши строки не заканчиваются нулем, и будет продолжать печатать символы, пока не найдет '\0'.

Чтобы не делать этого, вы можете указать максимальную ширину поля в спецификаторе формата:

printf("%.3s,%.3s,%.2s,%d\n", name, mode, op, bytes);
...