有没有办法在EF中使用ExecuteStoreCommand执行多个insert语句?我正在使用StringBuilder生成多个insert语句。代码基本上看起来像这样
StringBuilder _saveReasonQuery = new StringBuilder();
public void ApplyRules()
{
using(var context = new SpecializedDBEntities())
{
foreach(var item in list)
{
//Do something
SaveReason(item, reasonId);
}
context.GetObjectContext().ExecuteStoreCommand(_saveReasonQuery.ToString());
ctx.SaveChanges();
}
}
//Inside Save Reason
public void SaveReason(...)
{
_saveReasonQuery.Append("INSERT INTO ORDER_IMPLEM_DTL_REASON ");
_saveReasonQuery.Append("(ORDER_IMPLEM_REASON_ID, SALES_ORDER_IMPLEM_DTL_ID, REASON_ID, REASON_STAT_ID, SALES_ORDER_BRDCST_ID, CREA_BY, CREA_DT)");
_saveReasonQuery.Append(" VALUES ");
_saveReasonQuery.Append(String.Format("('0', '{0}', '{1}', '{2}', '{3}', '{4}', TO_DATE('{5}', 'YY-MM-DD HH:mi'));",
orderSpot.SALES_ORDER_IMPLEM_DTL_ID, reasonId, Common.Convert_Int64(orderSpot.CUESHT_STAT_ID),
orderSpot.SALES_ORDER_DTL_BRDCST_ID, userContext.User.USER_CD, DateTime.Now.ToUniversalTime().ToString("yy-MM-dd hh:mm")));
}
但是,当EF执行查询时,会出现无效字符错误。我做了一些研究,发现是分号产生了这个错误。但是我不能删除分号,因为它是Oracle的“终结”。有什么建议吗?
另外,我直接针对数据库从StringBuilder执行生成的字符串,它可以插入5行。
答案 0 :(得分:1)
我通过反复试验找到了解决方案。
事实证明,你需要将多个插入物包含在BEGIN ... END中;声明,使分号能够被读作行尾分隔符。
我现在的代码如下:
StringBuilder _saveReasonQuery = new StringBuilder();
public void ApplyRules()
{
using(var context = new SpecializedDBEntities())
{
_saveReasonQuery.Append("BEGIN ");
foreach(var item in list)
{
//Do something
SaveReason(item, reasonId);
}
_saveReasonQuery.Append("END;");
context.GetObjectContext().ExecuteStoreCommand(_saveReasonQuery.ToString());
ctx.SaveChanges();
}
}
//Inside Save Reason
public void SaveReason(...)
{
_saveReasonQuery.Append("INSERT INTO ORDER_IMPLEM_DTL_REASON ");
_saveReasonQuery.Append("(ORDER_IMPLEM_REASON_ID, SALES_ORDER_IMPLEM_DTL_ID, REASON_ID, REASON_STAT_ID, SALES_ORDER_BRDCST_ID, CREA_BY, CREA_DT)");
_saveReasonQuery.Append(" VALUES ");
_saveReasonQuery.Append(String.Format("('0', '{0}', '{1}', '{2}', '{3}', '{4}', TO_DATE('{5}', 'YY-MM-DD HH:mi'));",
orderSpot.SALES_ORDER_IMPLEM_DTL_ID, reasonId, Common.Convert_Int64(orderSpot.CUESHT_STAT_ID),
orderSpot.SALES_ORDER_DTL_BRDCST_ID, userContext.User.USER_CD, DateTime.Now.ToUniversalTime().ToString("yy-MM-dd hh:mm")));
}
答案 1 :(得分:0)
ObjectContext.ExecuteStoreCommand
和其名称中隐含的较新DbContext.ExecuteSqlCommand
只会执行一个命令。我建议你在List<>
更新语句中构建命令,并在它们准备就绪后按顺序执行它们......
public void ApplyRules()
{
var saveQueries = new List<Tuple<string, object[]>>();
foreach(var item in list)
{
//Do something
saveQueries.Add(SaveQuery(reasonId, ...));
}
using(var context = new SpecializedDBEntities())
{
foreach(var saveQuery in saveQueries)
{
testContext.Database.ExecuteSqlCommand(saveQuery.Item1, saveQuery.Item2);
}
ctx.SaveChanges();
}
}
private Tuple<string, object[]> SaveQuery(int reasonId, ...)
{
const string query =
"INSERT INTO ORDER_IMPLEM_DTL_REASON " +
"(ORDER_IMPLEM_REASON_ID, SALES_ORDER_IMPLEM_DTL_ID, REASON_ID, REASON_STAT_ID, SALES_ORDER_BRDCST_ID, CREA_BY, CREA_DT)" +
" VALUES {0}, {0}, {1}, {2}, {3}, {4}, {5}";
object[] values = new object[]{
orderSpot.SALES_ORDER_IMPLEM_DTL_ID,
reasonId,
Common.Convert_Int64(orderSpot.CUESHT_STAT_ID),
orderSpot.SALES_ORDER_DTL_BRDCST_ID,
userContext.User.USER_CD,
DateTime.Now.ToUniversalTime().ToString("yy-MM-dd hh:mm")
};
return new Tuple<string, object[]>(query, values);
}