SqlSugarTool.cs 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359
  1. using Newtonsoft.Json;
  2. using SKMC.Api.Common.File;
  3. using SKMC.Api.Common.Logger;
  4. using SqlSugar;
  5. using System;
  6. using System.Collections.Generic;
  7. using System.Data;
  8. using System.Linq;
  9. using System.Linq.Expressions;
  10. using System.Text;
  11. using DbType = SqlSugar.DbType;
  12. namespace SKMC.Api.Common.DB
  13. {
  14. /// <summary>
  15. /// SqlSugar的工具类
  16. /// </summary>
  17. public class SqlSugarTool
  18. {
  19. private static readonly ILogger log = LogFactory.Get();
  20. private static readonly AppConfig appConfig = AppConfig.Instance();
  21. /// <summary>
  22. /// 从配置库文件获取数据库实例
  23. /// </summary>
  24. /// <returns></returns>
  25. public static SqlSugarClient GetConfigDB() => GetDB("DB_CONFIG");
  26. /// <summary>
  27. /// 从运行库文件获取数据库实例
  28. /// </summary>
  29. /// <returns></returns>
  30. public static SqlSugarClient GetRuntimeDB() => GetDB("DB_RUNDATA");
  31. /// <summary>
  32. /// 从硬件库文件获取数据库实例
  33. /// </summary>
  34. /// <returns></returns>
  35. public static SqlSugarClient GetDeviceDB() => GetDB("DB_DEVICE");
  36. private static SqlSugarClient GetDB(string dbName)
  37. {
  38. SqlSugarClient db = new SqlSugarClient(new ConnectionConfig()
  39. {
  40. ConnectionString = String.Format(appConfig.ReadStringValue(dbName, "connection.datasource"), System.Environment.CurrentDirectory),
  41. DbType = DbType.Sqlite,
  42. // 自动释放数据务,如果存在事务,在事务结束后释放
  43. IsAutoCloseConnection = appConfig.ReadBoolValue(dbName, "connection.autoclosed"),
  44. // 从数据库系统表读取主键信息中(InitKeyType.Attribute从实体特性中读取主键自增列信息)
  45. InitKeyType = InitKeyType.Attribute
  46. // 暂不设置数据库缓存
  47. });
  48. // 调试SQL,AOP ,日志
  49. db.Aop.OnLogExecuted = (sql, pars) => // SQL执行完事件
  50. {
  51. // 获取执行后的总毫秒数
  52. double sqlExecutionTotalMilliseconds = db.Ado.SqlExecutionTime.TotalMilliseconds;
  53. };
  54. db.Aop.OnLogExecuting = (sql, pars) => // SQL执行前事件。可在这里查看生成的sql
  55. {
  56. string tempSQL = LookSQL(sql, pars);
  57. };
  58. db.Aop.OnError = (exp) =>// 执行SQL 错误事件
  59. {
  60. // exp.sql exp.parameters 可以拿到参数和错误Sql
  61. StringBuilder sb_SugarParameterStr = new StringBuilder("###SugarParameter参数为:");
  62. var parametres = exp.Parametres as SugarParameter[];
  63. if (parametres != null)
  64. {
  65. sb_SugarParameterStr.Append(JsonConvert.SerializeObject(parametres));
  66. }
  67. StringBuilder sb_error = new StringBuilder();
  68. sb_error.AppendLine("SqlSugarClient执行sql异常信息:" + exp.Message);
  69. sb_error.AppendLine("###赋值后sql:" + LookSQL(exp.Sql, parametres));
  70. sb_error.AppendLine("###带参数的sql:" + exp.Sql);
  71. sb_error.AppendLine("###参数信息:" + sb_SugarParameterStr.ToString());
  72. sb_error.AppendLine("###StackTrace信息:" + exp.StackTrace);
  73. log.Error(sb_error.ToString());
  74. };
  75. db.Aop.OnExecutingChangeSql = (sql, pars) => // SQL执行前 可以修改SQL
  76. {
  77. // 判断update或delete方法是否有where条件。如果真的想删除所有数据,请where(p=>true)或where(p=>p.id>0)
  78. if (sql.TrimStart().IndexOf("delete ", StringComparison.CurrentCultureIgnoreCase) == 0)
  79. {
  80. if (sql.IndexOf("where", StringComparison.CurrentCultureIgnoreCase) <= 0)
  81. {
  82. throw new Exception("delete删除方法需要有where条件!!");
  83. }
  84. }
  85. else if (sql.TrimStart().IndexOf("update ", StringComparison.CurrentCultureIgnoreCase) == 0)
  86. {
  87. if (sql.IndexOf("where", StringComparison.CurrentCultureIgnoreCase) <= 0)
  88. {
  89. throw new Exception("update更新方法需要有where条件!!");
  90. }
  91. }
  92. return new KeyValuePair<string, SugarParameter[]>(sql, pars);
  93. };
  94. // db.Aop.OnDiffLogEvent = it =>// 数据库操作前和操作后的数据变化。
  95. // {
  96. // var editBeforeData = it.BeforeData;
  97. // var editAfterData = it.AfterData;
  98. // var sql = it.Sql;
  99. // var parameter = it.Parameters;
  100. // var data = it.BusinessData;
  101. // var time = it.Time;
  102. // var diffType = it.DiffType;// 枚举值 insert 、update 和 delete 用来作业务区分
  103. // 日志方法
  104. // };
  105. // db.Ado.CommandTimeOut // 设置sql执行超时等待时间(毫秒单位)
  106. // db.Ado.Connection.ConnectionTimeout // 设置数据库连接等待时间(毫秒单位)
  107. return db;
  108. }
  109. ///<summary>
  110. ///查看赋值后的sql
  111. ///</summary>
  112. ///<param name="sql"></param>
  113. ///<param name="pars">参数</param>
  114. ///<returns></returns>
  115. private static string LookSQL(string sql, SugarParameter[] pars)
  116. {
  117. if (pars == null || pars.Length == 0) return sql;
  118. StringBuilder sb_sql = new StringBuilder(sql);
  119. var tempOrderPars = pars.Where(p => p.Value != null).OrderByDescending(p => p.ParameterName.Length).ToList();// 防止 @par1错误替换@par12
  120. for (var index = 0; index < tempOrderPars.Count; index++)
  121. {
  122. sb_sql.Replace(tempOrderPars[index].ParameterName, "'" + tempOrderPars[index].Value.ToString() + "'");
  123. }
  124. return sb_sql.ToString();
  125. }
  126. /// <summary>
  127. /// 获取一个实体
  128. /// </summary>
  129. /// <typeparam name="T"></typeparam>
  130. /// <param name="id">主键</param>
  131. /// <param name="db"></param>
  132. /// <returns></returns>
  133. public static T GetModel<T>(object id, SqlSugarClient db)
  134. {
  135. return db.Queryable<T>().InSingle(id);
  136. }
  137. /// <summary>
  138. /// 从缓存中获取一个实体
  139. /// </summary>
  140. /// <typeparam name="T"></typeparam>
  141. /// <param name="id">主键</param>
  142. /// <param name="db"></param>
  143. /// <returns></returns>
  144. public static T GetModel_WithCache<T>(object id, SqlSugarClient db)
  145. {
  146. return db.Queryable<T>().WithCache().InSingle(id);
  147. }
  148. /// <summary>
  149. /// 添加实体数据,并返回主键值(主键为自增int类型id,实体需要设置主键特性)(为null字段不更新,注意model有默认值的情况)
  150. /// </summary>
  151. /// <typeparam name="T"></typeparam>
  152. /// <param name="t"></param>
  153. /// <param name="db">(事务的db对象)</param>
  154. /// <returns></returns>
  155. public static int InsertModel<T>(T t, SqlSugarClient db) where T : class, new()
  156. {
  157. // Where(true/*不插入null值的列*/,false/*不插入主键值*/)
  158. return db.Insertable(t).
  159. // IgnoreColumns(p=>p.Equals("id",StringComparison.InvariantCultureIgnoreCase)).
  160. IgnoreColumns(true, false).ExecuteReturnIdentity();
  161. }
  162. /// <summary>
  163. /// 添加实体数据,并返回主键值(主键为自增long类型id,实体需要设置主键特性)(为null字段不更新,注意model有默认值的情况)
  164. /// </summary>
  165. /// <typeparam name="T"></typeparam>
  166. /// <param name="t"></param>
  167. /// <param name="db">(事务的db对象)</param>
  168. /// <returns></returns>
  169. public static long InsertModel_BigIdentity<T>(T t, SqlSugarClient db) where T : class, new()
  170. {
  171. // Where(true/*不插入null值的列*/,false/*不插入主键值*/)
  172. return db.Insertable(t).
  173. // IgnoreColumns(p=>p.Equals("id",StringComparison.InvariantCultureIgnoreCase)).
  174. IgnoreColumns(true, false).ExecuteReturnBigIdentity();
  175. }
  176. /// <summary>
  177. /// 添加实体数据,删除缓存,并返回主键值(主键为自增int类型id,实体需要设置主键特性)(为null字段不更新,注意model有默认值的情况)
  178. /// </summary>
  179. /// <typeparam name="T"></typeparam>
  180. /// <param name="t"></param>
  181. /// <param name="db">(事务的db对象)</param>
  182. /// <returns></returns>
  183. public static int InsertModel_RemoveDataCache<T>(T t, SqlSugarClient db) where T : class, new()
  184. {
  185. // Where(true/*不插入null值的列*/,false/*不插入主键值*/)
  186. return db.Insertable(t).
  187. // IgnoreColumns(p=>p.Equals("id",StringComparison.InvariantCultureIgnoreCase)).
  188. IgnoreColumns(true, false).RemoveDataCache().ExecuteReturnIdentity();
  189. }
  190. /// <summary>
  191. /// 添加实体数据,并返回主键值(主键为自增long类型id,实体需要设置主键特性)(为null字段不更新,注意model有默认值的情况)
  192. /// </summary>
  193. /// <typeparam name="T"></typeparam>
  194. /// <param name="t"></param>
  195. /// <param name="db">(事务的db对象)</param>
  196. /// <returns></returns>
  197. public static long InsertModel_BigIdentity_RemoveDataCache<T>(T t, SqlSugarClient db) where T : class, new()
  198. {
  199. // Where(true/*不插入null值的列*/,false/*不插入主键值*/)
  200. return db.Insertable(t).
  201. // IgnoreColumns(p=>p.Equals("id",StringComparison.InvariantCultureIgnoreCase)).
  202. IgnoreColumns(true, false).RemoveDataCache().ExecuteReturnBigIdentity();
  203. }
  204. /// <summary>
  205. /// 根据主键更新实体,返回影响条数(实体字段要有主键特性)
  206. /// </summary>
  207. /// <typeparam name="T"></typeparam>
  208. /// <param name="t"></param>
  209. /// <param name="db">(事务的db对象)</param>
  210. /// <returns></returns>
  211. public static int UpdateModelByKey<T>(T t, SqlSugarClient db) where T : class, new()
  212. {
  213. // 字段null,不进行更新
  214. return db.Updateable(t).IgnoreColumns(ignoreAllNullColumns: true).ExecuteCommand();
  215. }
  216. /// <summary>
  217. /// 根据条件表达式更新实体(为null字段不更新,注意model有默认值的情况),返回影响条数
  218. /// </summary>
  219. /// <typeparam name="T"></typeparam>
  220. /// <param name="t"></param>
  221. /// <param name="expressionWhere">条件表达式</param>
  222. /// <param name="db">(事务的db对象)</param>
  223. /// <returns></returns>
  224. public static int UpdateModelsIgnoreNull<T>(T t, Expression<Func<T, bool>> expressionWhere, SqlSugarClient db) where T : class, new()
  225. {
  226. return db.Updateable(t).IgnoreColumns(ignoreAllNullColumns: true).Where(expressionWhere).ExecuteCommand();
  227. }
  228. /// <summary>
  229. /// 根据条件表达式更新实体(指定要更新的列),返回影响条数
  230. /// </summary>
  231. /// <typeparam name="T"></typeparam>
  232. /// <param name="t"></param>
  233. /// <param name="columns"></param>
  234. /// <param name="expressionWhere">条件表达式</param>
  235. /// <param name="db">(事务的db对象)</param>
  236. /// <returns></returns>
  237. public static int UpdateModels<T>(T t, Expression<Func<T, object>> columns, Expression<Func<T, bool>> expressionWhere, SqlSugarClient db) where T : class, new()
  238. {
  239. return db.Updateable(t).UpdateColumns(columns).Where(expressionWhere).ExecuteCommand();
  240. }
  241. /// <summary>
  242. /// 动态更新
  243. /// </summary>
  244. /// <typeparam name="T"></typeparam>
  245. /// <param name="columns">要更新的字段</param>
  246. /// <param name="expressionWhere">where条件表达式</param>
  247. /// <param name="db"></param>
  248. /// <returns></returns>
  249. public static int Update<T>(Expression<Func<T, T>> columns, Expression<Func<T, bool>> expressionWhere, SqlSugarClient db) where T : class, new()
  250. {
  251. return db.Updateable<T>().SetColumns(columns).
  252. // IgnoreColumns(ignoreAllNullColumns: true).// 不能加此方法,会有“CommandText 属性尚未初始化”异常
  253. Where(expressionWhere).ExecuteCommand();
  254. /* 调用示例
  255. Update<Entity.SysAdmin>(p => new Entity.SysAdmin { photo = photo, Password = newPwd }
  256. ,p => p.ID == sm.id && p.Password == oldPwd ,db) > 0;
  257. */
  258. }
  259. /// <summary>
  260. /// 根据主键更新实体,返回影响条数(实体字段要有主键特性)并删除缓存,返回影响条数(实体字段要有主键特性)
  261. /// </summary>
  262. /// <typeparam name="T"></typeparam>
  263. /// <param name="t"></param>
  264. /// <param name="db">(事务的db对象)</param>
  265. /// <returns></returns>
  266. public static int UpdateModelByKey_RemoveDataCache<T>(T t, SqlSugarClient db) where T : class, new()
  267. {
  268. // 字段null,不进行更新
  269. return db.Updateable(t).IgnoreColumns(ignoreAllNullColumns: true).RemoveDataCache().ExecuteCommand();
  270. }
  271. /// <summary>
  272. /// 根据条件表达式更新实体(指定要更新的列)并删除缓存,返回影响条数
  273. /// </summary>
  274. /// <typeparam name="T"></typeparam>
  275. /// <param name="t"></param>
  276. /// <param name="columns"></param>
  277. /// <param name="expressionWhere">条件表达式</param>
  278. /// <param name="db">(事务的db对象)</param>
  279. /// <returns></returns>
  280. public static int UpdateModels_RemoveDataCache<T>(T t, Expression<Func<T, object>> columns, Expression<Func<T, bool>> expressionWhere, SqlSugarClient db) where T : class, new()
  281. {
  282. return db.Updateable(t).UpdateColumns(columns).Where(expressionWhere).RemoveDataCache().ExecuteCommand();
  283. }
  284. /// <summary>
  285. /// 动态更新,并删除缓存
  286. /// </summary>
  287. /// <typeparam name="T"></typeparam>
  288. /// <param name="columns">要更新的字段</param>
  289. /// <param name="expressionWhere">where条件表达式</param>
  290. /// <param name="db"></param>
  291. /// <returns></returns>
  292. public static int Update_RemoveDataCache<T>(Expression<Func<T, T>> columns, Expression<Func<T, bool>> expressionWhere, SqlSugarClient db) where T : class, new()
  293. {
  294. return db.Updateable<T>().SetColumns(columns).
  295. // IgnoreColumns(ignoreAllNullColumns: true).// 不能加此方法,会有“CommandText 属性尚未初始化”异常
  296. Where(expressionWhere).RemoveDataCache().ExecuteCommand();
  297. /* 调用示例
  298. Update<Entity.SysAdmin>(p => new Entity.SysAdmin { photo = photo, Password = newPwd }
  299. ,p => p.ID == sm.id && p.Password == oldPwd ,db) > 0;
  300. */
  301. }
  302. /// <summary>
  303. /// 删除ids集合条件的字符串(高效率写法,注意防止注入攻击)
  304. /// </summary>
  305. /// <param name="ids">ids为字符串 "1,2,3"或"1" 形式</param>
  306. /// <param name="db">(事务的db对象)</param>
  307. /// <param name="key">主键字段</param>
  308. /// <returns></returns>
  309. public static bool DeleteByWhereSql_ids<T>(string ids, SqlSugarClient db, string key = "id") where T : class, new()
  310. {
  311. return db.Deleteable<T>().Where(string.Format(" {0} IN ({1})", key, ids)).ExecuteCommand() > 0;
  312. }
  313. /// <summary>
  314. /// 根据条件表达式删除,返回影响条数
  315. /// </summary>
  316. /// <typeparam name="T"></typeparam>
  317. /// <param name="expression">表达式</param>
  318. /// <param name="db">(事务的db对象)</param>
  319. /// <returns></returns>
  320. public static int DeleteModels<T>(Expression<Func<T, bool>> expression, SqlSugarClient db) where T : class, new()
  321. {
  322. return db.Deleteable<T>().Where(expression).ExecuteCommand();
  323. }
  324. }
  325. }