概要
完成一个程序,作用是统计一个文件夹下面所有文件的代码行数。输入是一个文件夹的绝对路径,输出是代码行数。所以此程序的新特点有两个:
- 统计某一文件夹下的所有文件;
- 可以任意指定本机硬盘上任何位置的某一个文件夹。
前言
在上一篇随笔中熟悉了文件的基本操作。但仍然有改进的余地:统计特定文件时,还是需要手动输入文件名。如果文件数量很多怎么办?可不可以直接统计某个文件夹下面的所有文件的代码行数?今天解决的就是这个问题。
思考
我们已经解决了一个问题:写一个带有文件操作的C语言程序,输入文件名,输出此文件下的换行数。
考虑了一下现有的成果以及新增的目标,有需要更改的地方应该是输入,由文件名换成一个文件夹,然后通过程序扫描一遍这个文件夹下面所有的文件。然后读取所有的文件名,存放到数组里。再把字符串由fopen一个个调用,统计行数。细节差不多是这样,但总归要先扫描文件夹吧!
怎么扫描文件夹下面的文件?
搜索引擎搜索:C语言怎么扫描文件目录?
有找到几个链接,比较有用,可以参考:
比较可惜的是,上述网站提供的参考方法有使用函数opendir,查找了一下,发现这是一个在Linux系统下的函数。本人没有用过Linux。想了想,并没有直接开始搜索“安装Linux教程”,我心想:总有不用这个函数的方法吧!于是依旧在搜索引擎上不断地寻找。
找了接近一小时也无果,我打算换个问法。于是我在搜索引擎打入了:统计一个文件夹下文件
立刻映入我眼帘的就有一个黑科技,我惊呆了。
我立刻在桌面按照步骤进行实验。不得不说,看到结果,我非常满意。这不就我要的文件夹下面的所有文件名的字符串组合嘛!
立刻通过编程实现这一功能,效果拔群。
#include<stdio.h>
int main()
{
FILE *fp;
fp = fopen("TEXT.bat","w");
fputs("DIR *.* /B> LIST.TXT",fp);
fclose(fp);
return 0;
}
这里面有一行神秘的操作码,
DIR *.* /B> LIST.TXT
问题来了,这一行代码到底是什么呢?还是搜索引擎帮助你:)
大概知道了:
-
dir 显示目录中的文件和子目录列表;
-
/B 使用空格式(没有标题信息或摘要);
-
* . * (两个星号中间一个句点)就是说显示所有文件,文件名和扩展名没有限制。举例说明:如果是*.txt,则是显示所有扩展名为txt的文件;
-
最后面是字符串,很容易就知道是重定向输出的文件名。
好了。那么进行下一步的操作。我们考虑到一个文件夹只统计C语言代码的行数,所以这里我考虑只对.c后缀的文件进行这种操作。
#include<stdio.h>
int main()
{
FILE *fp;
fp = fopen("Text.bat","w");
fputs("DIR *.c /B> list.txt",fp);
fclose(fp);
fp=fopen("Text.bat","r");
fclose(fp);
return 0;
}
这儿遇到了第一个问题,运行可执行文件main.exe之后,文件夹内的文件情况如下图:
没有产生预期想要获得的文本文件list.txt。看来问题出在:
fp = fopen("Text1.bat","r");
这行语句上。因为,已经正常生成.bat文件,并且双击运行的话还是能生成文本文件,说明.bat文件没有问题,应该是打开文件的方式有问题。
立刻搜索:c语言打开bat文件。得到解决方法:
在程序中使用system() 函数。
假设bat文件的名称叫a.bat 即:
system("a.bat");
通过搜素来的方法,轻松解决这个问题。
#include<stdio.h>
int main()
{
FILE *fp;
fp = fopen("Text.bat","w");
fputs("DIR *.c /B> list.txt",fp);
fclose(fp);
system("Text.bat");
return 0;
}
编译后运行结果:
得到了我们想要的list.txt。也就是说.bat文件被正确地执行了。生成LIST的工作已经完成。现在已经有了存储文件名字符串的文本文件,下面只需要从这些文件中读取文件名,再调用上一篇随笔中实现的功能,即可实现这次需要添加的新功能了。
实现的简单的从list.txt中读取文件名。打开文件进行行数统计:
#include<stdio.h>
#include<Windows.h>
#include<string.h>
int main()
{
FILE *fp;
fp = fopen("Text.bat","w");
fputs("DIR *.c /B>list.txt",fp);
fclose(fp);
system("Text.bat");
/*~~~the end of creating file name list~~~*/
/*~~~the beginning of get .c file name from list~~~*/
static int count = 0;
FILE *fp1, *fp2;
fp1 = fopen("list.txt","r");
char s[100], singleline[1000];
while(fgets(s, 100, fp1))//get the name(one line) of a file from the list
{
// int len=strlen(s);
// if(s[len-1]=='
') s[len-1]=' '; what is this?
printf("%s: ",s);
fp2 = fopen(s,"r");//open the correct file, according to the file name
while(fgets(singleline, 1000, fp2))
{
count++;
}
printf("%d
", count);
fclose(fp2);
}
printf("
");
fclose(fp1);
system("pause");
return 0;
}
我们先不要管带有注释 what is this? 的那一行代码。
编译后执行,这次我在可执行文件.exe的目录下放了一些.c文件进行测试,看看运行结果如何:
我觉得代码意思都很明了了,然而执行结果还是失败的。
仔细观察结果,发现几个奇怪的点。
- .c文件是没问题的,统计代码行数的几行代码是由上一个版本引进的所以也应该是没问题的,但是结果却是0;
- 输出中有多余的空行。
找到问题所在了!由于fgets函数碰到' '是会停止输入的, 留在了缓冲区,下一次fgets时会导致文件读取失败(个人猜想,可能有误)所以我们要设法处理掉这个' '。
思考一下,数组s是存储文件名的。举"C++.txt"为例,假设造成读入失败的原因是字符串后面跟了一个
C | + | + | . | t | x | t |
---|