介绍

很久没有更新博客了,之前想更新但是发现博客园崩了,外加工作上的调换也比较忙,最近有了点时间我来继续更新下这个系列的文章。

今年3月份我带着我们研发组同事,将公司产品从老Abp重构到Abp vNext目前已经上线,我非常确认Abp vNext完全可以应用到生产环境,并且他足以支撑超复杂业务的系统开发。

很多人提到Abp就想到DDD,这很好,但是Abp并不是要求你一定要DDD,很多人在社区在群里说你不DDD你用Abp还不如不用,Abp你用来开发系统太重了,我其实很像问一下说出这些话的人,你深入用过Abp嘛?你用它开发过复杂的业务系统嘛?你能保证你现在使用的系统能够持久更新嘛?你说Abp很重我就想知道他重的点在哪里?(此处欢迎大佬找我探讨)

这次连载我将由浅入深的来给大家把 Abp vNext 过一遍,该项目作为老张的哲学Blog.Core博客系统姊妹篇,用比较简单的业务来演示你该如何开始使用Abp vNext,DDD我也会涉及到,如果时间允许IdentityServer和模块化我也会讲,反正就是过一遍让你不要太触怕这个东西,话不多说咱们还是直接上项目比较实在。

开始

Abp官网:https://abp.io/

直接创建项目,项目名称:Blog.Core.AbpvNext 我完全是为了看着方便这么叫,注意我选择的Ui是Angular,采用方案是将服务器端和授权分离为两个应用程序,如下图所示。

项目下载下来后修改DbMigrator、Host、IdentityServer的appsettings.json数据库连接字符串。

然后启动DbMigrator生成数据库,生成出来的数据库如下图所示,数据库表包含IdentityServer、用户、角色、组织、审计、Entity、安全、Feature、权限、日志、后台任务、设置等这几类。

项目介绍

我之前写过一个博客站点,如图所示我就根据这个博客站点来简单的分析下模型,模型如下图所示,后面我们会根据业务在进行修改。

创建模型

根据上面的模型把Entity创建一下,目录结构看下图

  public class SiteUser : FullAuditedAggregateRoot<Guid>
{
/// <summary>
/// 用户名
/// </summary>
public string UserName { get; set; }
/// <summary>
/// 用户
/// </summary>
public string Name { get; set; }
/// <summary>
/// 密码
/// </summary>
public string PassWord { get; set; }
/// <summary>
/// 头像
/// </summary>
public string HeadPortrait { get; set; }
/// <summary>
/// 邮箱
/// </summary>
public string Email { get; set; }
/// <summary>
/// 介绍
/// </summary>
public string Introduce { get; set; }
}
    public class Question:FullAuditedAggregateRoot<Guid>
{
/// <summary>
/// 标题
/// </summary>
public string Title { get; set; }
/// <summary>
/// 内容
/// </summary>
public string Content { get; set; }
/// <summary>
/// 类别
/// </summary>
public string Tag { get; set; }
/// <summary>
/// 访问量
/// </summary>
public int Traffic { get; set; } /// <summary>
/// 问答评论
/// </summary>
public virtual ICollection<QuestionComment> QuestionComments { get; set; }
}
    public class QuestionComment:FullAuditedAggregateRoot<Guid>
{
/// <summary>
/// 内容
/// </summary>
public string Content { get; set; } /// <summary>
/// 是否采纳
/// </summary>
public bool IsAdoption { get; set; } /// <summary>
/// 问答信息
/// </summary>
public virtual Question Question { get; set; }
}
    /// <summary>
/// 文章
/// </summary>
public class Article: FullAuditedAggregateRoot<Guid>
{
/// <summary>
/// 标题
/// </summary>
public string Title { get; set; }
/// <summary>
/// 封面
/// </summary>
public string Cover { get; set; }
/// <summary>
/// 内容
/// </summary>
public string Content { get; set; }
/// <summary>
/// 类别
/// </summary>
public string Tag { get; set; }
/// <summary>
/// 访问量
/// </summary>
public int Traffic { get; set; }
/// <summary>
/// 文章评论
/// </summary>
public virtual ICollection<ArticleComment> ArticleComments { get; set; } = new List<ArticleComment>(); }
    public class ArticleComment:FullAuditedAggregateRoot<Guid>
{
/// <summary>
/// 内容
/// </summary>
public string Content { get; set; } /// <summary>
/// 问答Id
/// </summary>
public Guid QuestionId { get; set; }
}

加入上下文并创建映射

在 EntityFrameworkCore层 AbpvNextDbContext 中 将Entity加入到上下文。


public class AbpvNextDbContext : AbpDbContext<AbpvNextDbContext>
{ public DbSet<Article> Articles { get; set; } public DbSet<ArticleComment> ArticleComments { get; set; } public DbSet<Question> Questions { get; set; } public DbSet<QuestionComment> QuestionComments { get; set; } public DbSet<SiteUser> SiteUsers { get; set; }
}

创建 EntityConfigurationGroup 文件夹,创建数据映射配置,如图所示

public class SiteUserCfg : IEntityTypeConfiguration<SiteUser>
{
public void Configure(EntityTypeBuilder<SiteUser> builder)
{
builder.ToTable(AbpvNextConsts.DbTablePrefix + "SiteUser", AbpvNextConsts.DbSchema);
builder.ConfigureByConvention(); builder.Property(e => e.UserName).HasMaxLength(128);
builder.Property(e => e.Name).HasMaxLength(128);
builder.Property(e => e.PassWord).HasMaxLength(256);
builder.Property(e => e.Email).HasMaxLength(128);
builder.Property(e => e.HeadPortrait).HasMaxLength(512);
builder.Property(e => e.Introduce).HasMaxLength(1024); }
}
 public  class ArticleCfg : IEntityTypeConfiguration<Article>
{
public void Configure(EntityTypeBuilder<Article> builder)
{
builder.ToTable(AbpvNextConsts.DbTablePrefix + "Article", AbpvNextConsts.DbSchema);
builder.ConfigureByConvention(); builder.Property(e => e.Title).HasMaxLength(128);
builder.Property(e => e.Cover).HasMaxLength(1024);
// builder.Property(e => e.Content).HasMaxLength(128);
builder.Property(e => e.Tag).HasMaxLength(128); builder.HasMany(e => e.ArticleComments).WithOne()
.HasForeignKey(x => x.ArticleId).IsRequired(false);
}
}
  public class ArticleCommentCfg : IEntityTypeConfiguration<ArticleComment>
{
public void Configure(EntityTypeBuilder<ArticleComment> builder)
{
builder.ToTable(AbpvNextConsts.DbTablePrefix + "ArticleComment", AbpvNextConsts.DbSchema);
builder.ConfigureByConvention(); builder.Property(e => e.Content).HasMaxLength(1024); }
}
    public class QuestionCfg : IEntityTypeConfiguration<Question>
{
public void Configure(EntityTypeBuilder<Question> builder)
{
builder.ToTable(AbpvNextConsts.DbTablePrefix + "Question", AbpvNextConsts.DbSchema);
builder.ConfigureByConvention(); builder.Property(e => e.Title).HasMaxLength(128);
// builder.Property(e => e.Content).HasMaxLength(128);
builder.Property(e => e.Tag).HasMaxLength(128); builder.HasMany(e => e.QuestionComments).WithOne()
.HasForeignKey(x => x.QuestionId).IsRequired(false);
}
}
    public class QuestionCommentCfg : IEntityTypeConfiguration<QuestionComment>
{
public void Configure(EntityTypeBuilder<QuestionComment> builder)
{ builder.ToTable(AbpvNextConsts.DbTablePrefix + "QuestionComment", AbpvNextConsts.DbSchema);
builder.ConfigureByConvention(); builder.Property(e => e.Content).HasMaxLength(1024);
}
}

加入到配置上下文

    public static class AbpvNextDbContextModelCreatingExtensions
{
public static void ConfigureAbpvNext(this ModelBuilder builder)
{
Check.NotNull(builder, nameof(builder)); // 文章
builder.ApplyConfiguration(new ArticleCfg()); builder.ApplyConfiguration(new ArticleCommentCfg()); // 问答
builder.ApplyConfiguration(new QuestionCfg()); builder.ApplyConfiguration(new QuestionCommentCfg()); // 用户
builder.ApplyConfiguration(new SiteUserCfg());
}
}

创建迁移选择 EntityFrameworkCore.DbMigrations 执行 Add-Migration Init_App_Db .

结语

该项目存放仓库在 https://github.com/BaseCoreVueProject/Blog.Core.AbpvNext

QQ群:867095512

项目开发过程中可能随时根据想法进行变化,加油!!!

知识全聚集 .Net Core 技术突破 丨ABP vNext 开始的更多相关文章

  1. 知识全聚集 .Net Core 技术突破 | 我用C#手把手教你玩微信自动化一

    知识全聚集 .Net Core 技术突破 | 我用C#手把手教你玩微信自动化一 教程 01 | 模块化方案一 02 | 模块化方案二 03 | 简单说说工作单元 其他教程预览 分库分表项目实战教程 G ...

  2. 知识全聚集 .Net Core 技术突破 | 简单说说工作单元

    知识全聚集 .Net Core 技术突破 | 简单说说工作单元 教程 01 | 模块化方案一 02 | 模块化方案二 其他教程预览 分库分表项目实战教程 Git地址: https://github.c ...

  3. 知识全聚集 .Net Core 技术突破 | 如何实现一个模块化方案一

    简介 模块化的介绍一共2篇 这一篇我们实现一个功能非常简单的StartupModules模块化. 第二篇我们来实现一个ABP的模块化效果. 思考 其实来简单想一下模块化的实验思路,写个接口=>模 ...

  4. 【HMS Core 6.0全球上线】Network Kit全链路网络加速技术,应用无惧网络拥塞

    HMS Core 6.0已于7月15日全球上线,本次版本向广大开发者开放了众多全新能力与技术.其中HMS Core Network Kit开放了全链路网络加速技术,助力开发者为用户提供低时延的畅快网络 ...

  5. In-Cell、On-Cell和OGS全贴合屏幕技术区别

    昨天刚发布的小米3用的是OGS全贴合屏幕技术,包括魅族MX3也是同样的技术,但是iPhone5是In-Cell屏幕技术,什么才是全贴合?它们之间到底有何区别?哪个好?小编今天就来普及一下全贴合屏幕技术 ...

  6. 基于Kubernetes 构建.NET Core技术中台

    今天下午在腾讯云+社区社区分享了<基于Kubernetes 构建.NET Core技术中台>,下面是演讲内容的文字实录. 我们为什么需要中台 我们现在处于企业信息化的新时代.为什么这样说呢 ...

  7. CCF虚拟现实与可视化技术专委会丨面向增强现实的可视计算技术研究进展概述

    https://mp.weixin.qq.com/s/I-rNwgXHEtwgdpkWzKtVXw 摘要 新一代增强现实技术需要依赖可视计算理论与方法解决大尺度复杂环境下的场景建模.内容生成.感知交互 ...

  8. 2019亚太内容分发大会,阿里云获CDN领袖奖、技术突破奖

    近日,亚太CDN产业联盟主办的2019亚太内容分发大会在上海召开.本次大会以"5G分发"为主题,集结了CDN领域近千名行业领袖.专家参与.在会上,阿里云斩获“CDN领袖奖”.“技术 ...

  9. 基于Kebernetes 构建.NET Core技术中台

    原文:基于Kebernetes 构建.NET Core技术中台 我们为什么需要中台 我们现在处于企业信息化的新时代.为什么这样说呢? 过去企业信息化的主流重心是企业内部信息化.但现在以及未来的企业信息 ...

随机推荐

  1. Linux - fuser 命令

    前言 之前连公司堡垒机的时候发现连不上,找运维排查是建立的链接数太多,很多超时链接没有断掉,导致不能再创建链接 此时,需要手动断开用户终端链接,然后百度搜到 fuser 可以断开用户终端链接 命令作用 ...

  2. Python+Selenium自动化-安装模块和浏览器驱动操作方法

    Python+Selenium自动化-安装模块和浏览器驱动操作方法 1.安装模块文件 pip install selenium 2.安装浏览器驱动 我们主要用的浏览器驱动有chrome浏览器.fire ...

  3. K8S集群etcd备份与恢复

    参考链接: K8S集群多master:Etcd v3备份与恢复 K8S集群单master:Kubernetes Etcd 数据备份与恢复 ETCD系列之一:简介:https://developer.a ...

  4. docker磁盘空间清理办法

    docker磁盘空间清理办法 前段时间遇到docker磁盘空间太少,无法写入数据的问题.起因是因为我在本地(Mac Pro)运行了多个mysql容器,并且导入了一部分线上数据,最后还没导入完毕就已经没 ...

  5. NVIDIA DeepStream 5.0构建智能视频分析应用程序

    NVIDIA DeepStream 5.0构建智能视频分析应用程序 无论是要平衡产品分配和优化流量的仓库,工厂流水线检查还是医院管理,要确保员工和护理人员在照顾病人的同时使用个人保护设备(PPE),就 ...

  6. 在NVIDIA A100 GPU中使用DALI和新的硬件JPEG解码器快速加载数据

    在NVIDIA A100 GPU中使用DALI和新的硬件JPEG解码器快速加载数据 如今,最流行的拍照设备智能手机可以捕获高达4K UHD的图像(3840×2160图像),原始数据超过25 MB.即使 ...

  7. 毫米波RADAR与LIDAR探秘

    毫米波RADAR与LIDAR探秘 说起激光雷达和毫米波雷达,相信业内人士并不陌生,激光雷达是以发射激光束探测目标的位置.速度等特征量的雷达系统.而毫米波雷达是指工作在毫米波波段探测的雷达.毫米波实质上 ...

  8. 在python_request 中 nb-log 日志模块的使用,应用到项目实际使用

    一.安装 pip install nb-log pycham 中安装: 二.基本使用 2.1 pycham中调整设置控制台日志打印出的颜色 2.2 设置完成后去掉console弹出的颜色设置 2.3  ...

  9. 总结springboot开启mybatis驼峰命名自动映射的三种方式

    方式一:通过springboot的配置文件application.yml mybatis: configuration: map-underscore-to-camel-case: true 此方式是 ...

  10. springMVC异常处理(自定义异常)HandlerExceptionResolver

    注:本篇的异常主要指的是controller.service和dao层中执行方法抛出的异常. 一.为什么要处理异常? 因为如果我们不处理异常,异常信息就会直接抛出给浏览器,于是浏览器页面就直接显示异常 ...