原文:2019-11-29-win10-uwp-手把手教你使用-asp-dotnet-core-做-cs-程序

title author date CreateTime categories
win10 uwp 手把手教你使用 asp dotnet core 做 cs 程序
lindexi
2019-11-29 10:16:12 +0800
2018-06-03 09:51:33 +0800
Win10 UWP dotnet

本文是一个非常简单的博客,让大家知道如何使用 asp dot net core 做后台,使用 UWP 或 WPF 等做前台。
本文因为没有什么业务,也不想做管理系统,所以看到起来是很简单。

VisualStudio

建议去官网下载最新的在线安装,然后安装,安装的时候需要选择通用平台开发、桌面开发、跨平台开发。如果你安装的不是中文版,请看下面图片找到差不多的位置,或者全部打勾。

点击安装,可以去睡觉,明天再看我的博客。因为 VisualStudio 的安装是很慢的,如果发现因为自己的网站无法访问,需要代理,可以发邮件给我,我发一个代理给你。

创建项目

现在一起创建三个项目,第一个项目是 UWP 项目,没有什么需要注意,点击创建空白UWP项目就可以。第二个是一个 dotnet standard 项目。第三个是 dotnet core 项目。

UWP

创建一个 UWP 程序是非常简单,点击C#,创建空白项目,随意给个名字,点击确定。注意尽量选择最高的平台,在项目开发是需要通过用户才知道需要选哪个平台。但是在开始学的时候,请尽量选择最新版本,这样可以减少因为版本问题让自己的时间花费在解决版本问题不兼容

如果完全没有开发过 UWP 程序,那么请看win10 uwp 如何开始写 uwp 程序 - CSDN博客

dotnet standard

创建一个 dotnet standard 可以让多个项目使用,在这里放一些类的定义。这个项目叫 Model ,名字不是随便叫的。

dotnet core

右击解决方案,选择新建项目,在弹出的窗口选择 dotnet core ,选择 asp.net core web 程序。

随意给这个项目一个名字,然后点击确定

在弹出的窗口选择 API ,注意不勾选 Docker 支持

现在就创建好了所有项目

引用项目

现在在创建好的 UWP 项目和 asp dotnet core 项目,右击引用,加上 Model 的引用

在 UWP 项目,引用的叫引用,在 asp dotnet core 项目,引用叫依赖项,只需要右击项目,点击添加就可以看到引用

现在使用新的 VisualStudio 打开相同的解决方案,这样才可以进行调试 asp dotnet core 的同时调试 UWP 项目

创建通用结构

现在打开 Model 项目,创建一个类,这个类是通用结构,在多个项目都需要使用。现在决定做这个项目是读取现有的小伙伴的网站和名字,添加新的名字和网站记录。

那么这里的类就需要包含小伙伴的名字和他的网站

    public class RoqawzemJajene
{
public int Id { get; set; } public string Name { get; set; } public string Url { get; set; }
}

可以看到上面的代码多了一个属性 Id ,现在我还不告诉大家为什么需要添加 id ,请在看到本文之后,自己尝试删除 id 然后重新运行项目。

设置控制器

现在打开 asp dotnet core 项目,通过 RoqawzemJajene 添加控制器。控制器就是访问 URL 时可以返回结果的类。通过控制器才可以使用 URL 访问,当然使用其他的方法也是可以做到,但是没有这个方法好用。

右击 Controler 添加,点击添加控制器

选择EF的控制器,请看下面图片。使用这个控制器,就会自动下载 EF 而且帮你设置好很多类,最简单的方法是这样写。如果想知道具体的每个类意思就需要自己去看文档,推荐ASP.NET Core 中文文档目录

这时需要告诉 VisualStudio 使用哪个模型类,和上下文,实际上如果刚才的引用已经写好,而且有 RoqawzemJajene 那么这一步是十分简单的。如果没有按照上面的方法来,那么这一步可能就无法继续。本文下面的也就无法继续。如果实在不知道怎么弄,欢迎在评论告诉我。

点击下拉,可以找到 RoqawzemJajene 这个类

点击了之后会看到没有数据上下文,点击最后的加号就可以自动帮你弄好

现在可以看到和下图差不多的界面,点击添加就可以等着 VisualStudio 帮你添加一些项目了

现在 VisualStudio 会帮你创建两个类

    public class TopetowLallteContext : DbContext
{
public TopetowLallteContext (DbContextOptions<TopetowLallteContext> options)
: base(options)
{
} public DbSet<Model.RoqawzemJajene> RoqawzemJajene { get; set; }
}
    [Produces("application/json")]
[Route("api/RoqawzemJajenes")]
public class RoqawzemJajenesController : Controller
{
private readonly TopetowLallteContext _context; public RoqawzemJajenesController(TopetowLallteContext context)
{
_context = context;
} // GET: api/RoqawzemJajenes
[HttpGet]
public IEnumerable<RoqawzemJajene> GetRoqawzemJajene()
{
return _context.RoqawzemJajene;
} // GET: api/RoqawzemJajenes/5
[HttpGet("{id}")]
public async Task<IActionResult> GetRoqawzemJajene([FromRoute] int id)
{
if (!ModelState.IsValid)
{
return BadRequest(ModelState);
} var roqawzemJajene = await _context.RoqawzemJajene.SingleOrDefaultAsync(m => m.Id == id); if (roqawzemJajene == null)
{
return NotFound();
} return Ok(roqawzemJajene);
} // PUT: api/RoqawzemJajenes/5
[HttpPut("{id}")]
public async Task<IActionResult> PutRoqawzemJajene([FromRoute] int id, [FromBody] RoqawzemJajene roqawzemJajene)
{
if (!ModelState.IsValid)
{
return BadRequest(ModelState);
} if (id != roqawzemJajene.Id)
{
return BadRequest();
} _context.Entry(roqawzemJajene).State = EntityState.Modified; try
{
await _context.SaveChangesAsync();
}
catch (DbUpdateConcurrencyException)
{
if (!RoqawzemJajeneExists(id))
{
return NotFound();
}
else
{
throw;
}
} return NoContent();
} // POST: api/RoqawzemJajenes
[HttpPost]
public async Task<IActionResult> PostRoqawzemJajene([FromBody] RoqawzemJajene roqawzemJajene)
{
if (!ModelState.IsValid)
{
return BadRequest(ModelState);
} _context.RoqawzemJajene.Add(roqawzemJajene);
await _context.SaveChangesAsync(); return CreatedAtAction("GetRoqawzemJajene", new { id = roqawzemJajene.Id }, roqawzemJajene);
} // DELETE: api/RoqawzemJajenes/5
[HttpDelete("{id}")]
public async Task<IActionResult> DeleteRoqawzemJajene([FromRoute] int id)
{
if (!ModelState.IsValid)
{
return BadRequest(ModelState);
} var roqawzemJajene = await _context.RoqawzemJajene.SingleOrDefaultAsync(m => m.Id == id);
if (roqawzemJajene == null)
{
return NotFound();
} _context.RoqawzemJajene.Remove(roqawzemJajene);
await _context.SaveChangesAsync(); return Ok(roqawzemJajene);
} private bool RoqawzemJajeneExists(int id)
{
return _context.RoqawzemJajene.Any(e => e.Id == id);
}
}

请看上图 RoqawzemJajenesController 就是控制器,TopetowLallteContext 就是数据,里面包含了数据定义。现代的程序员是很少写古老的 sql ,因为很难维护,而且容易写错。只需要直接对DbSet<RoqawzemJajene> RoqawzemJajene列表做修改就可以。对于很多个人项目,不使用 sql 来读写数据库也是没问题。但是对于大型的项目还是需要 sql ,所以有空去学一下还是可以。

因为 asp dotnet core 默认的数据库是 sql server ,如果直接运行项目会说没有找到数据库,所以本文告诉大家使用内存做数据库。

打开 Startup.cs 找到 ConfigureServices ,可以看到下面代码

        public void ConfigureServices(IServiceCollection services)
{
services.AddMvc(); services.AddDbContext<TopetowLallteContext>(options =>
options.UseSqlServer(Configuration.GetConnectionString("TopetowLallteContext")));
}

这里 UseSqlServer 就是使用 sql server 数据库,把他修改为options.UseInMemoryDatabase("lindexi")请看代码

        public void ConfigureServices(IServiceCollection services)
{
services.AddMvc(); services.AddDbContext<TopetowLallteContext>(options =>
options.UseInMemoryDatabase("lindexi"));
}

这时就可以使用内存放数据,当然这时如果关闭了应用,数据也会没了。

现在打开 RoqawzemJajenesController 构造函数,在第一次进入添加一些数据

        public RoqawzemJajenesController(TopetowLallteContext context)
{
_context = context;
if (!context.RoqawzemJajene.Any())
{
context.RoqawzemJajene.Add(new RoqawzemJajene()
{
Name = "lindexi",
Url = "https://lindexi.oschina.io"
}); context.SaveChanges();
}
}

在这里添加修改列表都需要调用 context.SaveChanges() 保存

运行网站

先右击 asp dotnet core 项目属性,点击调试,可以看到下面界面

因为不需要使用浏览器,所以去掉启动浏览器。需要记下端口,这个端口在下面的 UWP 项目使用。

请看上图,我的应用 URL 里面包含了端口 64043

http://localhost:64043/

现在右击设置 asp dotnet core 项目为启动项目,然后按下 F5 运行这个项目

UWP 连接

打开 UWP 程序,先创建一个 ViewModel 用于数据绑定。本文不会告诉大家很多关于 ViewModel 的方法,如果想了解这个写法,请看win10 uwp MVVM入门

ViewModel

现在需要定义一个数据结构,ObservableCollection 的列表,这样可以在界面绑定

    public class ViewModel
{
public ObservableCollection<RoqawzemJajene> RoqawzemJajeneList { get; set; } =
new ObservableCollection<RoqawzemJajene>();
}

绑定数据

界面绑定,首先需要添加 ViewModel 属性,打开 MainPage.xaml.cs 添加下面代码

        public MainPage()
{
this.InitializeComponent();
DataContext = ViewModel;
} public ViewModel ViewModel { get; } = new ViewModel();

界面

然后在 MainPage.xaml 添加一个列表,在这个 ListView 可以用来显示列表

   <Grid>
<Grid.RowDefinitions>
<RowDefinition />
<RowDefinition Height="Auto" />
</Grid.RowDefinitions>
<ListView ItemsSource="{x:Bind ViewModel.RoqawzemJajeneList}">
<ListView.ItemContainerStyle>
<Style TargetType="ListViewItem">
<Setter Property="HorizontalContentAlignment" Value="Stretch"></Setter>
</Style>
</ListView.ItemContainerStyle>
<ListView.ItemTemplate>
<DataTemplate x:DataType="model:RoqawzemJajene">
<Grid>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="100" />
<ColumnDefinition></ColumnDefinition>
</Grid.ColumnDefinitions>
<TextBlock Text="{x:Bind Name}"></TextBlock>
<TextBlock Grid.Column="1" Text="{x:Bind Url}"></TextBlock>
</Grid>
</DataTemplate>
</ListView.ItemTemplate>
</ListView>
<StackPanel Grid.Row="1" Orientation="Horizontal">
<Button Margin="10,10,10,10" Content="下载数据" Click="Button_OnClick"></Button>
</StackPanel>
</Grid>

因为使用了 DataType ,所以需要添加引用

    xmlns:model="using:Model"

因为现在添加了一个按钮下载数据,所以需要打开 MainPage.xaml.cs 添加下面代码

        private void Button_OnClick(object sender, RoutedEventArgs e)
{
// 更新数据
}

现在的 UWP 软件是可以编译通过的,但是运行是不会显示内容

下载列表

在 MainPage.xaml.cs 的 Button_OnClick 调用 ViewModel 的函数用来更新数据

        private void Button_OnClick(object sender, RoutedEventArgs e)
{
ViewModel.Update();
}

这时可能会觉得这样写不好,因为有 x:bind 可以在 xaml 绑定 ViewModel 的方法为什么还需要添加在按钮。因为这样写我可以在按钮点击下去同时做一些动画。

现在打开 ViewModel 先添加引用

using System.Net.Http;

然后打开 Update 函数,访问 URL 需要用到 HttpClient,还记得刚才的 RoqawzemJajenesController 里面有这样的代码

    [Route("api/RoqawzemJajenes")]
public class RoqawzemJajenesController : Controller

这里的 Route 就是路由,也就是在 URL 后面添加 api/RoqawzemJajenes 就可以访问这个控制器。默认的 Get 方法就是通过 GetRoqawzemJajene 实现。

        [HttpGet]
public IEnumerable<RoqawzemJajene> GetRoqawzemJajene()
{
return _context.RoqawzemJajene;
}

所以想调用 GetRoqawzemJajene 方法就需要访问 http://localhost:64043/api/RoqawzemJajenes,需要知道,这里的链接的端口是需要在上面右击 asp dotnet core 项目属性,调试那里才知道。

因为 asp dotnet core 程序返回的是 json ,可以通过设置让他返回 xml 。因为刚才没设置,默认返回的格式是 json 。

为了解析数据,需要在 Nuget 安装 json ,打开 Nuget 安装第一个就可以

通过简单的 Get 的方法访问 URL 可以使用下面代码,参见 win10 UWP GET Post - CSDN博客

        public async Task Update()
{
var httpClient = new HttpClient();
var url = "http://localhost:64043/"; url = url + "api/RoqawzemJajenes";
var str = await httpClient.GetStringAsync(url);
var roqawzemJajeneList = JsonConvert.DeserializeObject<List<RoqawzemJajene>>(str); RoqawzemJajeneList.Clear(); foreach (var temp in roqawzemJajeneList)
{
RoqawzemJajeneList.Add(temp);
}
}

上面的代码通过 Get 访问 URL 拿到数据只需要一句话

            var str = await httpClient.GetStringAsync(url);

因为这里使用了异步,所以修改了方法,需要在 MainPage.xaml.cs 的 Button_OnClick 修改

        private async void Button_OnClick(object sender, RoutedEventArgs e)
{
await ViewModel.Update();
}

现在尝试运行 asp dotnet core 项目,然后再运行 UWP 项目,这时点击一下 UWP 项目的下载数据,就可以看到如下图界面

现在就完成了 UWP 程序的连接

上传数据

现在尝试上传数据,因为写界面速度比较慢,所以直接添加一个按钮,里面把我的一个小伙伴的信息传上去。

虽然刚才的代码已经有一个函数可以用来上传数据

        public async Task<IActionResult> PostRoqawzemJajene([FromBody] RoqawzemJajene roqawzemJajene)

但是为了让大家知道如何使用路由,所以修改一下这个参数。

我希望访问 http://localhost:64043/api/RoqawzemJajenes/add 来添加一个小伙伴。

这时可以在 PostRoqawzemJajene 添加特性。打开 RoqawzemJajenesController ,进入 PostRoqawzemJajene ,在函数上面添加下面代码

      [Route("add")]
[HttpPost]
public async Task<IActionResult> PostRoqawzemJajene([FromBody] RoqawzemJajene roqawzemJajene)

现在打开 ViewModel 添加一个函数

       public async Task Add()
{
var roqawzemJajene = new RoqawzemJajene()
{
Name = "头像",
Url = "https://huangtengxiao.gitee.io/"
}; var url = Url + "api/RoqawzemJajenes/add"; var httpClient = new HttpClient(); var str = JsonConvert.SerializeObject(roqawzemJajene); var stringContent = new StringContent(str);
stringContent.Headers.ContentType.MediaType = "application/json"; await httpClient.PostAsync(url, stringContent);
}

这里的 Url 是因为在很多函数都需要使用,所以我把 http://localhost:64043/ 拿出来。在 UWP 使用 post 是很简单,只需要一句代码

            await httpClient.PostAsync(url, stringContent);

现在打开 MainPage.xaml 添加一个按钮

            <Button Margin="10,10,10,10" Content="上传" Click="Add_OnClick"></Button>

然后打开 MainPage.xaml.cs 添加函数

        private async void Add_OnClick(object sender, RoutedEventArgs e)
{
await ViewModel.Add();
await ViewModel.Update();
}

现在尝试运行 UWP 程序,然后点击上传,可以看到这个界面

如果刚才的代码有地方没有写对,如 stringContent 没有添加下面代码

            stringContent.Headers.ContentType.MediaType = "application/json";

或者写的 Url 不对,都可以在 asp dotnet core 项目的输出看到和下面差不多的输出

Application Insights Telemetry (unconfigured): {"name":"Microsoft.ApplicationInsights.Dev.Message","time":"2018-06-03T07:31:44.4397220Z","tags":{"ai.application.ver":"1.0.0.0","ai.operation.id":"d8ff58c-4a3bece1b1751287","ai.location.ip":"127.0.0.1","ai.internal.nodeName":"DESKTOP-KA1CD6M","ai.operation.name":"POST /api/add","ai.cloud.roleInstance":"DESKTOP-KA1CD6M","ai.internal.sdkVersion":"aspnet5c:2.1.1","ai.operation.parentId":"|d8ff58c-4a3bece1b1751287."},"data":{"baseType":"MessageData","baseData":{"ver":2,"message":"Request starting HTTP/1.1 POST http://localhost:64043/api/add  0","severityLevel":"Information","properties":{"CategoryName":"Microsoft.AspNetCore.Hosting.Internal.WebHost","Scheme":"http","Method":"POST","Host":"localhost:64043","ContentLength":"0","Path":"/api/add","Protocol":"HTTP/1.1","AspNetCoreEnvironment":"Development","DeveloperMode":"true"}}}}
Application Insights Telemetry (unconfigured): {"name":"Microsoft.ApplicationInsights.Dev.Message","time":"2018-06-03T07:31:44.4417985Z","tags":{"ai.application.ver":"1.0.0.0","ai.operation.id":"d8ff58c-4a3bece1b1751287","ai.location.ip":"127.0.0.1","ai.internal.nodeName":"DESKTOP-KA1CD6M","ai.operation.name":"POST /api/add","ai.cloud.roleInstance":"DESKTOP-KA1CD6M","ai.internal.sdkVersion":"aspnet5c:2.1.1","ai.operation.parentId":"|d8ff58c-4a3bece1b1751287."},"data":{"baseType":"MessageData","baseData":{"ver":2,"message":"Request finished in 2.0912ms 404","severityLevel":"Information","properties":{"CategoryName":"Microsoft.AspNetCore.Hosting.Internal.WebHost","StatusCode":"404","ElapsedMilliseconds":"2.0912","AspNetCoreEnvironment":"Development","DeveloperMode":"true"}}}}
Application Insights Telemetry (unconfigured): {"name":"Microsoft.ApplicationInsights.Dev.Request","time":"2018-06-03T07:31:44.4395938Z","tags":{"ai.location.ip":"127.0.0.1","ai.cloud.roleInstance":"DESKTOP-KA1CD6M","ai.internal.nodeName":"DESKTOP-KA1CD6M","ai.operation.id":"d8ff58c-4a3bece1b1751287","ai.internal.sdkVersion":"aspnet5c:2.1.1","ai.operation.name":"POST /api/add","ai.application.ver":"1.0.0.0"},"data":{"baseType":"RequestData","baseData":{"ver":2,"id":"|d8ff58c-4a3bece1b1751287.","name":"POST /api/add","duration":"00:00:00.0045034","success":false,"responseCode":"404","url":"http://localhost:64043/api/add","properties":{"httpMethod":"POST","AspNetCoreEnvironment":"Development","DeveloperMode":"true"}}}}

在这里,如何写一个简单的 cs 程序已经告诉了大家,建议完全复制代码来试试。如果发现还是无法运行,看到的界面和我不一样。那么尝试下载我的代码来试试。

如果遇到任何问题欢迎通过评论告诉我,或发邮件给我。

代码:手把手教你使用 asp dotnet core 做 cs 程序 1.1-CSDN下载

2019-11-29-win10-uwp-手把手教你使用-asp-dotnet-core-做-cs-程序的更多相关文章

  1. win10 uwp 手把手教你使用 asp dotnet core 做 cs 程序

    本文是一个非常简单的博客,让大家知道如何使用 asp dot net core 做后台,使用 UWP 或 WPF 等做前台. 本文因为没有什么业务,也不想做管理系统,所以看到起来是很简单. Visua ...

  2. win10 uwp 使用 asp dotnet core 做图床服务器客户端

    原文 win10 uwp 使用 asp dotnet core 做图床服务器客户端 本文告诉大家如何在 UWP 做客户端和 asp dotnet core 做服务器端来做一个图床工具   服务器端 从 ...

  3. vs2017 在win10下安装后开始运行asp.net core 项目时出错

    vs2017 在win10下安装后开始运行asp.net core 项目时出错 报找不到什么 解决方法: 下载 asp.net 2.2 安装好,重新启动电脑问题解决.

  4. 一篇文章教你学会ASP.Net Core LINQ基本操作

    一篇文章教你学会ASP.Net Core LINQ基本操作 为什么要使用LINQ LINQ中提供了很多集合的扩展方法,配合lambda能简化数据处理. 例如我们想要找出一个IEnumerable< ...

  5. ABP 教程文档 1-1 手把手引进门之 ASP.NET Core & Entity Framework Core(官方教程翻译版 版本3.2.5)

    本文是ABP官方文档翻译版,翻译基于 3.2.5 版本 官方文档分四部分 一. 教程文档 二.ABP 框架 三.zero 模块 四.其他(中文翻译资源) 本篇是第一部分的第一篇. 第一部分分三篇 1- ...

  6. 手把手教你开发BLE数据透传应用程序

    如何开发BLE数据透传应用程序?什么是BLE service和characteristic?如何开发自己的service和characteristic?如何区分ATT和GATT?有没有什么工具可以对B ...

  7. [转]教你实践ASP.NET Core Authorization

    本文转自:http://www.cnblogs.com/rohelm/p/Authorization.html 本文目录 Asp.net Core 对于授权的改动很友好,非常的灵活,本文以MVC为主, ...

  8. 2019.11.29 Mysql的数据操作

    为名为name的表增加数据(插入所有字段) insert into name values(1,‘张三’,‘男’,20); 为名为name的表增加数据(插入部分字段) insert into name ...

  9. 2019.11.29 SAP SMTP郵件服務器配置 發送端 QQ郵箱

    今天群裏的小夥伴問了如何配置郵件的問題,隨自己在sap裏面配置了一個 1.    RZ10配置參數 a)       参数配置前,先导入激活版本 执行完毕后返回 b)      输入参数文件DEFAU ...

  10. pycharm+anaconda在Mac上的配置方法 2019.11.29

    内心os: 听人说,写blog是加分项,那他就不是浪费时间的事儿了呗 毕竟自己菜还是留下来东西来自己欣赏吧 Mac小电脑上进行python数据开发环境的配置 首先下载Anaconda,一个超好用的数据 ...

随机推荐

  1. amanda安装

    下载地址 http://www.amanda.org/download.php http://www.zmanda.com/download-amanda.php 编译安装 编译出错: error: ...

  2. 项目进程中input的onblur事件挂不上去,失效问题解决记录

    一开始直接在js文件中写 var n=document.querySelector("input"); n.onblur=funcition(){ ///事件过程吧啦啦啦 }; 事 ...

  3. 【原创】Airflow 简介&如何部署一个健壮的 apache-airflow 调度系统

    声明 本文摘录了很多前辈的文章,原文如下: https://www.jianshu.com/p/2ecef979c606 Airflow 简介 Airflow是一个可编程,调度和监控的工作流平台,基于 ...

  4. Confluence 邮箱设置

    Confluence有两种方法设置邮箱 原理: confluence服务器配置好邮箱信息,用户触发邮件发送规则时,confluence服务使用已配置的邮箱信息登录到邮箱服务器,进行发件服务. 那么我们 ...

  5. Win7共享只看到部分文件

    把192.168.70.88这台机器的Android目录做为共享文件夹. 共享端:windows server 用户端:其它pc机安装windows7 或windows 10 在用户端只能看到部分文件 ...

  6. SQLAlchemy 多对多

    创建多对多表 from sqlalchemy.ext.declarative import declarative_base Base=declarative_base() from sqlalche ...

  7. python cookie登录DVWA,phpstudy搭建DVWA参考https://www.jianshu.com/p/97d874548300

    import requestsheader={"User-Agent": "Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleW ...

  8. 201871010131-张兴盼《面向对象程序设计(java)》第十四周学习总结

    项目 内容 这个作业属于哪个课程 https://www.cnblogs.com/nwnu-daizh/ 这个作业要求在哪里 https://www.cnblogs.com/lily-2018/p/1 ...

  9. xSS-Reflected

    反射性(非持久性XSS),藏在URL中 一般用户访问恶意链接执行 Low <?php header ("X-XSS-Protection: 0"); // Is there ...

  10. 多个页面进行爬虫 pycharm

    使用requests , lxml  ,xpath进行爬取并写入了TXT(也可以存入数据库).参考博客:https://blog.csdn.net/yexing_cts/article/details ...