C#实现简单的文件夹对比程序(续)

上一篇文章实现了可视化显示不同路径下相同文件夹内的文件和子文件夹差异,文件少时已经够用,但当有成百上千个子文件夹时,人工逐个查看文件夹差异就比较困难,因此本文实现循环查找文件夹内容差异并导出为本地Excel文件的功能。
  本文使用队列保存要差异对比的文件夹及子文件夹路径,没有使用递归,代码逻辑看着稍微有点别扭,主要代码如下:

csharp 复制代码
List<DirectoryCompareRecord> results= new List<DirectoryCompareRecord>();

Queue<string> subDirs= new Queue<string>();

string leftDir=txtLeftDir.Text.TrimEnd('\\');
string rightDir=txtRightDir.Text.TrimEnd('\\');
string curLeftDir = string.Empty;
string curRightDir = string.Empty;

List<string> subLeftDirs = new List<string>();
List<string> subRightDirs = new List<string>();
List<string> subLeftFiles = new List<string>();
List<string> subRightFiles = new List<string>();

string tmpFileName = string.Empty;
string tmpDirName = string.Empty;
string tmp = string.Empty;

FileInfo leftFile = null;
FileInfo rightFile = null;

do
{                
    curLeftDir = Path.Combine(leftDir, tmp);
    curRightDir = Path.Combine(rightDir, tmp);

    subLeftDirs.Clear();
    subRightDirs.Clear();
    subLeftFiles.Clear();
    subRightFiles.Clear();
    

    subLeftFiles.AddRange(Directory.GetFiles(curLeftDir));
    subRightFiles.AddRange(Directory.GetFiles(curRightDir));

    subLeftFiles.Sort();
    subRightFiles.Sort();

    foreach (string file in subLeftFiles)
    {
        tmpFileName = Path.GetFileName(file);
        tmpDirName = Path.Combine(curRightDir, tmpFileName);

        if (!subRightFiles.Contains(tmpDirName))
        {
            DirectoryCompareRecord record = new DirectoryCompareRecord();
            record.ObjectType = "文件";
            record.ResultType = "文件缺失";
            record.LeftDirectory = curLeftDir;
            record.LeftFile = tmpFileName;

            results.Add(record);
        }
        else
        {
            leftFile=new FileInfo(file);
            rightFile=new FileInfo(tmpDirName);

            if (leftFile.LastWriteTime != rightFile.LastWriteTime)
            {
                DirectoryCompareRecord record = new DirectoryCompareRecord();
                record.ObjectType = "文件";
                record.ResultType = "文件有差异";
                record.LeftDirectory = curLeftDir;
                record.LeftFile = tmpFileName;
                record.LeftFileLastModifyDate = leftFile.LastWriteTime.ToString();
                record.RightDirectory = curRightDir;
                record.RightFile = tmpFileName;
                record.RightFileLastModifyDate = rightFile.LastWriteTime.ToString();

                results.Add(record);
            }
            else
            {
                subRightFiles.Remove(tmpDirName);
            }                        
        }
    }

    foreach (string file in subRightFiles)
    {
        tmpFileName = Path.GetFileName(file);
        tmpDirName = Path.Combine(curLeftDir, tmpFileName);

        if (!subLeftFiles.Contains(tmpDirName))
        {
            DirectoryCompareRecord record = new DirectoryCompareRecord();
            record.ObjectType = "文件";
            record.ResultType = "文件缺失";
            record.RightDirectory = curRightDir;
            record.RightFile = tmpFileName;

            results.Add(record);
        }
    }

    subLeftDirs.AddRange(Directory.GetDirectories(curLeftDir));
    subRightDirs.AddRange(Directory.GetDirectories(curRightDir));

    subLeftDirs.Sort();
    subRightDirs.Sort();

    foreach (string dir in subLeftDirs)
    {
        tmpFileName = Path.GetFileName(dir);
        tmpDirName = Path.Combine(curRightDir, tmpFileName);

        if (!subRightDirs.Contains(tmpDirName))
        {
            DirectoryCompareRecord record = new DirectoryCompareRecord();
            record.ObjectType = "文件夹";
            record.ResultType = "文件夹缺失";
            record.LeftDirectory = curLeftDir;
            record.LeftFile = tmpFileName;

            results.Add(record);
        }
        else
        {
            subDirs.Enqueue(dir.Substring(leftDir.Length+1));
            subRightFiles.Remove(tmpDirName);                        
        }
    }

    foreach (string file in subRightDirs)
    {
        tmpFileName = Path.GetFileName(file);
        tmpDirName = Path.Combine(curLeftDir, tmpFileName);

        if (!subLeftDirs.Contains(tmpDirName))
        {
            DirectoryCompareRecord record = new DirectoryCompareRecord();
            record.ObjectType = "文件夹";
            record.ResultType = "文件夹缺失";
            record.RightDirectory = curRightDir;
            record.RightFile = tmpFileName;

            results.Add(record);
        }
    }

    if(subDirs.Count>0)
    {
        tmp=subDirs.Dequeue();
    }
    
} while (subDirs.Count >0);

对比结果导出Excel文件采用Sylvan.Data.Excel库,代码实现也比较简单,如下所示:

csharp 复制代码
SaveFileDialog saveFileDialog = new SaveFileDialog();
saveFileDialog.Filter = "XLSX|*.xlsx";
if (saveFileDialog.ShowDialog() != DialogResult.OK)
{
    return;
}

DbDataReader reader = ObjectDataReader.Create(results);
using (var w = ExcelDataWriter.Create(saveFileDialog.FileName))
{
    w.Write(reader);
}

最后是程序运行效果及导出文件的效果,代码地址:https://github.com/guochao2299/DirCompare

参考文献:

[1]https://blog.csdn.net/sd7o95o/article/details/142976888

[2]https://github.com/MarkPflug/Sylvan.Data.Excel

相关推荐
gu2011 分钟前
c#编程:学习Linq,重几个简单示例开始
开发语言·学习·c#·linq
pchmi4 小时前
CNN常用卷积核
深度学习·神经网络·机器学习·cnn·c#
yuanpan5 小时前
23种设计模式之《组合模式(Composite)》在c#中的应用及理解
开发语言·设计模式·c#·组合模式
滴_咕噜咕噜6 小时前
C#基础总结:常用的数据结构
开发语言·数据结构·c#
万兴丶9 小时前
Unity 适用于单机游戏的红点系统(前缀树 | 数据结构 | 设计模式 | 算法 | 含源码)
数据结构·unity·设计模式·c#
程序猿多布11 小时前
C#设计模式 学习笔记
设计模式·c#
软件黑马王子17 小时前
Unity游戏制作中的C#基础(5)条件语句和循环语句知识点全解析
游戏·unity·c#
shepherd枸杞泡茶17 小时前
第3章 3.3日志 .NET Core日志 NLog使用教程
c#·asp.net·.net·.netcore
Aimeast1 天前
关于选择最佳.NET Core SSH服务器库的全面分析
c#·ssh
蒋劲豪1 天前
WPF项目暴露WebApi接口;WinForm项目暴露WebApi接口;C#项目暴露WebApi接口;
开发语言·c#·wpf