14

SQL Server の FILESTREAM 列に大量のデータを書き込む際に問題が発生しています。具体的には、1.5 ~ 2 GB 程度の小さなファイルは正常に処理されますが、サイズが 6 GB 以上になると、転送の終わりに向かって断続的 IOExceptionに「ハンドルが無効です」というメッセージが表示されます。.CopyTo()

データをチャンクで書き込むことを考えましたが、SQL Server はフィールドにデータを追加する前にフィールドのバッキング ファイルをコピーするため、大きなファイルのパフォーマンスが完全に損なわれます。

コードは次のとおりです。

public long AddFragment (string location , string description = null) 
{
    const string sql = 
        @"insert into [Fragment] ([Description],[Data]) " +
            "values (@description,0x); " +
         "select [Id], [Data].PathName(), " +
             "GET_FILESTREAM_TRANSACTION_CONTEXT() " +
         "from " +
             "[Fragment] " +
         "where " +
             "[Id] = SCOPE_IDENTITY();";

    long id;

    using (var scope = new TransactionScope(
        TransactionScopeOption.Required, 
            new TransactionOptions {
                Timeout = TimeSpan.FromDays(1)
            })) 
    {
        using (var connection = new SqlConnection(m_ConnectionString)) 
        {
            connection.Open();

            byte[] serverTx;
            string serverLocation;

            using (var command = new SqlCommand (sql, connection)) 
            {
                command.Parameters.Add("@description", 
                    SqlDbType.NVarChar).Value = description;

                using (var reader = command.ExecuteReader ()) 
                {
                    reader.Read();
                    id = reader.GetSqlInt64(0).Value;
                    serverLocation = reader.GetSqlString (1).Value;
                    serverTx = reader.GetSqlBinary (2).Value;
                }
            }

            using (var source = new FileStream(location, FileMode.Open, 
                FileAccess.Read, FileShare.Read, 4096, 
                FileOptions.SequentialScan))
            using (var target = new SqlFileStream(serverLocation, 
                serverTx, FileAccess.Write))
            {
                source.CopyTo ( target );
            }
        }

        scope.Complete();
    }

    return id;
}
4

2 に答える 2

4

FileStreamクラスの周りにBufferedStreamクラスを使用することをお勧めします。

また、 SqlFileStreamクラスでWriteTimeOutプロパティを設定していることを確認してください。

ここに、SqlFileStreamについてのすべてを説明する本当に良い投稿がありますhttp://www.simple-talk.com/sql/learn-sql-server/an-introduction-to-sql-server-filestream/

于 2012-10-09T23:04:28.210 に答える
2

いくつかのコメントで示唆されているように、問題はおそらくトランザクションのタイムアウトです。これを確認するには、SQL Server プロファイラーを実行し、トランザクションがロールバックされるのを監視します。

特に指定しない限り、machine.config のデフォルトの maxTimeout は 10 分で、コードでオーバーライドすることはできません。最大タイムアウトを増やすには、machine.config の構成設定に以下を追加します。

<system.transactions>
  <machineSettings maxTimeout="00:30:00" />
</system.transactions>
于 2014-03-31T12:43:48.343 に答える