购物车Demo,前端使用AngularJS,后端使用ASP.NET Web API(1)--后端
原文:购物车Demo,前端使用AngularJS,后端使用ASP.NET Web API(1)--后端
chsakell分享了前端使用AngularJS,后端使用ASP.NET Web API的购物车案例,非常精彩,这里这里记录下对此项目的理解。
文章:
http://chsakell.com/2015/01/31/angularjs-feat-web-api/
http://chsakell.com/2015/03/07/angularjs-feat-web-api-enable-session-state/
源码:
https://github.com/chsakell/webapiangularjssecurity
本系列共三篇,本篇是第一篇。
购物车Demo,前端使用AngularJS,后端使用ASP.NET Web API(1)--后端
购物车Demo,前端使用AngularJS,后端使用ASP.NET Web API(2)--前端,以及前后端Session
购物车Demo,前端使用AngularJS,后端使用ASP.NET Web API(3)--Idetity,OWIN前后端验证
■ 配置EF
首先搞清模型之间的关系:
public class Gadget
{
public int GadgetID{get;set;}
... public int CategoryID{get;set;}
public Category Category{get;set;}
} public class Category
{
public int CategoryID{get;set;}
...
public List<Gadget> Gadgets{get;set;}
} public class Order
{
public int OrderID{get;set;}
...
public List<Gadget> Gadgets{get;set;}
} public class GadgetOrder
{
public int GadgetOrderID{get;set;} public int OrderID{get;set;}
public Order Order{get;set;} public int GadgetID{get;set;}
public Gadget Gadget{get;set;}
}
以上,Category和Gadget是1对多的关系,GadgetOrder是Order和Gadget的中间表。
接着需要通过EF Fluent API来配置领域,需要实现EntityTypeConfiguration<TModel>这个基类。比如:
public class OrderConfiguration : EntityTypeConfiguration<Order>
{
public OrderConfiguration()
{
Ignore(o => o.Gadgets);
}
}
然后就配置上下文,继承DbContext这个基类。
public class StoreContext : DbContext
{
protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
...
modelBuilder.Configurations.Add(new OrderConfiguration());
} public DbSet<Order> Orders{get;set;}
...
}
我们还希望在生成数据库的时候生成一些种子数据,需要继承DropCreateDatabaseIfModelChanges<TContext>这个泛型类。
public class StoreInitializer : DropCreateDatabaseIfModelChanges<StoreContext>
{
protected override void Seed(StoreContext context)
{
try
{
GetCategoreis().ForEach(c => context.Categories.Add(c));
...
}
catch(Exception ex)
{
Debug.WriteLine(ex.Message);
}
} private static List<Category> GetCategories()
{
...
}
}
如何调用数据库种子数据的类StoreInitializer呢?有一种方法使在项目全局文件中配置。(还有一种方法使在DbContext的构造函数中配置,还有一种在数据库迁移文件中配置,etc.)
void Application_Start(object sender, EventArgs e)
{
Database.SetInitializer(new StoreIntializer());
}
最后,关于EF的配置部分,需要在Web.config中配置,大致如下:
<connectionStrings>
<add name="ProductServiceContext" connectionString="Data Source=.;User=someusername;Password=somepassword;Initial Catalog=MyProductService;Integrated Security=True" providerName="System.Data.SqlClient"/>
</connectionStrings>
■ CategoriesController
先俯瞰。
public class CategoriesController : ApiController
{
private StoreContext db = new SotoreContext(); ... private bool CategoryExists(int id)
{
return db.Categories.Count(g => g.CategoryID=id) > ;
} protected override void Dispose(boo disposing)
{
if(disposing)
{
db.Dispose();
}
base.Dispose(disposing);
}
}
再细节。
/GET api/Categories
public IQueryable<Category> GetCategories()
{
return db.Categories;
} //GET api/Categories/5
[ResponseType(typeof(Category))]
public async Task<IHttpActionResult> GetCategory(int id)
{
Category category = await db.Categories.FindAsync(id);
if(category == null)
{
return NotFound();
}
return Ok(category);
} //put api/Categories/5
[ResponseType(typeof(void))]
public async Task<IHttpActionResult> PutCategory(int id, Category category)
{
if(!ModelState.IsValid)
{
return BadRequest(ModelState);
} if(id != category.CategoryID)
{
return BadRequest();
} db.Entry(category).State = EntityState.Modified; try
{
await db.SavheChangesAsync();
}
catch(DbUpdateConcurrencyException)
{
if(!CategoryExists(id))
{
return NotFound();
}
else
{
throw;
}
}
return StatusCode(HttpStatusCOde.NoContet);
} //post api/Categories
[ResponseType(typeof(Category))]
public async Task<IHttpActionResult> PostCategory(Category category)
{
if(!ModelState.IsValid)
{
return BadRequest(ModelState);
} db.Categories.Add(category);
await db.SaveChangesAsync(); //返回到指定的路由
return CreatedAtRoute("DefaultApi", new {id = category.CategoryID}, category);
} //delete api/Categoreis/5
[ResponseType(typeof(Category))]
public async Task<IHttpActionResult> DeleteCategory(int id)
{
Category category = await db.Categories.FindAsync(id);
if(category == null)
{
return NotFound();
} db.Categories.Remove(category);
await db.SaveChangesAsync();
return Ok(category);
}
■ GadgetsController
先俯瞰。
public class GadgetsController : ApiController
{
private StoreContext db = new StoreContext(); protected override void Dispose(bool disposing)
{
if(disposing)
{
db.Dispose();
} base.Dispose(disposing);
} private bool GadgetExists(int id)
{
return db.Gadgets.Count(g => g.GadgetID == ID) > ;
}
}
再细节。
//get api/Gadgets
public IQueryable<Gadget> GetGadgets()
{
return db.Gadgets;
} //get api/Gadgets/5
[ResponseType(typeof(Gadgets))]
public async Task<IHttpActionResult> GetGadget(int id)
{
Gadget gadget = await db.Gadgets.FindAsync(id);
if(gadget == null)
{
return NotFound();
}
return Ok(gadget);
} //put api/Gadgets/5
[ResponseType(typeof(void))]
public async Task<IHttpActionResult> PutGadget(int id, Gadget gadget)
{
if(!ModelState.IsValid)
{
return BadRequest(ModelState);
} if(d != gadget.GadgetID)
{
return BadRequest();
} db.Entry(gadget).State = EntityState.Modified; try
{
await db.SaveChangesAsync();
}
catch(DbUpdateConcurrencyException)
{
if(!GadgetExists(id))
{
return NotFound();
}
else
{
throw;
}
}
} //post api/Gadgets
[ResposneType(typeof(Gadget))]
public async Task<IHttpActionResult> PostGadget(Gadget gadget)
{
if(!ModelState.IsValid)
{
return BadRequest(ModelState);
}
db.Gadgets.Add(gadget);
await db.SaveChangesAsync(); return CreatedAtRoute("DefaultApi", new {id=gadget.GadgetID}, gadget)
} //delete api/Gadgets/5
[ResponseType(typeof(Gadget))]
public async Task<IHttpActionResult> DeleteGadget(int id)
{
Gadget gadget = await db.Gadgets.FindAsync(id);
if(gadget == null)
{
return NotFound();
}
db.Gadgets.Remove(gadget);
await db.SaveChangesAsync();
return Ok(gadget);
}
■ OrdersController
firstly overview.
public class OrdersController : ApiController
{
private StoreContext db = new StoreContext(); protected override void Dispose(bool dispoing)
{
if(disposing)
{
db.Dispose();
}
base.Dispose(disposing);
} private bool OrderExists(int id)
{
return db.Orders.Count(g => g.OrderID == id) > ;
}
}
then details.
// get api/Orders
public IQueryable<Order> GetOrders()
{
return db.Orders;
} //get api/Orders/5
[ResponseType(typeof(Order))]
public async Task<IHttpActionResult> GetOrder(int id)
{
Order order = await db.Orders.FindAsync(id);
if(order == null)
{
return NotFound();
}
return Ok(order);
} //put api/Orders/5
[ResponseType(typeof(void))]
public async Task<IHttpActionResult> PutOrder(int id, Order order)
{
if(!ModelState.IsValid)
{
return BadRequest(ModelState);
} if(id != order.OrderID)
{
return BadRequest();
} db.Entry(order).State = EntityState.Modified; try
{
await db.SaveChangesAsync();
}
catch(DbUpdateConcurrecyException)
{
if(!OrderExists(id))
{
return NotFound();
}
else
{
throw;
}
}
return StatusCode(HttpStatusCOde.NoContet);
} //post api/Orders
[RequestType(typeof(Order))]
public async Task<IHttpActionResult> PostOrder(Order order)
{
if(!ModelState.IsValid)
{
reuturn BadRequest(ModelState);
}
try
{
db.Orders.Add(order);
foreach(Gadget gadget in order.Gadgets)
{
db.GadgetOrders.Add(new GadgetOrder{
OrderID = order.OrderID,
GadgetID = gadget.GadgetID
})
} await db.SaveChangesAsync();
}
catch(Exception ex)
{
return BadRequest(ex.Message);
}
return CreatedAtRoutne("Default", new {controller = "Home", action="viewOrder", id = order.orderID}, order);
} //delete api/Orders/5
[ResponseType(typeof(Order))]
public async Task<IHttpActionResult> DeleteOrder(int id)
{
Order order = await db.Orders.FindAsync(id);
if(order == null)
{
return NotFound();
} db.Orders.Remoe(order);
await db.SaveChangesAsync();
return Ok(order);
}
待续~~
购物车Demo,前端使用AngularJS,后端使用ASP.NET Web API(1)--后端的更多相关文章
- 购物车Demo,前端使用AngularJS,后端使用ASP.NET Web API(3)--Idetity,OWIN前后端验证
原文:购物车Demo,前端使用AngularJS,后端使用ASP.NET Web API(3)--Idetity,OWIN前后端验证 chsakell分享了前端使用AngularJS,后端使用ASP. ...
- 购物车Demo,前端使用AngularJS,后端使用ASP.NET Web API(2)--前端,以及前后端Session
原文:购物车Demo,前端使用AngularJS,后端使用ASP.NET Web API(2)--前端,以及前后端Session chsakell分享了前端使用AngularJS,后端使用ASP.NE ...
- 对一个前端AngularJS,后端OData,ASP.NET Web API案例的理解
依然chsakell,他写了一篇前端AngularJS,后端OData,ASP.NET Web API的Demo,关于OData在ASP.NET Web API中的正删改查没有什么特别之处,但在前端调 ...
- 对一个前端使用AngularJS后端使用ASP.NET Web API项目的理解(4)
chsakell分享了一个前端使用AngularJS,后端使用ASP.NET Web API的项目. 源码: https://github.com/chsakell/spa-webapi-angula ...
- 对一个前端使用AngularJS后端使用ASP.NET Web API项目的理解(3)
chsakell分享了一个前端使用AngularJS,后端使用ASP.NET Web API的项目. 源码: https://github.com/chsakell/spa-webapi-angula ...
- 对一个前端使用AngularJS后端使用ASP.NET Web API项目的理解(2)
chsakell分享了一个前端使用AngularJS,后端使用ASP.NET Web API的项目. 源码: https://github.com/chsakell/spa-webapi-angula ...
- 对一个前端使用AngularJS后端使用ASP.NET Web API项目的理解(1)
chsakell分享了一个前端使用AngularJS,后端使用ASP.NET Web API的项目. 源码: https://github.com/chsakell/spa-webapi-angula ...
- 前端使用AngularJS的$resource,后端ASP.NET Web API,实现增删改查
AngularJS中的$resource服务相比$http服务更适合与RESTful服务进行交互.本篇后端使用ASP.NET Web API, 前端使用$resource,实现增删改查. 本系列包括: ...
- ASP.NET Web API 2 external logins with Facebook and Google in AngularJS app
转载:http://bitoftech.net/2014/08/11/asp-net-web-api-2-external-logins-social-logins-facebook-google-a ...
随机推荐
- IComparer<T> 接口Linq比较接口
IComparer<T>比较两个对象并返回一个值,指示一个对象是小于.等于还是大于另一个对象. 在Linq当中,很多扩展方法接受一个实现IComparer<T>接口的实例的对象 ...
- STLtoSVG,and SVG to Bmp
先用这两个工具: Slic3R或者Skeinforge:这个两个工具的作用就是把STL文件切片为叠加的矢量图(SVG格式) 因为SVG是分层的,一层一层的把每层都转换成一张Bmp文件 听说ImageM ...
- NSIS如何对一整个目录文件夹(包括子文件夹和其中的文件)压缩
原来不加/r参数,NSIS编译器就会不认识文件夹啊. File /r [dir] Reference: http://stackoverflow.com/questions/7973242/nsis- ...
- Android Toast简介
Toast是Android中一种提供给用户简短信息的视图,该视图已浮于应用程序之上的形式呈现给用户.因为它并不获得焦点,即使用户正在输入什么也不会受到影响.它的目标是尽可能以不显眼的方式,使用户看到你 ...
- (greedy)Best Time to Buy and Sell Stock II
Say you have an array for which the ith element is the price of a given stock on day i. Design an al ...
- HDU 4274 Spy's Work (树 DFS)
给定N个点,每个点都有一个唯一的前驱结点(点1为大boss),每个点的实际权值是子节点的求和值.现在给出某些点的权值的估算(> , = , < ),问这些估算是否会有冲突,现在保证每个点的 ...
- python3-day3(函数-参数)
1.参数 函数的核心是参数传值,其次是返回值,熟练这两这个技术即可灵活使用函数. 1>普通参数 def show(name): print(name) show('tom') 2>默认参 ...
- Android菜鸟的成长笔记(28)——Google官方对Andoird 2.x提供的ActionBar支持
在Google官方Android设计指南中(链接:http://www.apkbus.com/design/get-started/ui-overview.html)有一个新特性就是自我标识,也就是宣 ...
- JMeter基础概念
JMeter 介绍:一个非常优秀的开源的性能测试工具. 优点:你用着用着就会发现它的重多优点,当然不足点也会呈现出来. 从性能工具的原理划分: Jmeter工具和其他性能工具在原理上完全一致,工具包含 ...
- 解决Fetching android sdk component information加载过久问题
安装完成后,如果直接启动,Android Studio会去获取 android sdk 组件信息,这个过程相当慢,还经常加载失败,导致Android Studio启动不起开.解决办法就是不去获取and ...