.net core下验证码及二维码登录的实现_玄冰_殇的博客-程序员秘密

技术标签: c#  .net core  asp.net  验证码  后端  

在上一篇[.net core下访问控制层的实现]主要介绍了通过中间件实现逻辑层面的权限控制,本篇主要介绍下在 .net core下如何生成验证码和二维码。

生成验证码

验证码实现的逻辑比较简单,生成一个随机数的图片,然后将随机数保存至cookie中,用于客户端校验。

首先是写个生成随机数的方法,下面提供个简单的生成算法,不是特别严谨,但作为后台管理应用基本够用了。

private static string RndNum(int VcodeNum)
{
    //验证码可以显示的字符集合  
    string Vchar = "0,1,2,3,4,5,6,7,8,9,a,b,c,d,e,f,g,h,i,j,k,l,m,n,p" +
        ",q,r,s,t,u,v,w,x,y,z,A,B,C,D,E,F,G,H,I,J,K,L,M,N,P,P,Q" +
        ",R,S,T,U,V,W,X,Y,Z";
    string[] VcArray = Vchar.Split(new Char[] { ',' });//拆分成数组   
    string code = "";//产生的随机数  
    int temp = -1;//记录上次随机数值,尽量避避免生产几个一样的随机数  

    Random rand = new Random();
    //采用一个简单的算法以保证生成随机数的不同  
    for (int i = 1; i < VcodeNum + 1; i++)
    {
        if (temp != -1)
        {
            rand = new Random(i * temp * unchecked((int)DateTime.Now.Ticks));//初始化随机类  
        }
        int t = rand.Next(61);//获取随机数  
        if (temp != -1 && temp == t)
        {
            return RndNum(VcodeNum);//如果获取的随机数重复,则递归调用  
        }
        temp = t;//把本次产生的随机数记录起来  
        code += VcArray[t];//随机数的位数加一  
    }
    return code;
}

然后根据随机数生成图片流:

public static MemoryStream Create(out string code, int numbers = 4)
{
    code = RndNum(numbers);
    //Bitmap img = null;
    //Graphics g = null;
    MemoryStream ms = null;
    Random random = new Random();
    //验证码颜色集合  
    Color[] c = { Color.Black, Color.Red, Color.DarkBlue, Color.Green, Color.Orange, Color.Brown, Color.DarkCyan, Color.Purple };

    //验证码字体集合
    string[] fonts = { "Verdana", "Microsoft Sans Serif", "Comic Sans MS", "Arial", "宋体" };


    using (var img = new Bitmap((int)code.Length * 18, 32))
    {
        using (var g = Graphics.FromImage(img))
        {
            g.Clear(Color.White);//背景设为白色

            //在随机位置画背景点  
            for (int i = 0; i < 100; i++)
            {
                int x = random.Next(img.Width);
                int y = random.Next(img.Height);
                g.DrawRectangle(new Pen(Color.LightGray, 0), x, y, 1, 1);
            }
            //验证码绘制在g中  
            for (int i = 0; i < code.Length; i++)
            {
                int cindex = random.Next(7);//随机颜色索引值  
                int findex = random.Next(5);//随机字体索引值  
                Font f = new Font(fonts[findex], 15, FontStyle.Bold);//字体  
                Brush b = new SolidBrush(c[cindex]);//颜色  
                int ii = 4;
                if ((i + 1) % 2 == 0)//控制验证码不在同一高度  
                {
                    ii = 2;
                }
                g.DrawString(code.Substring(i, 1), f, b, 3 + (i * 12), ii);//绘制一个验证字符  
            }
            ms = new MemoryStream();//生成内存流对象  
            img.Save(ms, ImageFormat.Jpeg);//将此图像以Png图像文件的格式保存到流中  
        }
    }

    return ms;
}

最后将流输出,同时将随机数保存至cookie中:

/// <summary>
/// 获取图形验证码
/// </summary>
/// <returns></returns>
[HttpGet("VerifyCode")]
public async Task GetVerifyCode()
{
    Response.ContentType = "image/jpeg";
    using (var stream = VerifyCodeHelper.Create(out string code))
    {
        var buffer = stream.ToArray();

        // 将验证码的token放入cookie
        Response.Cookies.Append(VERFIY_CODE_TOKEN_COOKIE_NAME, await SecurityServices.GetVerifyCodeToken(code));

        await Response.Body.WriteAsync(buffer, 0, buffer.Length);
    }
}

这样就基本实现了验证码的生成,可以看下效果,输入对应的/VerifyCode就能出现对应的验证码:

1.png

生成二维码

在 .net core下生成二维码需要引入QRCoder.dll第三方组件,生成二维码代码就比较简单了:

/// <summary>
/// 
/// </summary>
/// <param name="url">存储内容</param>
/// <param name="pixel">像素大小</param>
/// <returns></returns>
public static Bitmap GetQRCode(string url, int pixel)
{

    QRCodeGenerator generator = new QRCodeGenerator();
    QRCodeData codeData = generator.CreateQrCode(url, QRCodeGenerator.ECCLevel.M, true);
    QRCode qrcode = new QRCode(codeData);

    Bitmap qrImage = qrcode.GetGraphic(pixel);

    return qrImage;
}

这样就可以将对应的登录地址放至二维码,返回至客户端了:

/// <summary>
/// 获取登录二维码
/// </summary>
/// <returns></returns>
[HttpGet("qrcode")]
public async Task GetQRCode()
{
    Response.ContentType = "image/jpeg";

    // 生成一个token并放入redis
    string qrToken = await SecurityServices.GetQRToken();

    var bitmap = QRCodeHelper.GetQRCode($"{domain}/Security/Login?token={qrToken}", 4);

    // 将二维码回调标识输出给cookie
    // 创建cookie
    Response.Cookies.Append(QRTOKEN_COOKIE_NAME, qrToken);

    bitmap.Save(Response.Body, ImageFormat.Jpeg);
}

这样输入地址/qrcode就能返回对应的二维码啦。

二维码登录实现

前面已经生成二维码给客户端了,如何实现登录呢,这里服务端还得提供一个接口给到客户端,用于二维码登录结果回调。

客户端轮询该接口,判断对应的token是否存在对应的登录记录,若存在则告诉客户端已经登录,客户端即可调转至首页了,若不存在,则等待。

当然你可以设置一个二维码失效时间,当二维码失效客户端自动跳转至账号密码登录页。

我们创建一个二维码登录结果回调服务:

/// <summary>
/// 二维码登录结果回调
/// </summary>
/// <returns></returns>
[HttpGet("qrresult")]
public async Task<LoginResult> GetQRResult()
{
string qrToken = Request.Cookies[QRTOKEN_COOKIE_NAME];
if (string.IsNullOrWhiteSpace(qrToken))
{
    return new LoginResult { Code = (int)LoginResultCode.InvalidAccess };
}

return await SecurityServices.QRResult(qrToken);
}

服务到redis中去找对应的登录记录,如果有则返回登录成功:

public static async Task<LoginResult> QRResult(string token)
{
    long temp = await redis.IncAsync(token);
    if (temp <= 1)
    {
        return new LoginResult { Code = (int)LoginResultCode.QRCodeExpired };
    }

    string sessionId = await redis.Get<string>($"{token}_1");
    if (string.IsNullOrWhiteSpace(sessionId))
    {
        return new LoginResult { Code = (int)LoginResultCode.QRCodeStandby };
    }

    return new LoginResult { Code = (int)LoginResultCode.Succeed, SessionId = sessionId };
}

这样就基本实现了二维码的登录啦。

总结

本篇主要讲验证码和二维码的实现和思路说了下,在小项目中基本够用,有兴趣的小伙伴可以尝试一下。

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/jayshehe/article/details/79808941

智能推荐

Leetcode-颠倒二进制位_二进制倒序排列 leetcode_chestnutllin的博客-程序员秘密

40.颠倒二进制位题目内容:代码及思路:看到题目的时候就想到,直接进行右移运算不就好了,但是参考了大家很多做法,不是很理解,希望请教大家,具体分析一下为什么是这样。class Solution {public: uint32_t reverseBits(uint32_t n) { uint32_t m(0); //m...

ON-LSTM:用有序神经元表达层次结构_sooner高的博客-程序员秘密

本文转载自 苏剑林[1]在科学空间上关于ON-LSTM非常好的解析, 中间加入一些细节的说明, 以期读者在理解原理之余, 对其实现也有概念. 下面开始…前言今天介绍一个有意思的LSTM变种:ON-LSTM,其中“ON”的全称是“Ordered Neurons”,即有序神经元,换句话说这种LSTM内部的神经元是经过特定排序的,从而能够表达更丰富的信息。ON-LSTM来自文章《Ordered ...

Oracle存储过程详解(一)_wangxy799的博客-程序员秘密

存储过程创建语法: create [or replace] procedure 存储过程名(param1 in type,param2 out type)as变量1 类型(值范围);变量2 类型(值范围);Begin Select count(*) into 变量1 from 表A where列名=param1; If (判断条件) then Select

iOS开发--iOS应用架构谈 view层的组织和调用方案_clean架构苹果开发_Wilson-Yuan的博客-程序员秘密

前言《iOS应用架构谈 开篇》出来之后,很多人来催我赶紧出第二篇。这一篇文章出得相当艰难,因为公司里的破事儿特别多,我自己又有点私事儿,以至于能用来写博客的时间不够充分。现在好啦,第二篇出来了。当我们开始设计View层的架构时,往往是这个App还没有开始开发,或者这个App已经发过几个版本了,然后此时需要做非常彻底的重构。一般也就是这两种时机会去做View层架构,基于这

Linux CFS调度器之唤醒抢占--Linux进程的管理与调度(三十)_尚先生的博客的博客-程序员秘密

原文链接:https://blog.csdn.net/gatieme/article/details/52068061日期 内核版本 架构 作者 GitHub CSDN 2016-0729 Linux-4.6 X86 &amp; arm gatieme LinuxDeviceDrivers Linux进程管理与调度 CFS负责处理普通非实时进程, 这类进程是我们linux中最普遍的进程1 前景回顾1.1 CFS调度算法CFS

利用NamingContainer属性获取GridView行号_极客行天下的博客-程序员秘密

在最近的一个项目中,用到在GridView模板列中添加有DropDownList控件,并开启其AutoPostback属性。当发生SelectedIndexChanged事件时,想同时获取其所在的行号,从而获取相应的行信息。由于DropDoweList与button不同,无法指定其CommandName,所以,也就没办法通过常规的方法捕获所在行的索引。颇费了些周折,后来找到用NamingCon

随便推点

fzu-神龙的难题 舞蹈链之可重复覆盖_舞蹈链可重复_圣诞老人家的博客-程序员秘密

这是个剑与魔法的世界.英雄和魔物同在,动荡和安定并存.但总的来说,库尔特王国是个安宁的国家,人民安居乐业,魔物也比较少.但是.总有一些魔物不时会进入城市附近,干扰人民的生活.就要有一些人出来守护居民们不被魔物侵害.魔法使艾米莉就是这样的一个人.她骑着她的坐骑,神龙米格拉一起消灭干扰人类生存的魔物,维护王国的安定.艾米莉希望能够在损伤最小的前提下完成任务.每次战斗前,她都用时间停止魔法停住时间,然后

Centos Stream 8虚拟机搭建单节点OpenStack_吹尽黄沙始到经的博客-程序员秘密

记录在Centos Stream 8上搭建all-in-one模式OpenStack的过程。

NBUT - 1586 - 买票回家啦 (最长公共子序列)_wang_hoho的博客-程序员秘密

[1586] 买票回家啦时间限制: 1000 ms 内存限制: 65535 K问题描述集训要结束了,同学们就准备回家了。好舍不得回家阿。(那就再待一个月嘛,就这么愉快地决定了。)超哥要回家了,可是他没有挤进12306官网, 可怜的他就随便找了个能代购车票的网站。结果,当他付钱时傻眼了,这个网站竟然要验证码。验证码嘛就照着样子输入就好了呀,哦不,这个网站管理员是蛇精病阿,

Django学习笔记(三十九):(BUG)ModuleNotFoundError: No module named 'django'_小火skr车的博客-程序员秘密

当执行python manage.py runerver报错如下,说没有找到module named ‘django’Error Reproduce:mamerunminmatoMacBook-Pro:learning_log RunMin$ source ll_env/bin/activate(ll_env) mamerunminmatoMacBook-Pro:learning_log ...

【模拟IC学习笔记】 Widlar电流源_虫谷ALL的博客-程序员秘密

Widlar电流源原理、反馈极性、环路增益

触动精灵 alilib_aita6163的博客-程序员秘密

--gethtmlfunction gethtml (url)local sz = require("sz")local http = require("szocket.http")local res, code = http.request(url);return resend-----------------------------------...

推荐文章

热门文章

相关标签