The number of characters of comments in a file (C programming)

前端 未结 3 1026
太阳男子
太阳男子 2021-01-25 11:52

I can\'t seem to get it right, tried everything, but..

int commentChars() {
char str[256], fileName[256];
FILE *fp;
int i;


do{
    long commentCount=0;
    ffl         


        
3条回答
  •  说谎
    说谎 (楼主)
    2021-01-25 12:30

    This basically trivial modification of your code deals with several problems in your code.

    1. You should not use feof() like that — `while (!feof(file)) is always wrong.
    2. You should not read data that is not part of the string just read.

    I've also refactored your code so that the function takes a file name, opens, counts and closes it, and reports on what it found.

    #include 
    #include 
    
    // Revised interface - process a given file name, reporting
    static void commentChars(char const *file)
    {
        char str[256];
        FILE *fp;
        long commentCount = 0;
    
        if (!(fp = fopen(file, "r")))
        {
            fprintf(stderr, "Error! File %s not found\n", file);
            return;
        }
    
        while (fgets(str, sizeof(str), fp) != 0)
        {
            int len = strlen(str);
            for (int i = 0; i <= len; i++)
            {
                if (str[i] == '/' && str[i + 1] == '/')
                {
                    commentCount += (strlen(str) - 2);
                    break;
                }
            }
        }
    
        fclose(fp);
    
        printf("%s: Number of characters contained in comments: %ld\n", file, commentCount);
    }
    
    int main(int argc, char **argv)
    {
        if (argc == 1)
            commentChars("/dev/stdin");
        else
        {
            for (int i = 1; i < argc; i++)
                commentChars(argv[i]);
        }
        return 0;
    }
    

    When run on the source code (ccc.c), it yields:

    ccc.c: Number of characters contained in comments: 58
    

    The comment isn't really complete (oops), but it serves to show what goes on. It counts the newline which fgets() preserves as part of the comment, though the // introducer is not counted.

    Dealing with /* comments is harder. You need to spot a slash followed by a star, and then read up to the next star slash character pair. This is probably more easily done using character by character input than line-by-line input; you will, at least, need to be able to interleave character analysis with line input.

    When you're ready for it, you can try this torture test on your program. It's what I use to check my comment stripper, SCC (which doesn't handle trigraphs — by conscious decision; if the source contains trigraphs, I have a trigraph remover which I use on the source first).

    /*
    @(#)File:            $RCSfile: scc.test,v $
    @(#)Version:         $Revision: 1.7 $
    @(#)Last changed:    $Date: 2013/09/09 14:06:33 $
    @(#)Purpose:         Test file for program SCC
    @(#)Author:          J Leffler
    */
    
    /*TABSTOP=4*/
    
    // -- C++ comment
    
    /*
    Multiline C-style comment
    #ifndef lint
    static const char sccs[] = "@(#)$Id: scc.test,v 1.7 2013/09/09 14:06:33 jleffler Exp $";
    #endif
    */
    
    /*
    Multi-line C-style comment
    with embedded /* in line %C% which should generate a warning
    if scc is run with the -w option
    Two comment starts /* embedded /* in line %C% should generate one warning
    */
    
    /* Comment */ Non-comment /* Comment Again */ Non-Comment Again /*
    Comment again on the next line */
    
    // A C++ comment with a C-style comment marker /* in the middle
    This is plain text under C++ (C99) commenting - but comment body otherwise
    // A C++ comment with a C-style comment end marker */ in the middle
    
    The following C-style comment end marker should generate a warning
    if scc is run with the -w option
    */
    Two of these */ generate */ one warning
    
    It is possible to have both warnings on a single line.
    Eg:
    */ /* /* */ */
    
    SCC has been trained to handle 'q' single quotes in most of
    the aberrant forms that can be used.  '\0', '\\', '\'', '\\
    n' (a valid variant on '\n'), because the backslash followed
    by newline is elided by the token scanning code in CPP before
    any other processing occurs.
    
    This is a legitimate equivalent to '\n' too: '\
    \n', again because the backslash/newline processing occurs early.
    
    The non-portable 'ab', '/*', '*/', '//' forms are handled OK too.
    
    The following quote should generate a warning from SCC; a
    compiler would not accept it.  '
    \n'
    
    " */ /* SCC has been trained to know about strings /* */ */"!
    "\"Double quotes embedded in strings, \\\" too\'!"
    "And \
    newlines in them"
    
    "And escaped double quotes at the end of a string\""
    
    aa '\\
    n' OK
    aa "\""
    aa "\
    \n"
    
    This is followed by C++/C99 comment number 1.
    // C++/C99 comment with \
    continuation character \
    on three source lines (this should not be seen with the -C flag)
    The C++/C99 comment number 1 has finished.
    
    This is followed by C++/C99 comment number 2.
    /\
    /\
    C++/C99 comment (this should not be seen with the -C flag)
    The C++/C99 comment number 2 has finished.
    
    This is followed by regular C comment number 1.
    /\
    *\
    Regular
    comment
    *\
    /
    The regular C comment number 1 has finished.
    
    /\
    \/ This is not a C++/C99 comment!
    
    This is followed by C++/C99 comment number 3.
    /\
    \
    \
    / But this is a C++/C99 comment!
    The C++/C99 comment number 3 has finished.
    
    /\
    \* This is not a C or C++  comment!
    
    This is followed by regular C comment number 2.
    /\
    */ This is a regular C comment *\
    but this is just a routine continuation *\
    and that was not the end either - but this is *\
    \
    /
    The regular C comment number 2 has finished.
    
    This is followed by regular C comment number 3.
    /\
    \
    \
    \
    * C comment */
    The regular C comment number 3 has finished.
    
    Note that \u1234 and \U0010FFF0 are legitimate Unicode characters
    (officially universal character names) that could appear in an
    id\u0065ntifier, a '\u0065' character constant, or in a "char\u0061cter\
     string".  Since these are mapped long after comments are eliminated,
    they cannot affect the interpretation of /* comments */.  In particular,
    none of \u0002A.  \U0000002A, \u002F and \U0000002F ever constitute part
    of a comment delimiter ('*' or '/').
    
    More double quoted string stuff:
    
        if (logtable_out)
        {
        sprintf(logtable_out,
            "insert into %s (bld_id, err_operation, err_expected, err_sql_stmt, err_sql_state)" 
            " values (\"%s\", \"%s\", \"%s\", \"", str_logtable, blade, operation, expected);
        /* watch out for embedded double quotes. */
        }
    
    
    /* Non-terminated C-style comment at the end of the file
    

提交回复
热议问题