![](/d/20211017/4495139fdc1e65c7c1bd50c706a04998.gif)
以上列表中的文件并不是來自于某個文件夾中的所有jpg文件,而是來自于
![](/d/20211017/2d9117a672764ce63a3a4479f1dad0c7.gif)
這個文件。
將多個文件合并為一個文件在許多應(yīng)用領(lǐng)域都十分有用。親自實現(xiàn)這樣一個程序一定不但過癮且在許多時候可以幫助我們構(gòu)建更高效的程序。這里我做了一個方案例分享給大家。
由于合并后的文件就像一個包裹,所以下文中都把這樣的文件稱為“包文件”
主構(gòu)思:
要把多個文件合并成一個包文件,還要可以區(qū)分其中的某個文件并提取出來。我們需要知道文件的名稱和這個文件在包文件中的位置及長度,也就是所謂的地址偏移。
由于包文件常常會比較大,所以不應(yīng)該讓它的內(nèi)容常駐于內(nèi)存,只應(yīng)該需要某部分的時候再從包文件中提取。
我是這樣做的:
![](/d/20211017/8ebc82d29897ffaa224a37f40c1b8703.gif)
一個管理器類,提供一些外圍的方法
_pathList用于存放要添加到包文件的文件路徑,通過調(diào)用AddSourceFile()方法添加
_pf 是具體的包文件,通過LoadPackFile() 生成實例,通過CurrentPackFile屬性返回
Build方法用于生成包文件
![](/d/20211017/e774e519ea1a0c56ccd1e506958fa9e5.gif)
PackFile類作為PackFileManager的嵌套類,它提供包文件的屬性和施工細(xì)節(jié)。
好了,我們先來看看PackFileManager.Build()方法
復(fù)制代碼 代碼如下:
public void Build(string path)
{
using (FileStream fs = new FileStream(path, FileMode.Create, FileAccess.Write))
{
BinaryWriter bw = new BinaryWriter(fs);
bw.Write("PackFile");
bw.Write(this._pathList.Count);
foreach (string f in this._pathList)
{
FileInfo fi = new FileInfo(f);
bw.Write(fi.Length);
fi = null;
}
foreach (string f in this._pathList)
{
bw.Write(Path.GetFileName(f));
}
foreach (string f in this._pathList)
{
bw.Write(File.ReadAllBytes(f));
bw.Flush();
}
}
}
1. 先寫個“PackFile”字符串到文件頭
2. 把以Int32為類型的,要輸出到包文件中的文件數(shù)量寫入
3. 把以long為類型的,要輸出到包文件中的每個文件的長度寫入。
4. 再把每個文件名寫入
5. 最后寫入每個文件的實體內(nèi)容。
由于在寫或讀時不頻繁在Write方法或ReadXXX方法的不同版本間頻繁切換,所以我想這樣組織文件結(jié)構(gòu)可以更高效一些。
疑問來了。在寫入文件名的時候,我們使用bw.Write(Path.GetFileName(f));
調(diào)用了BinaryWriter.Write(string value),傳入的是字符串,那么在讀取的時候要調(diào)用BinaryReader.ReadString()。這時它是如何區(qū)分兩個字符串邊界的。還好,Write方法會先將字符串長度作為一個四字節(jié)無符號整數(shù)寫入,于是在用BinaryReader.ReadString()的時候它會根據(jù)這個值來讀取特定長度的值,并理解為字符串。
這里列出幾個重要方法:
復(fù)制代碼 代碼如下:
PackFileManager的LoadPackFile方法
public void LoadPackFile(string path)
{
if (!File.Exists(path))
{
throw new FileNotFoundException(path);
}
if (_pf != null)
{
_pf.Close();
_pf = null;
}
FileStream fs = new FileStream(path, FileMode.Open, FileAccess.Read);
BinaryReader br = new BinaryReader(fs);
if (br.ReadString() != "PackFile")
{
throw new InvalidCoalescentFileException("該文件不是有效的包文件");
}
this._pf = new PackFile(fs,br);
}
此時,我們在生成時寫入的字符串"PackFile" 就有了明確的功能
PackFile的構(gòu)造函數(shù)
復(fù)制代碼 代碼如下:
internal PackFile(FileStream srcFile,BinaryReader br)
{
this._sourceFile = srcFile;
_br = br;
this._fileCount = _br.ReadInt32();//取文件數(shù)
for (int i = 1; i = _fileCount; i++)
{
this._fileLengthList.Add(_br.ReadInt64());
}
for (int i = 1; i = _fileCount; i++)
{
this._shortNameList.Add(_br.ReadString());
}
this._contentStartPos = _sourceFile.Position;//設(shè)置實體文件總起始位置
}
PackFile.GetBytes()
復(fù)制代碼 代碼如下:
public byte[] GetBytes(int index)
{
long startPos = this._contentStartPos;
for (int i = 0; i index; i++)
{
startPos += this._fileLengthList[i];
}
_sourceFile.Position = startPos; //設(shè)置某文件內(nèi)容的起始位置
return _br.ReadBytes((int)_fileLengthList[index]);
}
這只是一個草案,我們還可以加入壓縮、或是像ZIP文件那樣的嵌套文件夾功能,改進(jìn)后的代碼別忘與我分享哦。
您可能感興趣的文章:- asp.net 合并GridView中某列相同信息的行(單元格)
- asp.net 未能加載文件或程序集“XXX”或它的某一個依賴項。試圖加載格式不正確的程序。
- asp.net中rdlc 合并行的方法
- 未能加載文件或程序集“AspNetPager”或它的某一個依賴項。拒絕訪問
- asp.net中GridView和DataGrid相同列合并實現(xiàn)代碼
- ASP.NET GridView 實現(xiàn)課程表顯示(動態(tài)合并單元格)實現(xiàn)步驟
- PowerShell入門教程之訪問.Net程序集、COM和WMI實例
- SQL Server中調(diào)用C#類中的方法實例(使用.NET程序集)
- 如何合并多個 .NET 程序集