0

標準Cでコーディングした単純なスキャナーの lex バージョンを実装しています。私が問題を抱えているのは、I/O が lex ファイルで期待どおりに動作していないことです。fscanf文字列を保存せず、指定した変数に整数を 0 として保存します。fgetc は、テスト ファイルに存在しない文字を返します。以下の私のコードには、なぜこれが起こっているのかが飛び出すもの (または欠けているもの) がありますか? lex の使い方は完全に間違っていますか?

スキャナー-lf.l:

%{
#include <stdio.h>
#include <stdlib.h>
extern int lineno;
extern int number;
extern char string[];
extern FILE *yyin;
#define INTEGER 1
#define FLOAT 2
#define READ 3
#define WRITE 4
#define ID 5
#define LPAREN 6
#define RPAREN 7
#define PLUS 8
#define MINUS 9
#define MULT 10
#define ASSIGN 11
#define DIV 12
#define COMMENT 13
#define ERROR 14
%}

%%
[ \t] {
    printf("whitespace\n");
}
[0-9]* {
    fscanf(yyin, "%d", &number);
    printf("%d\n", number);
    int c = fgetc(yyin);
    printf("%c", c);
    /*return INTEGER;*/
}
[a-zA-Z][a-zA-Z0-9]* {
    fscanf(yyin, "%s", string);
    printf("%s\n", string);
    /*return ID;*/
}

スキャナー-lex.c:

#include <stdio.h>
#include <stdlib.h>
/* A couple of globals */
int lineno = 0;
int number;
char string[100];
FILE *yyin;
char charSet[] = { '(', ')', '+', '-', '*' };
#define INTEGER 1
#define FLOAT 2
#define READ 3
#define WRITE 4
#define ID 5
#define LPAREN 6
#define RPAREN 7
#define PLUS 8
#define MINUS 9
#define MULT 10
#define ASSIGN 11
#define DIV 12
#define COMMENT 13
#define ERROR 14

int yywrap(){
}

int main(int argc, char **argv){
    int rc;
    if (argc > 1){
        yyin = fopen(argv[1], "r");
        while ((rc=yylex())){
            switch (rc){
                case READ:
                printf("found a read\n");
                break;
                case WRITE:
                printf("found a write\n");
                break;
                case ID:
                printf("found an id\n");
                break;
                case LPAREN:
                printf("found an (\n");
                break;
                case RPAREN:
                printf("found an )\n");
                break;
                case PLUS:
                printf("found an +\n");
                break;
                case MINUS:
                printf("found an -\n");
                break;
                case MULT:
                printf("found an *\n");
                break;
                case ASSIGN:
                printf("found an assign\n");
                break;
                case DIV:
                printf("found a /\n");
                break;
                case INTEGER:
                printf("found an integer\n");
                break;
                case FLOAT:
                printf("found a float\n");
                break;
                case COMMENT:
                printf("found a comment\n");
                break;
                case ERROR:
                printf("Error\n");
                break;
            }
        }
    }
    return 0;
}
4

1 に答える 1

1

これは間違っています。

[0-9]* {
    fscanf(yyin, "%d", &number);
    printf("%d\n", number);
    int c = fgetc(yyin);
    printf("%c", c);
    /*return INTEGER;*/
}

Lexはファイルからその内部バッファーに読み取り、指定されたパターンが最大限に一致したときに指定されたアクションを実行します。

Flexマニュアルの例を見てみましょう:

{DIGIT}+    {
            printf( "An integer: %s (%d)\n", yytext,
                    atoi( yytext ) );
            }

ご覧のように、

  1. サンプルパターンは+の代わりにを使用し*ます。空の文字列を一致させたくないので、それも使用する必要があります+

  2. 一致したパターンの内容はに保存されyytextます。fscanf()一致したテキストはすでにlexによって読み取られているため、を使用してファイルからパターンを読み取ろうとしないでください。 だから、単にnumber = strtol(yytext, NULL, 10);あなたの行動で使用してください。(またはを使用しないでください。数値の先頭にゼロがある場合、これらは出力が悪くなります。)atoi(yytext)sscanf(yytext, "%i", &number)

于 2013-01-31T00:50:10.130 に答える