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

相关推荐
偶尔的鼠标人42 分钟前
Avalonia DataGrid 控件的LostFocus事件会多次触发
开发语言·c#
ytttr8731 小时前
C# 仿QQ聊天功能实现 (SQL Server数据库)
数据库·oracle·c#
future_studio3 小时前
聊聊 Unity(小白专享、C# 小程序 之 图片播放器)
unity·小程序·c#
c#上位机9 小时前
wpf中Grid的MouseDown 事件无法触发的原因
c#·wpf
CodeCraft Studio10 小时前
国产化PDF处理控件Spire.PDF教程:如何在 C# 中从 HTML 和 PDF 模板生成 PDF
pdf·c#·html·.net·spire.pdf·pdf文档开发·html创建模板pdf
ysdysyn11 小时前
.NET 10深度解析:性能革新与开发生态的全新篇章
c#·.net
L X..14 小时前
Unity 光照贴图异常修复笔记
unity·c#·游戏引擎
reasonsummer16 小时前
【办公类-115-06】20250920职称资料上传04——docx复制、docx转PDF(课程表11个)
开发语言·windows·python·c#
William_cl1 天前
一、前置基础(MVC学习前提)_核心特性_【C# 泛型入门】为什么说 List<T>是程序员的 “万能收纳盒“?避坑指南在此
学习·c#·mvc
c#上位机1 天前
wpf之命令
c#·wpf