在WEB API 2中使用属性路由创建REST API 之书籍管理系统

WEB API2支持新的属性路由,本章会使用到一些实际项目中,此API会支持以下的一些Action

Paste_Image.png

所有方法都是只读的HTTP请求
我们会使用Entity Framework,图书记录会包含以下字段
ID
Title
Genre
Publication date
Price
Description
AuthorID (foreign key to an Authors table)
首先创建一个WEB API空项目,此过程不做详细说明。


Paste_Image.png

然后右键项目文件夹 Models添加类 名称Author

Paste_Image.png

添加另外一个类,名称Book

Paste_Image.png

添加控制器 右键Controllers 文件夹,添加控制器

Paste_Image.png

在添加窗口中 名称是Books 模型类为Book,勾选使用异步控制器操作


Paste_Image.png

新建上下文 添加 确定


Paste_Image.png

会添加成功2个文件
Paste_Image.png

创建数据库 选择 工具栏 NuGet包管理器 程序包管理控制台
输入 enable-migrations

Paste_Image.png

生成了新的文件

Paste_Image.png

打开此文件,替换以下代码。

protected override void Seed(BooksAPI.Models.BooksAPIContext context)
{
    context.Authors.AddOrUpdate(new Author[] {
        new Author() { AuthorId = 1, Name = "Ralls, Kim" },
        new Author() { AuthorId = 2, Name = "Corets, Eva" },
        new Author() { AuthorId = 3, Name = "Randall, Cynthia" },
        new Author() { AuthorId = 4, Name = "Thurman, Paula" }
        });

    context.Books.AddOrUpdate(new Book[] {
        new Book() { BookId = 1,  Title= "Midnight Rain", Genre = "Fantasy", 
        PublishDate = new DateTime(2000, 12, 16), AuthorId = 1, Description =
        "A former architect battles an evil sorceress.", Price = 14.95M }, 

        new Book() { BookId = 2, Title = "Maeve Ascendant", Genre = "Fantasy", 
            PublishDate = new DateTime(2000, 11, 17), AuthorId = 2, Description =
            "After the collapse of a nanotechnology society, the young" +
            "survivors lay the foundation for a new society.", Price = 12.95M },

        new Book() { BookId = 3, Title = "The Sundered Grail", Genre = "Fantasy", 
            PublishDate = new DateTime(2001, 09, 10), AuthorId = 2, Description =
            "The two daughters of Maeve battle for control of England.", Price = 12.95M },

        new Book() { BookId = 4, Title = "Lover Birds", Genre = "Romance", 
            PublishDate = new DateTime(2000, 09, 02), AuthorId = 3, Description =
            "When Carla meets Paul at an ornithology conference, tempers fly.", Price = 7.99M },

        new Book() { BookId = 5, Title = "Splish Splash", Genre = "Romance", 
            PublishDate = new DateTime(2000, 11, 02), AuthorId = 4, Description =
            "A deep sea diver finds true love 20,000 leagues beneath the sea.", Price = 6.99M},
    });
}

在程序包管理器控制台窗口中,键入以下命令

add-migration Initial

update-database
PM> add-migration Initial
Scaffolding migration 'Initial'.
The Designer Code for this migration file includes a snapshot of your current Code First model. This snapshot is used to calculate the changes to your model when you scaffold the next migration. If you make additional changes to your model that you want to include in this migration, then you can re-scaffold it by running 'Add-Migration Initial' again.
PM> Add-Migration Initial
Re-scaffolding migration 'Initial'.
PM> update-database
Specify the '-Verbose' flag to view the SQL statements being applied to the target database.
Applying explicit migrations: [201701170713346_Initial].
Applying explicit migration: 201701170713346_Initial.
Running Seed method.

这些命令可以创建一个本地数据库,并且运行Seed方法填充数据。

添加DTO类 解决方案中添加新文件夹,命名为Dto,然后添加BookDto到该文件夹下

using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;

namespace BooksAPI.Dto
{
    public class BookDto
    {
        public string Title { get; set; }
        public string Author { get; set; }
        public string Genre { get; set; }
    }
}

添加另一个名为BookDetailDto的类.

using System;

namespace BooksAPI.DTOs
{
    public class BookDetailDto
    {
        public string Title { get; set; }
        public string Genre { get; set; }
        public DateTime PublishDate { get; set; }
        public string Description { get; set; }
        public decimal Price { get; set; }         
        public string Author { get; set; }
    }
}

以下是BookCtrl中的代码

using BooksAPI.Dto;
using BooksAPI.DTOs;
using BooksAPI.Models;
using System;
using System.Data.Entity;
using System.Linq;
using System.Linq.Expressions;
using System.Threading.Tasks;
using System.Web.Http;
using System.Web.Http.Description;

namespace BooksAPI.Controllers
{
    public class BooksController : ApiController
    {
        private BooksAPIContext db = new BooksAPIContext();

        // Typed lambda expression for Select() method. 
        private static readonly Expression<Func<Book, BookDto>> AsBookDto =
            x => new BookDto
            {
                Title = x.Title,
                Author = x.Author.Name,
                Genre = x.Genre
            };

        // GET api/Books
        public IQueryable<BookDto> GetBooks()
        {
            return db.Books.Include(b => b.Author).Select(AsBookDto);
        }

        // GET api/Books/5
        [ResponseType(typeof(BookDto))]
        public async Task<IHttpActionResult> GetBook(int id)
        {
            BookDto book = await db.Books.Include(b => b.Author)
                .Where(b => b.BookId == id)
                .Select(AsBookDto)
                .FirstOrDefaultAsync();
            if (book == null)
            {
                return NotFound();
            }

            return Ok(book);
        }

        protected override void Dispose(bool disposing)
        {
            db.Dispose();
            base.Dispose(disposing);
        }
    }
}

下来添加属性路由
首先给控制器添加RoutePrefix属性

Paste_Image.png

添加属性路由

Paste_Image.png

获取书籍详细信息

        [Route("{id:int}/Details")]
        [ResponseType(typeof(BookDetailDto))]
        public async Task<IHttpActionResult> GetBookDetail(int id)
        {
            var book = await (from b in db.Books.Include(b => b.Author)
                              where b.AuthorId == id
                              select new BookDetailDto
                              {
                                  Title = b.Title,
                                  Genre = b.Genre,
                                  PublishDate = b.PublishDate,
                                  Price = b.Price,
                                  Description = b.Description,
                                  Author = b.Author.Name
                              }).FirstOrDefaultAsync();
            if (book == null)
            {
                return NotFound();
            } else
            {
                return Ok(book);
            }
        }

获取指定分类的书籍

        [Route("{genre}")]
        public IQueryable<BookDto> GetBookByGenre(string genre)
        {
            return db.Books.Include(b => b.Author)
               .Where(b => b.Genre.Equals(genre, StringComparison.OrdinalIgnoreCase))
               .Select(AsBookDto);
        }

获取指定作者的书籍列表

        [Route("~/api/authors/{authorId}/books")]
        public IQueryable<BookDto> GetBooksByAuthor(int authorId)
        {
            return db.Books.Include(b => b.Author)
                .Where(b => b.AuthorId == authorId)
                .Select(AsBookDto);
        }

按出版日期获取书籍

        //此处使用正则表达式来匹配日期格式,并且匹配多个格式的时间
        [Route("date/{pubdate:datetime:regex(\\d{4}-\\d{2}-\\d{2})}")]
        [Route("date/{*pubdate:datetime:regex(\\d{4}/\\d{2}/\\d{2})}")] 
        public IQueryable<BookDto> GetBooks(DateTime pubdate)
        {
            return db.Books.Include(b => b.Author)
                .Where(b => DbFunctions.TruncateTime(b.PublishDate)
                == DbFunctions.TruncateTime(pubdate))
                .Select(AsBookDto);
            
        }

分别请求测试

获取所有书籍

Paste_Image.png

获取指定书籍

Paste_Image.png

获取指定书籍的详细信息

Paste_Image.png

根据分类获取

Paste_Image.png

根据作者ID获取书籍列表

Paste_Image.png

根据出版日期获取书籍列表

Paste_Image.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,973评论 19 139
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 173,466评论 25 708
  • Spring Boot 参考指南 介绍 转载自:https://www.gitbook.com/book/qbgb...
    毛宇鹏阅读 46,970评论 6 342
  • 跟着思绪,不知觉走到你楼下 看到你房间亮着灯光 心里多了一些踏实 温意心间流过~ 我不知未来如何,只想任何任何 都...
    高国恒阅读 755评论 0 0
  • 有些时日没有写东西了,这段时间说忙也忙,说不忙也不忙。很累倒是真的。 也算是在忙一些自己的事情了,从那么早开始筹划...
    Liao__阅读 396评论 0 1