前几天,按照AngularJS2的英雄指南教程走了一遍,教程网址是http://origin.angular.live/docs/ts/latest/tutorial/

在步骤完成后,又更进一步,在英雄增删改的时候,直接调用.net core的WebApi来实现后台数据的操作,替换教程中的模拟WebApi方式。在替换.net core WebApi时,还是遇到了一些坑的,这里记录一下。

先来看一下WebApi和AngularJS的源代码:

WebApi

     [Route("api/[controller]")]
public class ValuesController : Controller
{
private List<Hero> heroes; public ValuesController()
{
heroes = GetHeroes() } [HttpGet]
public IEnumerable<Hero> Get()
{
return heroes;
} [HttpGet("{id}")]
public Hero Get(int id)
{
return heroes.Single(h => h.Id == id);
} [HttpGet]
[Route("GetList")]
public IEnumerable<Hero> GetList(string key)
{
return heroes.Where(h => h.Name.Contains(key));
} [HttpPost]
public void Post([FromBody]Hero info)
{
Hero hero = new Hero(); hero.Id = heroes.Max(h => h.Id) + ;
hero.Name = info.Name; AddHeroes(hero);
} [HttpPut("{id}")]
public void Put(int id, [FromBody]Hero hero)
{
Hero x = heroes.Single(h => h.Id == id); x.Name = hero.Name; UpdateHeroes(x);
} [HttpDelete("{id}")]
public void Delete(int id)
{
Hero hero = heroes.Single(h => h.Id == id);
RemoveHeroes (hero);
}
}

AngularJS

     getHeroes(): Promise<Hero[]> {
return this.http.get(this.heroUrl).toPromise().then(response => response.json() as Hero[]).catch(this.handleError);
} getHero(id: number): Promise<Hero> {
return this.getHeroes().then(heroes => heroes.find(hero => hero.id === id));
} getHeroesSlowly(): Promise<Hero[]> {
return new Promise<Hero[]>(resolve =>
setTimeout(resolve, 2000)) // delay 2 seconds
.then(() => this.getHeroes());
} updateHero(hero: Hero): Promise<Hero> {
const url = `${this.heroUrl}/${hero.id}`;
return this.http
.put(url, JSON.stringify(hero), { headers: this.headers })
.toPromise()
.then(() => hero)
.catch(this.handleError);
} addHero(heroName: string): Promise<Hero> {
return this.http
.post(this.heroUrl, JSON.stringify({ name: name }), { headers: this.headers })
.toPromise()
.then(response => response.json())
.catch(this.handleError);
} deleteHero(heroId: number): Promise<void> {
const url = `${this.heroUrl}/${heroId}`;
return this.http
.delete(url, { headers: this.headers })
.toPromise()
.then(() => null)
.catch(this.handleError);
} search(term: string): Observable<Hero[]> {
return this.http.get(`${this.heroUrl}/GetList?key=${term}`).map((r: Response) => r.json() as Hero[]);
} private handleError(error: any): Promise<any> {
console.error('An error occurred', error); // for demo purposes only
return Promise.reject(error.message | error);
}

一、跨域访问(Cors)问题

在建好WebApi站点,启动运行,一切正常。我们仅显示前两个Action,结果如下:

然而,在AngularJS客户端,出现了问题,界面没有任何英雄出现。

再查Chrome的控制台,发现了问题所在,WebApi的Access-Control-Allow-Origin没有开,也就是跨域访问不通过。

跨域访问,就是JS访问其他网址的信息。例如这里的AngularJS的

     getHeroes(): Promise<Hero[]> {
return this.http.get(this.heroUrl).toPromise().then(response => response.json() as Hero[]).catch(this.handleError);
}

方法中,调用heroUrl(WebApi网址)。AngularJS的网址是http://localhost:3000/,而WebApi的网址是http://localhost:5000/api/values。只要协议、域名、端口有任何一个不同,都被当作是不同的域。默认情况下是不支持直接跨域访问的。为了在WebiApi中增加跨域访问,我们要在WebApi的Startup.cs中打开Access-Control-Allow-Origin:

         public void Configure(IApplicationBuilder app, IHostingEnvironment env, ILoggerFactory loggerFactory)
{
app.UseCors(builder =>
{
builder.AllowAnyOrigin();
}); app.UseMvc();
}

现在程序访问正常。

我们查看AngularJS网站的network情况,在values的headers—Response Headers中有Access-Control-Allow-Origin: *的字样,这样就解决了前面出现的"No Access-Control-Allow-Origin header is present…"错误。

再继续测试,又出现问题了,这次是在更新时

错误信息如下:

我们都已经打开Access-Control-Allow-Origin,怎么还报错:"No Access-Control-Allow-Origin header is present…"?我们回过来看AngularJS,调用的是Update:

     updateHero(hero: Hero): Promise<Hero> {
const url = `${this.heroUrl}/${hero.id}`;
return this.http
.put(url, JSON.stringify(hero), { headers: this.headers })
.toPromise()
.then(() => hero)
.catch(this.handleError);
}

对应WebApi,这次调用的是Put方法

         [HttpPut("{id}")]
public void Put(int id, [FromBody]Hero hero)
{
Hero x = heroes.Single(h => h.Id == id); x.Name = hero.Name; UpdateHeroes(hero.Name);
}

是不是由于Method不对啊,以前都是Post、Get,这次是Put。趁早,一不做,二不休,在WebApi的Startup.cs中,写成如下:

         public void Configure(IApplicationBuilder app, IHostingEnvironment env, ILoggerFactory loggerFactory)
{
app.UseCors(builder =>
{
builder.AllowAnyOrigin();
builder.AllowAnyHeader();
builder.AllowAnyMethod();
}); app.UseMvc();
}

允许所有Origin、Header和Method。这次彻底搞定,运行正常。

二、Update方法,必须使用类

其实上面Update时,还是遇到了部分问题的。就是按照最开始的程序,我们在WebApi的Put方法中,传入的是英雄名称name,而不是Hero对象。

         [HttpPut("{id}")]
public void Put(int id, [FromBody]string name)
{
Hero x = heroes.Single(h => h.Id == id); x.Name = name; UpdateHero(x);
}

同时AngularJS中如下:

     updateHero(hero: Hero): Promise<Hero> {
const url = `${this.heroUrl}/${hero.id}`;
return this.http
.put(url, JSON.stringify({name: name}), { headers: this.headers })
.toPromise()
.then(() => hero)
.catch(this.handleError);
}

在程序运行时,我们修改第13号英雄:

查看network,提交也是name: "Bombasto1111"。

然而,在WebApi中,调试情况下:

Id没有问题,但是name是null。[FromBody]要求前台传入的是Json串,然后提交到Action中。这时,Angular传入的是{name: "Bombasto1111"},通过FromBody转换后,是一个object,而不是一个string,因此name的值是null。

如果想用string name方式,AngularJS就要写成:

     updateHero(hero: Hero): Promise<Hero> {
const url = `${this.heroUrl}/${hero.id}`;
return this.http
.put(url, JSON.stringify(hero.name), { headers: this.headers })
.toPromise()
.then(() => hero)
.catch(this.handleError);
}

再看network

WebApi调试状态如下图,name有值,一切搞定。

也可以使用Hero作为参数的方式,如同我们文档最开始提供的代码一样。

     updateHero(hero: Hero): Promise<Hero> {
const url = `${this.heroUrl}/${hero.id}`;
return this.http
.put(url, JSON.stringify(hero), { headers: this.headers })
.toPromise()
.then(() => hero)
.catch(this.handleError);
}
        [HttpPut("{id}")]
public void Put(int id, [FromBody]Hero hero)
{
Hero x = heroes.Single(h => h.Id == id); x.Name = hero.Name; UpdateHeroes(x);
}

三、路由改变

我们在做Hero Search的时候,又要在后台WebApi中编写按照关键字查询的方法

         [HttpGet]
public IEnumerable<Hero> GetList(string key)
{
return heroes.Where(h => h.Name.Contains(key));
}

如果仅仅这样写,整个WebApi都不好用了。在调用Get时,出现了喜闻乐见的500错误。

而调用Get(int id)方法,就是localhost:5000/api/values/11时,一切正常。这是为什么呢?

因为按照微软的方式,如果有多个相同Method(例如HttpGet)的方法,WebApi路由就没法区分了。这里,Get()和GetList(string key)都是[HttpGet],路由都是localhost:5000/api/values。而Get(int id)方法由于写的是[HttpGet("{id}")],网址是localhost:5000/api/values/11,与Get()路由不同。因此,为了解决Get()和GetList(string key),我们在getlist方法上增加指定路由的方式。例如这里就是

         [HttpGet]
[Route("GetList")]
public IEnumerable<Hero> GetList(string key)
{
return heroes.Where(h => h.Name.Contains(key));
}

在AngularJS中,就可以使用localhost:5000/api/values/GetList?key=XXXX的方式调用,程序如下:

     search(term: string): Observable<Hero[]> {
return this.http.get(`${this.heroUrl}/GetList?key=${term}`).map((r: Response) => r.json() as Hero[]);
}

当然了,如果想更进一步,需要再修改整个路由方式,增加Action。这就需要在Controller上增加[Route("api/[controller]/[action]/")]的写法,并且将Action的[Route]删除。

[Route("api/[controller]/[action]/")]

public class ValuesController : Controller

这次是真的解决了。

最新文章

  1. PC网站应用接入微信登录
  2. c/c++中主线程退出,子线程也会退出
  3. AngularJS自定义指令(Directives)在IE8下的一个坑
  4. MinGW安装和使用总结
  5. ecshop标签大全
  6. Android 开发笔记“Eclipse 调试和快捷键”
  7. log4net学习目录
  8. HDU 4508 沼泽湿地系列故事——记住减肥I (2013腾讯编程马拉松预赛第一)
  9. FineReport使用总结
  10. Zernike不变矩
  11. spring 文件模板下载多种实现方式
  12. ubuntu cpu频率控制
  13. RabbitMQ通过Exchange.fanout、不同的队列绑定同一个Exchange实现多播处理
  14. API的HTTP Status Code
  15. http://code52.org/DownmarkerWPF/
  16. R中绘制聚类的离散图
  17. 关于_WIN32_WINNT的含义
  18. Kali-linux物理访问攻击
  19. windows下安装pthreads扩展注意问题
  20. Redis 为什么用跳表而不用平衡树

热门文章

  1. [WPF] 将普通的Library工程,改造成WPF Custom Control 的Library
  2. H264编码技术
  3. WebApi2官网学习记录---Html Form Data
  4. windows程序消息机制(Winform界面更新有关)--转
  5. ControlStyles(枚举)
  6. c#类初始化器
  7. 视图View
  8. 高级子查询【weber出品必属精品】
  9. Apache URL转发
  10. (二)backbone - DEMO - user list