Files
SqlSugar/SqlSugar/Abstract/SqlBuilderProvider/Detail/UpdateBuilder.cs

269 lines
9.7 KiB
C#
Raw Normal View History

2017-01-07 21:54:51 +08:00
using System;
using System.Collections.Generic;
using System.Linq;
2017-05-19 15:26:06 +08:00
using System.Linq.Expressions;
2017-01-07 21:54:51 +08:00
using System.Text;
namespace SqlSugar
{
public class UpdateBuilder : IDMLBuilder
{
2017-05-19 15:26:06 +08:00
public UpdateBuilder()
{
this.sql = new StringBuilder();
this.DbColumnInfoList = new List<DbColumnInfo>();
2017-05-19 20:28:51 +08:00
this.SetValues = new List<KeyValuePair<string, string>>();
2017-05-21 01:14:55 +08:00
this.WhereValues = new List<string>();
2017-05-19 15:26:06 +08:00
}
public SqlSugarClient Context { get; set; }
public ILambdaExpressions LambdaExpressions { get; set; }
public ISqlBuilder Builder { get; set; }
public StringBuilder sql { get; set; }
public List<SugarParameter> Parameters { get; set; }
public string TableName { get; set; }
public string TableWithString { get; set; }
public List<DbColumnInfo> DbColumnInfoList { get; set; }
2017-05-20 18:48:14 +08:00
public List<string> WhereValues { get; set; }
2017-05-19 20:28:51 +08:00
public List<KeyValuePair<string, string>> SetValues { get; set; }
2017-05-19 15:26:06 +08:00
public bool IsUpdateNull { get; set; }
2017-05-21 01:40:06 +08:00
public List<string> PrimaryKeys { get; set; }
2017-05-21 11:16:11 +08:00
public bool IsOffIdentity { get; set; }
2017-05-14 12:59:15 +08:00
2017-05-19 15:26:06 +08:00
public virtual string SqlTemplate
2017-01-07 21:54:51 +08:00
{
get
{
2017-05-21 01:14:55 +08:00
return @"UPDATE {0} SET
{1} {2}";
2017-01-07 21:54:51 +08:00
}
}
2017-05-19 15:26:06 +08:00
public virtual string SqlTemplateBatch
2017-04-30 22:49:41 +08:00
{
get
{
2017-05-21 11:16:11 +08:00
return @"UPDATE S SET {0} FROM {1} S {2} INNER JOIN ";
}
}
public virtual string SqlTemplateJoin
{
get
{
return @" (
{0}
2017-05-21 01:14:55 +08:00
2017-05-21 11:16:11 +08:00
) T ON {1}
2017-05-22 02:21:43 +08:00
; ";
2017-04-30 22:49:41 +08:00
}
}
2017-05-21 01:14:55 +08:00
public virtual string SqlTemplateBatchSet
{
get
{
return "{0} AS {1}";
}
}
2017-05-19 15:26:06 +08:00
public virtual string SqlTemplateBatchSelect
2017-01-07 21:54:51 +08:00
{
get
{
2017-05-19 15:26:06 +08:00
return "{0} AS {1}";
2017-01-07 21:54:51 +08:00
}
2017-05-19 15:26:06 +08:00
}
2017-01-07 21:54:51 +08:00
2017-05-19 15:26:06 +08:00
public virtual string SqlTemplateBatchUnion
{
get
2017-01-07 21:54:51 +08:00
{
2017-05-21 11:16:11 +08:00
return "\t\t\r\nUNION ALL ";
2017-01-07 21:54:51 +08:00
}
}
2017-05-19 15:26:06 +08:00
public virtual void Clear()
{
}
public virtual string GetTableNameString
2017-01-07 21:54:51 +08:00
{
get
{
2017-05-19 15:26:06 +08:00
var result = Builder.GetTranslationTableName(TableName);
result += PubConst.Space;
if (this.TableWithString.IsValuable())
{
result += TableWithString + PubConst.Space;
}
return result;
2017-01-07 21:54:51 +08:00
}
}
2017-05-21 02:40:13 +08:00
public virtual string GetTableNameStringNoWith
{
get
{
var result = Builder.GetTranslationTableName(TableName);
return result;
}
}
2017-05-21 11:16:11 +08:00
2017-05-21 01:14:55 +08:00
public virtual ExpressionResult GetExpressionValue(Expression expression, ResolveExpressType resolveType, bool isMapping = true)
2017-05-19 15:26:06 +08:00
{
ILambdaExpressions resolveExpress = this.LambdaExpressions;
this.LambdaExpressions.Clear();
2017-05-19 20:28:51 +08:00
if (isMapping)
{
resolveExpress.MappingColumns = Context.MappingColumns;
resolveExpress.MappingTables = Context.MappingTables;
resolveExpress.IgnoreComumnList = Context.IgnoreColumns;
}
2017-05-19 15:26:06 +08:00
resolveExpress.Resolve(expression, resolveType);
this.Parameters = new List<SugarParameter>();
this.Parameters.AddRange(resolveExpress.Parameters);
var reval = resolveExpress.Result;
return reval;
}
public virtual string ToSqlString()
2017-01-07 21:54:51 +08:00
{
2017-05-19 15:26:06 +08:00
var groupList = DbColumnInfoList.GroupBy(it => it.TableId).ToList();
var isSingle = groupList.Count() == 1;
if (isSingle)
{
2017-05-21 11:16:11 +08:00
return ToSingleSqlString(groupList);
}
else
{
return TomultipleSqlString(groupList);
}
}
private string TomultipleSqlString(List<IGrouping<int, DbColumnInfo>> groupList)
{
Check.Exception(PrimaryKeys == null || PrimaryKeys.Count == 0, " Update List<T> need Primary key");
int pageSize = 200;
int pageIndex = 1;
int totalRecord = groupList.Count;
int pageCount = (totalRecord + pageSize - 1) / pageSize;
StringBuilder batchUpdateSql = new StringBuilder();
while (pageCount >= pageIndex)
{
StringBuilder updateTable = new StringBuilder();
string setValues = string.Join(",", groupList.First().Where(it=>it.IsPrimarykey==false&&(it.IsIdentity==false||(IsOffIdentity&&it.IsIdentity))).Select(it =>
2017-05-21 02:14:27 +08:00
{
if (SetValues.IsValuable())
{
var setValue = SetValues.Where(sv => sv.Key == Builder.GetTranslationColumnName(it.DbColumnName));
if (setValue != null && setValue.Any())
{
return setValue.First().Value;
}
}
2017-05-21 11:16:11 +08:00
var result = string.Format("S.{0}=T.{0}", Builder.GetTranslationColumnName(it.DbColumnName));
2017-05-21 02:14:27 +08:00
return result;
}));
2017-05-21 11:16:11 +08:00
batchUpdateSql.AppendFormat(SqlTemplateBatch.ToString(), setValues, GetTableNameStringNoWith, TableWithString);
int i = 0;
foreach (var columns in groupList.Skip((pageIndex - 1) * pageSize).Take(pageSize).ToList())
{
var isFirst = i == 0;
if (!isFirst)
{
updateTable.Append(SqlTemplateBatchUnion);
}
updateTable.Append("\r\n SELECT " + string.Join(",", columns.Select(it => string.Format(SqlTemplateBatchSelect, FormatValue(it.Value), it.DbColumnName))));
++i;
}
pageIndex++;
updateTable.Append("\r\n");
2017-05-21 01:14:55 +08:00
string whereString = null;
if (this.WhereValues.IsValuable())
{
foreach (var item in WhereValues)
{
var isFirst = whereString == null;
2017-05-21 11:16:11 +08:00
whereString += (isFirst ? null : " AND ");
2017-05-21 01:14:55 +08:00
whereString += item;
}
}
2017-05-21 01:40:06 +08:00
else if (PrimaryKeys.IsValuable())
2017-05-21 01:14:55 +08:00
{
foreach (var item in PrimaryKeys)
{
var isFirst = whereString == null;
2017-05-21 11:16:11 +08:00
whereString += (isFirst ? null : " AND ");
whereString += string.Format("S.{0}=T.{0}", Builder.GetTranslationColumnName(item));
2017-05-21 01:14:55 +08:00
}
}
2017-05-21 11:16:11 +08:00
batchUpdateSql.AppendFormat(SqlTemplateJoin, updateTable, whereString);
2017-05-19 15:26:06 +08:00
}
2017-05-21 11:16:11 +08:00
return batchUpdateSql.ToString();
}
private string ToSingleSqlString(List<IGrouping<int, DbColumnInfo>> groupList)
{
string columnsString = string.Join(",", groupList.First().Where(it => it.IsPrimarykey == false && (it.IsIdentity == false || (IsOffIdentity && it.IsIdentity))).Select(it =>
2017-05-19 15:26:06 +08:00
{
2017-05-21 11:16:11 +08:00
if (SetValues.IsValuable())
2017-05-21 02:40:13 +08:00
{
2017-05-21 11:16:11 +08:00
var setValue = SetValues.Where(sv => it.IsPrimarykey == false && (it.IsIdentity == false || (IsOffIdentity && it.IsIdentity))).Where(sv => sv.Key == Builder.GetTranslationColumnName(it.DbColumnName));
if (setValue != null && setValue.Any())
2017-05-21 02:40:13 +08:00
{
2017-05-21 11:16:11 +08:00
return setValue.First().Value;
2017-05-21 02:40:13 +08:00
}
2017-05-21 11:16:11 +08:00
}
var result = Builder.GetTranslationColumnName(it.DbColumnName) + "=" + this.Context.Ado.SqlParameterKeyWord + it.DbColumnName;
return result;
}));
string whereString = null;
if (this.WhereValues.IsValuable())
{
foreach (var item in WhereValues)
2017-05-19 15:26:06 +08:00
{
2017-05-21 11:16:11 +08:00
var isFirst = whereString == null;
whereString += (isFirst ? " WHERE " : " AND ");
whereString += item;
2017-05-19 15:26:06 +08:00
}
}
2017-05-21 11:16:11 +08:00
else if (PrimaryKeys.IsValuable())
{
foreach (var item in PrimaryKeys)
{
var isFirst = whereString == null;
whereString += (isFirst ? " WHERE " : " AND ");
whereString += Builder.GetTranslationColumnName(item) + "=" + this.Context.Ado.SqlParameterKeyWord + item;
}
}
return string.Format(SqlTemplate, GetTableNameString, columnsString, whereString);
2017-01-07 21:54:51 +08:00
}
2017-05-19 15:26:06 +08:00
public object FormatValue(object value)
2017-01-07 21:54:51 +08:00
{
2017-05-19 15:26:06 +08:00
if (value == null)
{
return "NULL";
}
else
{
var type = value.GetType();
if (type == PubConst.DateType)
{
2017-05-21 11:16:11 +08:00
var date = value.ObjToDate();
if (date < Convert.ToDateTime("1900-1-1")) {
date = Convert.ToDateTime("1900-1-1");
}
return "'" + date.ToString("yyyy-MM-dd hh:mm:ss.fff") + "'";
2017-05-19 15:26:06 +08:00
}
else if (type == PubConst.StringType || type == PubConst.ObjType)
{
return "N'" + value.ToString().ToSqlFilter() + "'";
}
else
{
return "N'" + value.ToString() + "'";
}
}
2017-01-07 21:54:51 +08:00
}
}
}