Asp.NetCore之组件写法
本章内容和大家分享的是Asp.NetCore组件写法,在netcore中很多东西都以提供组件的方式来使用,比如MVC架构,Session,Cache,数据库引用等; 这里我也通过调用验证码接口来自定义个组件以此说明如何使用,以及使用时需要注意的场景;
Middleware之hello world
对于netcore来说,通常会在UseStartup<Startup>对应的Startup类中引用组件,这个Startup可以换成自己自定义的其实类,不过需要在UseStartup的时候指向她;这里还是以Startup类为例,通过vs自动生成的文件,在这个类中我们能看到Configure方法体里面包含了:app.UseMvc(),app.UseStaticFiles(),app.xxx()等一些列自带的组件,下面来看下自定义个hello world中组件实例,首先使用静态扩展方法扩展IApplicationBuilder:
public static class MiddlewareExtends
{
/// <summary>
/// 测试用例中间件
/// </summary>
/// <param name="builder"></param>
/// <returns></returns>
public static IApplicationBuilder UseTest(this IApplicationBuilder builder)
{
return builder.UseMiddleware<TestMiddleware>();
} }
使用 builder.UseMiddleware<TestMiddleware>() 来添加自定义组件,组件实现代码如下:
public class TestMiddleware
{
private RequestDelegate _requestDelegate;
public TestMiddleware(RequestDelegate requestDelegate)
{
_requestDelegate = requestDelegate;
} public Task Invoke(HttpContext context)
{ context.Items["TestMiddleware"] = "hello world,我是TestMiddleware。"; return _requestDelegate(context);
}
}
以上是最基础的组件格式;注:
1. 组件必须要有 public Task Invoke(HttpContext context) ,HttpContext是咋们http上下文,Invoke()委托方法,每次程序被访问时就会进入Invoke;
2. 要有 public delegate Task RequestDelegate(HttpContext context); 委托方法,来响应http请求;
到这里咋们hello world就完成了,为了测试方法,我们直接在action中写入如下代码:
public IActionResult About()
{ ViewData["Message"] = HttpContext.Items["TestMiddleware"];
return View();
}
运行结果:
组件异步写法
public class TestMiddleware
{
private RequestDelegate _requestDelegate;
public TestMiddleware(RequestDelegate requestDelegate)
{
_requestDelegate = requestDelegate;
} public async Task Invoke(HttpContext context)
{ context.Items["TestMiddleware"] = "hello world,我是asyncTestMiddleware。"; await _requestDelegate(context);
}
}
仅仅需要async 和 await 组合修饰就行了;
.netcore自定义验证码组件
/// <summary>
/// 文字验证码
/// </summary>
public class WenZiCodeMiddleware
{
private RequestDelegate _requestDelegate;
public WenZiCodeMiddleware(RequestDelegate requestDelegate)
{
_requestDelegate = requestDelegate;
} public async Task Invoke(HttpContext context)
{ var url = "http://localhost:1001/shenniuapi/WenZiValidateCode";
using (HttpClient client = new HttpClient())
{
client.Timeout = TimeSpan.FromSeconds(); var str = "{\"UserName\": \"神牛步行3\",\"UserPwd\": \"4297f44b13955235245b2497399d7a93\",\"Token\": \"008我是测试\"}";
var content = new StringContent(str, Encoding.UTF8, "application/x-www-form-urlencoded");
content.Headers.ContentType = new MediaTypeHeaderValue("application/json");
var result01 = client.PostAsync(url, content).Result;
var stream = await result01.Content.ReadAsStreamAsync();
using (var reader = new StreamReader(stream))
{
var result02 = await reader.ReadToEndAsync();
context.Items["codedata"] = result02;
}
} await _requestDelegate(context);
}
}
我们同样需要再静态扩展方法里面添加如下代码,来加入组件:
/// <summary>
/// 文字验证码中间件
/// </summary>
/// <param name="builder"></param>
/// <returns></returns>
public static IApplicationBuilder UseWenZiValidateCode(this IApplicationBuilder builder)
{
return builder.UseMiddleware<WenZiCodeMiddleware>();
}
在Configure方法中,引用组件: app.UseWenZiValidateCode(); ;Action中,使用组件:
public FileResult GetCode()
{
var data = HttpContext.Items["codedata"].ToString();
var code = JsonConvert.DeserializeObject<MoValidateCodeResponse>(data);
return File(code.CodeStream, "image/jpeg");
}
View试图中代码:
GetCode:<img src="/home/GetCode" data-src="/home/GetCode" />
效果展示:
这里需要考虑场景是,我们上面提及到的Invoke方法是任意请求都会进入,那验证码这种功能做成组件是否不是很合理,因为验证码也只有在登陆界面或验证的界面需要用到而已,如我们上面写的验证码组件,每次都会被程序执行这显然不合理,因此个人认为如果你需要自定义组件,那么需要考量:是否每次请求都需要进入您的组件服务,如果不需要的话,那其实没必要弄一个组件,当然感觉很高大上;因此这里我不得不使用静态扩展方法(当然还有其他方式)来重写获取验证码的方法;
静态扩展方法重写验证码组件
由于上面我们在添加组件时有一个静态类了,那么我们直接在上面补充扩展方法:
/// <summary>
/// 文字验证码
/// </summary>
/// <param name="context"></param>
/// <returns></returns>
public static async Task<MoValidateCodeResponse> WenZiCode(this HttpContext context)
{
var code = default(MoValidateCodeResponse);
try
{
var url = "http://localhost:1001/shenniuapi/WenZiValidateCode";
using (HttpClient client = new HttpClient())
{
client.Timeout = TimeSpan.FromSeconds(); var str = "{\"UserName\": \"神牛步行3\",\"UserPwd\": \"4297f44b13955235245b2497399d7a93\",\"Token\": \"008我是测试\"}";
var content = new StringContent(str, Encoding.UTF8, "application/x-www-form-urlencoded");
content.Headers.ContentType = new MediaTypeHeaderValue("application/json");
var result01 = client.PostAsync(url, content).Result;
var stream = await result01.Content.ReadAsStreamAsync();
using (var reader = new StreamReader(stream))
{
var result02 = await reader.ReadToEndAsync();
code = await JsonConvert.DeserializeObjectAsync<MoValidateCodeResponse>(result02);
}
}
}
catch (Exception ex)
{
}
return code;
}
对应的验证码实体类:
/// <summary>
/// 神牛接口返回基类
/// </summary>
public class MoShenNiuBaseResponse
{
/// <summary>
/// 返回状态: 0:失败 1:成功
/// </summary>
public int Status { get; set; } /// <summary>
/// 错误信息
/// </summary>
public string Msg { get; set; }
} /// <summary>
/// 验证码返回类
/// </summary>
public class MoValidateCodeResponse : MoShenNiuBaseResponse
{ public MoValidateCodeResponse()
{
this.ImgCode = new List<MoImgCode>();
} /// <summary>
/// 验证码类型
/// </summary>
public string Code { get; set; } /// <summary>
/// 验证码图片流
/// </summary>
public byte[] CodeStream { get; set; } /// <summary>
/// 图片验证坐标
/// </summary>
public List<MoImgCode> ImgCode;
} /// <summary>
/// 图片验证码坐标
/// </summary>
public class MoImgCode
{
public string Index { get; set; } public string IndexType { get; set; } public string ImgUrl { get; set; } public Point Point_A { get; set; } public Point Point_B { get; set; } public bool IsChoice { get; set; }
} public class Point
{
public int X { get; set; }
public int Y { get; set; }
}
这个时候同样来到Action中:
public async Task<FileResult> GetCodeAsync()
{
var code = await HttpContext.WenZiCode(); return File(code.CodeStream, "image/jpeg");
}
修改view试图代码,增加点击验证码图片重新获取新的验证码:
<style type="text/css">
img{cursor:pointer}
</style>
<h3>@ViewData["Message"]</h3>
<h3>@ViewData["codedata"]</h3>
GetCode:<img src="/home/GetCode" data-src="/home/GetCode" />
GetCodeAsync:<img src="/home/GetCodeAsync" data-src="/home/GetCodeAsync" /> <script src="~/lib/jquery/dist/jquery.js"></script>
<script>
$(function () {
$("img").on("click", function () {
var img = this;
var nowTime = new Date().getTime();
var src = $(img).attr("data-src") + "?t=" + nowTime;
$(img).attr("src", src);
});
})
</script>
效果图:
以上就是本篇的所有内容,旨在分享怎么写一个组件和什么时候用组件合适,谢谢大家支持和点赞。
Asp.NetCore之组件写法的更多相关文章
- ASP.NET自定义控件组件开发 第三章 为控件添加事件 后篇
原文:ASP.NET自定义控件组件开发 第三章 为控件添加事件 后篇 第三章 为控件添加事件 后篇 前一篇文章只是简单的说了下事件,但是大家应该方法,在ASP.NET自定义控件中只是简单那么定义事件是 ...
- ASP.NET自定义控件组件开发 第一章 第二篇 接着待续
原文:ASP.NET自定义控件组件开发 第一章 第二篇 接着待续 ASP.NET自定义控件组件开发 第一章 第二篇 接着待续 很感谢大家给我的第一篇ASP.NET控件开发的支持!在写这些之前,我也看了 ...
- 『Asp.Net 组件』第一个 Asp.Net 服务器组件:自己的文本框控件
代码: using System.Web; using System.Web.UI; using System.Web.UI.WebControls; namespace DemoWebControl ...
- ASP.NETCore学习记录(一)
ASP.NETCore学习记录(一) asp.net core介绍 Startup.cs ConfigureServices Configure 0. ASP.NETCore 介绍 ASP.N ...
- ASP.NETCORE MVC模块化
ASP.NETCORE MVC模块化编程 前言 记得上一篇博客中跟大家分享的是基于ASP.NETMVC5,实际也就是基于NETFRAMEWORK平台实现的这么一个轻量级插件式框架.那么今天我主要分享的 ...
- 目录---Asp.NETCore轻松学系列【目录】
随笔分类 - Asp.NETCore轻松学系列 Asp.NETCore轻松学系列阅读指引目录 摘要: 耗时两个多月,坚持写这个入门系列文章,就是想给后来者更好更快的上手体验,这个系列可以说是从入门到进 ...
- 【目录】Asp.NETCore轻松学系列
随笔分类 - Asp.NETCore轻松学系列 Asp.NETCore轻松学系列阅读指引目录 摘要: 耗时两个多月,坚持写这个入门系列文章,就是想给后来者更好更快的上手体验,这个系列可以说是从入门到进 ...
- 壹佰文章最全总结| 《关于ASP.NETCore的分享之路》
学习路线图 (关于学习ASP.NET Core需要了解和掌握的知识点图) 一言不合就来图,各位博客园小伙伴大家好,感觉好久没有写文章了,自从春节开始,中间经历种种,慢慢的就开始微信公众号发文了,原因有 ...
- 彻底解决Asp.netCore WebApi 3.1 跨域时的预检查204 options重复请求的问题
Asp.netCore WebApi 3.1 跨域的预检查options问题 1:我们直接使用core跨域的中间件 ,注入跨域服务, services.AddCors(options => { ...
随机推荐
- Hadoop基本开发环境搭建(原创,已实践)
软件包: hadoop-2.7.2.tar.gz hadoop-eclipse-plugin-2.7.2.jar hadoop-common-2.7.1-bin.zip eclipse jdk1.8 ...
- 什么是Git?
其实我在写这篇随笔的时候连Git是什么都不知道,只是听说过,也注册了一个GitHub的账号,但并不会玩. 我也是查看了半天的网页才明白一个大概,但我觉得以后肯定会经常用到它. 简单的来说, Git 是 ...
- H5缓存-Manifest
在app中更新h5页面一直有缓存问题.默认什么都不做的情况下,app有一定的空间缓存页面.一开始更新之后会马上加载,等到app缓存空间上来之后更新就无法下载了.安卓能够清理缓存空间,ios就只能卸载重 ...
- fastcgi的介绍,原理及配置
fastcgi介绍: CGI全称是“公共网关接口”(Common Gateway Interface),HTTP服务器与你的或其它机器上的程序进行“交谈”的一种工具,其程序一般运行在网络服务器上. C ...
- rpm的用法
rpm是由红帽公司开发的软件包管理方式,使用rpm我们可以方便的进行软件的安装.查询.卸载.升级等工作.但是rpm软件包之间的依赖性问题往往会很繁琐,尤其是软件由多个rpm包组成时. yum基於RPM ...
- SSIS 数据流的连接和查找转换
在SSIS的数据流组件中,SSIS引擎使用Merge Join组件和 Lookup组件实现TSQL语句中的inner join 和 outer join 功能,Lookup查找组件的功能更类似TSQL ...
- 云服务器spark集群搭建
---恢复内容开始--- 1:去官网下载spark http://spark.apache.org 2:解压,然后在自己的机器上编译conf中的两个文件 mv slaves.template slav ...
- 双系统win7和ubuntu14.04进入了grub rescue>
可以跳过的废话:最近在学习caffe,需要在linux下安装cuda,sudo apt-get install cuda后,出现了由于根目录/空间不足而失败的情况. 于是想把win7下80G的一个盘格 ...
- YARN学习笔记(一)——YARN的简介
YARN的简介 什么是YARN MRv1的架构和缺陷 经典MapReduce的局限性 解决可伸缩性问题 YARN的架构 一个可运行任何分布式应用程序的集群 YARN中的应用程序提交 YARN的其他特性 ...
- 《深入理解Java虚拟机》学习笔记之字节码执行引擎
Java虚拟机的执行引擎不管是解释执行还是编译执行,根据概念模型都具有统一的外观:输入的是字节码文件,处理过程是字节码解析的等效过程,输出的是执行结果. 运行时栈帧结构 栈帧(Stack Frame) ...