基于.NET Core 3.1 网站开发和部署的方法

一、准备开发环境

1.主要开发工具的选择

  • vscode
  • .NET Core command-line interface (CLI) tools
  • Dbeaver

这里选择vscode + .net core cli 是因为不管在Windows还是Linux和Mac上都能使用这一套工具,而且命令行工具也非常强大。

2.vscode安装C#插件

在vscode插件市场中搜索安装即可

新手还可以去这里了解vscode的强大之处

3.安装数据库

这里使用Centos7,因为.NET Core 3.1只支持7及以上版本

配置网络

nmcli conn
nmcli conn add ifname ens34 con-name ens34 type enthernet autoconnect yes ip4 192.168.30.110/24

安装mariadb-server

使用XShell连接服务器

cd /etc/yum.repos.d/
mkdir bak
mv Cen* ./bak/
curl -o /etc/yum.repos.d/CentOS-Base.repo http://mirrors.aliyun.com/repo/Centos-7.repo
yum install mariadb-server.x86_64

启动服务

systemctl enable mariadb.service
systemctl start mariadb.service
systemctl status mariadb

执行安全加强脚本

mysql_secure_installation

创建数据库并授权访问帐号

MariaDB [(none)]> create database HotelWebDb default charset utf8 collate utf8_general_ci;
MariaDB [(none)]> grant all on HotelWebDb.* to sa@'192.168.30.%' identified by '110';
MariaDB [(none)]> show databases;
+--------------------+
| Database      |
+--------------------+
| information_schema |
| HotelWebDb     |
| mysql       |
| performance_schema |
+--------------------+
4 rows in set (0.00 sec)

4.使用Dbeaver连接数据库创建数据表和测试数据

编写Sql脚本

-- 管理员登录表
CREATE TABLE if not EXISTS SysAdmins
(
  LoginId INT UNSIGNED PRIMARY KEY AUTO_INCREMENT,
  LoginName VARCHAR(20) NOT NULL,
  LoginPwd VARCHAR(20) NOT NULL
)engine=innodb DEFAULT charset=utf8;
ALTER TABLE SysAdmins AUTO_INCREMENT=10000;

-- 新闻分类表
CREATE TABLE if not EXISTS NewsCategory
(
  CategoryId INT UNSIGNED PRIMARY KEY AUTO_INCREMENT,
  CategoryName VARCHAR(20) NOT NULL
)engine=innodb DEFAULT charset=utf8;

-- 新闻表
CREATE TABLE if not EXISTS News
(
  Id INT UNSIGNED PRIMARY KEY AUTO_INCREMENT,
  NewsTitle VARCHAR(100),
  NewsContent TEXT,
  PublishTime TIMESTAMP DEFAULT now(),
  CategoryId INT UNSIGNED ,
  FOREIGN KEY(CategoryId) REFERENCES NewsCategory (CategoryId)
)engine=innodb DEFAULT charset=utf8;
ALTER TABLE News AUTO_INCREMENT=1000;

-- 菜品分类表
CREATE TABLE if not EXISTS DishCategory
(
  CategoryId INT UNSIGNED PRIMARY KEY AUTO_INCREMENT,
  CategoryName VARCHAR(20)
)engine=innodb DEFAULT charset=utf8;

-- 菜品表
CREATE TABLE if not EXISTS Dishes
(
  Id INT UNSIGNED PRIMARY KEY AUTO_INCREMENT,
  DishName VARCHAR(100),
  UnitPrice NUMERIC(18, 2),
  CategoryId INT UNSIGNED,
  FOREIGN KEY(CategoryId) REFERENCES DishCategory (CategoryId)
)engine=innodb DEFAULT charset=utf8;
ALTER TABLE Dishes AUTO_INCREMENT=10000;

-- 菜品预订表
CREATE TABLE if not EXISTS DishBook
(
  Id INT UNSIGNED PRIMARY KEY AUTO_INCREMENT,
  HotelName VARCHAR(50),
  ConsumeTime datetime,
  ConsumePersons INT UNSIGNED,
  RoomType VARCHAR(20),
  CustomerPhone VARCHAR(20),
  CustomerName VARCHAR(20),
  CustomerEmail VARCHAR(100),
  Comments VARCHAR(500),
  BookTime TIMESTAMP DEFAULT now(),
  BookStatus INT DEFAULT 0 -- ( 0 表示未审核,1 表示审核通过,2 表示消费完成,-1 表示撤销订单)
)engine=innodb DEFAULT charset=utf8;
ALTER TABLE DishBook AUTO_INCREMENT=1000;

-- 招聘信息表
CREATE TABLE if not EXISTS Recruitment
(
  PostId INT UNSIGNED PRIMARY KEY AUTO_INCREMENT,
  PostName NVARCHAR(100),
  PostType CHAR(4), -- (全职或兼职)
  WorkPlace NVARCHAR(50),
  PostDesc TEXT,
  PostRequire TEXT,
  Experience NVARCHAR(100),
  EduBackground NVARCHAR(100),
  RequireCount INT,
  PublishTime TIMESTAMP DEFAULT now(),
  Manager VARCHAR(20),
  PhoneNumber VARCHAR(20),
  Email VARCHAR(100)
)engine=innodb DEFAULT charset=utf8;
ALTER TABLE Recruitment AUTO_INCREMENT=100000;

-- 投诉建议表
CREATE TABLE if not EXISTS Suggestion
(
  Id INT UNSIGNED PRIMARY KEY AUTO_INCREMENT,
  CustomerName VARCHAR(20),
  ConsumeDesc TEXT,
  SuggestionDesc TEXT,
  SuggestionTime TIMESTAMP DEFAULT now(),
  PhoneNumber VARCHAR(20),
  Email VARCHAR(20),
  StatusId INT DEFAULT 0 -- (0:未受理;1:已经受理)
)engine=innodb DEFAULT charset=utf8;
ALTER TABLE Suggestion AUTO_INCREMENT=10000;

创建测试数据

-- 插入测试数据
-- 管理员信息
insert into SysAdmins(LoginPwd,LoginName)values('123456','李浩');
insert into SysAdmins(LoginPwd,LoginName)values('123456','赵雪伶');
-- 新闻分类
insert into NewsCategory(CategoryName)values('公司新闻');
insert into NewsCategory(CategoryName)values('社会新闻');
-- 菜品分类
insert into DishCategory(CategoryName)values('川菜');
insert into DishCategory(CategoryName)values('湘菜');
insert into DishCategory(CategoryName)values('鲁菜');
insert into DishCategory(CategoryName)values('海鲜类');
insert into DishCategory(CategoryName)values('其他');
-- 新闻
insert into News(NewsTitle,NewsContent,CategoryId)values('迎接十一海鲜大促销','最新鲜的鱼类全面上市,欢迎新老顾客品尝。',1);
insert into News(NewsTitle,NewsContent,CategoryId)values('本店正在热招加盟商','如果您愿意在酒店行业有所突破,请加入我们。',1);
insert into News(NewsTitle,NewsContent,CategoryId)values('互联网的消费正在兴起','网上购物已经成为人们生活必不可少的一部分。',2);
insert into News(NewsTitle,NewsContent,CategoryId)values('本店正在热招加盟商','如果您愿意在酒店行业有所突破,请加入我们。',1);
insert into News(NewsTitle,NewsContent,CategoryId)values('互联网的消费正在兴起','网上购物已经成为人们生活必不可少的一部分。',2);
-- 菜品信息
insert into Dishes(DishName,UnitPrice,CategoryId)values('水煮鱼',50,1);
insert into Dishes(DishName,UnitPrice,CategoryId)values('回锅肉',85,1);
insert into Dishes(DishName,UnitPrice,CategoryId)values('剁椒鱼头',75,2);
insert into Dishes(DishName,UnitPrice,CategoryId)values('红椒腊牛肉',40,2);
insert into Dishes(DishName,UnitPrice,CategoryId)values('糖醋鲤鱼',70,3);
insert into Dishes(DishName,UnitPrice,CategoryId)values('玉记扒鸡',60,3);
insert into Dishes(DishName,UnitPrice,CategoryId)values('汤爆双脆',90,3);
insert into Dishes(DishName,UnitPrice,CategoryId)values('赤贝',80,4);
-- 预定信息
insert into DishBook(HotelName,ConsumeTime,ConsumePersons,RoomType,CustomerName,CustomerPhone,CustomerEmail,Comments)
values('天津南开店','2014-09-11 12:30',5,'包间','李丽','13589011222','lilivip@163.com','希望房间敞亮些');
insert into DishBook(HotelName,ConsumeTime,ConsumePersons,RoomType,CustomerName,CustomerPhone,CustomerEmail,Comments)
values('天津和平店','2014-10-11 14:30',5,'包间','王鑫新','13889018888','wangxinxin@qq.com','希望房间安静些');
insert into DishBook(HotelName,ConsumeTime,ConsumePersons,RoomType,CustomerName,CustomerPhone,CustomerEmail,Comments)
values('北京朝阳点','2014-12-10 17:30',5,'散座','刘花雨','13689011999','liuhuayu@126.com','房间靠里面点儿');
-- 招聘信息
insert into Recruitment(PostName,PostType,WorkPlace,PostDesc,PostRequire,Experience,EduBackground,RequireCount,Manager,PhoneNumber,Email)
values('大堂经理','全职','天津','负责一层楼的管理','要求具备该职位3年以上经营管理经验','3年','本科',2,'李超阳','15689011231','lichaoyang@hyl.com');
insert into Recruitment(PostName,PostType,WorkPlace,PostDesc,PostRequire,Experience,EduBackground,RequireCount,Manager,PhoneNumber,Email)
values('接待员','全职','北京','负责客户的接待礼仪','要求具备该职位1年以上经验','1年','高中',5,'李超阳','15689011231','lichaoyang@hyl.com');
insert into Recruitment(PostName,PostType,WorkPlace,PostDesc,PostRequire,Experience,EduBackground,RequireCount,Manager,PhoneNumber,Email)
values('总经理助理','全职','天津','负责日常的文秘工作','要求具备该职位3年以上经营管理经验','3年','本科',1,'李超阳','15689011231','lichaoyang@hyl.com');
-- 投诉建议
insert into Suggestion(CustomerName,ConsumeDesc,SuggestionDesc,PhoneNumber,Email)
values('杜小杰','在该店举行一次婚礼','感觉总体服务不到位,菜品味道没有以前的好。','15687423456','duxiaojie@qq.com');
insert into Suggestion(CustomerName,ConsumeDesc,SuggestionDesc,PhoneNumber,Email)
values('柳钢','在本店聚会一次','感觉上菜有点慢,希望后续改进。','15686623456','liugang@qq.com');

二、搭建项目框架

1.使用vscode创建一个mvc项目

进入vscode,打开终端创建项目的根目录
快捷键:Ctrl + `

创建项目根目录

D:\dotnet_core>mkdir HotelWebMVC
D:\dotnet_core>dir
 驱动器 D 中的卷是 补充
 卷的序列号是 0004-524D

 D:\dotnet_core 的目录

2019/12/10 09:33  <DIR>     .
2019/12/10 09:33  <DIR>     ..
2019/08/30 16:31  <DIR>     .vscode
2018/05/07 20:25  <DIR>     helloworld
2019/12/10 09:33  <DIR>     HotelWebMVC
2019/12/09 20:22  <DIR>     netcore_mvc

安装 .NET Core SDK

下载地址:https://dotnet.microsoft.com/download

查看微软的教程:https://docs.microsoft.com/en-us/aspnet/core/tutorials/first-mvc-app/start-mvc?view=aspnetcore-3.1&tabs=visual-studio-code

查看命令帮助

D:\dotnet_core\HotelWebMVC>dotnet new --help
用法: new [选项]

选项:
 -h, --help     Displays help for this command.
 -l, --list     Lists templates containing the specified name. If no name is specified, lists all templates.
 -n, --name     The name for the output being created. If no name is specified, the name of the current directory is used.
 -o, --output    Location to place the generated output.
 -i, --install    Installs a source or a template pack.
 -u, --uninstall   Uninstalls a source or a template pack.
 --nuget-source   Specifies a NuGet source to use during install.
 --type       Filters templates based on available types. Predefined values are "project", "item" or "other".
 --dry-run      Displays a summary of what would happen if the given command line were run if it would result in a template creation.
 --force       Forces content to be generated even if it would change existing files.
 -lang, --language  Filters templates based on language and specifies the language of the template to create.
 --update-check   Check the currently installed template packs for updates.
 --update-apply   Check the currently installed template packs for update, and install the updates.

创建项目

以上操作都是可以使用命令完成
比如添加解决方案

D:\dotnet_core\HotelWebMVC>dotnet sln ./HotelWebMVC.sln add DAL\DAL.csproj
已将项目“DAL\DAL.csproj”添加到解决方案中。

D:\dotnet_core\HotelWebMVC>dotnet sln ./HotelWebMVC.sln add BLL\BLL.csproj
已将项目“BLL\BLL.csproj”添加到解决方案中。

D:\dotnet_core\HotelWebMVC>dotnet sln list
项目
--
HotelWebMVC\HotelWebMVC.csproj
Models\Models.csproj
DAL\DAL.csproj
BLL\BLL.csproj

添加启动项目和类库项目之间的引用

Models -->DAL

D:\dotnet_core\HotelWebMVC\DAL>dotnet add reference ..\Models\Models.csproj
已将引用“..\Models\Models.csproj”添加到项目。

DAL —> BLL

D:\dotnet_core\HotelWebMVC\DAL>dotnet add ..\bll\BLL.csproj reference .\DAL.csproj
已将引用“..\DAL\DAL.csproj”添加到项目。

HotelWebMVC --> BLL

D:\dotnet_core\HotelWebMVC\DAL>dotnet add ..\HotelWebMVC\HotelWebMVC.csproj reference ..\bll\BLL.csproj
已将引用“..\bll\BLL.csproj”添加到项目。

2.生成Model

① 使用EF Core 生成模型

DB First模式

全局安装ef工具

dotnet tool install --global dotnet-ef

测试工具是否安装成功

D:\dotnet_core\HotelWebMVC>dotnet ef

           _/\__
        ---==/  \\
     ___ ___  |.  \|\
    | __|| __| | )  \\\
    | _| | _|  \_/ | //|\\
    |___||_|    /  \\\/\\

Entity Framework Core .NET Command-line Tools 3.1.0

Usage: dotnet ef [options] [command]

Options:
 --version    Show version information
 -h|--help    Show help information
 -v|--verbose   Show verbose output.
 --no-color    Don't colorize output.
 --prefix-output Prefix output with level.

Commands:
 database  Commands to manage the database.
 dbcontext  Commands to manage DbContext types.
 migrations Commands to manage migrations.

Use "dotnet ef [command] --help" for more information about a command.

只要看到有这个独角兽就代表安装成功了。

添加需要的NuGet包

dotnet add package Microsoft.EntityFrameworkCore.Design
dotnet add package Pomelo.EntityFrameworkCore.MySql

使用EF工具

查看帮助

D:\dotnet_core\HotelWebMVC\DAL>dotnet ef dbcontext scaffold --help

Usage: dotnet ef dbcontext scaffold [arguments] [options]

Arguments:
 <CONNECTION> The connection string to the database.
 <PROVIDER>  The provider to use. (E.g. Microsoft.EntityFrameworkCore.SqlServer)

Options:
 -d|--data-annotations         Use attributes to configure the model (where possible). If omitted, only the fluent API is used.
 -c|--context <NAME>          The name of the DbContext.
 --context-dir <PATH>          The directory to put DbContext file in. Paths are relative to the project directory.
 -f|--force               Overwrite existing files.
 -o|--output-dir <PATH>         The directory to put files in. Paths are relative to the project directory.
 --schema <SCHEMA_NAME>...       The schemas of tables to generate entity types for.
 -t|--table <TABLE_NAME>...       The tables to generate entity types for.
 --use-database-names          Use table and column names directly from the database.
 --json                 Show JSON output.
 -p|--project <PROJECT>         The project to use.
 -s|--startup-project <PROJECT>     The startup project to use.
 --framework <FRAMEWORK>        The target framework.
 --configuration <CONFIGURATION>    The configuration to use.
 --runtime <RUNTIME_IDENTIFIER>     The runtime to use.
 --msbuildprojectextensionspath <PATH> The MSBuild project extensions path. Defaults to "obj".
 --no-build               Don't build the project. Only use this when the build is up-to-date.
 -h|--help               Show help information
 -v|--verbose              Show verbose output.
 --no-color               Don't colorize output.
 --prefix-output            Prefix output with level.

连接数据库生成模型数据

D:\dotnet_core\HotelWebMVC\DAL>dotnet ef dbcontext scaffold "Server=192.168.30.110,3306;DataBase=HotelWebDb;User=sa;Pwd=110;" "Pomelo.EntityFrameworkCore.MySql" -s ..\HotelWebMVC\HotelWebMVC.csproj
Build started...
Build succeeded.

修改连接字符串的位置

修改在appsettings.json文件中添加连接字符串

{
 "Logging": {
  "LogLevel": {
   "Default": "Information",
   "Microsoft": "Warning",
   "Microsoft.Hosting.Lifetime": "Information"
  }
 },
 "AllowedHosts": "*",
 "ConnectionStrings":{
  "HotelWeb":"Server=192.168.30.110,3306;DataBase=HotelWebDb;User=sa;Pwd=110;"
 }
}

然后在Sartup.cs文件获取连接字符串

public void ConfigureServices(IServiceCollection services)
{
	services.AddControllersWithViews();
	string connString=Configuration.GetConnectionString("HotelWeb");
	services.AddDbContext<DAL.HotelWebDbContext>(options=>options.UseMySql(connString, x => x.ServerVersion("5.5.64-mariadb")));
}

最后DbContenxt类中配置就可以删除了

移动DAL生成的实体类到Models模块
修改实体类中的命名空间

② 另一种使用ADO.NET实现

不是这次的重点,所以略过。

三、编写Model部分的代码

1.编写一个简单的Helper类

using Microsoft.EntityFrameworkCore;

namespace DAL
{
  class EFCoreHelper
  {
    private DbContext dbContext = null;
    public EFCoreHelper(DbContext context)
    {
      this.dbContext = context;
    }

    /// <summary>
    /// 添加实体
    /// </summary>
    /// <typeparam name="T"></typeparam>
    public int Add<T>(T entity) where T : class
    {
      dbContext.Entry(entity).State=EntityState.Added;
      return dbContext.SaveChanges();
    }

    /// <summary>
    /// 修改实体的全部属性
    /// </summary>
    /// <typeparam name="T"></typeparam>
    public int Modify<T>(T entity) where T:class
    {
      dbContext.Entry(entity).State=EntityState.Modified;
      return dbContext.SaveChanges();
    }

    /// <summary>
    /// 删除实体
    /// </summary>
    /// <typeparam name="T"></typeparam>
    public int Delete<T>(T entity) where T:class
    {
      dbContext.Entry(entity).State=EntityState.Deleted;
      return dbContext.SaveChanges();
    }
  }
}

2.完成新闻后台数据访问类

数据访问类

using System;
using System.Linq;
using System.Collections.Generic;
using Models;

namespace DAL
{
  public class NewsService
  {
    private EFCoreHelper helper = new EFCoreHelper(new HotelWebDbContext());

    /// <summary>
    /// 添加新闻
    /// </summary>
    /// <param name="news"></param>
    /// <returns></returns>
    public int AddNews(News news) => helper.Add(news);

    /// <summary>
    /// 修改新闻
    /// </summary>
    /// <param name="news"></param>
    /// <returns></returns>
    public int ModifyNews(News news) => helper.Modify(news);

    /// <summary>
    /// 删除新闻
    /// </summary>
    /// <param name="newssId"></param>
    /// <returns></returns>
    public int DeleteNews(string newssId)
    {
      News news = new News() { Id = Convert.ToUInt32(newssId) };
      return helper.Delete(news);
    }

    /// <summary>
    /// 获取指定数量的新闻列表
    /// </summary>
    /// <param name="count"></param>
    /// <returns></returns>
    public List<News> GetNews(int count)
    {
      using (HotelWebDbContext dbContext = new HotelWebDbContext())
      {
        return (from n in dbContext.News orderby n.PublishTime descending select n).Take(count).ToList();
      }
    }

    /// <summary>
    /// 根据ID获取新闻信息
    /// </summary>
    /// <param name="newssId"></param>
    /// <returns></returns>
    public News GetNewsById(string newssId)
    {
      uint id = Convert.ToUInt32(newssId);
      using (HotelWebDbContext dbContext = new HotelWebDbContext())
      {
        return (from n in dbContext.News where n.Id == id select n).FirstOrDefault();
      }
    }

    /// <summary>
    /// 获取所有的新闻分类
    /// </summary>
    /// <returns></returns>
    public List<NewsCategory> GetCategories()
    {
      using (HotelWebDbContext dbContext = new HotelWebDbContext())
      {
        return (from c in dbContext.NewsCategory select c).ToList();
      }
    }
  }
}

业务逻辑部分

using System.Collections.Generic;
using DAL;
using Models;

namespace BLL
{
  public class NewsManager
  {
    private NewsService objService=new NewsService();
    /// <summary>
    /// 添加新闻
    /// </summary>
    /// <param name="news"></param>
    /// <returns></returns>
    public int AddNews(News news) => objService.AddNews(news);

    /// <summary>
    /// 修改新闻
    /// </summary>
    /// <param name="news"></param>
    /// <returns></returns>
    public int ModifyNews(News news) => objService.ModifyNews(news);

    /// <summary>
    /// 删除新闻
    /// </summary>
    /// <param name="newssId"></param>
    /// <returns></returns>
    public int DeleteNews(string newssId) => objService.DeleteNews(newssId);

    /// <summary>
    /// 获取指定数量的新闻列表
    /// </summary>
    /// <param name="count"></param>
    /// <returns></returns>
    public List<News> GetNews(int count) => objService.GetNews(count);

    /// <summary>
    /// 根据ID获取新闻信息
    /// </summary>
    /// <param name="newssId"></param>
    /// <returns></returns>
    public News GetNewsById(string newssId) => objService.GetNewsById(newssId);

    /// <summary>
    /// 获取所有的新闻分类
    /// </summary>
    /// <returns></returns>
    public List<NewsCategory> GetCategories() => objService.GetCategories();
  }
}

3.添加一个控制台项目用来测试

添加需要的引用

dotnet add package Microsoft.EntityFrameworkCore.Design
dotnet add package Pomelo.EntityFrameworkCore.MySql

DbContext中的数据库连接字符串添加回去

   protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
    {
      if(!optionsBuilder.IsConfigured)
      {
        optionsBuilder.UseMySql("Server=192.168.30.110;DataBase=HotelWebDb;User=sa;Pwd=110;",x => x.ServerVersion("5.5.64-mariadb"));
      }
    }

编写测试代码

 News news=new News()
      {
        NewsContent="你好这是一个测试新闻内容",
        NewsTitle="测试新闻",
        CategoryId=1
      };

      Console.WriteLine(objNews.AddNews(news));

启动调试
选择启动项目有两种方法

①通过solution explorer 插件选择

②通过配置launch.json 文件启动

然后修改启动程序入口就可以了

 {
      "name": ".NET Core Launch (console)",
      "type": "coreclr",
      "request": "launch",
      "preLaunchTask": "build",
      "program": "${workspaceFolder}/UnitTestPro/bin/Debug/netcoreapp3.1/UnitTestPro.dll",
      "args": [],
      "cwd": "${workspaceFolder}",
      "stopAtEntry": false,
      "console": "internalConsole"
    },

还要修改task文件,否则测试项目中新加的代码不能被执行。

结果查验

结果符合预期

注意事项
修改新闻调用的方法不支持部分属性修改,如果对象属性不设置,那么没有设置的字段被设置为空。
后面有字段部分修改的方法。

   News news=new News()
      {
        Id=1008,
        NewsContent="修改新闻的内容",
        NewsTitle="这是被修改的新闻标题",
      };
      Console.WriteLine(objNews.ModifyNews(news));

4.编写菜品预订

5.编写招聘

6.编写投诉和建议

7.管理员登录

类似不再贴代码

四、前端UI实现

1.完成前端Html代码的编写

不多说

2.完成MVC项目中控制器和视图的文件添加

这个只能手动添加,不像VS有模板可用

3.复制网站的静态资源

asp.net core 网站静态资源都是放在wwwroot目录的,并且文件名一般是小写。

4.在视图中引用资源

MVC框架可以直接识别在wwwroot中的静态资源,不用写显示的写出目录名。

 <link rel="stylesheet" href="~/lib/bootstrap/dist/css/bootstrap.min.css" rel="external nofollow" rel="external nofollow" />
  <link rel="stylesheet" href="~/css/site.css" rel="external nofollow" rel="external nofollow" />

5.编写动作方法

  public IActionResult Index()
    {
      return View();
    }

6.添加视图

7.启动调试

首页效果图

8.视图与控制器之间传递数据

使用ViewData

视图的网页标题可以使用这种方式传递

  public IActionResult Index()
    {
      ViewData["title"]="好运来酒店";
      ViewBag.list=new NewsManager().GetNews(4);
      return View();
    }

视图中引用数据

<!doctype html>
<html lang="zh">

<head>
  <!-- Required meta tags -->
  <meta charset="utf-8">
  <meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no">

  <!-- Bootstrap CSS -->
  <link rel="stylesheet" href="~/lib/bootstrap/dist/css/bootstrap.min.css" rel="external nofollow" rel="external nofollow" />
  <link rel="stylesheet" href="~/css/site.css" rel="external nofollow" rel="external nofollow" />

  <title>@ViewData["title"]-舒适快捷经济</title>
</head>

使用ViewBag

这是所谓的动态类型,直接无中生有造一个属性

ViewBag.list=new NewsManager().GetNews(4);

视图中引用

 @foreach (News item in @ViewBag.list) {
    <div class="d-flex flex-row justify-content-between newslist">
      <div class="p-2">
        <a href="/News/NewsDetail?id=@item.Id" rel="external nofollow" rel="external nofollow" >@item.NewsTitle</a>
      </div>
      <div class="p-2">
        @item.PublishTime.ToShortDateString()
      </div>
    </div>

使用viewmodel

控制器中使用View的重载方法传递viewmodel

   public IActionResult Index()
    {
      ViewData["title"]="好运来酒店";
      return View(new NewsManager().GetNews(4));
    }

视图中先声明后使用
Specify a model using the @model directive. Use the model with @Model:

@model list<News>
...
    @foreach (News item in @Model) {
    <div class="d-flex flex-row justify-content-between newslist">
      <div class="p-2">
        <a href="/News/NewsDetail?id=@item.Id" rel="external nofollow" rel="external nofollow" >@item.NewsTitle</a>
      </div>
      <div class="p-2">
        @item.PublishTime.ToShortDateString()
      </div>
    </div>

修改后的首页

9.分部视图

创建分部视图
分部视图的创建和其他视图的创建没有任何区别,只是它作为其他视图的一部分来视图用。
避免视图代码的重复。

在其他视图中使用分部视图

引用分部视图使用特殊的标签 <partial name="..." />

@model List<News>

<div class="row">
  <div class="col-md-4">
    <partial name="../Shared/_LeftContentPartial.cshtml" />
  </div>
  <div class="col-md-8">
    <div class="lefttitle righttitle">您现在所在的位置:中式餐厅酒店&gt;新闻动态</div>
    <div class="bg-light">
      @foreach (News item in @Model) {
      <div class="d-flex flex-row justify-content-between">
        <div class="p-2"><a href="NewsDetail?id=@item.Id" rel="external nofollow" >@item.NewsTitle</a></div>
        <div class="p-2">@item.PublishTime.ToShortDateString()</div>
      </div>
      }
    </div>
  </div>
</div>

10.使用Section布局定义

@RenderSection("js",required:false)

使用

 @section js{
    <script type="text/javascript">
      function changeFontSize(fontSize) {
        //获取要变化的对象
        var divContent = $(".row .m-4");
        divContent.css("font-size",fontSize);
      }
    </script>
  }

11.表单验证

使用模型验证方式实现

添加验证特性

public partial class DishBook
  {
    public uint Id { get; set; }

    public string HotelName { get; set; }

    [Required(ErrorMessage = "{0}不能为空")]
    [Display(Name = "消费时间")]
    public DateTime? ConsumeTime { get; set; }

    [Required(ErrorMessage = "{0}不能为空")]
    [RegularExpression(@"^\d+$", ErrorMessage = "请输入正确的{0}")]
    [Display(Name = "消费人数")]
    public uint? ConsumePersons { get; set; }

    public string RoomType { get; set; }

    [Required(ErrorMessage = "{0}不能为空")]
    [RegularExpression(@"^(13[0-9]|14[5|7]|15[0|1|2|3|5|6|7|8|9]|18[0|1|2|3|5|6|7|8|9])\d{8}$", ErrorMessage = "请输入正确的{0}")]
    [Display(Name = "手机号码")]
    public string CustomerPhone { get; set; }

    [Required(ErrorMessage = "{0}不能为空")]
    [Display(Name = "姓名")]
    public string CustomerName { get; set; }

    [EmailAddress(ErrorMessage = "请输入正确的{0}")]
    [Display(Name = "电子邮件")]
    public string CustomerEmail { get; set; }

    public string Comments { get; set; }

    public DateTime BookTime { get; set; }
    public int? BookStatus { get; set; }

    // 扩展属性
    [Required(ErrorMessage = "{0}不能为空")]
    [Remote(action:"CheckVcode",controller:"Dish")]
    [Display(Name = "验证码")]
    public string ValidationCode { get; set; }
  }

使用 Tag Helper 完成前端代码编写

@model DishBook

@section js{
  <partial name="../Shared/_ValidationScriptsPartial.cshtml" />
  <script src="~/lib/My97DatePicker/WdatePicker.js"></script>
}

<div class="row">
  <div class="col-md-4">
    <partial name="../Shared/_DishPartial.cshtml" />
  </div>
  <div class="col-md-8">
    <div class="lefttitle righttitle">您现在所在的位置:中式餐厅酒店&gt;在线预订</div>
    <form class="bg-light p-3" asp-controller="Dish" asp-action="Booking" method="post">
      <div class="form-group">
        <label for="HotelName">酒店名称:</label>
        <select class="form-control w-25" name="HotelName">
          <option value="天津南开店">天津南开店</option>
          <option value="天津和平店">天津和平店</option>
          <option value="北京朝阳店">北京朝阳店</option>
        </select>
      </div>
      <div class="form-group">
        <span><label asp-for="ConsumeTime"></label> :</span>
        <div class="d-flex flex-row align-items-center">
          <input class="form-control w-50" asp-for="ConsumeTime" onclick="WdatePicker({dateFmt:'yyyy-MM-dd HH:mm:ss'})">
          <span class="p-2 text-danger">*</span>
          <span asp-validation-for="ConsumeTime"></span>
        </div>
      </div>
      <div class="form-group">
        <span><label asp-for="ConsumePersons"></label>:</span>
        <div class="d-flex flex-row align-items-center">
          <input class="form-control w-50" type="text" asp-for="ConsumePersons">
          <span class="p-2 text-danger text-center">*</span>
          <span asp-validation-for="ConsumePersons"></span>
        </div>
      </div>
      <div class="form-group">
        <label for="RoomType"> 选择包间类型: </label>
        <select class="form-control w-25" name="RoomType">
          <option value="包间">包间</option>
          <option value="散座">散座</option>
        </select>
      </div>
      <div class="form-group">
        <span>您的<label asp-for="CustomerName"></label>:</span>
        <div class="d-flex flex-row align-items-center">
          <input class="form-control w-50" asp-for="CustomerName">
          <span class="p-2 text-danger text-center">*</span>
          <span asp-validation-for="CustomerName"></span>
        </div>
      </div>
      <div class="form-group">
        <span><label asp-for="CustomerPhone"></label>:</span>
        <div class="d-flex flex-row align-items-center">
          <input class="form-control w-50" asp-for="CustomerPhone">
          <span class="p-2 text-danger text-center">*</span>
          <span asp-validation-for="CustomerPhone"></span>
        </div>
      </div>
      <div class="form-group">
        <span><label asp-for="CustomerEmail"></label>:</span>
        <div class="d-flex flex-row align-items-center">
          <input class="form-control w-50" asp-for="CustomerEmail">
          <span asp-validation-for="CustomerEmail"></span>
        </div>
      </div>
      <div class="form-group">
        <label for="Comments"> 备注事项: </label>
        <textarea class="form-control" cols="20" name="Comments" rows="4" placeholder="备注"></textarea>
      </div>
      <div class="form-group">
        <span><label asp-for="ValidationCode"></label>:</span>
        <div class="d-flex flex-row align-items-center">
          <input class="form-control w-50" asp-for="ValidationCode">
          <span class="p-2 text-danger text-center">*</span>
          <span asp-validation-for="ValidationCode"></span>
        </div>
      </div>
      <div class="form-group">
        <img alt="验证码图片" title="看不清?点击换一个" src='@Url.Action("ValidationCode","Dish")' onclick="this.src=this.src+'?'" />
      </div>
      <div class="form-group">
        <input class="btn btn-success" type="submit" value="马上预定" />
      </div>
    </form>
  </div>
</div>

引入jQuery验证插件

这个是插件MVC项目时,模板自动生成的

@section js{
  <partial name="../Shared/_ValidationScriptsPartial.cshtml" />
  <script src="~/lib/My97DatePicker/WdatePicker.js"></script>
}

Tag Helper自动根据数据类型生成对应的输入框类型
可以手动设置 type 来覆盖

验证码提交到服务器验证
使用 [Remote] 特性
详细文档参考:https://docs.microsoft.com/en-us/aspnet/core/mvc/models/validation?view=aspnetcore-3.1

The jQuery Validate remote method expects a JSON response:

  • true means the input data is valid.
  • false, undefined, or null means the input is invalid. Display the default error message.
  • Any other string means the input is invalid. Display the string as a custom error message.

注意:The [Remote] attribute is in the Microsoft.AspNetCore.Mvc namespace.

五、使用区域项目完成后台管理系统

1.创建区域项目

建立区域项目目录结构

- Project name
  Areas
   Admin //分区项目1
    Controllers
    Views
   Services //分区项目2
    Controllers
    Views

添加路由规则

在Startup.cs文件中添加区域项目路由规则

  app.UseEndpoints(endpoints =>
  {
    endpoints.MapControllerRoute(
      name: "Admin",
      pattern: "{area:exists}/{controller=Home}/{action=Index}/{id?}"

    );
    endpoints.MapControllerRoute(
      name: "default",
      pattern: "{controller=Home}/{action=Index}/{id?}"
    );
  });

测试页面
如果有和主项目同名的控制器使用区域特性区分

 using Microsoft.AspNetCore.Mvc;

  namespace HotelWebMVC.Areas.Admin.Controllers
  {
    [Area("Admin")]
    public class HomeController : Controller
    {
      public IActionResult Index()
      {
        return View();
      }

      public IActionResult Welcome()
      {
        ViewData["Message"] = "Your welcome message";

        return View();
      }
    }
  }

如果不打区域标记区分,启动项目会包错。

2.使用Cookie保存登录凭证

添加依赖包

dotnet add package Microsoft.AspNetCore.Authentication.Cookies

配置服务

public void ConfigureServices(IServiceCollection services)
    {
      services.AddControllersWithViews();

      services.AddDistributedMemoryCache();
      services.AddSession(options =>
      {
        // Set a short timeout for easy testing.
        options.IdleTimeout = TimeSpan.FromSeconds(300);
        options.Cookie.HttpOnly = true;
        // Make the session cookie essential
        options.Cookie.IsEssential = true;
      });
      services.AddAuthentication(CookieAuthenticationDefaults.AuthenticationScheme)
      .AddCookie(options=>{
          options.Cookie.HttpOnly=true;
          options.ExpireTimeSpan=TimeSpan.FromMinutes(5);
          options.LoginPath="/admin";
          // options.AccessDeniedPath="/403.html";
        }
      );

      string connString = Configuration.GetConnectionString("HotelWeb");
      services.AddDbContext<DAL.HotelWebDbContext>(options => options.UseMySql(connString, x => x.ServerVersion("5.5.64-mariadb")));
    }

启用认证服务添加的位置有要求:必须在app.UseRouting 和 app.UseEndpoints 之间。

    app.UseRouting();

      app.UseSession();

      app.UseAuthentication();

      app.UseAuthorization();

      app.UseEndpoints(endpoints =>
      {
        endpoints.MapControllerRoute(
          name: "admin",
          pattern: "{area:exists}/{controller=SysAdmin}/{action=Index}/{id?}"

        );
        endpoints.MapControllerRoute(
          name: "default",
          pattern: "{controller=Home}/{action=Index}/{id?}");
      });

登录成功后签发凭证

using System.Security.Claims;
using Microsoft.AspNetCore.Mvc;
using Microsoft.AspNetCore.Authentication;
using Microsoft.AspNetCore.Authentication.Cookies;
List<Claim> claims = new List<Claim>() { new Claim("username", admin.LoginName) };

ClaimsIdentity claimsIdentity = new ClaimsIdentity(claims, CookieAuthenticationDefaults.AuthenticationScheme);

AuthenticationProperties properties = new AuthenticationProperties()
{
	IsPersistent = true
};

await HttpContext.SignInAsync
(
	CookieAuthenticationDefaults.AuthenticationScheme,
	new ClaimsPrincipal(claimsIdentity),
	properties
);

其他需登录后才能访问的资源,在控制器上添加 [Authorize]标记

[Authorize]
public class HomeController : Controller
{
  public IActionResult Index()
  {
    return View();
  }

  public IActionResult Welcome()
  {
    ViewData["Message"] = "Your welcome message";

    return View();
  }
}

3.使用ajax提交表单

在dotnet core 中不再支持@Ajax.Form方式,而是使用jquery插件的方式支持了。
通过定义data-* 属性来支持类似的功能

和Ajax助手的对照

======================================================
AjaxOptions	      HTML attribute
======================================================
Confirm	        data-ajax-confirm
HttpMethod	      data-ajax-method
InsertionMode	     data-ajax-mode
LoadingElementDuration	data-ajax-loading-duration
LoadingElementId	   data-ajax-loading
OnBegin	        data-ajax-begin
OnComplete	      data-ajax-complete
OnFailure	       data-ajax-failure
OnSuccess	       data-ajax-success
UpdateTargetId	    data-ajax-update
Url	          data-ajax-url
======================================================

这个特性只能在form和a标签上起作用
使用方法:

①下载插件并引用到项目中
地址:https://github.com/aspnet/jquery-ajax-unobtrusive/releases
将src文件夹中的js文件拷贝到项目对应的存放位置

②编写需要的js函数
编写回调函数

	<script>
    var onSuccess=function(data){
      alert(data);
      $("#mainForm")[0].reset();
      dishImg.src = "/images/default.png";
    };

    var onFailed=function(data){
      alert(data);
    };
  </script>

③使用data属性改写标签

这里注意:要使用ajax提交表单,data-ajax="true"必须要设置为true。
data-ajax-confirm=“确认要提交吗?” 这里是弹出框的内容,不是具体的函数名
data-ajax-begin
data-ajax-complete
data-ajax-failure
data-ajax-success
这些属性值就是回调函数的名称。

4.CKeditor使用

推荐使用ckeditor4,因为5版本中文输入有问题。
使用步骤:

下载编辑器的软件包

在页面中引入它的js脚本

 <script src="../../third_files/ckeditor4/ckeditor.js"></script>

使用texterea作为目标
编辑器的高度可以config.js文件中设置

<textarea id="editor" name="editor" rows="20"></textarea>

在js中创建

<script>
  CKEDITOR.replace( 'editor' );
</script>

自定义配置
修改配置文件config.js,推荐直接在默认的文件中添加需要的配置。

CKEDITOR.editorConfig = function( config ) {
 config.language = 'es';
 config.uiColor = '#F7B42C';
 config.height = 300;
 config.toolbarCanCollapse = true;
};

获取编辑器的内容
用于提交前验证是否有内容,NewsContent是编辑器textarea的id

var content=CKEDITOR.instances.NewsContent.getData();

注意此时数据验证通过,使用js提交表单的话,编辑器并没有替换原来的内容,需要手动替换
$("#NewsContent").html(content);
不然提交到控制器中是没有值的。

var onSubmit=function(){
  var content=CKEDITOR.instances.NewsContent.getData();

  if(content==""){
    alert("新闻内容不能为空");
  }
  else{
    $("#NewsContent").html(content);
    $("#mainForm").submit();
  }
}

清空编辑器的内容
CKEDITOR.instances.NewsContent.setData("");

In rare cases it may happen that the server or application configuration
will reject submitted HTML content if it is not encoded first (e.g. ASP.NET ValidateRequest).
In such case check the config.htmlEncodeOutput option.

config.htmlEncodeOutput = true;

上传图片设置
需要外部插件:file browser,popup,filetools
配置config.js文件

config.filebrowserBrowseUrl = ‘/browser/browse.php';
config.filebrowserUploadUrl = ‘/uploader/upload.php';

控制器中的参数使用 (IFormFile upload) 使用的是upload的参数名。
如何给一个响应,放弃不用这个,使用filebrowserBrowseUrl这个功能,里面也有上传

具体方法是:

  • 构建一个图片上传和浏览的页面
  • 完成图片上传功能
  • 图片选装功能

响应的js代码如下:

<script>
  // Helper function to get parameters from the query string.
  function getUrlParam( paramName ) {
    var reParam = new RegExp( '(?:[\?&]|&)' + paramName + '=([^&]+)', 'i' );
    var match = window.location.search.match( reParam );
    return ( match && match.length > 1 ) ? match[1] : null;
  }

  var upload=function(){
    var file=$(".btnUpload>input")[0].files[0];
    if(file==null){
      alert("请选择上传图片");
    }
    else{
      var formData=new FormData();
      formData.append("upload",file);
      // 实例化一个AJAX对象
      var xhr = new XMLHttpRequest();
      xhr.onload = function() {
        $(".border3").first().before('<div class="w-25 p-3 border3"><span class="d-inline-block"><h3><span class="badge badge-info rotated">new</span></h3><img class="img-thumbnail" src="/images/news/'+xhr.responseText+'" onclick="select(this)"></span><div class="text-center p-2">'+xhr.responseText+'</div></div>');
      }
      xhr.open("post",'@Url.Action("UploadImage","News")',true);

      // 发送表单数据
      xhr.send(formData);
    }
  }

  var selectedImg=null;
  var select=function(img){
    if(selectedImg!=null){
      selectedImg.parents(".border3").removeClass("selected");
    }
    selectedImg=$(img);
    selectedImg.parents(".border3").addClass("selected");
  }

  var choose=function(){
    if(selectedImg!=null){
      var funcNum = getUrlParam( 'CKEditorFuncNum' );
      var fileUrl = selectedImg.attr("src");
      window.opener.CKEDITOR.tools.callFunction( funcNum, fileUrl );
      window.close();
    }
    else{
      alert("请选装图片");
    }
  }
</script>

5.其他功能省略代码

六、使用依赖注入改进项目

1.抽取接口

通过vs来抽取接口,效率高。

2.重新组织项目结构

新增IDAL、IBLL、DBUtility
UI–>IBLL–>IDAL–>Models
BLL–>IBLL、IDAL
IDAL–>Models
DAL–>IDAL、DBUtility

3.注册依赖服务

public void ConfigureServices(IServiceCollection services)
  {
   //其他代码省略了
   ...

    services.AddDbContext<DBUtility.HotelWebDbContext>(
      options => options.UseMySql(
        Configuration.GetConnectionString("HotelWeb"),
        x => x.ServerVersion("5.5.64-mariadb")
      )
    );

    services.AddTransient<INewsManager, NewsManager>();
    services.AddTransient<IDishManager, DishManager>();
    services.AddTransient<IDishBookManager, DishBookManager>();
    services.AddTransient<ISuggestionManager, SuggestionManager>();
    services.AddTransient<IRecruitmentManager, RecruitmentManager>();
    services.AddTransient<ISysAdminManager, SysAdminManager>();

    services.AddTransient<INewsService,NewsService>();
    services.AddTransient<IDishService,DishService>();
    services.AddTransient<IDishBookService,DishBookService>();
    services.AddTransient<ISuggestionService,SuggestionService>();
    services.AddTransient<IRecruitmentService,RecruitmentService>();
    services.AddTransient<ISysAdminService,SysAdminService>();
  }

4.修改代码使用依赖注入

public class HomeController : Controller
  {
    private readonly INewsManager newsManager;
    private readonly ISuggestionManager suggestionManager;
    private readonly IRecruitmentManager recruitmentManager;
    public HomeController(
     INewsManager newsManager,
     ISuggestionManager suggestionManager,
     IRecruitmentManager recruitmentManager)
    {
      this.newsManager = newsManager;
      this.suggestionManager = suggestionManager;
      this.recruitmentManager = recruitmentManager;
    }

    // ...
  }

5.测试

启动项目也没有什么问题

七、项目发布

1.项目配置

dotnet core 中没有Web.conf文件了。
查看文档,都是通过Startup.cs中配置项目的。
暂时放弃配置。

2.命令行发布项目

CLI 提供了发布项目的相关命令

dotnet publish -c Release --no-self-contained -o /path/to/save/project/

3.另一种方式使用vs发布

很简单,下一步下一步做就好了。

八、通过Nginx部署到Linux服务器

1.在Centos7 上安装运行时

Register Microsoft key and feed

sudo rpm -Uvh https://packages.microsoft.com/config/centos/7/packages-microsoft-prod.rpm

Install the ASP.NET Core runtime

sudo yum install dotnet-sdk-3.1

2.安装libgdiplus

因为项目中使用验证码,需要用到这个命名空间:System.Drawing.Common
速度太慢,放弃。

3.将项目目录上传到linux

使用xshell 的ftp 轻松完成。

4.测试项目是否运行

cd /var/www/HotelWeb/
dotnet HotelWeb.dll

[root@centos7 HotelWeb]# dotnet HotelWebMVC.dll
info: Microsoft.Hosting.Lifetime[0]
   Now listening on: http://localhost:5000
info: Microsoft.Hosting.Lifetime[0]
   Application started. Press Ctrl+C to shut down.
info: Microsoft.Hosting.Lifetime[0]
   Hosting environment: Production
info: Microsoft.Hosting.Lifetime[0]
   Content root path: /var/www/HotelWeb

5.安装Nginx并配置

yum install nginx
vim /etc/nginx/nginx.conf
---------------
# 删除nginx默认的server,添加下面两个

server {
  listen  80 default_server;
  return  444;
}

server {
  listen    80;
  server_name  *.hotel.com;
  location / {
    proxy_pass     http://localhost:5000;
    proxy_http_version 1.1;
    proxy_set_header  Upgrade $http_upgrade;
    proxy_set_header  Connection keep-alive;
    proxy_set_header  Host $host;
    proxy_cache_bypass $http_upgrade;
    proxy_set_header  X-Forwarded-For $proxy_add_x_forwarded_for;
    proxy_set_header  X-Forwarded-Proto $scheme;
  }
}

----------------

6.启动nginx

systemctl start nginx
nginx -t
# 配置确认没有问题后,重新载入配置
nginx -s reload

7.浏览器测试

修改win7的host

www.hotel.com 192.168.30.110

8.最后的问题

因为libgdiplus这个库没有安装,所以进入验证码的页面会有报错。

fail: Microsoft.AspNetCore.Diagnostics.ExceptionHandlerMiddleware[1]
An unhandled exception has occurred while executing the request.
System.TypeInitializationException: The type initializer for 'Gdip' threw an exception.
---> System.DllNotFoundException: Unable to load shared library 'libgdiplus' or one of its dependencies.

9.修改dotnet监听端口

在Program.cs 文件中修改

   public static IHostBuilder CreateHostBuilder(string[] args) =>
      Host.CreateDefaultBuilder(args)
        .ConfigureWebHostDefaults(webBuilder =>
        {
          webBuilder.ConfigureKestrel(serverOptions =>
          {
            serverOptions.Listen(IPAddress.Any, 5000);
          })
          .UseStartup<Startup>();
        });
  }

修改后

[root@centos7 HotelWeb]# dotnet HotelWebMVC.dll
info: Microsoft.Hosting.Lifetime[0]
   Now listening on: http://0.0.0.0:5000
info: Microsoft.Hosting.Lifetime[0]
   Application started. Press Ctrl+C to shut down.
info: Microsoft.Hosting.Lifetime[0]
   Hosting environment: Production
info: Microsoft.Hosting.Lifetime[0]
   Content root path: /var/www/HotelWeb

[root@centos7 ~]# ss -tna
State    Recv-Q Send-Q          Local Address:Port                  Peer Address:Port
LISTEN   0   128                  *:5000                       *:*
LISTEN   0   50                  *:3306                       *:*
LISTEN   0   128                  *:111                        *:*
LISTEN   0   128                  *:80                        *:*
LISTEN   0   128                  *:22                        *:*
LISTEN   0   100              127.0.0.1:25                        *:*
ESTAB    0   0            192.168.30.110:22                  192.168.30.88:1581
ESTAB    0   52            192.168.30.110:22                  192.168.30.88:1516
LISTEN   0   128                 :::111                       :::*
LISTEN   0   128                 :::22                        :::*
LISTEN   0   100                 ::1:25                        :::*    

参考文档

连接字符串-EF

Entity Framework Core tools reference - .NET CLI

Using a Separate Migrations Project

Overview of ASP.NET Core MVC

The Form Tag Helper

Model validation in ASP.NET Core MVC

Use cookie authentication without ASP.NET Core Identity

Layout in ASP.NET Core

Architectural principles

Dependency injection in ASP.NET Core

Configuration in ASP.NET Core

CentOS 7 Package Manager - Install .NET Core

Add the Mono repository to your system

到此这篇关于基于.NET Core 3.1 网站开发和部署的方法的文章就介绍到这了,更多相关.NET Core 3.1 网站开发部署内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • 从ASP.NET Core3.1迁移到5.0的方法

    3月中旬,微软官方已经发布了dotnet 5的第一个预览版:5.0.0-preview.1. dotnet core经过前几个版本的发展和沉淀,到3.1已经基本趋于稳定. 所以从.net core 3.1升级到.net 5将非常简单,下面咱们就开始吧: (注:需要 Visual Studio 2019 16.6 或更高版本) 第一步:在 global.json 中更新 .NET Core SDK 版本 { "sdk": { "version": "5.0.

  • .NET Core3.1发布(翻译)

    .NET Core3.1发布 我们很高兴宣布.NET Core 3.1的发布.实际上,这只是对我们两个多月前发布的.NET Core 3.0的一小部分修复和完善.最重要的是.NET Core 3.1是长期支持(LTS)版本,并且将支持三年.和过去一样,我们希望花一些时间来发布下一个LTS版本.额外的两个月(在.NET Core 3.0之后)使我们能够选择和实施在已经非常稳定的基础上进行的正确改进. 您可以下载适用于Windows,macOS和Linux的.NET Core 3.1: .NET C

  • .NET Core3.1编写混合C++程序

    前言 随着 .NET Core 3.1 的第二个预览版本发布,微软正式将 C++/CLI 移植到 .NET Core 上,从此可以使用 C++ 编写 .NET Core 的程序了. 由于目前仅有 MSVC 支持编译此类混合代码,并且由于涉及到非托管代码,因此 C++/CLI 目前不能跨平台,只支持 Windows. 如果需要跨平台,除了微软的工作之外,还另外需要 gcc/clang 大量跟进,工作量较大且进度不可控,目前微软暂无使 C++/CLI 跨平台的计划. 先决条件 Visual Stud

  • .Net Core 2.2升级3.1的避坑指南(小结)

    写在前面 微软在更新.Net Core版本的时候,动作往往很大,使得每次更新版本的时候都得小心翼翼,坑实在是太多.往往是悄咪咪的移除了某项功能或者组件,或者不在支持XX方法,这就很花时间去找回需要的东西了,下面是个人在迁移.Net Core WebApi项目过程中遇到的问题汇总: 开始迁移 1. 修改*.csproj项目文件 <TargetFramework>netcoreapp2.2</TargetFramework> 修改为 <TargetFramework>net

  • 在阿里云函数计算上部署.NET Core 3.1的方法

    使用阿里云ECS或者其他常见的VPS服务部署应用的时候,需要手动配置环境,并且监测ECS的行为,做补丁之类的,搞得有点复杂.好在很多云厂商(阿里云.Azure等)提供了Serverless服务,借助于Serverless,开发人员可以更加专注于代码的开发,减少运维的成本. Azure的部署直接集成在了VS中,非常方便,本文主要介绍一下使用ASP.NET CORE 3.1部署在阿里云Serverless(函数计算)的内容. 准备 阿里云的函数计算提供了很多运行库,对.NET的支持现在到ASP.NE

  • asp.net core3.1 引用的元包dll版本兼容性问题解决方案

    自从.netcore 3.1出来后,大家都想立马升级到最新版本.我也是如此,微软也对.netcore 3.1 的官方组件不断升级,几乎每隔几天就会有部分元包可以升级.每次打开Nuget包管理器,"更新"的tab处总会有个数字暗示着你快点升级!一向有代码洁癖的同学(包括我),都会毫不犹豫的点击"全部升级".我们总是幻想着使用更高的版本,可以获取更牛的功能和更快的性能. 可惜,梦想是好的,结局却是残酷的.升级后,却发现有很多错误..netcore 3.1的确在性能方面有

  • .net core 3.1在iis上发布的踩坑记录

    前言 写这篇文章的目的是希望像我一样喜欢.net 的人在发布 core到 iis上时少走点弯路 网上找了些资料,其实实际操作比较简单,就是有几个坑很恶心 踩坑记录 首先是你的服务器需要有core 的运行环境,安装前先关闭iis dotnet-hosting-3.1.4-win.exe 可以去微软的官网找最新的版本(去微软的官网找你要的版本就好了) 安装成功后,第一个坑出现了,启动iis,发现原来在iis上的网站都报503错误了. 直接玩大了,最后发现就是这个东西搞的鬼,你卸载它iis之前的网站就

  • 1个文件如何轻松搞定Asp.net core 3.1动态页面转静态页面

    前言 最近一个Asp.net core项目需要静态化页面,百度查找了一下,没有发现合适的.原因如下 配置麻烦. 类库引用了第三方类,修改起来麻烦. 有只支持MVC,不支持PageModel. 继承ActionFilterAttribute类,只重写了OnActionExecutionAsync,看似静态化了,其实运行时该查数据库还是查数据库,没有真正静态化. 缺少灵活性,没有在线更新静态文件方法,不能测试查看实时页面,没有进行Html压缩,没有使用gzip.br压缩文件. 于是我开始了页面静态化

  • 基于.NET Core 3.1 网站开发和部署的方法

    一.准备开发环境 1.主要开发工具的选择 vscode .NET Core command-line interface (CLI) tools Dbeaver 这里选择vscode + .net core cli 是因为不管在Windows还是Linux和Mac上都能使用这一套工具,而且命令行工具也非常强大. 2.vscode安装C#插件 在vscode插件市场中搜索安装即可 新手还可以去这里了解vscode的强大之处 3.安装数据库 这里使用Centos7,因为.NET Core 3.1只支

  • Python基于whois模块简单识别网站域名及所有者的方法

    本文实例讲述了Python基于whois模块简单识别网站域名及所有者的方法.分享给大家供大家参考,具体如下: 对于一些网站,我们可能会关心其所有者是谁.为了找到网站的所有者,我们可以使用WHOIS协议查询域名的注册者是谁.Python中有一个对该协议的封装库.我们可以通过pip进行安装. pip install python-whois 补充:本机安装了Python2与Python3两个版本,这里就使用了pip2安装python-whois模块,如下图所示: 本机Python3环境下适用pip3

  • 基于.net core微服务的另一种实现方法

    前言 基于.net core 的微服务,网上很多介绍都是千篇一律基于类似webapi,通过http请求形式进行访问,但这并不符合大家使用习惯.如何像形如[ GetService<IOrderService>().SaveOrder(orderInfo)]的方式, 调用远程的服务,如果你正在为此苦恼, 本文或许是一种参考. 背景 原项目基于传统三层模式组织代码逻辑,随着时间的推移,项目内各模块逻辑互相交织,互相依赖,维护起来较为困难.为此我们需要引入一种新的机制来尝试改变这个现状,在考察了 Ja

  • JSP动态网站开发环境配置详细方法第1/2页

    下面就以Tomcat作为JSP引擎,配合Tomcat.Apache.IIS这三种Web服务器来讲述3种搭建JSP运行环境的方案. 一.相关软件介绍 1. J2SDK:Java2的软件开发工具,是Java应用程序的基础.JSP是基于Java技术的,所以配置JSP环境之前必须要安装J2SDK. 2. Apache服务器:Apache组织开发的一种常用Web服务器,提供Web服务. 3. Tomcat服务器:Apache组织开发的一种JSP引擎,本身具有Web服务器的功能,可以作为独立的Web服务器来

  • asp.net core下给网站做安全设置的方法详解

    前言 本文主要介绍了关于asp.net core给网站做安全设置的相关内容,分享出来供大家参考学习,下面话不多说了,来一起看看详细的介绍吧 设置方法如下 首先,我们来看下stack overflow网站的请求头文件: 可以看到一些我们熟悉或是陌生的HTTP头部文件字段. 在这里我们在对HTTP输入流的头部文件中,做一些基本的防护.首先要明确,既然我们是对HTTP头部做处理,那么就需要在Startup.cs类的 Configuration方法中做处理,因为这里就是处理HTTP输入流的. 首先做一些

  • 基于JSP的动态网站开发技术

    随着Web技术的发展和电子商务时代的到来,人们不再满足于建立各种静态地发布信息的网站,更多的时候需要能与用户进行交互,并能提供后台数据库的管理和控制等服务的动态网站. 动态网站开发技术 早期的动态网站开发技术使用的是CGI-BIN接口.开发人员编写与接口相关的单独的程序和基于Web的应用程序,后者通过Web服务器来调用前者.这种开发技术存在着严重的扩展性问题--每一个新的CGI程序要求在服务器上新增一个进程.如果多个用户并发地访问该程序,这些进程将耗尽该Web服务器所有的可用资源,直至其崩溃.

  • ASP.NET Core开发Docker部署

    Docker 常用命令 docker info 检查Docker的安装是否正确,如果没有找到此命令,则表示Docker没有正确安装 docker pull busybox 拉取一个预建的镜像 sample_job=$(docker run -d busybox /bin/sh -c "while true; do echo Docker; sleep 1; done") 以后台进程的方式运行hello docker sample_job命令会隔一秒打印一次Docker,使用Docker

  • 基于NET Core 的Nuget包制作、发布和运用流程解析(完整过程)

    目录 前期准备 Nuget包制作 上传Nuget包 Nuget包使用 (一).准备配置信息 (二).业务层调用 开发缘由:公司需要调用天眼查-开放平台 ,验证客户的的营业执照信息是否在存续期,并将企业基本信息返回,之后和使用百度图文识别的企业信息进行对照是否一致. 前期准备 在网站中注册后,需要够买套餐.之后点击个人中心进入,点击左侧的我的接口,找到申请接口点击选择你要使用的接口,此次使用的是企业基本信息. 存放Nuget包目前我已知的是1.NuGet官网https://www.nuget.or

  • Ubuntu 搭建基于Docker的LNMP+Redis的开发环境(图文)

    Ubuntu 搭建基于Docker的LNMP+Redis的开发环境 服务器环境:Ubuntu 14.04 1.安装Docker 1.1 执行update命令,和服务器同步软件包,执行apt-get install * 时可以下载最新的软件. 1.2 安装Docker和创建软链接   1.3 启用Docker服务 2. 获取搭建环境所需镜像 2.1 MySQL镜像 2.2 Redis镜像   2.3 nginx-php-fpm镜像 2.4 查看已下载的镜像 对于Docker初学者来说,可以使用现有

  • Oracle Portal及其门户网站开发概述

    正在看的ORACLE教程是:Oracle Portal及其门户网站开发概述.摘要: Portal是IT领域的新技术,是企业信息化工作的发展方向之一.本文首先介绍了Oracle Portal的定义.特点,接着阐述了portal的体系结构.随后本文介绍了Oracle9iAS Portal的功能特点及基于它的企业门户网站的开发.最后本文简要探讨了基于portal的门户网站开发的问题. 关键词: oracle portal,门户,网站,信息入口,企业入口 引言 Oracle Portal为企业提供了一个

随机推荐