用VSCode开发一个基于asp.net core 2.0/sql server linux(docker)/ng5/bs4的项目(3)
第一部分: 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:
using System.Collections.Generic;
using System.Threading.Tasks;
using Microsoft.EntityFrameworkCore;
using Tv.Models; namespace Tv.Database
{
public class TvRepository : ITvRepostitory
{
private readonly TvContext context; public TvRepository(TvContext context)
{
this.context = context;
} public async Task<List<TvNetwork>> GetTvNetworksAsync()
{
return await context.TvNetworks.Include(x => x.TvShows).ToListAsync();
} public async Task<TvNetwork> GetTvNetworkByIdAsync(int id, bool includeRelated = true)
{
if (includeRelated)
{
return await context.TvNetworks.Include(x => x.TvShows).SingleOrDefaultAsync(x => x.Id == id);
}
return await context.TvNetworks.FindAsync(id);
} public void AddTvNetwork(TvNetwork model)
{
context.TvNetworks.Add(model);
} public void RemoveTvNetwork(TvNetwork model)
{
context.TvNetworks.Remove(model);
}
}
}
Database/ITvRepository.cs:
using System.Collections.Generic;
using System.Threading.Tasks;
using Tv.Models; namespace Tv.Database
{
public interface ITvRepostitory
{
Task<List<TvNetwork>> GetTvNetworksAsync();
Task<TvNetwork> GetTvNetworkByIdAsync(int id, bool includeRelated = true);
void AddTvNetwork(TvNetwork model);
void RemoveTvNetwork(TvNetwork model);
}
}
Database/UnitOfWork.cs:
using System.Threading.Tasks; namespace Tv.Database
{
public class UnitOfWork : IUnitOfWork
{
private readonly TvContext context; public UnitOfWork(TvContext context)
{
this.context = context;
} public async Task SaveAsync()
{
await context.SaveChangesAsync();
}
}
}
Database/IUnitOfWork.cs:
using System.Threading.Tasks; namespace Tv.Database
{
public interface IUnitOfWork
{
Task SaveAsync();
}
}
Startup.cs:
public void ConfigureServices(IServiceCollection services)
{
services.AddAutoMapper();
// services.AddDbContext<TvContext>(opt => opt.UseSqlServer(Configuration["ConnectionStrings:Default"]));
services.AddDbContext<TvContext>(opt => opt.UseSqlServer(Configuration.GetConnectionString("Default")));
services.AddScoped<ITvRepostitory, TvRepository>();
services.AddScoped<IUnitOfWork, UnitOfWork>();
services.AddMvc();
}
TvController.cs:
using System.Collections.Generic;
using System.Threading.Tasks;
using AutoMapper;
using Microsoft.AspNetCore.Mvc;
using Microsoft.EntityFrameworkCore;
using Tv.Database;
using Tv.Models;
using Tv.ViewModels; namespace Tv.Controllers
{
public class TvController : Controller
{
private readonly ITvRepostitory repostiory;
private readonly IUnitOfWork unitOfWork;
private readonly IMapper mapper; public TvController(ITvRepostitory repostiory, IUnitOfWork unitOfWork, IMapper mapper)
{
this.repostiory = repostiory;
this.unitOfWork = unitOfWork;
this.mapper = mapper;
} [HttpGet("api/tvnetworks")]
public async Task<IEnumerable<TvNetworkViewModel>> GetTvNetworks()
{
var models = await repostiory.GetTvNetworksAsync();
var vms = mapper.Map<List<TvNetwork>, List<TvNetworkViewModel>>(models);
return vms;
} [HttpGet("api/tvnetworks/{id}")]
public async Task<IActionResult> Get(int id)
{
var model = await repostiory.GetTvNetworkByIdAsync(id);
var vm = mapper.Map<TvNetwork, TvNetworkViewModel>(model);
return Ok(vm);
} [HttpPost("api/tvnetworks")]
public async Task<IActionResult> Post([FromBody]TvNetworkUpdateViewModel vm)
{
if (!ModelState.IsValid)
{
return BadRequest(ModelState);
}
var model = mapper.Map<TvNetworkUpdateViewModel, TvNetwork>(vm);
repostiory.AddTvNetwork(model);
await unitOfWork.SaveAsync();
var result = mapper.Map<TvNetwork, TvNetworkViewModel>(model);
return Ok(result);
} [HttpPut("api/tvnetworks/{id}")]
public async Task<IActionResult> Put(int id, [FromBody]TvNetworkUpdateViewModel vm)
{
if (!ModelState.IsValid)
{
return BadRequest(ModelState);
}
var dbModel = await repostiory.GetTvNetworkByIdAsync(id);
if (dbModel == null)
{
return NotFound();
}
var model = mapper.Map<TvNetworkUpdateViewModel, TvNetwork>(vm, dbModel);
await unitOfWork.SaveAsync();
var result = mapper.Map<TvNetwork, TvNetworkViewModel>(model);
return Ok(result);
} [HttpDelete("api/tvnetworks/{id}")]
public async Task<IActionResult> Delete(int id)
{
var model = await repostiory.GetTvNetworkByIdAsync(id, includeRelated: false);
if (model == null)
{
return NotFound();
}
repostiory.RemoveTvNetwork(model);
await unitOfWork.SaveAsync();
return NoContent();
}
}
}
再操作下画面, 没有任何问题.
今天先写到这, VSCode的开发速度还是非常快的.
还剩下最后一部分--上传文件.
用VSCode开发一个基于asp.net core 2.0/sql server linux(docker)/ng5/bs4的项目(3)的更多相关文章
- 用VSCode开发一个基于asp.net core 2.0/sql server linux(docker)/ng5/bs4的项目(1)
最近使用vscode比较多. 学习了一下如何在mac上使用vscode开发asp.netcore项目. 这里是我写的关于vscode的一篇文章: https://www.cnblogs.com/cgz ...
- 用VSCode开发一个基于asp.net core 2.0/sql server linux(docker)/ng5/bs4的项目(2)
第一部分: http://www.cnblogs.com/cgzl/p/8478993.html 为Domain Model添加约束 前一部分, 我们已经把数据库创建出来了. 那么我们先看看这个数据库 ...
- [译]基于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没有新功能.它只是升级到 ...
- 基于ASP.NET Core 3.0快速搭建Razor Pages Web应用
前言 虽然说学习新的开发框架是一项巨大的投资,但是作为一个开发人员,不断学习新的技术并快速上手是我们应该掌握的技能,甚至是一个.NET Framework开发人员,学习.NET Core 新框架可以更 ...
- 基于ASP.NET Core 6.0的整洁架构
大家好,我是张飞洪,感谢您的阅读,我会不定期和你分享学习心得,希望我的文章能成为你成长路上的垫脚石,让我们一起精进. 本节将介绍基于ASP.NET Core的整洁架构的设计理念,同时基于理论落地的代码 ...
- 打造静态分析器(二)基于Asp.Net Core 3.0的AspectCore组件检测
上一篇,我们打造了一个简单的分析器,但是我们实际使用分析器就是为了对项目做分析检测,增加一些非语法的自检的 比如Asp.Net Core 3.0的替换依赖注入检测 设计分析 我们创建一个默认的Asp. ...
- 基于Asp.Net Core 5.0依赖Quartz.Net框架编写的任务调度web管理平台
源码地址: https://github.com/246850/Calamus.TaskScheduler 演示地址:http://47.101.47.193:1063/ 1.Quartz.NET框架 ...
- ASP.NET Core 监听SQL Server数据库的实时信息
1.开发环境: 开发工具:Visual Studio 2019 数据库:SQL Server2012 开发环境:.Net Core 3.1 2.使用技术: Signalr:实现消息推送 SqlDepe ...
- Asp.Net Core采用MailKit部署到Linux Docker连接邮件服务器报错
前段时间看文章了解到发邮件的SmtpClient已经过时了,微软官方推荐大家用其他解决方案,例如MailKit. https://docs.microsoft.com/zh-cn/dotnet/api ...
随机推荐
- ZOJ1450 BZOJ1136 BZOJ1137 HDU3932[最小圆覆盖]
Minimal Circle Time Limit: 5 Seconds Memory Limit: 32768 KB You are to write a program to find ...
- POJ 1584 A Round Peg in a Ground Hole[判断凸包 点在多边形内]
A Round Peg in a Ground Hole Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 6682 Acc ...
- HibernateTemplate#setMaxResults()的坑
如题所示,因为maxResults为实例属性,一经修改,则会应用于所有查询! 不是很清楚,Spring为什么要挖这个坑? 我的解决方案:如需设置返回结果集的大小,则通过HibernateCallbac ...
- 在.NetCore中使用Myrmec检测文件真实格式
Myrmec 是什么? Myrmec 是一个用于检测文件格式的库,Myrmec不同于其它库或者手写检测代码,Myrmec不依赖文件扩展名(在实际使用中,你的用户很可能使用虚假的扩展名欺骗你的应用程序) ...
- 小甲鱼OD学习第2讲
这次我们的任务是让我们输入任意用户名密码判断正确 我们输入fishc和111111,显示错误 我们猜测这是用GetDlgItemTextW来收集账号密码的输入值 我们找到了两个函数,给这两个函数都下断 ...
- php语言基础(一)
一.php标记风格 1.1xml风格(常用) <?php echo "xml风格": ?> 1.2脚本风格 <script language="php& ...
- [工具]Microsoft To-Do,简约还是简陋?
1. 简介 微软收购奇妙清单后,由奇妙清单的原班人马打造了一个全新的待办事项应用,就叫"To-Do"(简单粗暴,好像新浪微博直接就叫"微博"的感觉).这个应该刚 ...
- .addClass(),.removeClass(),.toggleClass()的区别
.addClass("className")方法是用来给指定元素增加类名,也就是说给指定的元素追加样式: 可以同时添加多个类名,空格符隔开 $("selector&quo ...
- java4 - 函数(方法)
一.学习大纲: 1. 定义函数可以将功能封装 2. 函数的级别都是同级别的,不能进行函数套用 3. 便于对该功能进行复用 4. 函数只有被调用才能被执行 5. 函数的出现提高了代码的复用性 6. 函数 ...
- nyoj 1022 合纵连横 经典并查集
思路:关键在于并查集的删点操作. 给每个诸侯国一个另外的编号,比如box[i]表示诸侯国i现在处于第box[i]个联盟,可以随时改变它的联盟编号,并且让box[i] = k, 实现删除操作.以前联盟中 ...