我有一个简单的控制台应用程序,执行一系列存储过程大约150,000次。它的目的是将初始数据库数据导入到数据库的新实例中。
它在工作中运行正常,但在家里我正在尝试使用Sql Server Express,一切看起来都很好,但完全没有完成25.76%sql连接进入破坏状态并生成一个InvalidOperation异常,说明连接已断开并且不能用于执行下一个命令。
我理解这意味着什么,但我无法理解为什么它会被破坏以及为什么它会在每次尝试运行控制台应用程序的同时发生。
它每次都以25.76%完成进入破坏状态,总是在那个位置,并且总是在执行的同一行SQL上。
发生异常时正在执行的行如下:
EXEC [geo].[addUpdateRegion] @countryCode = N'CG', @regionCode = N'08', @regionName = N'Plateaux', @initData = 1
它会导致InvalidOperationException:BeginExecuteNonQuery需要一个开放且可用的连接。连接的当前状态:已损坏。“
我很困惑,因为我完全禁用了超时。所以我想知道SQLExpress是否可以限制你可以在同一个连接上执行多少命令?
这就是代码的样子:
using log4net;
using System;
using System.Collections.Generic;
using System.Configuration;
using System.Data.SqlClient;
using System.IO;
using System.Linq;
using System.Text;
using System.Text.RegularExpressions;
using System.Threading;
using System.Threading.Tasks;
namespace XYZ.SetupData
{
class Program
{
static ILog logger = null;
static string connectionString = null;
static SqlConnection conn = null;
static double lineCount = 1;
static double lineIndex = 1;
static Program()
{
logger = LogManager.GetLogger(typeof(Program));
log4net.Config.XmlConfigurator.Configure();
var connSettings = ConfigurationManager.ConnectionStrings["theDb"];
if (connSettings == null || string.IsNullOrEmpty(connSettings.ConnectionString))
throw new ConfigurationErrorsException("connectionString theDbwas not found or is empty.");
connectionString = connSettings.ConnectionString;
conn = new SqlConnection(connectionString);
conn.Open();
}
static void Main(string[] args)
{
var baseDir = new DirectoryInfo(AppDomain.CurrentDomain.BaseDirectory + "sqlFiles");
var files = baseDir.GetFiles("*.sql", SearchOption.TopDirectoryOnly).Select(file => file.FullName).ToArray();
lineCount = (double)GetTotalFileLines(files);
foreach (var file in files)
{
ExecuteFileLines(file);
}
Console.ReadKey(true);
}
#region Utility
static void WriteProgress(string command)
{
double percent = (lineIndex / lineCount) * 100;
Console.Write("\r Percent Complete: {0}% ", Math.Round(percent, 2));
++lineIndex;
}
static int GetTotalFileLines(params string[] fileNames)
{
int total = 0;
foreach (var fileName in fileNames)
total += (File.ReadLines(fileName).Where(line => !string.IsNullOrEmpty(line) && !line.StartsWith("--")).Count());
return total;
}
static void ExecuteFileLines(string fileName)
{
TryRun(() =>
{
if (string.IsNullOrEmpty(fileName))
throw new ArgumentNullException("fileName");
if (!File.Exists(fileName))
throw new FileNotFoundException("file: " + fileName + " was not found!");
IEnumerable<string> fileLines = File.ReadLines(fileName).Where(line => !string.IsNullOrEmpty(line) && !line.StartsWith("--"));
LogInfo("--Staring Execution: " + fileName);
foreach (var line in fileLines)
{
RunSqlConnection(conn =>
{
LogInfo("RUNNING | " + line);
WriteProgress(line);
try
{
SqlCommand cmd = new SqlCommand(line, conn);
cmd.BeginExecuteNonQuery();
}
catch (Exception ex)
{
int i = 0;
}
});
}
});
}
static void RunSqlConnection(Action<SqlConnection> callBack)
{
try
{
callBack(conn);
}
catch (Exception ex)
{
LogError(ex);
throw ex;
}
}
static void TryRun(Action callBack)
{
try
{
callBack();
}
catch (Exception ex)
{
LogError(ex);
throw ex;
}
}
static void LogError(Exception ex)
{
logger.Error(ex.Message, ex);
Console.WriteLine(ex.ToString());
}
static void LogInfo(string message, params object[] parameters)
{
logger.Info(string.Format(message, parameters));
}
static void LogDebug(string message, params object[] parameters)
{
logger.Debug(string.Format(message, parameters));
}
#endregion
}
}
答案 0 :(得分:3)
您正在尝试处理多个线程上的工作。有一个错误是您同时使用相同的连接。这是不允许的。由于这是竞争条件,任何事情都可能在内部发生。
我不确定你为什么要使用APM模式,因为它已经过时了。我认为PLINQ plus同步IO加上每个工作项的一个连接实际上很适合您的场景。
答案 1 :(得分:1)
您应该检查Sql Server Editions的比较。
对于Sql Express Edition,您可能会达到每GB实例内存限制1 GB。