Загрузка больших файлов, сохраненных в базе данных
У нас есть много файлов, сохраненных как двоичные файлы в нашей базе данных SQL Server. Я сделал файл .ashx, который доставляет эти файлы пользователям. К сожалению, когда файлы становятся довольно большими, они не будут работать со следующей ошибкой:
Overflow or underflow in the arithmetic operation
Я предполагаю, что это исчерпывает память, поскольку я загружаю двоичный файл в байт [].
Итак, мой вопрос, как я могу сделать эту функциональность, читать в блоках (может быть?), Когда это из таблицы базы данных? Также кажется, что Response.TransmitFile () - хороший вариант, но, опять же, как это будет работать с базой данных?
DB.GetReposFile () в приведенном ниже коде получает файл из базы данных. Существуют различные поля для ввода: Имя файла, ContentType, метки даты и FileContent как varbinary.
Это моя функция, чтобы доставить файл:
context.Response.Clear();
try
{
if (!String.IsNullOrEmpty(context.Request.QueryString["id"]))
{
int id = Int32.Parse(context.Request.QueryString["id"]);
DataTable dtbl = DB.GetReposFile(id);
string FileName = dtbl.Rows[0]["FileName"].ToString();
string Extension = FileName.Substring(FileName.LastIndexOf('.')).ToLower();
context.Response.ContentType = ReturnExtension(Extension);
context.Response.AddHeader("Content-Disposition", "attachment; filename=" + FileName);
byte[] buffer = (byte[])dtbl.Rows[0]["FileContent"];
context.Response.OutputStream.Write(buffer, 0, buffer.Length);
}
else
{
context.Response.ContentType = "text/html";
context.Response.Write("<p>Need a valid id</p>");
}
}
catch (Exception ex)
{
context.Response.ContentType = "text/html";
context.Response.Write("<p>" + ex.ToString() + "</p>");
}
Update: Функция, с которой я закончил, это та, что указана ниже. Как упоминает Тим, DB.GetReposFileSize () просто получает длину данных контента. Я вызываю эту функцию в исходном коде вместо этих двух строк:
byte[] buffer = (byte[])dtbl.Rows[0]["FileContent"];
context.Response.OutputStream.Write(buffer, 0, buffer.Length);
Новая функция загрузки:
private void GetFileInChunks(HttpContext context, int ID)
{
//string path = @"c:\somefile.txt";
//FileInfo file = new FileInfo(path);
int len = DB.GetReposFileSize(ID);
context.Response.AppendHeader("content-length", len.ToString());
context.Response.Buffer = false;
//Stream outStream = (Stream)context.Response.OutputStream;
SqlConnection conn = null;
string strSQL = "select FileContent from LM_FileUploads where ID=@ID";
try
{
DB.OpenDB(ref conn, DB.DatabaseConnection.PDM);
SqlCommand cmd = new SqlCommand(strSQL, conn);
cmd.Parameters.AddWithValue("@ID", ID);
SqlDataReader reader = cmd.ExecuteReader(CommandBehavior.SequentialAccess);
reader.Read();
byte[] buffer = new byte[1024];
int bytes;
long offset = 0;
while ((bytes = (int)reader.GetBytes(0, offset, buffer, 0, buffer.Length)) > 0)
{
// TODO: do something with `bytes` bytes from `buffer`
context.Response.OutputStream.Write(buffer, 0, buffer.Length);
offset += bytes;
}
}
catch (Exception ex)
{
throw ex;
}
finally
{
DB.CloseDB(ref conn);
}
}