wc.exe是一个常见的工具,它能统计文本文件的字符数、单词数和行数。这个项目要求写一个命令行程序,模仿已有的wc.exe的功能,并加以扩充,给出某程序设计源语言文件的字符数、单词数和行数。
给实现一个统计程序,它能正确统计程序文件的字符数、单词数、行数,以及其他扩展功能,并能够快速的处理多个文件。
就这样,代码如下:
#include <stdlib.h>
#include <string.h>
#include <stdio.h>
#include <ctype.h>
#include <io.h>
void open_file(int a,int b,int c,int d,char data[]);
FILE *letter(FILE *fp1,char ch1);
FILE *message(FILE *fp1,char ch2);
FILE *null(FILE *fp1,char ch3);
void printDir( const char* path );
char str1[],str3[];
int word_N=,line_N=,ch_N=;//记录单词数、行数、字符数
int null_h=,code_h=,mess_h=;//记录空行、代码行、注释行
int s_ch=,s_word=,s_line=,hang_a=,s_s=;//记录输入的查询数据
int main ()
{
char s[];
gets(s);
while()
{
memset(str1, , sizeof(str1));
memset(str3, , sizeof(str3));
word_N=,line_N=,ch_N=,null_h=,code_h=,mess_h=;
s_ch=,s_word=,s_line=,hang_a=,s_s=;
int length_s=;
length_s=strlen(s);
int i=;
for(i=;i<length_s;++i)
{
if((s[i]=='-')&&(s[i+]=='c'))
{
s_ch=;
}
if((s[i]=='-')&&(s[i+]=='w'))
{
s_word=;
}
if((s[i]=='-')&&(s[i+]=='l'))
{
s_line=;
}
if((s[i]=='-')&&(s[i+]=='a'))
{
hang_a=;
}
if((s[i]=='-')&&(s[i+]=='s'))
{
s_s=;
}
if(s[i]=='F'){
strncpy(str1,s+i,length_s-i);
strcpy(str3,str1);
break;
}//获取文件名
}
if(s_s==)
{
s_ch=;
s_word=;
s_line=;
hang_a=;
printDir(str1);
}
else open_file(s_ch,s_word,s_line,hang_a,str1); printf("\n");
gets(s);
}
return ;
}
void printDir( const char* path )
{
struct _finddata_t data; long hnd = _findfirst( path, &data ); // 查找文件名与正则表达式chRE的匹配第一个文件
if ( hnd < )
{
perror( path );
}
int nRet = (hnd < ) ? - : ;
while ( nRet >= )
{
if ( data.attrib == _A_SUBDIR ) // 如果是目录
{
printf("文件名:[%s]*\n", data.name );
open_file(s_ch,s_word,s_line,hang_a,data.name );
}
else
{
printf("文件名:[%s]\n", data.name );
open_file(s_ch,s_word,s_line,hang_a,data.name);
}
nRet = _findnext( hnd, &data );
}
_findclose( hnd ); // 关闭当前句柄
}
void open_file(int a,int b,int c,int d,char data[])
{
word_N=,line_N=,ch_N=,null_h=,code_h=,mess_h=;
int i=,length_str3=,j=;
length_str3=strlen(str3);
char str2[];
memset(str1, , sizeof(str1));
strcpy(str1,str3);
for(i=;i<length_str3;++i)
{
if(str3[]==data[])
{
strcpy(str2,str3);
break;
}//当未输入指令-s时,直接打开str1
else if(str3[i]=='*')
{
for(;i<length_str3;++i)
str1[i]='\0';
strcat(str1,data);
printf("文件位置:%s\n", str1);
strcpy(str2,str1);
break;
}
}
FILE *fp;
fp=fopen(str2,"r");
if(fp==NULL){
printf("the file not found\n");
exit();
}//打开地址文件
char ch;
ch=fgetc(fp);
ch_N++;
if(ch==EOF)
ch_N--;
while(ch!=EOF)
{
switch(ch)
{
case ' ':fp=null(fp,ch);break;
case '{':fp=null(fp,ch);break;
case '}':fp=null(fp,ch);break;
case '/': fp=message(fp,ch);break;
default :fp=letter(fp,ch);break;
}
ch=fgetc(fp);
ch_N++;
if(ch==EOF)
ch_N--;
} if(a==)
printf("字符数:%d\n",ch_N);
if(b==)
printf("单词数:%d\n",word_N);
if(c==)
printf("总行数:%d\n",line_N);
if(d==)
{
printf("代码行:%d\n",code_h);
printf("注释行:%d\n",mess_h);
printf("空行:%d\n",null_h);
}
}
FILE *message(FILE *fp1,char ch2)
{
char ch_2;
ch_2=fgetc(fp1);
ch_N++;
if(ch_2=='/')
{
ch_2=fgetc(fp1);
ch_N++;
while((ch_2!='\n')&&(ch_2!=EOF))
{
ch_2=fgetc(fp1);
ch_N++;
}//换行时输出
ch_2=fgetc(fp1);
ch_N++;
mess_h++;
line_N++;//注释行+1,总行数+1,获取\n之后的下一个字符;
if(ch_2==EOF)
{
ch_N--;
return fp1;
}
}
fseek(fp1,-,);
ch_N--;
return fp1;
}//注释函数
FILE *null(FILE *fp1,char ch3)
{
char ch_3;
ch_3=ch3; if((ch_3=='{')||(ch_3=='}'))
{
ch_3=fgetc(fp1);
ch_N++;
if(ch_3=='\n')
{
null_h++;
line_N++;
ch_3=fgetc(fp1);
ch_N++;
}
}
if(ch_3==EOF)
{
ch_N--;
null_h++;
line_N++;
return fp1;
} while((ch_3==' '))
{
ch_3=fgetc(fp1);
ch_N++;
}
if(ch_3=='\n')
{
null_h++;
line_N++;
ch_3=fgetc(fp1);
ch_N++;
}
fseek(fp1,-,);
ch_N--;
return fp1;
}//空行函数
FILE *letter(FILE *fp1,char ch1)
{
char ch_1;
ch_1=ch1;
if(isalpha(ch_1))
{
ch_1=fgetc(fp1);
ch_N++;
while((isalpha(ch_1))||(isdigit(ch_1)))
{
ch_1=fgetc(fp1);
ch_N++;
}
word_N++;
}//字母型单词
else if(isdigit(ch_1))
{
ch_1=fgetc(fp1);
ch_N++;
while((isdigit(ch_1))||(isalpha(ch_1)))
{
ch_1=fgetc(fp1);
ch_N++;
}
word_N++;}//数字型单词
else if(ch_1=='\n')
{
code_h++;
line_N++;
ch_1=fgetc(fp1);
ch_N++;
}
else if(ch_1==EOF)
{
ch_N--;
return fp1;
}
else
{
ch_1=fgetc(fp1);
ch_N++;
}
fseek(fp1,-,);
ch_N--;
return fp1;
}