第一部分: http://www.cnblogs.com/cgzl/p/8478993.html

第二部分: http://www.cnblogs.com/cgzl/p/8481825.html

由于本文主要是讲VSCode开发等, 所以相关等一些angular/.net core的知识就相对少讲点.

我把需求改一下, 如图:

由于efcore目前还是使用中间表来建立多对多的关系, 所以分类标签这个需求就没什么特别的了, 就去掉吧. 还有电视剧分季这个技术上也是重复的, 也删掉.

目前只剩下电视台和电视剧的维护, 还剩下的知识点是:

  • 集合的CRUD操作
  • 对项目结构进行整理, 使用Unit Of Work 以及 Repository 模式
  • 上传文件

由于CRUD画面做起来比较简单, 我相信大家使用该技术的都会, 所以我直接把我写完的代码传上来. 此时页面的操作效果请看视频:

这时的代码:

https://pan.baidu.com/s/1egCmuNT0OxJNwkz0OQ72kA

这里面又一个比较常见的问题, 就是

针对集合的增删改:

上述业务的电视剧的增删改就会出现这种情况:

数据库里面原来有4条数据, 而前台操作完, 删除了第3条数据, 并且增加了一条数据, 其余数据的内容可能有变化.

这种集合类增删改的思路应该是这样的:

1.从传进来的集合找出所有新添加的数据(新添加的数据Id通常并不存在, 这是条件), 放在一个临时的集合, 然后添加到context.

2.从数据库集合找出所有需要删除的数据(有一些id在传进来的集合找不到的就是), 放在一个集合, 然后从conetxt一次性移除.

3.两个集合都有的数据就是只需要修改内容的数据, 更新即可.

下面开始实现这个功能:

首先确保Put方法里, 把电视台下所有的电视剧都读取出来:

接下来, 找到MappingProfile.cs, 使用AutoMapper来实现这个功能.

首先要忽略默认的集合映射:

然后调用AfterMap方法做后期手动处理, 代码的意思大概是这样:

或者, 也可以用Linq做一些重构:

回到画面试一下编辑功能:

然后添加和删除:

查看结果:

OK.

项目重构

下面, 我们使用Unit Of Work以及Repository Pattern对项目进行重构:

由于这部分也很简单, 并且也不是这篇文章的重点, 我就直接贴出重构后的代码吧:

Database/TvRepostiory.cs:

  1. using System.Collections.Generic;
  2. using System.Threading.Tasks;
  3. using Microsoft.EntityFrameworkCore;
  4. using Tv.Models;
  5.  
  6. namespace Tv.Database
  7. {
  8. public class TvRepository : ITvRepostitory
  9. {
  10. private readonly TvContext context;
  11.  
  12. public TvRepository(TvContext context)
  13. {
  14. this.context = context;
  15. }
  16.  
  17. public async Task<List<TvNetwork>> GetTvNetworksAsync()
  18. {
  19. return await context.TvNetworks.Include(x => x.TvShows).ToListAsync();
  20. }
  21.  
  22. public async Task<TvNetwork> GetTvNetworkByIdAsync(int id, bool includeRelated = true)
  23. {
  24. if (includeRelated)
  25. {
  26. return await context.TvNetworks.Include(x => x.TvShows).SingleOrDefaultAsync(x => x.Id == id);
  27. }
  28. return await context.TvNetworks.FindAsync(id);
  29. }
  30.  
  31. public void AddTvNetwork(TvNetwork model)
  32. {
  33. context.TvNetworks.Add(model);
  34. }
  35.  
  36. public void RemoveTvNetwork(TvNetwork model)
  37. {
  38. context.TvNetworks.Remove(model);
  39. }
  40. }
  41. }

Database/ITvRepository.cs:

  1. using System.Collections.Generic;
  2. using System.Threading.Tasks;
  3. using Tv.Models;
  4.  
  5. namespace Tv.Database
  6. {
  7. public interface ITvRepostitory
  8. {
  9. Task<List<TvNetwork>> GetTvNetworksAsync();
  10. Task<TvNetwork> GetTvNetworkByIdAsync(int id, bool includeRelated = true);
  11. void AddTvNetwork(TvNetwork model);
  12. void RemoveTvNetwork(TvNetwork model);
  13. }
  14. }

Database/UnitOfWork.cs:

  1. using System.Threading.Tasks;
  2.  
  3. namespace Tv.Database
  4. {
  5. public class UnitOfWork : IUnitOfWork
  6. {
  7. private readonly TvContext context;
  8.  
  9. public UnitOfWork(TvContext context)
  10. {
  11. this.context = context;
  12. }
  13.  
  14. public async Task SaveAsync()
  15. {
  16. await context.SaveChangesAsync();
  17. }
  18. }
  19. }

Database/IUnitOfWork.cs:

  1. using System.Threading.Tasks;
  2.  
  3. namespace Tv.Database
  4. {
  5. public interface IUnitOfWork
  6. {
  7. Task SaveAsync();
  8. }
  9. }

Startup.cs:

  1. public void ConfigureServices(IServiceCollection services)
  2. {
  3. services.AddAutoMapper();
  4. // services.AddDbContext<TvContext>(opt => opt.UseSqlServer(Configuration["ConnectionStrings:Default"]));
  5. services.AddDbContext<TvContext>(opt => opt.UseSqlServer(Configuration.GetConnectionString("Default")));
  6. services.AddScoped<ITvRepostitory, TvRepository>();
  7. services.AddScoped<IUnitOfWork, UnitOfWork>();
  8. services.AddMvc();
  9. }

TvController.cs:

  1. using System.Collections.Generic;
  2. using System.Threading.Tasks;
  3. using AutoMapper;
  4. using Microsoft.AspNetCore.Mvc;
  5. using Microsoft.EntityFrameworkCore;
  6. using Tv.Database;
  7. using Tv.Models;
  8. using Tv.ViewModels;
  9.  
  10. namespace Tv.Controllers
  11. {
  12. public class TvController : Controller
  13. {
  14. private readonly ITvRepostitory repostiory;
  15. private readonly IUnitOfWork unitOfWork;
  16. private readonly IMapper mapper;
  17.  
  18. public TvController(ITvRepostitory repostiory, IUnitOfWork unitOfWork, IMapper mapper)
  19. {
  20. this.repostiory = repostiory;
  21. this.unitOfWork = unitOfWork;
  22. this.mapper = mapper;
  23. }
  24.  
  25. [HttpGet("api/tvnetworks")]
  26. public async Task<IEnumerable<TvNetworkViewModel>> GetTvNetworks()
  27. {
  28. var models = await repostiory.GetTvNetworksAsync();
  29. var vms = mapper.Map<List<TvNetwork>, List<TvNetworkViewModel>>(models);
  30. return vms;
  31. }
  32.  
  33. [HttpGet("api/tvnetworks/{id}")]
  34. public async Task<IActionResult> Get(int id)
  35. {
  36. var model = await repostiory.GetTvNetworkByIdAsync(id);
  37. var vm = mapper.Map<TvNetwork, TvNetworkViewModel>(model);
  38. return Ok(vm);
  39. }
  40.  
  41. [HttpPost("api/tvnetworks")]
  42. public async Task<IActionResult> Post([FromBody]TvNetworkUpdateViewModel vm)
  43. {
  44. if (!ModelState.IsValid)
  45. {
  46. return BadRequest(ModelState);
  47. }
  48. var model = mapper.Map<TvNetworkUpdateViewModel, TvNetwork>(vm);
  49. repostiory.AddTvNetwork(model);
  50. await unitOfWork.SaveAsync();
  51. var result = mapper.Map<TvNetwork, TvNetworkViewModel>(model);
  52. return Ok(result);
  53. }
  54.  
  55. [HttpPut("api/tvnetworks/{id}")]
  56. public async Task<IActionResult> Put(int id, [FromBody]TvNetworkUpdateViewModel vm)
  57. {
  58. if (!ModelState.IsValid)
  59. {
  60. return BadRequest(ModelState);
  61. }
  62. var dbModel = await repostiory.GetTvNetworkByIdAsync(id);
  63. if (dbModel == null)
  64. {
  65. return NotFound();
  66. }
  67. var model = mapper.Map<TvNetworkUpdateViewModel, TvNetwork>(vm, dbModel);
  68. await unitOfWork.SaveAsync();
  69. var result = mapper.Map<TvNetwork, TvNetworkViewModel>(model);
  70. return Ok(result);
  71. }
  72.  
  73. [HttpDelete("api/tvnetworks/{id}")]
  74. public async Task<IActionResult> Delete(int id)
  75. {
  76. var model = await repostiory.GetTvNetworkByIdAsync(id, includeRelated: false);
  77. if (model == null)
  78. {
  79. return NotFound();
  80. }
  81. repostiory.RemoveTvNetwork(model);
  82. await unitOfWork.SaveAsync();
  83. return NoContent();
  84. }
  85. }
  86. }

再操作下画面, 没有任何问题.

今天先写到这, VSCode的开发速度还是非常快的.

还剩下最后一部分--上传文件.

用VSCode开发一个基于asp.net core 2.0/sql server linux(docker)/ng5/bs4的项目(3)的更多相关文章

  1. 用VSCode开发一个基于asp.net core 2.0/sql server linux(docker)/ng5/bs4的项目(1)

    最近使用vscode比较多. 学习了一下如何在mac上使用vscode开发asp.netcore项目. 这里是我写的关于vscode的一篇文章: https://www.cnblogs.com/cgz ...

  2. 用VSCode开发一个基于asp.net core 2.0/sql server linux(docker)/ng5/bs4的项目(2)

    第一部分: http://www.cnblogs.com/cgzl/p/8478993.html 为Domain Model添加约束 前一部分, 我们已经把数据库创建出来了. 那么我们先看看这个数据库 ...

  3. [译]基于ASP.NET Core 3.0的ABP v0.21已发布

    基于ASP.NET Core 3.0的ABP v0.21已发布 在微软发布仅仅一个小时后, 基于ASP.NET Core 3.0的ABP v0.21也紧跟着发布了. v0.21没有新功能.它只是升级到 ...

  4. 基于ASP.NET Core 3.0快速搭建Razor Pages Web应用

    前言 虽然说学习新的开发框架是一项巨大的投资,但是作为一个开发人员,不断学习新的技术并快速上手是我们应该掌握的技能,甚至是一个.NET Framework开发人员,学习.NET Core 新框架可以更 ...

  5. 基于ASP.NET Core 6.0的整洁架构

    大家好,我是张飞洪,感谢您的阅读,我会不定期和你分享学习心得,希望我的文章能成为你成长路上的垫脚石,让我们一起精进. 本节将介绍基于ASP.NET Core的整洁架构的设计理念,同时基于理论落地的代码 ...

  6. 打造静态分析器(二)基于Asp.Net Core 3.0的AspectCore组件检测

    上一篇,我们打造了一个简单的分析器,但是我们实际使用分析器就是为了对项目做分析检测,增加一些非语法的自检的 比如Asp.Net Core 3.0的替换依赖注入检测 设计分析 我们创建一个默认的Asp. ...

  7. 基于Asp.Net Core 5.0依赖Quartz.Net框架编写的任务调度web管理平台

    源码地址: https://github.com/246850/Calamus.TaskScheduler 演示地址:http://47.101.47.193:1063/ 1.Quartz.NET框架 ...

  8. ASP.NET Core 监听SQL Server数据库的实时信息

    1.开发环境: 开发工具:Visual Studio 2019 数据库:SQL Server2012 开发环境:.Net Core 3.1 2.使用技术: Signalr:实现消息推送 SqlDepe ...

  9. Asp.Net Core采用MailKit部署到Linux Docker连接邮件服务器报错

    前段时间看文章了解到发邮件的SmtpClient已经过时了,微软官方推荐大家用其他解决方案,例如MailKit. https://docs.microsoft.com/zh-cn/dotnet/api ...

随机推荐

  1. 重写equals()和hashCode()

    什么时候需要重写equals()? 只有当一个实例等于它本身的时候,equals()才会返回true值.通俗地说,此时比较的是两个引用是否指向内存中的同一个对象,也可以称做是否实例相 等.而我们在使用 ...

  2. linux scp远程拷贝文件及文件夹

    [http://www.jb51.net/LINUXjishu/73131.html] 1.拷贝本机/home/administrator/test整个目录至远程主机192.168.1.100的/ro ...

  3. iOS实现微信外部H5支付完成后返回原APP

    看到微信最近放开了微信H5支付,公司决定把H5集成到多款APP上.下面记录下了开发过程. 由于是微信新推出的支付方式,在网上搜索到的相关资料并不多,其中有一篇文件(点此跳转)对我的整个开发过程起到了很 ...

  4. springboot入门_模板

    springboot中已经不推荐使用jsp,而是推荐使用模板,如freemarker,thymeleaf等,本文记录在sprigboot中使用模板. 创建一个maven的springboot工程, f ...

  5. jQuery(function(){...})与(function($){...})(jQuery)的“兄弟”情结

    记得那时在学习写基于jQuery的插件时,了解到(function($){...})(jQuery)的代码结构,一开始还没发觉,后来百度了解它的语意时,从搜索结果中发现了jQuery(function ...

  6. Java上传Excel并解析

    1.上传: public String uploadFile(CommonsMultipartFile file,String uploadPath,String realUploadPath){ I ...

  7. 深入研究Sphinx的底层原理和高级使用

    深入研究Sphinx的底层原理和高级使用

  8. PHP中单引号与双引号的区别

    在PHP中,字符串的定义可以使用英文单引号' ',也可以使用英文双引号" ". 一般情况下两者是通用的.但双引号内部变量会解析,单引号则不解析. PHP允许我们在双引号串中直接包含 ...

  9. 基于epoll实现简单的web服务器

    1. 简介 epoll 是 Linux 平台下特有的一种 I/O 复用模型实现,于 2002 年在 Linux kernel 2.5.44 中被引入.在 epoll 之前,Unix/Linux 平台下 ...

  10. 4.3 lambda表达式

    函数,封装的代码块可以很复杂,也可以很简单.当函数的代码块简单到只有一个表达式,就可以考虑用lambda表达式,也称匿名函数. 1 lambda表达式基础 #首先需要声明一点:lambda是表达式而非 ...