实现验证码效果

前篇:

  验证码功能在进行注册和登录功能模块的时间是不可或缺的一部分,最近了解有关验证码的相关知识:

1.了解StringGenerator类

  StringGenerator是一个自编写的一个字符串生成器,分为创建随机密钥、生成随机数字、生成随机字母与数字、生成随机纯字母随机数和指定字符串长度、线程休眠、字符列表,创建验证码五部分组成。

  

 /// <summary>
    /// 字符串生成器
    /// </summary>
    public class StringGenerator
    {
        /// <summary>
        /// create a random key
        /// </summary>
        static readonly Random Random = new Random(~unchecked((int)DateTime.Now.Ticks));
        static readonly char[] NumberList = { '1', '2', '3', '4', '5', '6', '7', '8', '9' };
        static readonly char[] CharList = { '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' };
        static readonly char[] MixedList = { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'J', 'K', 'L', 'M', 'N', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z' }; //remove I & O

        /// <summary>
        /// 生成随机数字
        /// </summary>
        /// <param name="Length">生成长度</param>
        public static string Number(int Length)
        {
            return Create(Length, false, NumberList);
        }

        /// <summary>
        /// 生成随机字母与数字
        /// </summary>
        /// <param name="Length">生成长度</param>
        public static string Mixed(int Length)
        {
            return Create(Length, false, MixedList);
        }

        /// <summary>
        /// 生成随机纯字母随机数
        /// </summary>
        /// <param name="Length">生成长度</param>
        public static string Char(int Length)
        {
            return Create(Length, false, CharList);
        }

        /// <summary>
        /// 指定字符串长度、线程休眠、字符列表,创建验证码
        /// </summary>
        /// <param name="Length">Length.</param>
        /// <param name="Sleep">If set to <c>true</c> sleep.</param>
        /// <param name="List">List create CAPTCHA based on</param>
        /// <returns>The create.</returns>
        private static string Create(int Length, bool Sleep, char[] List)
        {
            if (Sleep)
                Thread.Sleep(3);
            char[] Pattern = List;
            string result = string.Empty;
            int n = Pattern.Length;
            for (int i = 0; i < Length; i++)
            {
                int rnd = Random.Next(0, n);
                result += Pattern[rnd];
            }
            return result;
        }
    }
StringGenerator类

2.了解CaptchaGenerator类

  CaptchaGenerator为一个图片验证码生成器,定义私有字段、构造器、构造方法。

   

 public class CaptchaGenerator
    {
        #region 私有字段

        public string Text { get; private set; }
        public Bitmap Image { get; private set; }
        private int LetterCount { set; get; }  // 验证码位数
        private int Type { set; get; }
        private int letterWidth = 16;  // 单个字体的宽度范围
        private int letterHeight = 20; // 单个字体的高度范围
        private static Random Random = new Random(~unchecked((int)DateTime.Now.Ticks));
        private Font[] fonts =
        {
            new Font(new FontFamily("Times New Roman"),10 + Random.Next(1),FontStyle.Regular),
            new Font(new FontFamily("Georgia"), 10 + Random.Next(1),FontStyle.Regular),
            new Font(new FontFamily("Arial"), 10 + Random.Next(1),FontStyle.Regular),
            new Font(new FontFamily("Comic Sans MS"), 10 + Random.Next(1),FontStyle.Regular)
        };

        #endregion

        #region 构造器

        /// <summary>
        /// 实例化 <see cref="Common.CaptchaGenerator"/> 类的一个新实例,默认为4个数字长度
        /// </summary>
        public CaptchaGenerator()
        {
            HttpContext.Current.Response.Expires = 0;
            HttpContext.Current.Response.Buffer = true;
            HttpContext.Current.Response.ExpiresAbsolute = DateTime.Now.AddSeconds(-1);
            HttpContext.Current.Response.AddHeader("pragma", "no-cache");
            HttpContext.Current.Response.CacheControl = "no-cache";
            LetterCount = 4;
            Type = 0;
            InitText();
            CreateImage();
        }

        /// <summary>
        /// 指定数字长度实例化 <see cref="T:AiXiu.Common.CaptchaGenerator"/> 类的一个新实例
        /// </summary>
        /// <param name="Length">Length.</param>
        public CaptchaGenerator(int Length)
        {
            HttpContext.Current.Response.Expires = 0;
            HttpContext.Current.Response.Buffer = true;
            HttpContext.Current.Response.ExpiresAbsolute = DateTime.Now.AddSeconds(-1);
            HttpContext.Current.Response.AddHeader("pragma", "no-cache");
            HttpContext.Current.Response.CacheControl = "no-cache";
            LetterCount = Length;
            Text = StringGenerator.Number(LetterCount);
            CreateImage();
        }

        /// <summary>
        /// 实例化 <see cref="T:AiXiu.Common.CaptchaGenerator"/> 类的一个新实例
        /// </summary>
        /// <param name="Length">Length.</param>
        /// <param name="type">Type 0 number , 1 char , 2 mixed.</param>
        public CaptchaGenerator(int Length, int type)
        {
            HttpContext.Current.Response.Expires = 0;
            HttpContext.Current.Response.Buffer = true;
            HttpContext.Current.Response.ExpiresAbsolute = DateTime.Now.AddSeconds(-1);
            HttpContext.Current.Response.AddHeader("pragma", "no-cache");
            HttpContext.Current.Response.CacheControl = "no-cache";
            LetterCount = Length;
            Type = type;
            InitText();
            CreateImage();
        }

        #endregion


        #region 私有方法

        /// <summary>
        /// 初始化文本
        /// </summary>
        private void InitText()
        {
            switch (Type)
            {
                case 0: Text = StringGenerator.Number(LetterCount); break;
                case 1: Text = StringGenerator.Char(LetterCount); break;
                case 2: Text = StringGenerator.Mixed(LetterCount); break;
                default: break;
            }
        }

        /// <summary>
        /// 绘制验证码
        /// </summary>
        private void CreateImage()
        {
            int ImageWidth = this.Text.Length * letterWidth;
            Bitmap Img = new Bitmap(ImageWidth, letterHeight);
            Graphics g = Graphics.FromImage(Img);
            g.Clear(Color.White);
            for (int i = 0; i < 2; i++)
            {
                int x1 = Random.Next(Img.Width - 1);
                int x2 = Random.Next(Img.Width - 1);
                int y1 = Random.Next(Img.Height - 1);
                int y2 = Random.Next(Img.Height - 1);
                g.DrawLine(new Pen(Color.Silver), x1, y1, x2, y2);
            }
            int _x = -12, _y;
            for (int int_index = 0; int_index < this.Text.Length; int_index++)
            {
                _x += Random.Next(12, 16);
                _y = Random.Next(-2, 2);
                string str_char = this.Text.Substring(int_index, 1);
                str_char = Random.Next(1) == 1 ? str_char.ToLower() : str_char.ToUpper();
                Brush newBrush = new SolidBrush(GetRandomColor());
                Point thePos = new Point(_x, _y);
                g.DrawString(str_char, fonts[Random.Next(fonts.Length - 1)], newBrush, thePos);
            }
            for (int i = 0; i < 10; i++)
            {
                int x = Random.Next(Img.Width - 1);
                int y = Random.Next(Img.Height - 1);
                Img.SetPixel(x, y, Color.FromArgb(Random.Next(0, 255), Random.Next(0, 255), Random.Next(0, 255)));
            }
            Img = TwistImage(Img, true, Random.Next(1, 3), Random.Next(4, 6));
            g.DrawRectangle(new Pen(Color.LightGray, 1), 0, 0, ImageWidth - 1, (letterHeight - 1));
            Image = Img;
        }


        /// <summary>
        /// 字体随机颜色
        /// </summary>
        private Color GetRandomColor()
        {
            Random RandomNum_First = new Random((int)DateTime.Now.Ticks);
            Thread.Sleep(RandomNum_First.Next(50));
            Random RandomNum_Sencond = new Random((int)DateTime.Now.Ticks);
            int int_Red = RandomNum_First.Next(180);
            int int_Green = RandomNum_Sencond.Next(180);
            int int_Blue = (int_Red + int_Green > 300) ? 0 : 400 - int_Red - int_Green;
            int_Blue = (int_Blue > 255) ? 255 : int_Blue;
            return Color.FromArgb(int_Red, int_Green, int_Blue);
        }

        /// <summary>
        /// 正弦曲线Wave扭曲图片
        /// </summary>
        /// <param name="srcBmp">图片路径</param>
        /// <param name="bXDir">如果扭曲则选择为True</param>
        /// <param name="dMultValue">波形的幅度倍数,越大扭曲的程度越高,一般为3</param>
        /// <param name="dPhase">波形的起始相位,取值区间[0-2*PI)</param>
        private Bitmap TwistImage(Bitmap srcBmp, bool bXDir, double dMultValue, double dPhase)
        {
            double PI = 6.283185307179586476925286766559;
            Bitmap destBmp = new Bitmap(srcBmp.Width, srcBmp.Height);
            Graphics graph = Graphics.FromImage(destBmp);
            graph.FillRectangle(new SolidBrush(Color.White), 0, 0, destBmp.Width, destBmp.Height);
            graph.Dispose();
            double dBaseAxisLen = bXDir ? (double)destBmp.Height : (double)destBmp.Width;
            for (int i = 0; i < destBmp.Width; i++)
            {
                for (int j = 0; j < destBmp.Height; j++)
                {
                    double dx;
                    dx = bXDir ? (PI * (double)j) / dBaseAxisLen : (PI * (double)i) / dBaseAxisLen;
                    dx += dPhase;
                    double dy = Math.Sin(dx);
                    int nOldX = 0, nOldY = 0;
                    nOldX = bXDir ? i + (int)(dy * dMultValue) : i;
                    nOldY = bXDir ? j : j + (int)(dy * dMultValue);

                    Color color = srcBmp.GetPixel(i, j);
                    if (nOldX >= 0 && nOldX < destBmp.Width
                     && nOldY >= 0 && nOldY < destBmp.Height)
                    {
                        destBmp.SetPixel(nOldX, nOldY, color);
                    }
                }
            }
            srcBmp.Dispose();
            return destBmp;
        }

        #endregion
    }
CaptchaGenerator类

3.创建一个CaptchaHander用于在主页面生成验证码

using System.Data.Common;
using System;
using System.Collections.Generic;
using System.Drawing;
using System.Drawing.Imaging;
using System.Linq;
using System.Web;

using System.Web.SessionState;
using HIPI.AiXiu.Common;

namespace HIPI.AiXiu.ASPh
{
    /// <summary>
    /// CaptchaHander 的摘要说明
    /// </summary>
    public class CaptchaHander : IHttpHandler,IRequiresSessionState
    {

        public void ProcessRequest(HttpContext context)
        {
            //生成验证码
            CaptchaGenerator capt = new CaptchaGenerator();
            var number = capt.Text;
            Bitmap image = capt.Image;
            context.Session.Add("yzm",number);
            image.Save(context.Response.OutputStream,ImageFormat.Jpeg);
            context.Response.ContentType = "image/Jpeg";
            context.Response.End();
        }

        public bool IsReusable
        {
            get
            {
                return false;
            }
        }
    }
}
CaptchaHander

4.创建主页面

<%@ Page Language="C#" AutoEventWireup="true" CodeBehind="WebForm1.aspx.cs" Inherits="yanzhengma.WebForm1" %>

<!DOCTYPE html>

<html xmlns="http://www.w3.org/1999/xhtml">
<head runat="server">
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/>
    <title></title>
</head>
<body>
    <form id="form1" runat="server">
        <div>
            验证码:<input id="txtyzm" type="text" /> <img src="ASPX/CaptchaHander.ashx" class="captcha" /><br />
            <input id="btn" type="button" value="确定" />
        </div>
    </form>
</body>
</html>
主页面

注意:在主页面中引入img的路径要与CaptchaHander的路径一致

5.实现点击刷新功能

在主页面编写jQuery:

 <script>
        // 图形验证码点击刷新
            var captcha = $$(".captcha");
            var captchaSrc = captcha.attr('src');
            captcha.on('click', function () {
                $$(this).attr("src", captchaSrc + "?r=" + Math.random());
        })
    </script>
验证码js代码

注意:需要引入相关jQuery包.

6.在后台验证是否输入成功

protected void Button1_Click(object sender, EventArgs e)
        {
            if (Page.IsValid)
            {
                //验证码
                string code = txtCaptcha.Text.Trim();
                if (string.IsNullOrWhiteSpace(code))
                {
                    PageExtensions.Alert(this, "reg", "请输入验证码");
                }
                if (!Session["yzm"].Equals(code) || Session["yzm"] == null)
                {
                    PageExtensions.Alert(this, "reg", "图片验证码不正确");
                    return;
                }
                PageExtensions.Alert(this, "reg", "验证码正确");
            }
            }
验证验证码是否输入成功

7.效果图:

 

 

posted @ 2022-09-25 20:36  大白白2  阅读(69)  评论(0)    收藏  举报