mirror of
https://github.com/nsnail/FreeSql.git
synced 2025-04-22 18:52:50 +08:00
- 增加 低代码扩展包 FreeSql.Extensions.ZoreEntity;
This commit is contained in:
parent
2368377b0d
commit
b814921a6a
@ -0,0 +1,43 @@
|
||||
<Project Sdk="Microsoft.NET.Sdk">
|
||||
|
||||
<PropertyGroup>
|
||||
<TargetFrameworks>netstandard2.0;net45;net40</TargetFrameworks>
|
||||
<GeneratePackageOnBuild>true</GeneratePackageOnBuild>
|
||||
<Authors>FreeSql;ncc;YeXiangQin</Authors>
|
||||
<Description>FreeSql 扩展包,实现 低代码、零实体类型,并且支持导航属性,级联保存 等功能.</Description>
|
||||
<PackageProjectUrl>https://github.com/2881099/FreeSql</PackageProjectUrl>
|
||||
<RepositoryUrl>https://github.com/2881099/FreeSql</RepositoryUrl>
|
||||
<RepositoryType>git</RepositoryType>
|
||||
<PackageLicenseExpression>MIT</PackageLicenseExpression>
|
||||
<PackageTags>FreeSql;ORM;低代码;</PackageTags>
|
||||
<PackageId>$(AssemblyName)</PackageId>
|
||||
<PackageIcon>logo.png</PackageIcon>
|
||||
<Title>$(AssemblyName)</Title>
|
||||
<IsPackable>true</IsPackable>
|
||||
<GenerateAssemblyInfo>true</GenerateAssemblyInfo>
|
||||
<SignAssembly>true</SignAssembly>
|
||||
<AssemblyOriginatorKeyFile>key.snk</AssemblyOriginatorKeyFile>
|
||||
<DelaySign>false</DelaySign>
|
||||
<Version>3.2.806-preview20231214</Version>
|
||||
<PackageReadmeFile>readme.md</PackageReadmeFile>
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<None Include="../../readme.md" Pack="true" PackagePath="\"/>
|
||||
<None Include="../../logo.png" Pack="true" PackagePath="\" />
|
||||
</ItemGroup>
|
||||
|
||||
<PropertyGroup Condition="'$(Configuration)|$(TargetFramework)|$(Platform)'=='Debug|netstandard2.0|AnyCPU'">
|
||||
<DocumentationFile>FreeSql.Extensions.ZoreEntity.xml</DocumentationFile>
|
||||
<WarningLevel>3</WarningLevel>
|
||||
</PropertyGroup>
|
||||
|
||||
<PropertyGroup Condition="'$(TargetFramework)' == 'net40'">
|
||||
<DefineConstants>net40</DefineConstants>
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<ProjectReference Include="..\..\FreeSql\FreeSql.csproj" />
|
||||
</ItemGroup>
|
||||
|
||||
</Project>
|
@ -0,0 +1,67 @@
|
||||
<?xml version="1.0"?>
|
||||
<doc>
|
||||
<assembly>
|
||||
<name>FreeSql.Extensions.ZoreEntity</name>
|
||||
</assembly>
|
||||
<members>
|
||||
<member name="P:FreeSql.Extensions.ZoreEntity.ZoreDbContext.Select">
|
||||
<summary>
|
||||
【有状态管理】自动 Include 查询
|
||||
</summary>
|
||||
</member>
|
||||
<member name="M:FreeSql.Extensions.ZoreEntity.ZoreDbContext.SelectNoTracking(System.String)">
|
||||
<summary>
|
||||
【无状态管理】指定表查询
|
||||
</summary>
|
||||
</member>
|
||||
<member name="P:FreeSql.Extensions.ZoreEntity.ZoreDbContext.ChangeReport.ChangeInfo.BeforeObject">
|
||||
<summary>
|
||||
Type = Update 的时候,获取更新之前的对象
|
||||
</summary>
|
||||
</member>
|
||||
<member name="P:FreeSql.Extensions.ZoreEntity.ZoreDbContext.ChangeReport.Report">
|
||||
<summary>
|
||||
实体变化记录
|
||||
</summary>
|
||||
</member>
|
||||
<member name="P:FreeSql.Extensions.ZoreEntity.ZoreDbContext.ChangeReport.OnChange">
|
||||
<summary>
|
||||
实体变化事件
|
||||
</summary>
|
||||
</member>
|
||||
<member name="M:FreeSql.Extensions.ZoreEntity.ZoreDbContext.SelectImpl.LeftJoin(System.String,System.String[])">
|
||||
<summary>
|
||||
举例1:LeftJoin("table1", "id", "user.id") -> LEFT JOIN [table1] b ON b.[id] = a.[id]<para></para>
|
||||
举例2:LeftJoin("table1", "id", "user.id", "xid", "user.xid") -> LEFT JOIN [table1] b ON b.[id] = [a].id] AND b.[xid] = a.[xid]
|
||||
</summary>
|
||||
</member>
|
||||
<member name="M:FreeSql.Extensions.ZoreEntity.ZoreDbContext.SelectImpl.InnerJoin(System.String,System.String[])">
|
||||
<summary>
|
||||
举例1:InnerJoin("table1", "id", "user.id") -> INNER JOIN [table1] b ON b.[id] = a.[id]<para></para>
|
||||
举例2:InnerJoin("table1", "id", "user.id", "xid", "user.xid") -> INNER JOIN [table1] b ON b.[id] = [a].id] AND b.[xid] = a.[xid]
|
||||
</summary>
|
||||
</member>
|
||||
<member name="M:FreeSql.Extensions.ZoreEntity.ZoreDbContext.SelectImpl.RightJoin(System.String,System.String[])">
|
||||
<summary>
|
||||
举例1:RightJoin("table1", "id", "user.id") -> RIGTH JOIN [table1] b ON b.[id] = a.[id]<para></para>
|
||||
举例2:RightJoin("table1", "id", "user.id", "xid", "user.xid") -> RIGTH JOIN [table1] b ON b.[id] = [a].id] AND b.[xid] = a.[xid]
|
||||
</summary>
|
||||
</member>
|
||||
<member name="M:FreeSql.Extensions.ZoreEntity.ZoreDbContext.SelectImpl.Where(System.Collections.Generic.IEnumerable{System.Collections.Generic.Dictionary{System.String,System.Object}})">
|
||||
<summary>
|
||||
WHERE [Id] IN (...)
|
||||
</summary>
|
||||
</member>
|
||||
<member name="M:FreeSql.Extensions.ZoreEntity.ZoreDbContext.SelectImpl.Where(System.Object)">
|
||||
<summary>
|
||||
Where(new { Year = 2017, CategoryId = 198, IsPublished = true })<para></para>
|
||||
WHERE [Year] = 2017 AND [CategoryId] = 198 AND [IsPublished] = 1
|
||||
</summary>
|
||||
</member>
|
||||
<member name="M:FreeSql.Extensions.ZoreEntity.ZoreDbContext.SelectImpl.Where(System.String,System.Object)">
|
||||
<summary>
|
||||
WHERE [field] = ..
|
||||
</summary>
|
||||
</member>
|
||||
</members>
|
||||
</doc>
|
@ -0,0 +1,915 @@
|
||||
using FreeSql.DataAnnotations;
|
||||
using FreeSql.Internal;
|
||||
using FreeSql.Internal.CommonProvider;
|
||||
using FreeSql.Internal.Model;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Data;
|
||||
using System.Data.Common;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
using T = System.Collections.Generic.Dictionary<string, object>;
|
||||
|
||||
namespace FreeSql.Extensions.ZoreEntity
|
||||
{
|
||||
partial class ZoreDbContext
|
||||
{
|
||||
public class SelectImpl
|
||||
{
|
||||
ZoreDbContext _dbcontext;
|
||||
IFreeSql _orm => _dbcontext._orm;
|
||||
List<ZoreTableInfo> _tables => _dbcontext._tables;
|
||||
int _mainTableIndex = -1;
|
||||
List<TableAliasInfo> _tableAlias;
|
||||
ISelect<TestDynamicFilterInfo> _select;
|
||||
Select0Provider _selectProvider;
|
||||
string _field;
|
||||
Dictionary<string, string> _fieldAlias;
|
||||
List<Action<DbDataReaderContext>> _fieldReader;
|
||||
string _groupBy;
|
||||
List<DbParameter> _params = new List<DbParameter>();
|
||||
CommonUtils _common => _selectProvider._commonUtils;
|
||||
bool _useStates = true;
|
||||
bool _includeAll = false;
|
||||
|
||||
SelectImpl() { }
|
||||
internal SelectImpl(ZoreDbContext dbcontext, string tableName)
|
||||
{
|
||||
_dbcontext = dbcontext;
|
||||
var tableIndex = _tables.FindIndex(a => a.CsName.ToLower() == tableName?.ToLower());
|
||||
if (tableIndex == -1) throw new Exception($"未定义表名 {tableName}");
|
||||
|
||||
_mainTableIndex = tableIndex;
|
||||
_tableAlias = new List<TableAliasInfo>();
|
||||
_select = _orm.Select<TestDynamicFilterInfo>()
|
||||
.AsTable((t, old) => _tables[tableIndex].DbName);
|
||||
_selectProvider = _select as Select0Provider;
|
||||
_fieldAlias = new Dictionary<string, string>();
|
||||
_fieldReader = new List<Action<DbDataReaderContext>>();
|
||||
FlagFetchResult(_tables[_mainTableIndex], "a", "");
|
||||
}
|
||||
|
||||
public SelectImpl NoTracking()
|
||||
{
|
||||
_useStates = false;
|
||||
return this;
|
||||
}
|
||||
public SelectImpl IncludeAll()
|
||||
{
|
||||
var ignores = new Dictionary<string, bool>();
|
||||
_includeAll = true;
|
||||
LocalAutoInclude(_tables[_mainTableIndex], "a");
|
||||
return this;
|
||||
|
||||
void LocalAutoInclude(ZoreTableInfo table, string alias, string navPath = "")
|
||||
{
|
||||
if (ignores.ContainsKey(table.CsName)) return;
|
||||
ignores.Add(table.CsName, true);
|
||||
TableAliasInfo tableAlias = null;
|
||||
if (table != _tables[_mainTableIndex])
|
||||
tableAlias = FlagFetchResult(table, alias.ToString(), navPath);
|
||||
|
||||
var leftJoins = table.Navigates.Where(a => a.Value.RefType == TableRefType.ManyToOne || a.Value.RefType == TableRefType.OneToOne).ToArray();
|
||||
foreach (var join in leftJoins)
|
||||
{
|
||||
var joinTable = join.Value.RefTable;
|
||||
if (ignores.ContainsKey(joinTable.CsName)) return;
|
||||
|
||||
var joinAlias = GetMaxAlias();
|
||||
var joinOn = string.Join(" AND ", join.Value.RefColumns.Select((bname, idx) =>
|
||||
$"{joinAlias}.{_common.QuoteSqlName(join.Value.RefTable.ColumnsByCs[bname].Attribute.Name)} = {alias}.{_common.QuoteSqlName(join.Value.Table.ColumnsByCs[join.Value.Columns[idx]].Attribute.Name)}"));
|
||||
_select.LeftJoin($"{_common.QuoteSqlName(join.Value.RefTable.DbName)} {joinAlias} ON {joinOn}");
|
||||
|
||||
LocalAutoInclude(joinTable, joinAlias, $"{navPath}.{join.Key}");
|
||||
}
|
||||
if (tableAlias == null) tableAlias = _tableAlias.Where(a => a.Alias == alias).First();
|
||||
var includeManys = table.Navigates.Where(a => a.Value.RefType == TableRefType.OneToMany || a.Value.RefType == TableRefType.ManyToMany).ToArray();
|
||||
foreach (var includeMany in includeManys)
|
||||
tableAlias.IncludeMany.Add(NativeTuple.Create(includeMany.Key, (Action<SelectImpl>)null));
|
||||
}
|
||||
}
|
||||
public SelectImpl Include(string navigate, Action<SelectImpl> then = null)
|
||||
{
|
||||
var alias = _tableAlias[0];
|
||||
var navPath = navigate.Split('.');
|
||||
var navPath2 = "";
|
||||
for (var navIdx = 0; navIdx < navPath.Length; navIdx++)
|
||||
{
|
||||
if (alias.Table.Navigates.TryGetValue(navPath[navIdx], out var nav) == false) throw new Exception($"{alias.Table.CsName} 未定义导航属性 {navPath[navIdx]}");
|
||||
if (nav.RefType == TableRefType.OneToMany || nav.RefType == TableRefType.ManyToMany)
|
||||
{
|
||||
if (navIdx < navPath.Length - 1) throw new Exception($"导航属性 OneToMany/ManyToMany {navPath[navIdx]} 不能处于中间位置");
|
||||
if (alias.IncludeMany.Where(a => a.Item1 == nav.NavigateKey).Any() == false)
|
||||
alias.IncludeMany.Add(NativeTuple.Create(nav.NavigateKey, then));
|
||||
break;
|
||||
}
|
||||
navPath2 = navIdx == 0 ? nav.NavigateKey : $"{navPath2}.{nav.NavigateKey}";
|
||||
var navAlias = _tableAlias.Where(a => string.Join(".", a.NavPath) == navPath2).FirstOrDefault();
|
||||
if (navAlias == null)
|
||||
{
|
||||
var joinAlias = GetMaxAlias();
|
||||
var joinOn = string.Join(" AND ", nav.RefColumns.Select((bname, idx) =>
|
||||
$"{joinAlias}.{_common.QuoteSqlName(nav.RefTable.ColumnsByCs[bname].Attribute.Name)} = {alias.Alias}.{_common.QuoteSqlName(nav.Table.ColumnsByCs[nav.Columns[idx]].Attribute.Name)}"));
|
||||
_select.LeftJoin($"{_common.QuoteSqlName(nav.RefTable.DbName)} {joinAlias} ON {joinOn}");
|
||||
navAlias = FlagFetchResult(nav.RefTable, joinAlias, navPath2);
|
||||
}
|
||||
alias = navAlias;
|
||||
}
|
||||
return this;
|
||||
}
|
||||
//public SelectImpl IncludeSubQuery(string resultKey, string tableName, Action<SelectImpl> then)
|
||||
//{
|
||||
// var query = _dbcontext.SelectNoTracking(tableName);
|
||||
// query._tableAlias.AddRange(_tableAlias);
|
||||
// return this;
|
||||
//}
|
||||
|
||||
/// <summary>
|
||||
/// 举例1:LeftJoin("table1", "id", "user.id") -> LEFT JOIN [table1] b ON b.[id] = a.[id]<para></para>
|
||||
/// 举例2:LeftJoin("table1", "id", "user.id", "xid", "user.xid") -> LEFT JOIN [table1] b ON b.[id] = [a].id] AND b.[xid] = a.[xid]
|
||||
/// </summary>
|
||||
public SelectImpl LeftJoin(string tableName, params string[] onFields) => Join("LEFT JOIN", tableName, onFields);
|
||||
/// <summary>
|
||||
/// 举例1:InnerJoin("table1", "id", "user.id") -> INNER JOIN [table1] b ON b.[id] = a.[id]<para></para>
|
||||
/// 举例2:InnerJoin("table1", "id", "user.id", "xid", "user.xid") -> INNER JOIN [table1] b ON b.[id] = [a].id] AND b.[xid] = a.[xid]
|
||||
/// </summary>
|
||||
public SelectImpl InnerJoin(string tableName, params string[] onFields) => Join("INNER JOIN", tableName, onFields);
|
||||
/// <summary>
|
||||
/// 举例1:RightJoin("table1", "id", "user.id") -> RIGTH JOIN [table1] b ON b.[id] = a.[id]<para></para>
|
||||
/// 举例2:RightJoin("table1", "id", "user.id", "xid", "user.xid") -> RIGTH JOIN [table1] b ON b.[id] = [a].id] AND b.[xid] = a.[xid]
|
||||
/// </summary>
|
||||
public SelectImpl RightJoin(string tableName, params string[] onFields) => Join("RIGTH JOIN", tableName, onFields);
|
||||
SelectImpl Join(string joinType, string tableName, params string[] onFields)
|
||||
{
|
||||
if (onFields == null || onFields.Length == 0) throw new Exception($"{joinType} 参数 {nameof(onFields)} 不能为空");
|
||||
if (onFields.Length % 2 != 0) throw new Exception($"{joinType} 参数 {nameof(onFields)} 数组长度必须为偶数,正确:.LeftJoin(\"table1\", \"id\", \"user.id\")");
|
||||
|
||||
var table = _tables.Where(a => a.CsName.ToLower() == tableName?.ToLower()).FirstOrDefault();
|
||||
if (table == null) throw new Exception($"未定义表名 {tableName}");
|
||||
var alias = GetMaxAlias();
|
||||
var navKey = tableName;
|
||||
for (var a = 2; true; a++)
|
||||
{
|
||||
if (_tables[_mainTableIndex].ColumnsByCs.ContainsKey(navKey))
|
||||
{
|
||||
navKey = $"{tableName}{a}";
|
||||
continue;
|
||||
}
|
||||
if (_tableAlias.Any(b => b.NavPath.Length == 1 && b.NavPath.Last() == navKey))
|
||||
{
|
||||
navKey = $"{tableName}{a}";
|
||||
continue;
|
||||
}
|
||||
break;
|
||||
}
|
||||
FlagFetchResult(table, alias, navKey);
|
||||
var joinOn = new string[onFields.Length / 2];
|
||||
for (var a = 0; a < onFields.Length; a += 2)
|
||||
{
|
||||
var field1 = ParseField(table, alias, onFields[a]);
|
||||
if (field1 == null) throw new Exception($"未匹配字段名 {onFields[a]}");
|
||||
var field2 = ParseField(table, alias, onFields[a + 1]);
|
||||
if (field2 == null) throw new Exception($"未匹配字段名 {onFields[a + 1]}");
|
||||
joinOn[a / 2] = $"{field1.Item1} = {field2.Item1}";
|
||||
}
|
||||
_select.RawJoin($"{joinType} {_common.QuoteSqlName(table.DbName)} {alias} ON {string.Join(" AND ", joinOn)}");
|
||||
return this;
|
||||
}
|
||||
|
||||
class TableAliasInfo
|
||||
{
|
||||
public string Alias { get; set; }
|
||||
public ZoreTableInfo Table { get; set; }
|
||||
public string[] NavPath { get; set; }
|
||||
public List<NativeTuple<string, Action<SelectImpl>>> IncludeMany { get; set; } = new List<NativeTuple<string, Action<SelectImpl>>>();
|
||||
}
|
||||
class DbDataReaderContext
|
||||
{
|
||||
public DbDataReader Reader { get; set; }
|
||||
public int Index { get; set; }
|
||||
public T Result { get; set; }
|
||||
}
|
||||
string GetMaxAlias()
|
||||
{
|
||||
var max = (int)_tableAlias.Where(a => a.Alias.Length == 1).Max(a => a.Alias[0]);
|
||||
if (max < 'a') max = 'a';
|
||||
for (var a = 1; true; a++)
|
||||
{
|
||||
var alias = ((char)(max + a)).ToString();
|
||||
if (_tableAlias.Where(b => b.Alias == alias).Any()) continue;
|
||||
return alias;
|
||||
}
|
||||
}
|
||||
TableAliasInfo FlagFetchResult(ZoreTableInfo table, string alias, string navPath)
|
||||
{
|
||||
var tableAlias = _tableAlias.Where(a => a.Alias == alias).FirstOrDefault();
|
||||
if (tableAlias == null)
|
||||
{
|
||||
var navPathArray = navPath.Split('.').Where(a => string.IsNullOrWhiteSpace(a) == false).ToArray();
|
||||
_tableAlias.Add(tableAlias = new TableAliasInfo
|
||||
{
|
||||
Alias = alias,
|
||||
Table = table,
|
||||
NavPath = navPathArray
|
||||
});
|
||||
}
|
||||
var sbfield = new StringBuilder();
|
||||
if (string.IsNullOrEmpty(_field) == false) sbfield.Append(", ").Append(_field);
|
||||
foreach (var col in table.Columns.Values)
|
||||
{
|
||||
var colName = col.Attribute.Name;
|
||||
for (var a = 2; true; a++)
|
||||
{
|
||||
if (_fieldAlias.ContainsKey(colName)) colName = $"{col.Attribute.Name}{a}";
|
||||
else break;
|
||||
}
|
||||
_fieldAlias.Add(colName, col.Attribute.Name);
|
||||
sbfield.Append(", ").Append(alias).Append(".").Append(_common.QuoteSqlName(col.Attribute.Name));
|
||||
if (colName != col.Attribute.Name) sbfield.Append(_common.FieldAsAlias(colName));
|
||||
}
|
||||
_field = sbfield.Remove(0, 2).ToString();
|
||||
sbfield.Clear();
|
||||
|
||||
_fieldReader.Add(dr =>
|
||||
{
|
||||
var pkIsNull = false;
|
||||
foreach (var col in table.Columns.Values)
|
||||
{
|
||||
if (pkIsNull == false && col.Attribute.IsPrimary)
|
||||
{
|
||||
pkIsNull = dr.Reader.IsDBNull(dr.Index);
|
||||
if (pkIsNull) dr.Result.Clear();
|
||||
}
|
||||
if (pkIsNull == false) dr.Result[col.CsName] = Utils.GetDataReaderValue(col.CsType, dr.Reader.GetValue(dr.Index));
|
||||
dr.Index++;
|
||||
}
|
||||
});
|
||||
return tableAlias;
|
||||
}
|
||||
T FetchResult(DbDataReader reader)
|
||||
{
|
||||
var fieldIndex = 0;
|
||||
var result = new T();
|
||||
for (var aliasIndex = 0; aliasIndex < _tableAlias.Count; aliasIndex++)
|
||||
{
|
||||
var navValue = result;
|
||||
var drctx = new DbDataReaderContext { Index = fieldIndex, Reader = reader };
|
||||
if (aliasIndex == 0)
|
||||
{
|
||||
drctx.Result = result;
|
||||
}
|
||||
else
|
||||
{
|
||||
var isNull = false;
|
||||
for (var navidx = 0; navidx < _tableAlias[aliasIndex].NavPath.Length - 1; navidx++)
|
||||
{
|
||||
var navKey = _tableAlias[aliasIndex].NavPath[navidx];
|
||||
if (navValue.ContainsKey(navKey) == false)
|
||||
{
|
||||
isNull = true;
|
||||
break;
|
||||
}
|
||||
navValue = navValue[navKey] as T;
|
||||
if (navValue == null)
|
||||
{
|
||||
isNull = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (isNull)
|
||||
{
|
||||
fieldIndex += _tableAlias[aliasIndex].Table.Columns.Count;
|
||||
continue;
|
||||
}
|
||||
drctx.Result = new T();
|
||||
navValue[_tableAlias[aliasIndex].NavPath.LastOrDefault()] = drctx.Result;
|
||||
}
|
||||
_fieldReader[aliasIndex](drctx);
|
||||
fieldIndex = drctx.Index;
|
||||
if (aliasIndex > 0 && drctx.Result.Any() == false) navValue.Remove(_tableAlias[aliasIndex].NavPath.LastOrDefault());
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
void IncludeMany(TableAliasInfo alias, NativeTuple<string, Action<SelectImpl>> navMany, List<T> list, List<string> flagIndexs = null)
|
||||
{
|
||||
if (list?.Any() != true) return;
|
||||
if (flagIndexs == null) flagIndexs = new List<string>();
|
||||
flagIndexs.Add(alias.Table.CsName);
|
||||
|
||||
var nav = alias.Table.Navigates[navMany.Item1];
|
||||
if (_includeAll && flagIndexs.Contains(nav.RefTable.CsName)) return;
|
||||
|
||||
if (nav.RefType == TableRefType.OneToMany)
|
||||
{
|
||||
var subTable = nav.RefTable;
|
||||
var subSelect = new SelectImpl(_dbcontext, subTable.CsName);
|
||||
if (_includeAll) subSelect.IncludeAll();
|
||||
|
||||
Func<Dictionary<string, bool>> getWhereDic = () =>
|
||||
{
|
||||
var sbDic = new Dictionary<string, bool>();
|
||||
for (var y = 0; y < list.Count; y++)
|
||||
{
|
||||
var sbWhereOne = new StringBuilder();
|
||||
sbWhereOne.Append("(");
|
||||
for (var z = 0; z < nav.Columns.Count; z++)
|
||||
{
|
||||
if (z > 0) sbWhereOne.Append(" AND ");
|
||||
var refcol = nav.RefTable.ColumnsByCs[nav.RefColumns[z]];
|
||||
var val = Utils.GetDataReaderValue(refcol.Attribute.MapType, list[y][nav.Columns[z]]);
|
||||
sbWhereOne.Append(_common.FormatSql($"a.{_common.QuoteSqlName(refcol.Attribute.Name)}={{0}}", val));
|
||||
}
|
||||
sbWhereOne.Append(")");
|
||||
var whereOne = sbWhereOne.ToString();
|
||||
sbWhereOne.Clear();
|
||||
if (sbDic.ContainsKey(whereOne) == false) sbDic.Add(whereOne, true);
|
||||
}
|
||||
return sbDic;
|
||||
};
|
||||
if (nav.Columns.Count == 1)
|
||||
{
|
||||
var refcol = nav.RefTable.ColumnsByCs[nav.RefColumns[0]];
|
||||
var args1 = $"a.{_common.QuoteSqlName(refcol.Attribute.Name)}";
|
||||
var left = _common.FormatSql("{0}", new object[] { list.Select(a => a[nav.Columns[0]]).Distinct().ToArray() });
|
||||
subSelect._select.Where($"({args1} in {left.Replace(", \r\n \r\n", $") \r\n OR {args1} in (")})");
|
||||
}
|
||||
else
|
||||
{
|
||||
var sbDic = getWhereDic();
|
||||
var sbWhere = new StringBuilder();
|
||||
foreach (var sbd in sbDic)
|
||||
sbWhere.Append(" OR ").Append(sbd.Key);
|
||||
subSelect._select.Where(sbWhere.Remove(0, 4).ToString());
|
||||
sbWhere.Clear();
|
||||
sbDic.Clear();
|
||||
}
|
||||
navMany.Item2?.Invoke(subSelect);
|
||||
var subList = subSelect.ToListPrivate(null, null, flagIndexs);
|
||||
foreach (var item in list)
|
||||
{
|
||||
item[nav.NavigateKey] = subList.Where(a =>
|
||||
{
|
||||
for (var z = 0; z < nav.Columns.Count; z++)
|
||||
if (CompareEntityPropertyValue(nav.Table.ColumnsByCs[nav.Columns[z]].Attribute.MapType, item[nav.Columns[z]], a[nav.RefColumns[z]]) == false)
|
||||
return false;
|
||||
return true;
|
||||
}).ToList();
|
||||
}
|
||||
subList.Clear();
|
||||
}
|
||||
else if (nav.RefType == TableRefType.ManyToMany)
|
||||
{
|
||||
var subTable = nav.RefTable;
|
||||
var subSelect = new SelectImpl(_dbcontext, subTable.CsName);
|
||||
if (_includeAll) subSelect.IncludeAll();
|
||||
|
||||
var middleJoinOn = string.Join(" AND ", nav.RefColumns.Select((bname, idx) =>
|
||||
$"midtb.{_common.QuoteSqlName(nav.RefMiddleTable.ColumnsByCs[nav.MiddleColumns[nav.Columns.Count + idx]].Attribute.Name)} = a.{_common.QuoteSqlName(nav.RefTable.ColumnsByCs[bname].Attribute.Name)}"));
|
||||
subSelect._select.InnerJoin($"{_common.QuoteSqlName(nav.RefMiddleTable.DbName)} midtb ON {middleJoinOn}");
|
||||
|
||||
Func<Dictionary<string, bool>> getWhereDic = () =>
|
||||
{
|
||||
var sbDic = new Dictionary<string, bool>();
|
||||
for (var y = 0; y < list.Count; y++)
|
||||
{
|
||||
var sbWhereOne = new StringBuilder();
|
||||
sbWhereOne.Append("(");
|
||||
for (var z = 0; z < nav.Columns.Count; z++)
|
||||
{
|
||||
if (z > 0) sbWhereOne.Append(" AND ");
|
||||
var midcol = nav.RefMiddleTable.ColumnsByCs[nav.MiddleColumns[z]];
|
||||
var val = Utils.GetDataReaderValue(midcol.Attribute.MapType, list[y][nav.Columns[z]]);
|
||||
sbWhereOne.Append(_common.FormatSql($"midtb.{_common.QuoteSqlName(midcol.Attribute.Name)}={{0}}", val));
|
||||
}
|
||||
sbWhereOne.Append(")");
|
||||
var whereOne = sbWhereOne.ToString();
|
||||
sbWhereOne.Clear();
|
||||
if (sbDic.ContainsKey(whereOne) == false) sbDic.Add(whereOne, true);
|
||||
}
|
||||
return sbDic;
|
||||
};
|
||||
if (nav.Columns.Count == 1)
|
||||
{
|
||||
var midcol = nav.RefMiddleTable.ColumnsByCs[nav.MiddleColumns[0]];
|
||||
var args1 = $"midtb.{_common.QuoteSqlName(midcol.Attribute.Name)}";
|
||||
var left = _common.FormatSql("{0}", new object[] { list.Select(a => a[nav.Columns[0]]).Distinct().ToArray() });
|
||||
subSelect._select.Where($"({args1} in {left.Replace(", \r\n \r\n", $") \r\n OR {args1} in (")})");
|
||||
}
|
||||
else
|
||||
{
|
||||
var sbDic = getWhereDic();
|
||||
var sbWhere = new StringBuilder();
|
||||
foreach (var sbd in sbDic)
|
||||
sbWhere.Append(" OR ").Append(sbd.Key);
|
||||
subSelect._select.Where(sbWhere.Remove(0, 4).ToString());
|
||||
sbWhere.Clear();
|
||||
sbDic.Clear();
|
||||
}
|
||||
navMany.Item2?.Invoke(subSelect);
|
||||
var subList = subSelect.ToListPrivate(
|
||||
string.Join(", ", nav.MiddleColumns.Select((a, idx) => $"midtb.{_common.QuoteSqlName(nav.RefMiddleTable.ColumnsByCs[a].Attribute.Name)}{_common.FieldAsAlias($"midtb_field__{idx}")}")),
|
||||
(dict, dr) =>
|
||||
{
|
||||
var fieldCount = dr.FieldCount - nav.MiddleColumns.Count;
|
||||
for (var z = 0; z < nav.MiddleColumns.Count; z++)
|
||||
dict[$"midtb_field__{z}"] = Utils.GetDataReaderValue(nav.RefMiddleTable.ColumnsByCs[nav.MiddleColumns[z]].CsType, dr.GetValue(fieldCount + z));
|
||||
}, flagIndexs);
|
||||
foreach (var item in list)
|
||||
{
|
||||
item[nav.NavigateKey] = subList.Where(a =>
|
||||
{
|
||||
for (var z = 0; z < nav.Columns.Count; z++)
|
||||
if (CompareEntityPropertyValue(nav.Table.ColumnsByCs[nav.Columns[z]].Attribute.MapType, item[nav.Columns[z]], a[$"midtb_field__{z}"]) == false)
|
||||
return false;
|
||||
return true;
|
||||
}).ToList();
|
||||
}
|
||||
foreach (var subItem in subList)
|
||||
for (var z = 0; z < nav.MiddleColumns.Count; z++)
|
||||
subItem.Remove($"midtb_field__{z}");
|
||||
subList.Clear();
|
||||
}
|
||||
}
|
||||
|
||||
public string ToSql(string field = null)
|
||||
{
|
||||
if (string.IsNullOrWhiteSpace(field)) return _select.ToSql(_field);
|
||||
return _select.ToSql(field);
|
||||
}
|
||||
List<T> ToListPrivate(string otherField, Action<T, DbDataReader> otherReader, List<string> flagIndexs = null)
|
||||
{
|
||||
var sql = string.IsNullOrWhiteSpace(otherField) ? this.ToSql() : this.ToSql($"{_field},{otherField}");
|
||||
var ret = new List<T>();
|
||||
var dbParms = _params.ToArray();
|
||||
var before = new Aop.CurdBeforeEventArgs(_tables[_mainTableIndex].Type, _tables[_mainTableIndex], Aop.CurdType.Select, sql, dbParms);
|
||||
_orm.Aop.CurdBeforeHandler?.Invoke(this, before);
|
||||
Exception exception = null;
|
||||
try
|
||||
{
|
||||
_orm.Ado.ExecuteReader(_dbcontext._transaction?.Connection, _dbcontext._transaction, fetch =>
|
||||
{
|
||||
var item = FetchResult(fetch.Object);
|
||||
otherReader?.Invoke(item, fetch.Object);
|
||||
ret.Add(item);
|
||||
}, CommandType.Text, sql, _dbcontext._commandTimeout, dbParms);
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
exception = ex;
|
||||
throw;
|
||||
}
|
||||
finally
|
||||
{
|
||||
var after = new Aop.CurdAfterEventArgs(before, exception, ret);
|
||||
_orm.Aop.CurdAfterHandler?.Invoke(this, after);
|
||||
}
|
||||
foreach (var join in _tableAlias)
|
||||
{
|
||||
if (join.IncludeMany.Any() == false) continue;
|
||||
var list = new List<T>();
|
||||
if (join.Alias == "a") list = ret;
|
||||
else
|
||||
{
|
||||
foreach (var obj in ret)
|
||||
{
|
||||
T item = obj;
|
||||
foreach (var navKey in join.NavPath)
|
||||
{
|
||||
if (string.IsNullOrWhiteSpace(navKey)) continue;
|
||||
item.TryGetValue(navKey, out var obj2);
|
||||
item = obj2 as T;
|
||||
if (item == null) break;
|
||||
}
|
||||
if (item != null) list.Add(item);
|
||||
}
|
||||
}
|
||||
foreach(var navMany in join.IncludeMany)
|
||||
IncludeMany(join, navMany, list, flagIndexs);
|
||||
}
|
||||
if (_useStates && flagIndexs == null)
|
||||
foreach (var item in ret)
|
||||
_dbcontext.AttachCascade(_tables[_mainTableIndex], item, true);
|
||||
return ret;
|
||||
}
|
||||
public List<T> ToList() => ToListPrivate(null, null);
|
||||
public T ToOne() => ToListPrivate(null, null).FirstOrDefault();
|
||||
public T First() => ToOne();
|
||||
public bool Any() => _select.Any();
|
||||
public long Count() => _select.Count();
|
||||
public SelectImpl Count(out long count)
|
||||
{
|
||||
_select.Count(out count);
|
||||
return this;
|
||||
}
|
||||
public SelectImpl WithTransaction(DbTransaction transaction)
|
||||
{
|
||||
_select.WithTransaction(transaction);
|
||||
return this;
|
||||
}
|
||||
public SelectImpl WithConnection(DbConnection connection)
|
||||
{
|
||||
_select.WithConnection(connection);
|
||||
return this;
|
||||
}
|
||||
public SelectImpl CommandTimeout(int timeout)
|
||||
{
|
||||
_select.CommandTimeout(timeout);
|
||||
return this;
|
||||
}
|
||||
public SelectImpl Distinct()
|
||||
{
|
||||
_select.Distinct();
|
||||
return this;
|
||||
}
|
||||
public SelectImpl Master()
|
||||
{
|
||||
_select.Master();
|
||||
return this;
|
||||
}
|
||||
public SelectImpl ForUpdate(bool nowait = false)
|
||||
{
|
||||
_select.ForUpdate(nowait);
|
||||
return this;
|
||||
}
|
||||
|
||||
NativeTuple<string, ColumnInfo> ParseField(ZoreTableInfo firstTable, string firstTableAlias, string property)
|
||||
{
|
||||
if (string.IsNullOrEmpty(property)) return null;
|
||||
var field = property.Split('.').Select(a => a.Trim()).ToArray();
|
||||
|
||||
if (field.Length == 1)
|
||||
{
|
||||
if (firstTable != null && firstTable.ColumnsByCs.TryGetValue(field[0], out var col2) == true)
|
||||
return NativeTuple.Create($"{firstTableAlias}.{_common.QuoteSqlName(col2.Attribute.Name)}", col2);
|
||||
|
||||
foreach (var ta2 in _tableAlias)
|
||||
{
|
||||
if (ta2.Table.ColumnsByCs.TryGetValue(field[0], out col2))
|
||||
return NativeTuple.Create($"{ta2.Alias}.{_common.QuoteSqlName(col2.Attribute.Name)}", col2);
|
||||
}
|
||||
}
|
||||
else if (field.Length == 2)
|
||||
{
|
||||
if (firstTable != null && firstTable.CsName.ToLower() == field[0].ToLower() && firstTable.ColumnsByCs.TryGetValue(field[1], out var col2) == true)
|
||||
return NativeTuple.Create($"{firstTableAlias}.{_common.QuoteSqlName(col2.Attribute.Name)}", col2);
|
||||
|
||||
var ta2s = _tableAlias.Where(a => a.Table.CsName.ToLower() == field[0].ToLower()).ToArray();
|
||||
if (ta2s.Length == 1 && ta2s[0].Table.ColumnsByCs.TryGetValue(field[1], out col2) == true)
|
||||
return NativeTuple.Create($"{ta2s[0].Alias}.{_common.QuoteSqlName(col2.Attribute.Name)}", col2);
|
||||
if (ta2s.Length > 1)
|
||||
{
|
||||
ta2s = _tableAlias.Where(a => a.Table.CsName == field[0]).ToArray();
|
||||
if (ta2s.Length == 1 && ta2s[0].Table.ColumnsByCs.TryGetValue(field[1], out col2) == true)
|
||||
return NativeTuple.Create($"{ta2s[0].Alias}.{_common.QuoteSqlName(col2.Attribute.Name)}", col2);
|
||||
}
|
||||
if (_tableAlias.Where(a => a.Alias == field[0]).FirstOrDefault()?.Table.ColumnsByCs.TryGetValue(field[1], out col2) == true)
|
||||
return NativeTuple.Create($"{field[0]}.{_common.QuoteSqlName(col2.Attribute.Name)}", col2);
|
||||
}
|
||||
|
||||
var navPath = string.Join(".", field.Skip(1).Take(field.Length - 1));
|
||||
var ta = _tableAlias.Where(a => string.Join(".", a.NavPath) == navPath).FirstOrDefault();
|
||||
if (ta?.Table.ColumnsByCs.TryGetValue(field.Last(), out var col) == true)
|
||||
return NativeTuple.Create($"{ta.Alias}.{_common.QuoteSqlName(col.Attribute.Name)}", col);
|
||||
throw new Exception(CoreStrings.Cannot_Match_Property(property));
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// WHERE [Id] IN (...)
|
||||
/// </summary>
|
||||
public SelectImpl Where(IEnumerable<T> items)
|
||||
{
|
||||
var alias = _tableAlias.Where(a => a.Table == _tables[_mainTableIndex]).FirstOrDefault()?.Alias;
|
||||
if (!string.IsNullOrWhiteSpace(alias)) alias = $"{alias}.";
|
||||
var where = _common.WhereItems(_tables[_mainTableIndex].Primarys, alias, items);
|
||||
_select.Where(where);
|
||||
return this;
|
||||
}
|
||||
/// <summary>
|
||||
/// Where(new { Year = 2017, CategoryId = 198, IsPublished = true })<para></para>
|
||||
/// WHERE [Year] = 2017 AND [CategoryId] = 198 AND [IsPublished] = 1
|
||||
/// </summary>
|
||||
public SelectImpl Where(object multipleFields)
|
||||
{
|
||||
if (multipleFields == null) return this;
|
||||
foreach (var prop in multipleFields.GetType().GetProperties())
|
||||
WhereDynamicFilter(new DynamicFilterInfo { Field = prop.Name, Operator = DynamicFilterOperator.Eq, Value = prop.GetValue(multipleFields, null) });
|
||||
return this;
|
||||
}
|
||||
/// <summary>
|
||||
/// WHERE [field] = ..
|
||||
/// </summary>
|
||||
public SelectImpl Where(string field, object value) => WhereDynamicFilter(new DynamicFilterInfo { Field = field, Operator = DynamicFilterOperator.Eq, Value = value });
|
||||
public SelectImpl Where(string field, string @operator, object value)
|
||||
{
|
||||
switch (@operator?.ToLower().Trim())
|
||||
{
|
||||
case "=":
|
||||
case "==":
|
||||
case "eq":
|
||||
return WhereDynamicFilter(new DynamicFilterInfo { Field = field, Operator = DynamicFilterOperator.Eq, Value = value });
|
||||
case "!=":
|
||||
case "<>":
|
||||
return WhereDynamicFilter(new DynamicFilterInfo { Field = field, Operator = DynamicFilterOperator.NotEqual, Value = value });
|
||||
case ">":
|
||||
return WhereDynamicFilter(new DynamicFilterInfo { Field = field, Operator = DynamicFilterOperator.GreaterThan, Value = value });
|
||||
case ">=":
|
||||
return WhereDynamicFilter(new DynamicFilterInfo { Field = field, Operator = DynamicFilterOperator.GreaterThanOrEqual, Value = value });
|
||||
case "<":
|
||||
return WhereDynamicFilter(new DynamicFilterInfo { Field = field, Operator = DynamicFilterOperator.LessThan, Value = value });
|
||||
case "<=":
|
||||
return WhereDynamicFilter(new DynamicFilterInfo { Field = field, Operator = DynamicFilterOperator.LessThanOrEqual, Value = value });
|
||||
case "like":
|
||||
case "contains":
|
||||
return WhereDynamicFilter(new DynamicFilterInfo { Field = field, Operator = DynamicFilterOperator.Contains, Value = value });
|
||||
case "!like":
|
||||
case "notlike":
|
||||
case "not like":
|
||||
return WhereDynamicFilter(new DynamicFilterInfo { Field = field, Operator = DynamicFilterOperator.NotContains, Value = value });
|
||||
case "in":
|
||||
return WhereDynamicFilter(new DynamicFilterInfo { Field = field, Operator = DynamicFilterOperator.Any, Value = value });
|
||||
case "!in":
|
||||
case "notin":
|
||||
case "not in":
|
||||
return WhereDynamicFilter(new DynamicFilterInfo { Field = field, Operator = DynamicFilterOperator.Any, Value = value });
|
||||
}
|
||||
throw new Exception($"未实现 {@operator}");
|
||||
}
|
||||
public SelectImpl WhereColumns(string field1, string @operator, string field2)
|
||||
{
|
||||
var field1Result = ParseField(null, null, field1);
|
||||
if (field1Result == null) throw new Exception($"未匹配字段名 {field1}");
|
||||
var field2Result = ParseField(null, null, field2);
|
||||
if (field2Result == null) throw new Exception($"未匹配字段名 {field2}");
|
||||
switch (@operator?.ToLower().Trim())
|
||||
{
|
||||
case "=":
|
||||
case "==":
|
||||
case "eq":
|
||||
_select.Where($"{field1Result.Item1} = {field2Result.Item1}");
|
||||
return this;
|
||||
case "!=":
|
||||
case "<>":
|
||||
_select.Where($"{field1Result.Item1} <> {field2Result.Item1}");
|
||||
return this;
|
||||
case ">":
|
||||
_select.Where($"{field1Result.Item1} > {field2Result.Item1}");
|
||||
return this;
|
||||
case ">=":
|
||||
_select.Where($"{field1Result.Item1} >= {field2Result.Item1}");
|
||||
return this;
|
||||
case "<":
|
||||
_select.Where($"{field1Result.Item1} < {field2Result.Item1}");
|
||||
return this;
|
||||
case "<=":
|
||||
_select.Where($"{field1Result.Item1} <= {field2Result.Item1}");
|
||||
return this;
|
||||
}
|
||||
throw new Exception($"未实现 {@operator}");
|
||||
}
|
||||
public SelectImpl WhereDynamicFilter(DynamicFilterInfo filter)
|
||||
{
|
||||
var sql = ParseDynamicFilter(filter);
|
||||
_selectProvider._where.Append(sql);
|
||||
return this;
|
||||
}
|
||||
string ParseDynamicFilter(DynamicFilterInfo filter)
|
||||
{
|
||||
var replacedFilter = new DynamicFilterInfo();
|
||||
var replacedMap = new List<NativeTuple<string, string>>();
|
||||
LocalCloneFilter(filter, replacedFilter);
|
||||
var oldWhere = _selectProvider._where.ToString();
|
||||
var newWhere = "";
|
||||
try
|
||||
{
|
||||
_selectProvider._where.Clear();
|
||||
_select.WhereDynamicFilter(replacedFilter);
|
||||
newWhere = _selectProvider._where.ToString();
|
||||
}
|
||||
finally
|
||||
{
|
||||
_selectProvider._where.Clear().Append(oldWhere);
|
||||
}
|
||||
foreach (var rm in replacedMap)
|
||||
{
|
||||
var find = $"a.{_common.QuoteSqlName(rm.Item1)}";
|
||||
var idx = newWhere.IndexOf(find);
|
||||
if (idx != -1) newWhere = $"{newWhere.Substring(0, idx)}{rm.Item2}{newWhere.Substring(idx + find.Length)}";
|
||||
}
|
||||
return newWhere;
|
||||
|
||||
void LocalCloneFilter(DynamicFilterInfo source, DynamicFilterInfo target)
|
||||
{
|
||||
target.Field = source.Field;
|
||||
target.Operator = source.Operator;
|
||||
target.Value = source.Value;
|
||||
target.Logic = source.Logic;
|
||||
if (string.IsNullOrWhiteSpace(source.Field) == false)
|
||||
{
|
||||
var parseResult = ParseField(null, null, source.Field);
|
||||
if (parseResult != null)
|
||||
{
|
||||
if (TestDynamicFilterInfo._dictTypeToPropertyname.TryGetValue(parseResult.Item2.Attribute.MapType, out var pname))
|
||||
target.Field = pname;
|
||||
else
|
||||
target.Field = TestDynamicFilterInfo._dictTypeToPropertyname[typeof(string)];
|
||||
replacedMap.Add(NativeTuple.Create(target.Field, parseResult.Item1));
|
||||
}
|
||||
}
|
||||
if (source.Filters?.Any() == true)
|
||||
{
|
||||
target.Filters = new List<DynamicFilterInfo>();
|
||||
foreach (var sourceChild in source.Filters)
|
||||
{
|
||||
var targetChild = new DynamicFilterInfo();
|
||||
target.Filters.Add(targetChild);
|
||||
LocalCloneFilter(sourceChild, targetChild);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
public class SubQuery
|
||||
{
|
||||
internal SelectImpl _parentQuery;
|
||||
internal SubQuery() { }
|
||||
public SelectImpl From(string tableName)
|
||||
{
|
||||
var query = _parentQuery._dbcontext.SelectNoTracking(tableName);
|
||||
query._selectProvider._tables[0].Alias =
|
||||
query._tableAlias[0].Alias = $"sub_{_parentQuery._tableAlias[0].Alias}";
|
||||
query._tableAlias.AddRange(_parentQuery._tableAlias);
|
||||
return query;
|
||||
}
|
||||
}
|
||||
public SelectImpl WhereExists(Func<SubQuery, SelectImpl> q)
|
||||
{
|
||||
var query = q?.Invoke(new SubQuery { _parentQuery = this });
|
||||
switch (_orm.Ado.DataType)
|
||||
{
|
||||
case DataType.Oracle:
|
||||
case DataType.OdbcOracle:
|
||||
case DataType.CustomOracle:
|
||||
case DataType.Dameng:
|
||||
case DataType.OdbcDameng:
|
||||
case DataType.GBase:
|
||||
query.Limit(-1);
|
||||
break;
|
||||
default:
|
||||
query.Limit(1); //#462 ORACLE rownum <= 2 会影响索引变慢
|
||||
break;
|
||||
}
|
||||
_selectProvider._where.Append($" AND EXISTS({query.ToSql("1").Replace(" \r\n", " \r\n ")})");
|
||||
return this;
|
||||
}
|
||||
public SelectImpl GroupByRaw(string sql)
|
||||
{
|
||||
if (string.IsNullOrWhiteSpace(sql)) return this;
|
||||
_useStates = false;
|
||||
_groupBy = $"{_groupBy}, {sql}";
|
||||
_select.GroupBy(_groupBy);
|
||||
return this;
|
||||
}
|
||||
public SelectImpl GroupBy(string[] fields)
|
||||
{
|
||||
var count = 0;
|
||||
for (var a = 0; a < fields.Length; a++)
|
||||
{
|
||||
if (string.IsNullOrWhiteSpace(fields[a])) continue;
|
||||
var field1 = ParseField(null, null, fields[a]);
|
||||
if (field1 == null) throw new Exception($"未匹配字段名 {fields[a]}");
|
||||
_groupBy = $"{_groupBy}, {field1.Item1}";
|
||||
count++;
|
||||
}
|
||||
if (count > 0)
|
||||
{
|
||||
_useStates = false;
|
||||
_select.GroupBy(_groupBy);
|
||||
}
|
||||
return this;
|
||||
}
|
||||
public SelectImpl HavingRaw(string sql)
|
||||
{
|
||||
_select.Having(sql);
|
||||
return this;
|
||||
}
|
||||
public SelectImpl OrderByRaw(string sql)
|
||||
{
|
||||
_select.OrderBy(sql);
|
||||
return this;
|
||||
}
|
||||
SelectImpl OrderBy(bool isdesc, string[] fields)
|
||||
{
|
||||
for (var a = 0; a < fields.Length; a ++)
|
||||
{
|
||||
if (string.IsNullOrWhiteSpace(fields[a])) continue;
|
||||
var field1 = ParseField(null, null, fields[a]);
|
||||
if (field1 == null) throw new Exception($"未匹配字段名 {fields[a]}");
|
||||
if (isdesc) _select.OrderBy($"{field1.Item1} DESC");
|
||||
else _select.OrderBy(field1.Item1);
|
||||
}
|
||||
return this;
|
||||
}
|
||||
public SelectImpl OrderBy(params string[] fields) => OrderBy(false, fields);
|
||||
public SelectImpl OrderByDescending(params string[] fields) => OrderBy(true, fields);
|
||||
public SelectImpl Offset(int offset)
|
||||
{
|
||||
_select.Offset(offset);
|
||||
return this;
|
||||
}
|
||||
public SelectImpl Limit(int limit)
|
||||
{
|
||||
_select.Limit(limit);
|
||||
return this;
|
||||
}
|
||||
public SelectImpl Skip(int offset) => Offset(offset);
|
||||
public SelectImpl Take(int limit) => Limit(limit);
|
||||
public SelectImpl Page(int pageNumber, int pageSize)
|
||||
{
|
||||
_select.Page(pageNumber, pageSize);
|
||||
return this;
|
||||
}
|
||||
|
||||
TResult InternalQuerySingle<TResult>(string field) => _orm.Ado.CommandFluent(this.ToSql(field))
|
||||
.WithConnection(_selectProvider._connection)
|
||||
.WithTransaction(_selectProvider._transaction).QuerySingle<TResult>();
|
||||
public decimal Sum(string field)
|
||||
{
|
||||
var field1 = ParseField(null, null, field);
|
||||
if (field1 == null) throw new Exception($"未匹配字段名 {field}");
|
||||
return InternalQuerySingle<decimal>($"sum({field1.Item1})");
|
||||
}
|
||||
public TMember Min<TMember>(string field)
|
||||
{
|
||||
var field1 = ParseField(null, null, field);
|
||||
if (field1 == null) throw new Exception($"未匹配字段名 {field}");
|
||||
return InternalQuerySingle<TMember>($"min({field1.Item1})");
|
||||
}
|
||||
public TMember Max<TMember>(string field)
|
||||
{
|
||||
var field1 = ParseField(null, null, field);
|
||||
if (field1 == null) throw new Exception($"未匹配字段名 {field}");
|
||||
return InternalQuerySingle<TMember>($"max({field1.Item1})");
|
||||
}
|
||||
public double Avg(string field)
|
||||
{
|
||||
var field1 = ParseField(null, null, field);
|
||||
if (field1 == null) throw new Exception($"未匹配字段名 {field}");
|
||||
return InternalQuerySingle<double>($"avg({field1.Item1})");
|
||||
}
|
||||
|
||||
|
||||
[Table(DisableSyncStructure = true)]
|
||||
class TestDynamicFilterInfo
|
||||
{
|
||||
public Guid DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf00 { get; set; }
|
||||
public bool DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf01 { get; set; }
|
||||
public string DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf02 { get; set; }
|
||||
public char DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf03 { get; set; }
|
||||
public DateTime DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf04 { get; set; }
|
||||
public DateTimeOffset DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf05 { get; set; }
|
||||
public TimeSpan DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf06 { get; set; }
|
||||
|
||||
public int DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf07 { get; set; }
|
||||
public long DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf08 { get; set; }
|
||||
public short DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf09 { get; set; }
|
||||
public sbyte DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf10 { get; set; }
|
||||
|
||||
public uint DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf11 { get; set; }
|
||||
public ulong DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf12 { get; set; }
|
||||
public ushort DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf13 { get; set; }
|
||||
public byte DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf14 { get; set; }
|
||||
public byte[] DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf15 { get; set; }
|
||||
|
||||
public double DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf16 { get; set; }
|
||||
public float DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf17 { get; set; }
|
||||
public decimal DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf18 { get; set; }
|
||||
|
||||
internal static Dictionary<Type, string> _dictTypeToPropertyname = new Dictionary<Type, string>
|
||||
{
|
||||
[typeof(Guid)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf00),
|
||||
[typeof(bool)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf01),
|
||||
[typeof(string)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf02),
|
||||
[typeof(char)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf03),
|
||||
[typeof(DateTime)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf04),
|
||||
[typeof(DateTimeOffset)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf05),
|
||||
[typeof(TimeSpan)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf06),
|
||||
|
||||
[typeof(int)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf07),
|
||||
[typeof(long)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf08),
|
||||
[typeof(short)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf09),
|
||||
[typeof(sbyte)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf10),
|
||||
|
||||
[typeof(uint)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf11),
|
||||
[typeof(ulong)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf12),
|
||||
[typeof(ushort)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf13),
|
||||
[typeof(byte)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf14),
|
||||
[typeof(byte[])] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf15),
|
||||
|
||||
[typeof(double)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf16),
|
||||
[typeof(float)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf17),
|
||||
[typeof(decimal)] = nameof(DynamicField_4bf98fbe2b4d4d14bbb3fc66fa08bf18),
|
||||
};
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
1476
Extensions/FreeSql.Extensions.ZoreEntity/ZoreDbContext.cs
Normal file
1476
Extensions/FreeSql.Extensions.ZoreEntity/ZoreDbContext.cs
Normal file
File diff suppressed because it is too large
Load Diff
97
Extensions/FreeSql.Extensions.ZoreEntity/ZoreDescriptor.cs
Normal file
97
Extensions/FreeSql.Extensions.ZoreEntity/ZoreDescriptor.cs
Normal file
@ -0,0 +1,97 @@
|
||||
using FreeSql.DataAnnotations;
|
||||
using FreeSql.Internal.Model;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
|
||||
namespace FreeSql.Extensions.ZoreEntity
|
||||
{
|
||||
public class TableDescriptor
|
||||
{
|
||||
public string Name { get; set; }
|
||||
public string DbName { get; set; }
|
||||
public string AsTable { get; set; }
|
||||
public bool DisableSyncStructure { get; set; }
|
||||
public string Comment { get; set; }
|
||||
public List<ColumnDescriptor> Columns { get; } = new List<ColumnDescriptor>();
|
||||
public List<NavigateDescriptor> Navigates { get; } = new List<NavigateDescriptor>();
|
||||
public List<IndexDescriptor> Indexes { get; } = new List<IndexDescriptor>();
|
||||
|
||||
public class ColumnDescriptor
|
||||
{
|
||||
public string Name { get; set; }
|
||||
public string DbType { get; set; }
|
||||
bool? _IsPrimary, _IsIdentity, _IsNullable, _IsVersion;
|
||||
public bool IsPrimary { get => _IsPrimary ?? false; set => _IsPrimary = value; }
|
||||
public bool IsIdentity { get => _IsIdentity ?? false; set => _IsIdentity = value; }
|
||||
public bool IsNullable { get => _IsNullable ?? false; set => _IsNullable = value; }
|
||||
public bool IsVersion { get => _IsVersion ?? false; set => _IsVersion = value; }
|
||||
public Type MapType { get; set; }
|
||||
public DateTimeKind ServerTime { get; set; }
|
||||
public string InsertValueSql { get; set; }
|
||||
int? _StringLength;
|
||||
public int StringLength { get => _StringLength ?? 0; set => _StringLength = value; }
|
||||
int? _Precision;
|
||||
public int Precision { get => _Precision ?? 0; set => _Precision = value; }
|
||||
int? _Scale;
|
||||
public int Scale { get => _Scale ?? 0; set => _Scale = value; }
|
||||
public string Comment { get; set; }
|
||||
|
||||
public ColumnAttribute ToAttribute()
|
||||
{
|
||||
var attr = new ColumnAttribute
|
||||
{
|
||||
Name = Name,
|
||||
DbType = DbType,
|
||||
MapType = MapType,
|
||||
ServerTime = ServerTime,
|
||||
InsertValueSql = InsertValueSql,
|
||||
};
|
||||
if (_IsPrimary != null) attr.IsPrimary = IsPrimary;
|
||||
if (_IsIdentity != null) attr.IsIdentity = IsIdentity;
|
||||
if (_IsNullable != null) attr.IsNullable = IsNullable;
|
||||
if (_IsVersion != null) attr.IsVersion = IsVersion;
|
||||
if (_StringLength != null) attr.StringLength = StringLength;
|
||||
if (_Precision != null) attr.Precision = Precision;
|
||||
if (_Scale != null) attr.Scale = Scale;
|
||||
return attr;
|
||||
}
|
||||
}
|
||||
public class IndexDescriptor
|
||||
{
|
||||
public string Name { get; set; }
|
||||
public string Fields { get; set; }
|
||||
public bool IsUnique { get; set; }
|
||||
public IndexMethod IndexMethod { get; set; }
|
||||
}
|
||||
public class NavigateDescriptor
|
||||
{
|
||||
public string Name { get; set; }
|
||||
public NavigateType Type { get; set; }
|
||||
public string RelTable { get; set; }
|
||||
public string Bind { get; set; }
|
||||
public string ManyToMany { get; set; }
|
||||
}
|
||||
public enum NavigateType
|
||||
{
|
||||
OneToOne, ManyToOne, OneToMany, ManyToMany
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
class ZoreTableRef
|
||||
{
|
||||
internal string NavigateKey { get; set; }
|
||||
public TableRefType RefType { get; set; }
|
||||
internal ZoreTableInfo Table { get; set; }
|
||||
internal ZoreTableInfo RefTable { get; set; }
|
||||
internal ZoreTableInfo RefMiddleTable { get; set; }
|
||||
|
||||
public List<string> Columns { get; set; } = new List<string>();
|
||||
public List<string> MiddleColumns { get; set; } = new List<string>();
|
||||
public List<string> RefColumns { get; set; } = new List<string>();
|
||||
}
|
||||
class ZoreTableInfo : TableInfo
|
||||
{
|
||||
public Dictionary<string, ZoreTableRef> Navigates { get; set; } = new Dictionary<string, ZoreTableRef>(StringComparer.OrdinalIgnoreCase);
|
||||
}
|
||||
}
|
BIN
Extensions/FreeSql.Extensions.ZoreEntity/key.snk
Normal file
BIN
Extensions/FreeSql.Extensions.ZoreEntity/key.snk
Normal file
Binary file not shown.
@ -89,6 +89,8 @@ Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "FreeSql.Provider.QuestDb",
|
||||
EndProject
|
||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "FreeSql.Provider.Xugu", "Providers\FreeSql.Provider.Xugu\FreeSql.Provider.Xugu.csproj", "{353F3732-0704-40F2-972B-036E9CC01881}"
|
||||
EndProject
|
||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "FreeSql.Extensions.ZoreEntity", "Extensions\FreeSql.Extensions.ZoreEntity\FreeSql.Extensions.ZoreEntity.csproj", "{4367B7AC-604F-4503-A1D4-643ADBFCF703}"
|
||||
EndProject
|
||||
Global
|
||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||
Debug|Any CPU = Debug|Any CPU
|
||||
@ -495,6 +497,18 @@ Global
|
||||
{353F3732-0704-40F2-972B-036E9CC01881}.Release|x64.Build.0 = Release|Any CPU
|
||||
{353F3732-0704-40F2-972B-036E9CC01881}.Release|x86.ActiveCfg = Release|Any CPU
|
||||
{353F3732-0704-40F2-972B-036E9CC01881}.Release|x86.Build.0 = Release|Any CPU
|
||||
{4367B7AC-604F-4503-A1D4-643ADBFCF703}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{4367B7AC-604F-4503-A1D4-643ADBFCF703}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{4367B7AC-604F-4503-A1D4-643ADBFCF703}.Debug|x64.ActiveCfg = Debug|Any CPU
|
||||
{4367B7AC-604F-4503-A1D4-643ADBFCF703}.Debug|x64.Build.0 = Debug|Any CPU
|
||||
{4367B7AC-604F-4503-A1D4-643ADBFCF703}.Debug|x86.ActiveCfg = Debug|Any CPU
|
||||
{4367B7AC-604F-4503-A1D4-643ADBFCF703}.Debug|x86.Build.0 = Debug|Any CPU
|
||||
{4367B7AC-604F-4503-A1D4-643ADBFCF703}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{4367B7AC-604F-4503-A1D4-643ADBFCF703}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{4367B7AC-604F-4503-A1D4-643ADBFCF703}.Release|x64.ActiveCfg = Release|Any CPU
|
||||
{4367B7AC-604F-4503-A1D4-643ADBFCF703}.Release|x64.Build.0 = Release|Any CPU
|
||||
{4367B7AC-604F-4503-A1D4-643ADBFCF703}.Release|x86.ActiveCfg = Release|Any CPU
|
||||
{4367B7AC-604F-4503-A1D4-643ADBFCF703}.Release|x86.Build.0 = Release|Any CPU
|
||||
EndGlobalSection
|
||||
GlobalSection(SolutionProperties) = preSolution
|
||||
HideSolutionNode = FALSE
|
||||
@ -529,10 +543,11 @@ Global
|
||||
{9D7EA01A-110A-4A0C-A46B-9A0FBC88DD3D} = {94C8A78D-AA15-47B2-A348-530CD86BFC1B}
|
||||
{FEE501EB-60D1-4370-BC65-F939BCA7F32E} = {2A381C57-2697-427B-9F10-55DA11FD02E4}
|
||||
{353F3732-0704-40F2-972B-036E9CC01881} = {2A381C57-2697-427B-9F10-55DA11FD02E4}
|
||||
{4367B7AC-604F-4503-A1D4-643ADBFCF703} = {4A92E8A6-9A6D-41A1-9CDA-DE10899648AA}
|
||||
EndGlobalSection
|
||||
GlobalSection(ExtensibilityGlobals) = postSolution
|
||||
SolutionGuid = {089687FA-5D21-40AC-BA8A-AA0D1E1H7F98}
|
||||
RESX_PrefixTranslations = True
|
||||
RESX_NeutralResourcesLanguage = en-US
|
||||
RESX_PrefixTranslations = True
|
||||
SolutionGuid = {089687FA-5D21-40AC-BA8A-AA0D1E1H7F98}
|
||||
EndGlobalSection
|
||||
EndGlobal
|
||||
|
@ -1104,82 +1104,6 @@
|
||||
</summary>
|
||||
<returns></returns>
|
||||
</member>
|
||||
<member name="T:FreeSql.Extensions.DynamicEntity.DynamicCompileBuilder">
|
||||
<summary>
|
||||
动态创建实体类型
|
||||
</summary>
|
||||
</member>
|
||||
<member name="M:FreeSql.Extensions.DynamicEntity.DynamicCompileBuilder.#ctor(IFreeSql,System.String,System.Attribute[])">
|
||||
<summary>
|
||||
配置Class
|
||||
</summary>
|
||||
<param name="className">类名</param>
|
||||
<param name="attributes">类标记的特性[Table(Name = "xxx")] [Index(xxxx)]</param>
|
||||
<returns></returns>
|
||||
</member>
|
||||
<member name="M:FreeSql.Extensions.DynamicEntity.DynamicCompileBuilder.Property(System.String,System.Type,System.Attribute[])">
|
||||
<summary>
|
||||
配置属性
|
||||
</summary>
|
||||
<param name="propertyName">属性名称</param>
|
||||
<param name="propertyType">属性类型</param>
|
||||
<param name="attributes">属性标记的特性-支持多个</param>
|
||||
<returns></returns>
|
||||
</member>
|
||||
<member name="M:FreeSql.Extensions.DynamicEntity.DynamicCompileBuilder.Property(System.String,System.Type,System.Boolean,System.Attribute[])">
|
||||
<summary>
|
||||
配置属性
|
||||
</summary>
|
||||
<param name="propertyName">属性名称</param>
|
||||
<param name="propertyType">属性类型</param>
|
||||
<param name="isOverride">该属性是否重写父类属性</param>
|
||||
<param name="attributes">属性标记的特性-支持多个</param>
|
||||
<returns></returns>
|
||||
</member>
|
||||
<member name="M:FreeSql.Extensions.DynamicEntity.DynamicCompileBuilder.Property(System.String,System.Type,System.Boolean,System.Object,System.Attribute[])">
|
||||
<summary>
|
||||
配置属性
|
||||
</summary>
|
||||
<param name="propertyName">属性名称</param>
|
||||
<param name="propertyType">属性类型</param>
|
||||
<param name="isOverride">该属性是否重写父类属性</param>
|
||||
<param name="defaultValue">属性默认值</param>
|
||||
<param name="attributes">属性标记的特性-支持多个</param>
|
||||
<returns></returns>
|
||||
</member>
|
||||
<member name="M:FreeSql.Extensions.DynamicEntity.DynamicCompileBuilder.Extend(System.Type)">
|
||||
<summary>
|
||||
配置父类
|
||||
</summary>
|
||||
<param name="superClass">父类类型</param>
|
||||
<returns></returns>
|
||||
</member>
|
||||
<member name="M:FreeSql.Extensions.DynamicEntity.DynamicCompileBuilder.OverrideProperty(System.Reflection.Emit.TypeBuilder@,System.Reflection.Emit.MethodBuilder,FreeSql.Extensions.DynamicEntity.DynamicCompileBuilder.PropertyMethodEnum,System.String)">
|
||||
<summary>
|
||||
Override属性
|
||||
</summary>
|
||||
<param name="typeBuilder"></param>
|
||||
</member>
|
||||
<member name="M:FreeSql.Extensions.DynamicEntity.DynamicCompileBuilder.Build">
|
||||
<summary>
|
||||
Emit动态创建出Class - Type
|
||||
</summary>
|
||||
<returns></returns>
|
||||
</member>
|
||||
<member name="M:FreeSql.Extensions.DynamicEntity.DynamicCompileBuilder.FirstCharToLower(System.String)">
|
||||
<summary>
|
||||
首字母小写
|
||||
</summary>
|
||||
<param name="input"></param>
|
||||
<returns></returns>
|
||||
</member>
|
||||
<member name="M:FreeSql.Extensions.DynamicEntity.DynamicCompileBuilder.FirstCharToUpper(System.String)">
|
||||
<summary>
|
||||
首字母大写
|
||||
</summary>
|
||||
<param name="input"></param>
|
||||
<returns></returns>
|
||||
</member>
|
||||
<member name="M:FreeSql.Extensions.EntityUtil.EntityUtilExtensions.GetEntityKeyString(IFreeSql,System.Type,System.Object,System.Boolean,System.String)">
|
||||
<summary>
|
||||
获取实体的主键值,以 "*|_,[,_|*" 分割,当任意一个主键属性无值时,返回 null
|
||||
@ -5884,28 +5808,6 @@
|
||||
请使用 fsql.InsertDict(dict) 方法插入字典数据
|
||||
</summary>
|
||||
</member>
|
||||
<member name="M:FreeSqlGlobalDynamicEntityExtensions.DynamicEntity(FreeSql.ICodeFirst,System.String,System.Attribute[])">
|
||||
<summary>
|
||||
动态构建Class Type
|
||||
</summary>
|
||||
<returns></returns>
|
||||
</member>
|
||||
<member name="M:FreeSqlGlobalDynamicEntityExtensions.CreateInstance(FreeSql.Internal.Model.TableInfo,System.Collections.Generic.Dictionary{System.String,System.Object})">
|
||||
<summary>
|
||||
根据字典,创建 table 对应的实体对象
|
||||
</summary>
|
||||
<param name="table"></param>
|
||||
<param name="dict"></param>
|
||||
<returns></returns>
|
||||
</member>
|
||||
<member name="M:FreeSqlGlobalDynamicEntityExtensions.CreateDictionary(FreeSql.Internal.Model.TableInfo,System.Object)">
|
||||
<summary>
|
||||
根据实体对象,创建 table 对应的字典
|
||||
</summary>
|
||||
<param name="table"></param>
|
||||
<param name="instance"></param>
|
||||
<returns></returns>
|
||||
</member>
|
||||
<member name="M:FreeSqlGlobalExpressionCallExtensions.Between(System.DateTime,System.DateTime,System.DateTime)">
|
||||
<summary>
|
||||
C#: that >= between && that <= and<para></para>
|
||||
@ -6422,3 +6324,115 @@
|
||||
</member>
|
||||
</members>
|
||||
</doc>
|
||||
`0})">
|
||||
<summary>
|
||||
插入数据,传入实体集合
|
||||
</summary>
|
||||
<typeparam name="T1"></typeparam>
|
||||
<param name="source"></param>
|
||||
<returns></returns>
|
||||
</member>
|
||||
<member name="M:IFreeSql.InsertOrUpdate``1">
|
||||
<summary>
|
||||
插入或更新数据,此功能依赖数据库特性(低版本可能不支持),参考如下:<para></para>
|
||||
MySql 5.6+: on duplicate key update<para></para>
|
||||
PostgreSQL 9.4+: on conflict do update<para></para>
|
||||
SqlServer 2008+: merge into<para></para>
|
||||
Oracle 11+: merge into<para></para>
|
||||
Sqlite: replace into<para></para>
|
||||
达梦: merge into<para></para>
|
||||
人大金仓:on conflict do update<para></para>
|
||||
神通:merge into<para></para>
|
||||
MsAccess:不支持<para></para>
|
||||
注意区别:FreeSql.Repository 仓储也有 InsertOrUpdate 方法(不依赖数据库特性)
|
||||
</summary>
|
||||
<typeparam name="T1"></typeparam>
|
||||
<returns></returns>
|
||||
</member>
|
||||
<member name="M:IFreeSql.Update``1">
|
||||
<summary>
|
||||
修改数据
|
||||
</summary>
|
||||
<typeparam name="T1"></typeparam>
|
||||
<returns></returns>
|
||||
</member>
|
||||
<member name="M:IFreeSql.Update``1(System.Object)">
|
||||
<summary>
|
||||
修改数据,传入动态条件,如:主键值 | new[]{主键值1,主键值2} | TEntity1 | new[]{TEntity1,TEntity2} | new{id=1}
|
||||
</summary>
|
||||
<typeparam name="T1"></typeparam>
|
||||
<param name="dywhere">主键值、主键值集合、实体、实体集合、匿名对象、匿名对象集合</param>
|
||||
<returns></returns>
|
||||
</member>
|
||||
<member name="M:IFreeSql.Select``1">
|
||||
<summary>
|
||||
查询数据
|
||||
</summary>
|
||||
<typeparam name="T1"></typeparam>
|
||||
<returns></returns>
|
||||
</member>
|
||||
<member name="M:IFreeSql.Select``1(System.Object)">
|
||||
<summary>
|
||||
查询数据,传入动态条件,如:主键值 | new[]{主键值1,主键值2} | TEntity1 | new[]{TEntity1,TEntity2} | new{id=1}
|
||||
</summary>
|
||||
<typeparam name="T1"></typeparam>
|
||||
<param name="dywhere">主键值、主键值集合、实体、实体集合、匿名对象、匿名对象集合</param>
|
||||
<returns></returns>
|
||||
</member>
|
||||
<member name="M:IFreeSql.Delete``1">
|
||||
<summary>
|
||||
删除数据
|
||||
</summary>
|
||||
<typeparam name="T1"></typeparam>
|
||||
<returns></returns>
|
||||
</member>
|
||||
<member name="M:IFreeSql.Delete``1(System.Object)">
|
||||
<summary>
|
||||
删除数据,传入动态条件,如:主键值 | new[]{主键值1,主键值2} | TEntity1 | new[]{TEntity1,TEntity2} | new{id=1}
|
||||
</summary>
|
||||
<typeparam name="T1"></typeparam>
|
||||
<param name="dywhere">主键值、主键值集合、实体、实体集合、匿名对象、匿名对象集合</param>
|
||||
<returns></returns>
|
||||
</member>
|
||||
<member name="M:IFreeSql.Transaction(System.Action)">
|
||||
<summary>
|
||||
开启事务(不支持异步)<para></para>
|
||||
v1.5.0 关闭了线程事务超时自动提交的机制
|
||||
</summary>
|
||||
<param name="handler">事务体 () => {}</param>
|
||||
</member>
|
||||
<member name="M:IFreeSql.Transaction(System.Data.IsolationLevel,System.Action)">
|
||||
<summary>
|
||||
开启事务(不支持异步)<para></para>
|
||||
v1.5.0 关闭了线程事务超时自动提交的机制
|
||||
</summary>
|
||||
<param name="isolationLevel"></param>
|
||||
<param name="handler">事务体 () => {}</param>
|
||||
</member>
|
||||
<member name="P:IFreeSql.Ado">
|
||||
<summary>
|
||||
数据库访问对象
|
||||
</summary>
|
||||
</member>
|
||||
<member name="P:IFreeSql.Aop">
|
||||
<summary>
|
||||
所有拦截方法都在这里
|
||||
</summary>
|
||||
</member>
|
||||
<member name="P:IFreeSql.CodeFirst">
|
||||
<summary>
|
||||
CodeFirst 模式开发相关方法
|
||||
</summary>
|
||||
</member>
|
||||
<member name="P:IFreeSql.DbFirst">
|
||||
<summary>
|
||||
DbFirst 模式开发相关方法
|
||||
</summary>
|
||||
</member>
|
||||
<member name="P:IFreeSql.GlobalFilter">
|
||||
<summary>
|
||||
全局过滤设置,可默认附加为 Select/Update/Delete 条件
|
||||
</summary>
|
||||
</member>
|
||||
</members>
|
||||
</doc>
|
||||
|
Loading…
x
Reference in New Issue
Block a user