写在开头

FreeSql 是 .NET 开源生态下的 ORM 轮子,在一些人眼里属于重复造轮子:不看也罢。就像昨天有位朋友截图某培训直播发给我看,内容为:“FreeSQL(个人产品),自己玩可以,不要商用。ORM框架:1.安全、稳定(更新稳定、有BUG有人修复,有人升级)”。

这突出其来的“关爱”,让我的内心毫无波澜,确实是毫无波澜,比起当初 FreeSql 初出茅庐之时的讽刺友好得多。写在开头的这些内容并不祈求这部分人改变观念,该黑的请继续黑,黑总比没有关注好,是吧?我无所谓你,但是别人呢?麻烦你们不要无脑抨击,你们这种行为不知道挽杀了多少社区项目。

2018 年 12 月份开发 FreeSql 到现在,1859 颗星,412 Issues,18 PR,170K 包下载量。说明还是有开发者关注和喜爱,只要有人关注,就不会停更不修 BUG 一说。大家有兴趣可以看看更新记录,看看我们的代码提交量,4700+ 单元测试不说非常多,我个人觉得已经超过很多国产项目,有兴趣的再去隔壁“国产第一” ORM 上看看,对比对比!如果不更新了,请把位置让出来;如果有BUG修复不了,请让 FreeSql 来;如果不好用,就不要搞一堆 SEO 害人入坑;如果。。。如果。。。

这不是挑衅,看到对方的 issues 实在不忍,看到对方的源码,哇哦,单元测试在哪里?好了不废话了。。

20个月了,FreeSql 还活着,而且生命力顽强见下图:

预告:年底发布 2.0.0 版本将冻结新功能开发,不再制造新 BUG,一心修复老功能引出的 BUG,完善文档。

本文将介绍在过去的三个月完成的一些有意义的功能介绍。

入戏准备

FreeSql 是 .Net ORM,能支持 .NetFramework4.0+、.NetCore、Xamarin、XAUI、Blazor、以及还有说不出来的运行平台,因为代码绿色无依赖,支持新平台非常简单。目前单元测试数量:4700+,Nuget下载数量:170K+,源码几乎每天都有提交。值得高兴的是 FreeSql 加入了 ncc 开源社区:https://github.com/dotnetcore/FreeSql,加入组织之后社区责任感更大,需要更努力做好品质,为开源社区出一份力。

QQ群:4336577(已满)、8578575(在线)

为什么要重复造轮子?

FreeSql 主要优势在于易用性上,基本是开箱即用,在不同数据库之间切换兼容性比较好。作者花了大量的时间精力在这个项目,肯请您花半小时了解下项目,谢谢。

FreeSql 整体的功能特性如下:

  • 支持 CodeFirst 对比结构变化迁移;
  • 支持 DbFirst 从数据库导入实体类;
  • 支持 丰富的表达式函数,自定义解析;
  • 支持 批量添加、批量更新、BulkCopy;
  • 支持 导航属性,贪婪加载、延时加载、级联保存;
  • 支持 读写分离、分表分库,租户设计;
  • 支持 MySql/SqlServer/PostgreSQL/Oracle/Sqlite/达梦/神通/人大金仓/MsAccess;

1.5.0 -> 1.8.0-preview 更新的重要功能如下:

一、增加 $"{a.Code}_{a.Id}" lambda 解析;

二、增加 lambda 表达式树解析子查询 ToList + string.Join() 产生 类似 group_concat 的效果;

三、增加 SqlExt 常用开窗函数的自定义表达式解析;

四、完善 WhereDynamicFilter 动态过滤查询;

五、增加 BeginEdit/EndEdit 批量编辑数据的功能;

六、增加 人大金仓/神通 数据库的访问支持;

七、增加 父子表(树表)递归查询、删除功能;

FreeSql 使用非常简单,只需要定义一个 IFreeSql 对象即可:

  1. static IFreeSql fsql = new FreeSql.FreeSqlBuilder()
  2. .UseConnectionString(FreeSql.DataType.MySql, connectionString)
  3. .UseAutoSyncStructure(true) //自动同步实体结构到数据库
  4. .Build(); //请务必定义成 Singleton 单例模式

增加 $"{a.Code}_{a.Id}" lambda 解析;

在之前查询数据的时候,$"" 这种语法糖神器居然不能使用在 lambda 表达式中,实属遗憾。现在终于可以了,如下:

  1. var item = fsql.GetRepository<Topic>().Insert(new Topic { Clicks = 101, Title = "我是中国人101", CreateTime = DateTime.Parse("2020-7-5") });
  2. var sql = fsql.Select<Topic>().WhereDynamic(item).ToSql(a => new
  3. {
  4. str = $"x{a.Id + 1}z-{a.CreateTime.ToString("yyyyMM")}{a.Title}{a.Title}"
  5. });
  6. Assert.Equal($@"SELECT concat('x',ifnull((a.`Id` + 1), ''),'z-',ifnull(date_format(a.`CreateTime`,'%Y%m'), ''),'',ifnull(a.`Title`, ''),'',ifnull(a.`Title`, ''),'') as1
  7. FROM `tb_topic` a
  8. WHERE (a.`Id` = {item.Id})", sql);

再次说明:都是亲儿子,并且都有对应的单元测试,兄台大可放心用在不同的数据库中

增加 lambda 表达式树解析子查询 ToList + string.Join() 产生 类似 group_concat 的效果;

v1.8.0+ string.Join + ToList 实现将子查询的多行结果,拼接为一个字符串,如:"1,2,3,4"

  1. fsql.Select<Topic>().ToList(a => new {
  2. id = a.Id,
  3. concat = string.Join(",", fsql.Select<StringJoin01>().ToList(b => b.Id))
  4. });
  5. //SELECT a.`Id`, (SELECT group_concat(b.`Id` separator ',')
  6. // FROM `StringJoin01` b)
  7. //FROM `Topic` a

该语法,在不同数据库都作了相应的 SQL 翻译。

增加 SqlExt 常用的自定义表达式树解析;

SqlExt.cs 定义了一些常用的表达式树解析,如下:

  1. fsql.Select<T1, T2>()
  2. .InnerJoin((a, b) => b.Id == a.Id)
  3. .ToList((a, b) => new
  4. {
  5. Id = a.Id,
  6. EdiId = b.Id,
  7. over1 = SqlExt.Rank().Over().OrderBy(a.Id).OrderByDescending(b.EdiId).ToValue(),
  8. case1 = SqlExt.Case()
  9. .When(a.Id == 1, 10)
  10. .When(a.Id == 2, 11)
  11. .When(a.Id == 3, 12)
  12. .When(a.Id == 4, 13)
  13. .When(a.Id == 5, SqlExt.Case().When(b.Id == 1, 10000).Else(999).End())
  14. .End(), //这里因为复杂才这样,一般使用三元表达式即可:a.Id == 1 ? 10 : 11
  15. groupct1 = SqlExt.GroupConcat(a.Id).Distinct().OrderBy(b.EdiId).Separator("_").ToValue()
  16. });

本功能利用 FreeSql 自定义解析实现常用表达式树解析,欢迎 PR 补充

完善 WhereDynamicFilter 动态过滤查询

是否见过这样的高级查询功能,WhereDynamicFilter 在后端可以轻松完成这件事情,前端根据 UI 组装好对应的 json 字符串传给后端就行,如下:

  1. DynamicFilterInfo dyfilter = JsonConvert.DeserializeObject<DynamicFilterInfo>(@"
  2. {
  3. ""Logic"" : ""Or"",
  4. ""Filters"" :
  5. [
  6. {
  7. ""Field"" : ""Code"",
  8. ""Operator"" : ""NotContains"",
  9. ""Value"" : ""val1"",
  10. ""Filters"" :
  11. [
  12. {
  13. ""Field"" : ""Name"",
  14. ""Operator"" : ""NotStartsWith"",
  15. ""Value"" : ""val2"",
  16. }
  17. ]
  18. },
  19. {
  20. ""Field"" : ""Parent.Code"",
  21. ""Operator"" : ""Equals"",
  22. ""Value"" : ""val11"",
  23. ""Filters"" :
  24. [
  25. {
  26. ""Field"" : ""Parent.Name"",
  27. ""Operator"" : ""Contains"",
  28. ""Value"" : ""val22"",
  29. }
  30. ]
  31. }
  32. ]
  33. }
  34. ");
  35. fsql.Select<VM_District_Parent>().WhereDynamicFilter(dyfilter).ToList();
  36. //SELECT a.""Code"", a.""Name"", a.""ParentCode"", a__Parent.""Code"" as4, a__Parent.""Name"" as5, a__Parent.""ParentCode"" as6
  37. //FROM ""D_District"" a
  38. //LEFT JOIN ""D_District"" a__Parent ON a__Parent.""Code"" = a.""ParentCode""
  39. //WHERE (not((a.""Code"") LIKE '%val1%') AND not((a.""Name"") LIKE 'val2%') OR a__Parent.""Code"" = 'val11' AND (a__Parent.""Name"") LIKE '%val22%')

ISelect.WhereDynamicFilter 方法实现动态过滤条件(与前端交互),支持的操作符:

  • Contains/StartsWith/EndsWith/NotContains/NotStartsWith/NotEndsWith:包含/不包含,like '%xx%',或者 like 'xx%',或者 like '%xx'
  • Equal/NotEqual:等于/不等于
  • GreaterThan/GreaterThanOrEqual:大于/大于等于
  • LessThan/LessThanOrEqual:小于/小于等于
  • Range:范围查询
  • DateRange:日期范围,有特殊处理 value[1] + 1
  • Any/NotAny:是否符合 value 中任何一项(直白的说是 SQL IN)

增加 BeginEdit/EndEdit 批量编辑数据的功能;

场景:winform 加载表数据后,一顿添加、修改、删除操作之后,点击【保存】

  1. [Fact]
  2. public void BeginEdit()
  3. {
  4. fsql.Delete<BeginEdit01>().Where("1=1").ExecuteAffrows();
  5. var repo = fsql.GetRepository<BeginEdit01>();
  6. var cts = new[] {
  7. new BeginEdit01 { Name = "分类1" },
  8. new BeginEdit01 { Name = "分类1_1" },
  9. new BeginEdit01 { Name = "分类1_2" },
  10. new BeginEdit01 { Name = "分类1_3" },
  11. new BeginEdit01 { Name = "分类2" },
  12. new BeginEdit01 { Name = "分类2_1" },
  13. new BeginEdit01 { Name = "分类2_2" }
  14. }.ToList();
  15. repo.Insert(cts);
  16. repo.BeginEdit(cts); //开始对 cts 进行编辑
  17. cts.Add(new BeginEdit01 { Name = "分类2_3" });
  18. cts[0].Name = "123123";
  19. cts.RemoveAt(1);
  20. Assert.Equal(3, repo.EndEdit());
  21. }
  22. class BeginEdit01
  23. {
  24. public Guid Id { get; set; }
  25. public string Name { get; set; }
  26. }

上面的代码 EndEdit 方法执行的时候产生 3 条 SQL 如下:

  1. INSERT INTO "BeginEdit01"("Id", "Name") VALUES('5f26bf07-6ac3-cbe8-00da-7dd74818c3a6', '分类2_3')
  2. UPDATE "BeginEdit01" SET "Name" = '123123'
  3. WHERE ("Id" = '5f26bf00-6ac3-cbe8-00da-7dd01be76e26')
  4. DELETE FROM "BeginEdit01" WHERE ("Id" = '5f26bf00-6ac3-cbe8-00da-7dd11bcf54dc')

提醒:该操作只对 tags 有效,不是针对全表对比更新。

增加 人大金仓/神通 数据库的访问支持

天津神舟通用数据技术有限公司(简称“神舟通用公司”),隶属于中国航天科技集团(CASC)。是国内从事数据库、大数据解决方案和数据挖掘分析产品研发的专业公司。公司获得了国家核高基科技重大专项重点支持,是核高基专项的牵头承担单位。自1993年在航天科技集团开展数据库研发以来,神通数据库已历经27年的发展历程。公司核心产品主要包括神通关系型数据库、神通KStore海量数据管理系统、神通商业智能套件等系列产品研发和市场销售。基于产品组合,可形成支持交易处理、MPP数据库集群、数据分析与处理等解决方案,可满足多种应用场景需求。产品通过了国家保密局涉密信息系统、公安部等保四级、军B +级等安全评测和认证。

北京人大金仓信息技术股份有限公司(以下简称“人大金仓”)是具有自主知识产权的国产数据管理软件与服务提供商。人大金仓由中国人民大学一批最早在国内开展数据库教学、科研、开发的专家于1999年发起创立,先后承担了国家“863”、“核高基”等重大专项,研发出了具有国际先进水平的大型通用数据库产品。2018年,人大金仓申报的“数据库管理系统核心技术的创新与金仓数据库产业化”项目荣获2018年度国家科学技术进步二等奖,产学研的融合进一步助力国家信息化建设。

随着华为、中兴事务,国产数据库市场相信是未来是趋势走向,纵观 .net core 整个圈子对国产神舟通用、人大金仓数据库的支持几乎为 0,今天 FreeSql ORM 可以使用 CodeFirst/DbFirst 两种模式进行开发。

并且声称:FreeSql 对各数据库没有亲儿子一说,除了 MsAcces 其他全部是亲儿子,在功能提供方面一碗水端平。

众所周知 EFCore for oracle 问题多,并且现在才刚刚更新到 3.x,在这样的背景下,一个国产数据库更不能指望谁实现好用的 EFCore。目前看来除了 EFCore for sqlserver 我们没把握完全占优势,起码在其他数据库肯定是我们更接地气。

使用 FreeSql 访问人大金仓/神通 数据库,只需要修改代码如下即可:

  1. static IFreeSql fsql = new FreeSql.FreeSqlBuilder()
  2. .UseConnectionString(FreeSql.DataType.ShenTong, connectionString) //修改 DataType 设置切换数据库
  3. .UseAutoSyncStructure(true) //自动同步实体结构到数据库
  4. .Build(); //请务必定义成 Singleton 单例模式

增加 父子表(树表)递归查询、删除功能;

无限级分类(父子)是一种比较常用的表设计,每种设计方式突出优势的同时也带来缺陷,如:

  • 方法1:表设计中只有 parent_id 字段,困扰:查询麻烦(本文可解决);
  • 方法2:表设计中冗余子级id便于查询,困扰:添加/更新/删除的时候需要重新计算;
  • 方法3:表设计中存储左右值编码,困扰:同上;

方法1设计最简单,我们正是解决它设计简单,使用复杂的问题。

首先,按照导航属性的定义,定义好父子属性:

  1. public class Area
  2. {
  3. [Column(IsPrimary = true)]
  4. public string Code { get; set; }
  5. public string Name { get; set; }
  6. public virtual string ParentCode { get; set; }
  7. [Navigate(nameof(ParentCode))]
  8. public Area Parent { get; set; }
  9. [Navigate(nameof(ParentCode))]
  10. public List<Area> Childs { get; set; }
  11. }

定义 Parent 属性,在表达式中可以这样:

  1. fsql.Select<Area>().Where(a => a.Parent.Parent.Parent.Name == "中国").First();

定义 Childs 属性,在表达式中可以这样(子查询):

  1. fsql.Select<Area>().Where(a => a.Childs.AsSelect().Any(c => c.Name == "北京")).First();

定义 Childs 属性,还可以使用【级联保存】【贪婪加载】 等等操作。

功能1:ToTreeList

配置好父子属性之后,就可以这样用了:

  1. var t1 = fsql.Select<Area>().ToTreeList();
  2. Assert.Single(t1);
  3. Assert.Equal("100000", t1[0].Code);
  4. Assert.Single(t1[0].Childs);
  5. Assert.Equal("110000", t1[0].Childs[0].Code);
  6. Assert.Equal(2, t1[0].Childs[0].Childs.Count);
  7. Assert.Equal("110100", t1[0].Childs[0].Childs[0].Code);
  8. Assert.Equal("110101", t1[0].Childs[0].Childs[1].Code);

查询数据本来是平面的,ToTreeList 方法将返回的平面数据在内存中加工为树型 List 返回。

功能2:AsTreeCte 递归删除

很常见的无限级分类表功能,删除树节点时,把子节点也处理一下。

  1. fsql.Select<Area>()
  2. .Where(a => a.Name == "中国")
  3. .AsTreeCte()
  4. .ToDelete()
  5. .ExecuteAffrows(); //删除 中国 下的所有记录

如果软删除:

  1. fsql.Select<Area>()
  2. .Where(a => a.Name == "中国")
  3. .AsTreeCte()
  4. .ToUpdate()
  5. .Set(a => a.IsDeleted, true)
  6. .ExecuteAffrows(); //软删除 中国 下的所有记录

功能3:AsTreeCte 递归查询

若不做数据冗余的无限级分类表设计,递归查询少不了,AsTreeCte 正是解决递归查询的封装,方法参数说明:

参数 描述
(可选) pathSelector 路径内容选择,可以设置查询返回:中国 -> 北京 -> 东城区
(可选) up false(默认):由父级向子级的递归查询,true:由子级向父级的递归查询
(可选) pathSeparator 设置 pathSelector 的连接符,默认:->
(可选) level 设置递归层级

通过测试的数据库:MySql8.0、SqlServer、PostgreSQL、Oracle、Sqlite、达梦、人大金仓

姿势一:AsTreeCte() + ToTreeList

  1. var t2 = fsql.Select<Area>()
  2. .Where(a => a.Name == "中国")
  3. .AsTreeCte() //查询 中国 下的所有记录
  4. .OrderBy(a => a.Code)
  5. .ToTreeList(); //非必须,也可以使用 ToList(见姿势二)
  6. Assert.Single(t2);
  7. Assert.Equal("100000", t2[0].Code);
  8. Assert.Single(t2[0].Childs);
  9. Assert.Equal("110000", t2[0].Childs[0].Code);
  10. Assert.Equal(2, t2[0].Childs[0].Childs.Count);
  11. Assert.Equal("110100", t2[0].Childs[0].Childs[0].Code);
  12. Assert.Equal("110101", t2[0].Childs[0].Childs[1].Code);
  13. // WITH "as_tree_cte"
  14. // as
  15. // (
  16. // SELECT 0 as cte_level, a."Code", a."Name", a."ParentCode"
  17. // FROM "Area" a
  18. // WHERE (a."Name" = '中国')
  19. // union all
  20. // SELECT wct1.cte_level + 1 as cte_level, wct2."Code", wct2."Name", wct2."ParentCode"
  21. // FROM "as_tree_cte" wct1
  22. // INNER JOIN "Area" wct2 ON wct2."ParentCode" = wct1."Code"
  23. // )
  24. // SELECT a."Code", a."Name", a."ParentCode"
  25. // FROM "as_tree_cte" a
  26. // ORDER BY a."Code"

姿势二:AsTreeCte() + ToList

  1. var t3 = fsql.Select<Area>()
  2. .Where(a => a.Name == "中国")
  3. .AsTreeCte()
  4. .OrderBy(a => a.Code)
  5. .ToList();
  6. Assert.Equal(4, t3.Count);
  7. Assert.Equal("100000", t3[0].Code);
  8. Assert.Equal("110000", t3[1].Code);
  9. Assert.Equal("110100", t3[2].Code);
  10. Assert.Equal("110101", t3[3].Code);
  11. //执行的 SQL 与姿势一相同

姿势三:AsTreeCte(pathSelector) + ToList

设置 pathSelector 参数后,如何返回隐藏字段?

  1. var t4 = fsql.Select<Area>()
  2. .Where(a => a.Name == "中国")
  3. .AsTreeCte(a => a.Name + "[" + a.Code + "]")
  4. .OrderBy(a => a.Code)
  5. .ToList(a => new {
  6. item = a,
  7. level = Convert.ToInt32("a.cte_level"),
  8. path = "a.cte_path"
  9. });
  10. Assert.Equal(4, t4.Count);
  11. Assert.Equal("100000", t4[0].item.Code);
  12. Assert.Equal("110000", t4[1].item.Code);
  13. Assert.Equal("110100", t4[2].item.Code);
  14. Assert.Equal("110101", t4[3].item.Code);
  15. Assert.Equal("中国[100000]", t4[0].path);
  16. Assert.Equal("中国[100000] -> 北京[110000]", t4[1].path);
  17. Assert.Equal("中国[100000] -> 北京[110000] -> 北京市[110100]", t4[2].path);
  18. Assert.Equal("中国[100000] -> 北京[110000] -> 东城区[110101]", t4[3].path);
  19. // WITH "as_tree_cte"
  20. // as
  21. // (
  22. // SELECT 0 as cte_level, a."Name" || '[' || a."Code" || ']' as cte_path, a."Code", a."Name", a."ParentCode"
  23. // FROM "Area" a
  24. // WHERE (a."Name" = '中国')
  25. // union all
  26. // SELECT wct1.cte_level + 1 as cte_level, wct1.cte_path || ' -> ' || wct2."Name" || '[' || wct2."Code" || ']' as cte_path, wct2."Code", wct2."Name", wct2."ParentCode"
  27. // FROM "as_tree_cte" wct1
  28. // INNER JOIN "Area" wct2 ON wct2."ParentCode" = wct1."Code"
  29. // )
  30. // SELECT a."Code" as1, a."Name" as2, a."ParentCode" as5, a.cte_level as6, a.cte_path as7
  31. // FROM "as_tree_cte" a
  32. // ORDER BY a."Code"

更多姿势...请根据代码注释进行尝试

写在最后

作者的努力,喜欢能打动到你,希望正在使用的、善良的您能动一动小手指,把文章转发一下,让更多人知道 .NET 有这样一个好用的 ORM 存在。谢谢了!!

FreeSql 开源协议 MIT https://github.com/dotnetcore/FreeSql,可以商用,文档齐全。QQ群:4336577(已满)、8578575(在线)

如果你有好的 ORM 实现想法,欢迎给作者留言讨论,谢谢观看!

[开源] .Net ORM FreeSql 1.8.0-preview 最新动态播报(番号:我还活着)的更多相关文章

  1. [开源] .Net ORM FreeSql 1.10.0 稳步向行

    写在开头 FreeSql 是 .NET 开源生态下的 ORM 轮子,转眼快两年了,说真的开源不容易(只有经历过才明白).今天带点干货和湿货给大家,先说下湿货. 认识我的人,知道 CSRedisCore ...

  2. [开源] .Net orm FreeSql 1.5.0 最新版本(番号:好久不见)

    废话开头 这篇文章是我有史以来编辑最长时间的,历时 4小时!!!原本我可以利用这 4小时编写一堆胶水代码,真心希望善良的您点个赞,谢谢了!! 很久很久没有写文章了,上一次还是在元旦发布 1.0 版本的 ...

  3. .NET 开源SqlServer ORM框架 SqlSugar 3.0 API

    3.1.x ,将作为3.X系统的最后一个版本,下面将会开发 全新的功能 更新列表:https://github.com/sunkaixuan/SqlSugar/releases 优点: SqlSuga ...

  4. LitepalNewDemo【开源数据库ORM框架-LitePal2.0.0版本的使用】

    版权声明:本文为HaiyuKing原创文章,转载请注明出处! 前言 本Demo使用的是LitePal2.0.0版本,对于旧项目如何升级到2.0.0版本,请阅读<赶快使用LitePal 2.0版本 ...

  5. C#.NET ORM FreeSql 读取使用 US7ASCII 的 Oracle 数据库中文显示乱码问题

    前言 关于 Oracle US7ASCII 中文乱码的问题,Ado.Net 和 Odbc 无法解决.包括最新的.Net Core..NET6..NET7 都无法解决这个问题. FreeSql 对 Or ...

  6. Blazor WebAssembly 3.2.0 Preview 4 如期发布

    ASP.NET团队如期3.16在官方博客发布了 Blazor WebAssembly 3.2.0 Preview 4:https://devblogs.microsoft.com/aspnet/bla ...

  7. [Android]Android端ORM框架——RapidORM(v2.0)

    以下内容为原创,欢迎转载,转载请注明 来自天天博客:http://www.cnblogs.com/tiantianbyconan/p/5626716.html [Android]Android端ORM ...

  8. fir.im Weekly - 关于 Log Guru 开源、Xcode 探索和 Android7.0 适配

    本期 fir.im Weekly 整理了最近的一些技术分享,包括关于 Log Guru 开源.Xcode 探索. Android7.0 适配等等 iOS/Android 相关的工具.源码分享和技术文章 ...

  9. .Net开源SqlServer ORM框架SqlSugar整理

    一.链接整理 官方Git源代码地址: https://github.com/sunkaixuan/SqlSugar 最新发布版更新地址:当前版本Release 3.5.2.1 https://gith ...

随机推荐

  1. 浅谈服务治理、微服务与Service Mesh(三) Service Mesh与Serverless

    作为本系列文章的第三篇(前两篇<浅谈服务治理.微服务与Service Mesh(一)Dubbo的前世今生>,<浅谈服务治理.微服务与Service Mesh(二) Spring Cl ...

  2. 一起学Blazor WebAssembly 开发(2)

    上篇文章讲了Blazor的两种模式的区别及各自的使用场景,本篇就开始学习WebAssembly模式,本篇主要学习的是创建项目及认识项目结构: 创建项目 选择Blazor应用 选择WebAssembly ...

  3. 前端css 同级元素 设置不同样式 :first-child :nth-child() 的操作收藏

    说明:最近在写前端vue  调样式的时候遇到了一个问题 同一个div下对多个同级别的<span>标签进行 边距设置 <div class="shuju-div"& ...

  4. HRNet + Object Contextual Representation

    文章内容来自CCF-CV走进高校报告会中MSRA王井东老师的报告"Learning high-resolution and object-contextual representations ...

  5. Andriod开发---《横竖屏切换时 Activity的生命周期的总结》

    横屏切换竖屏Activity的生命周期详解,下面分析一下切换时具体的生命周期: 1.新建一个Activity,并把各个生命周期打印出来 2.运行Activity,得到如下信息 onCreate--&g ...

  6. Nginx实现JWT验证-基于OpenResty实现

    介绍 权限认证是接口开发中不可避免的问题,权限认证包括两个方面 接口需要知道调用的用户是谁 接口需要知道该用户是否有权限调用 第1个问题偏向于架构,第2个问题更偏向于业务,因此考虑在架构层解决第1个问 ...

  7. JAVA JDBC Template的使用

    JAVA JDBC Template的使用 什么是Template? Spring框架对JDBC的简单封装.提供了一个JDBCTemplate对象简化JDBC的开发 Template使用步骤 导入ja ...

  8. arcgis for js 如何用contains过滤数据

    添加全部数据 // 构建map容器 var view = new MapView({ container: 'mapId', map: map }); /******************** * ...

  9. Day01_企业权限管理(SSM整合)

    学于黑马程序员和传智播客联合做的教学项目 感谢 黑马程序员官网 传智播客官网 个人根据教程的每天的工作进度的代码和资料 密码:cti5 b站在线视频 微信搜索"艺术行者",关注并回 ...

  10. PHP array_intersect_key() 函数

    实例 比较两个数组的键名,并返回交集: <?php$a1=array("a"=>"red","b"=>"gree ...