圖片驗證碼

借鑑博客:

1..net core ZKWeb.System.Drawing 生成圖片驗證碼

2.Asp.NET Core+ABP框架+IdentityServer4+MySQL+Ext JS之驗證碼

功能主要使用類VerifyCodeHelper借鑑第一個博客,其餘爲自己設計,如果有更好的方案歡迎留言交流,謝謝。

1.在應用層添加包ZKWeb.System.Drawing。

通常使用System.Drawing.Common。這是一個執行圖形相關任務的流行的API,在這裏我們使用ZKWeb.System.Drawing代替他

2.在創建二維碼的使用類

 

using System;
using System.DrawingCore;
using System.DrawingCore.Drawing2D;
using System.DrawingCore.Imaging;
using System.IO;
 
namespace XXXX.Common.Core.Helper
{
    public sealed class VerifyCodeHelper
    {
        #region 單例模式  
        //創建私有化靜態obj鎖    
        private static readonly object _ObjLock = new object();
        //創建私有靜態字段,接收類的實例化對象    
        private static VerifyCodeHelper _VerifyCodeHelper = null;
        //構造函數私有化    
        private VerifyCodeHelper() { }
        //創建單利對象資源並返回    
        public static VerifyCodeHelper GetSingleObj()
        {
            if (_VerifyCodeHelper == null)
            {
                lock (_ObjLock)
                {
                    if (_VerifyCodeHelper == null)
                        _VerifyCodeHelper = new VerifyCodeHelper();
                }
            }
            return _VerifyCodeHelper;
        }
        #endregion
 
        #region 生產驗證碼  
        public enum VerifyCodeType { NumberVerifyCode, AbcVerifyCode, MixVerifyCode };
 
        /// <summary>  
        /// 1.數字驗證碼  
        /// </summary>  
        /// <param name="length"></param>  
        /// <returns></returns>  
        private string CreateNumberVerifyCode(int length)
        {
            int[] randMembers = new int[length];
            int[] validateNums = new int[length];
            string validateNumberStr = "";
            //生成起始序列值    
            int seekSeek = unchecked((int)DateTime.Now.Ticks);
            Random seekRand = new Random(seekSeek);
            int beginSeek = seekRand.Next(0, Int32.MaxValue - length * 10000);
            int[] seeks = new int[length];
            for (int i = 0; i < length; i++)
            {
                beginSeek += 10000;
                seeks[i] = beginSeek;
            }
            //生成隨機數字    
            for (int i = 0; i < length; i++)
            {
                Random rand = new Random(seeks[i]);
                int pownum = 1 * (int)Math.Pow(10, length);
                randMembers[i] = rand.Next(pownum, Int32.MaxValue);
            }
            //抽取隨機數字    
            for (int i = 0; i < length; i++)
            {
                string numStr = randMembers[i].ToString();
                int numLength = numStr.Length;
                Random rand = new Random();
                int numPosition = rand.Next(0, numLength - 1);
                validateNums[i] = Int32.Parse(numStr.Substring(numPosition, 1));
            }
            //生成驗證碼    
            for (int i = 0; i < length; i++)
            {
                validateNumberStr += validateNums[i].ToString();
            }
            return validateNumberStr;
        }
 
        /// <summary>  
        /// 2.字母驗證碼  
        /// </summary>  
        /// <param name="length">字符長度</param>  
        /// <returns>驗證碼字符</returns>  
        private string CreateAbcVerifyCode(int length)
        {
            char[] verification = new char[length];
            char[] dictionary = { 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', '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', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z'
            };
            Random random = new Random();
            for (int i = 0; i < length; i++)
            {
                verification[i] = dictionary[random.Next(dictionary.Length - 1)];
            }
            return new string(verification);
        }
 
        /// <summary>  
        /// 3.混合驗證碼  
        /// </summary>  
        /// <param name="length">字符長度</param>  
        /// <returns>驗證碼字符</returns>  
        private string CreateMixVerifyCode(int length)
        {
            char[] verification = new char[length];
            char[] dictionary = { 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z',
                '0', '1', '2', '3', '4', '5', '6', '7', '8', '9',
                'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z'
            };
            Random random = new Random();
            for (int i = 0; i < length; i++)
            {
                verification[i] = dictionary[random.Next(dictionary.Length - 1)];
            }
            return new string(verification);
        }
 
        /// <summary>  
        /// 產生驗證碼(隨機產生4-6位)  
        /// </summary>  
        /// <param name="type">驗證碼類型:數字,字符,符合</param>  
        /// <returns></returns>  
        public string CreateVerifyCode(VerifyCodeType type)
        {
            string verifyCode = string.Empty;
            Random random = new Random();
            int length = 4;// random.Next(4, 6);
            switch (type)
            {
                case VerifyCodeType.NumberVerifyCode:
                    verifyCode = GetSingleObj().CreateNumberVerifyCode(length);
                    break;
                case VerifyCodeType.AbcVerifyCode:
                    verifyCode = GetSingleObj().CreateAbcVerifyCode(length);
                    break;
                case VerifyCodeType.MixVerifyCode:
                    verifyCode = GetSingleObj().CreateMixVerifyCode(length);
                    break;
            }
            return verifyCode;
        }
        #endregion
 
        #region 驗證碼圖片  
        /// <summary>  
        /// 驗證碼圖片 => Bitmap  
        /// </summary>  
        /// <param name="verifyCode">驗證碼</param>  
        /// <param name="width">寬</param>  
        /// <param name="height">高</param>  
        /// <returns>Bitmap</returns>  
        public Bitmap CreateBitmapByImgVerifyCode(string verifyCode, int width, int height)
        {
            Font font = new Font("Arial", 14, (FontStyle.Bold | FontStyle.Italic));
            Brush brush;
            Bitmap bitmap = new Bitmap(width, height);
            Graphics g = Graphics.FromImage(bitmap);
            SizeF totalSizeF = g.MeasureString(verifyCode, font);
            SizeF curCharSizeF;
            PointF startPointF = new PointF(0, (height - totalSizeF.Height) / 2);
            Random random = new Random(); //隨機數產生器  
            g.Clear(Color.White); //清空圖片背景色    
            for (int i = 0; i < verifyCode.Length; i++)
            {
                brush = new LinearGradientBrush(new Point(0, 0), new Point(1, 1), Color.FromArgb(random.Next(255), random.Next(255), random.Next(255)), Color.FromArgb(random.Next(255), random.Next(255), random.Next(255)));
                g.DrawString(verifyCode[i].ToString(), font, brush, startPointF);
                curCharSizeF = g.MeasureString(verifyCode[i].ToString(), font);
                startPointF.X += curCharSizeF.Width;
            }
 
            //畫圖片的干擾線    
            for (int i = 0; i < 10; i++)
            {
                int x1 = random.Next(bitmap.Width);
                int x2 = random.Next(bitmap.Width);
                int y1 = random.Next(bitmap.Height);
                int y2 = random.Next(bitmap.Height);
                g.DrawLine(new Pen(Color.Silver), x1, y1, x2, y2);
            }
 
            //畫圖片的前景干擾點    
            for (int i = 0; i < 100; i++)
            {
                int x = random.Next(bitmap.Width);
                int y = random.Next(bitmap.Height);
                bitmap.SetPixel(x, y, Color.FromArgb(random.Next()));
            }
 
            g.DrawRectangle(new Pen(Color.Silver), 0, 0, bitmap.Width - 1, bitmap.Height - 1); //畫圖片的邊框線    
            g.Dispose();
            return bitmap;
        }
 
        /// <summary>  
        /// 驗證碼圖片 => byte[]  
        /// </summary>  
        /// <param name="verifyCode">驗證碼</param>  
        /// <param name="width">寬</param>  
        /// <param name="height">高</param>  
        /// <returns>byte[]</returns>  
        public byte[] CreateByteByImgVerifyCode(string verifyCode, int width, int height)
        {
            Font font = new Font("Arial", 14, (FontStyle.Bold | FontStyle.Italic));
            Brush brush;
            Bitmap bitmap = new Bitmap(width, height);
            Graphics g = Graphics.FromImage(bitmap);
            SizeF totalSizeF = g.MeasureString(verifyCode, font);
            SizeF curCharSizeF;
            PointF startPointF = new PointF(0, (height - totalSizeF.Height) / 2);
            Random random = new Random(); //隨機數產生器  
            g.Clear(Color.White); //清空圖片背景色    
            for (int i = 0; i < verifyCode.Length; i++)
            {
                brush = new LinearGradientBrush(new Point(0, 0), new Point(1, 1), Color.FromArgb(random.Next(255), random.Next(255), random.Next(255)), Color.FromArgb(random.Next(255), random.Next(255), random.Next(255)));
                g.DrawString(verifyCode[i].ToString(), font, brush, startPointF);
                curCharSizeF = g.MeasureString(verifyCode[i].ToString(), font);
                startPointF.X += curCharSizeF.Width;
            }
 
            //畫圖片的干擾線    
            for (int i = 0; i < 10; i++)
            {
                int x1 = random.Next(bitmap.Width);
                int x2 = random.Next(bitmap.Width);
                int y1 = random.Next(bitmap.Height);
                int y2 = random.Next(bitmap.Height);
                g.DrawLine(new Pen(Color.Silver), x1, y1, x2, y2);
            }
 
            //畫圖片的前景干擾點    
            for (int i = 0; i < 100; i++)
            {
                int x = random.Next(bitmap.Width);
                int y = random.Next(bitmap.Height);
                bitmap.SetPixel(x, y, Color.FromArgb(random.Next()));
            }
 
            g.DrawRectangle(new Pen(Color.Silver), 0, 0, bitmap.Width - 1, bitmap.Height - 1); //畫圖片的邊框線    
            g.Dispose();
 
            //保存圖片數據    
            MemoryStream stream = new MemoryStream();
            bitmap.Save(stream, ImageFormat.Jpeg);
            //輸出圖片流    
            return stream.ToArray();
 
        }
        #endregion
    }
}

3.創建存儲二維碼的類

[Table("VerifyCode")]
    public class VerifyCode : Entity<long>, IHasCreationTime
    {
        public const int MaxCodeLength = 6;

        [Required]
        public string Md5Key { get; set; }


        [Required]
        [MaxLength(MaxCodeLength)]
        public string Code { get; set; }

        /// <summary>
        /// 是否使用
        /// </summary>
        public bool IsUsed { get; set; }

        public DateTime CreationTime { get; set; }
    }

4.創建獲取驗證碼的接口

使用VerifyCodeHelper工具類獲取隨機碼,並獲取隨機碼圖片,轉爲base64圖片格式,記錄到數據庫中。

 public async Task<VerifyCodeListDto> GetVerify()
        {
            var code = VerifyCodeHelper.GetSingleObj().CreateVerifyCode(VerifyCodeType.MixVerifyCode);                //取隨機碼
            var imgCode = VerifyCodeHelper.GetSingleObj().CreateByteByImgVerifyCode(code, 100, 60);
            var image = $"data:image/jpeg;base64,{Convert.ToBase64String(imgCode)}";
            var verifyCode = new VerifyCode()
            {
                Md5Key = image,
                Code = code,
                IsUsed = false
            };
            var imgEntity =  await _verifyCodeRepository.InsertAsync(verifyCode);
            await CurrentUnitOfWork.SaveChangesAsync();
            return imgEntity.MapTo<VerifyCodeListDto>();
        }

僅發送驗證碼的Id和圖片數據。

public class VerifyCodeListDto:EntityDto<long>
    {
        [Required]
        public string Md5Key { get; set; }

    }

5.接收到請求時檢查驗證碼,並設置驗證碼爲已使用。

public async Task Check(CreateOrUpdateMemberInput input)
        {
            //檢查code是否正確
            var Code = await _verifyCodeRepository.GetAsync(input.Verify.Id);
            TimeSpan dt = DateTime.Now - Code.CreationTime;
            if(dt.Minutes>10)
                throw new UserFriendlyException("驗證碼十分鐘內有效,該碼已過期");
            if (Code.IsUsed ==true)
                throw new UserFriendlyException("驗證碼不能重複使用,此碼無效");
            if (Code.Code != input.Verify.Code)
                throw new UserFriendlyException("驗證碼錯誤");
                Code.IsUsed =true;
              await _verifyCodeRepository.UpdateAsync(Code)  
        }

 

 

 

 

發佈了23 篇原創文章 · 獲贊 7 · 訪問量 7909
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章