• c#实现验证码功能(多种模式下分别实现验证功能)详细,带注释


    网上找了很多验证相关的代码,发现有很多瑕疵。现在本人整理测试了一个实现验证码功能的代码,里面有纯数字,纯英文,英文和数字混合等三种模式。并且在必要地方都已经备有注释,希望可以帮到那些需要的人。

    验证码功能一般是用于防止批量注册的,不少网站为了防止用户利用机器人自动注册、登录、灌水,都采用了验证码技术。所谓验证码,就是将一串随机产生的数字或字母或符号或文字,生成一幅图片, 图片里加上一些干扰象素(防止OCR),由用户肉眼识别其中的验证码信息,输入表单提交网站验证,验证成功后才能使用某项功能。

    前台代码:
    最好注意一下生成出来的图片样式,我个人是做了“margin:-5px;”的处理;
    <body> 
        <form id="form1" runat="server">
           <div>  
           <asp:TextBox ID="TextBox1" runat="server"></asp:TextBox>  
            <asp:Image ID="Image1" CssClass="csss" runat="server" ImageUrl="png.aspx" /><br />  
            <br />  
           <asp:Button ID="btnOK" runat="server" OnClick="btnOK_Click" Text="确认" />  
            <asp:DropDownList ID="DropDownList1" runat="server" AutoPostBack="True" OnSelectedIndexChanged="DropDownList1_SelectedIndexChanged">  
                <asp:ListItem Value="3">默认</asp:ListItem>  
                <asp:ListItem Value="1">文字</asp:ListItem>  
                <asp:ListItem Value="2">数字</asp:ListItem>  
                <asp:ListItem Value="3">混合</asp:ListItem>  
            </asp:DropDownList></div>  
    
        </form>
    </body>

    后台代码:
        protected void DropDownList1_SelectedIndexChanged(object sender, EventArgs e)
        {
            switch (DropDownList1.SelectedValue)
            {
                case "1":
                    Image1.ImageUrl = "png.aspx?aa=1";
                    break;
                case "2":
                    Image1.ImageUrl = "png.aspx?aa=2";
                    break;
                case "3":
                    Image1.ImageUrl = "png.aspx?aa=3";
                    break;
            }
        }
        protected void btnOK_Click(object sender, EventArgs e)
        {
            if (TextBox1.Text == Session["gif"].ToString())
            {
                Response.Write("OK,正确");
            }
            else
                Response.Write("验证码不符合");
        }

    另外,记得新建一个png.aspx页面,此页面主要是用来生成一个图片格式的文件。aspx页面为空,然后在aspx.cs页面添加一下代码。
    //记得加上以下引用;
    using System.Text;
    using System.Drawing;
    
    protected void Page_Load(object sender, EventArgs e)
        {
            switch (Request.QueryString["aa"])
            {
                case "1":
                    Session["gif"] = stxt(4);
                    break;
                case "2":
                    Session["gif"] = GetRandomint(4);
                    break;
                case "3":
                    Session["gif"] = CreateRandomCode(4);
                    break;
                default:
                    Session["gif"] = CreateRandomCode(4);
                    break;
            }
            CreateImage(Session["gif"].ToString());
        }
    
        /// <summary>
        /// 这个是使用数字模式
        /// </summary>
        /// <param name="VcodeNum"></param>
        /// <returns></returns>
        private String GetRandomint(int codeCount)
        {
            Random random = new Random();
            string min = "";
            string max = "";
            for (int i = 0; i < codeCount; i++)
            {
                min += "1";
                max += "9";
            }
            return (random.Next(Convert.ToInt32(min), Convert.ToInt32(max)).ToString());
        }
    
        /// <summary>
        /// 这个是使用文字模式
        /// </summary>
        /// <param name="VcodeNum"></param>
        /// <returns></returns>
        private string stxt(int num)
        {
            Encoding gb = Encoding.GetEncoding("gb2312");
    
            //调用函数产生10个随机中文汉字编码 
            object[] bytes = CreateRegionCode(num);
            string strtxt = "";
    
            //根据汉字编码的字节数组解码出中文汉字 
            for (int i = 0; i < num; i++)
            {
                strtxt += gb.GetString((byte[])Convert.ChangeType(bytes[i], typeof(byte[])));
            }
            return strtxt;
        }
        /* 
            此函数在汉字编码范围内随机创建含两个元素的十六进制字节数组,每个字节数组代表一个汉字,并将 
            四个字节数组存储在object数组中。 
            参数:strlength,代表需要产生的汉字个数 
        */
        public static object[] CreateRegionCode(int strlength)
        {
            //定义一个字符串数组储存汉字编码的组成元素 
            string[] rBase = new String[16] { "0", "1", "2", "3", "4", "5", "6", "7", "8", "9", "a", "b", "c", "d", "e", "f" };
            Random rnd = new Random();
            //定义一个object数组用来 
            object[] bytes = new object[strlength];
    
            /**/
            /*每循环一次产生一个含两个元素的十六进制字节数组,并将其放入bject数组中 
         每个汉字有四个区位码组成 
         区位码第1位和区位码第2位作为字节数组第一个元素 
         区位码第3位和区位码第4位作为字节数组第二个元素 
        */
            for (int i = 0; i < strlength; i++)
            {
                //区位码第1位 
                int r1 = rnd.Next(11, 14);
                string str_r1 = rBase[r1].Trim();
    
                //区位码第2位 
                rnd = new Random(r1 * unchecked((int)DateTime.Now.Ticks) + i);//更换随机数发生器的种子避免产生重复值 
                int r2;
                if (r1 == 13)
                {
                    r2 = rnd.Next(0, 7);
                }
                else
                {
                    r2 = rnd.Next(0, 16);
                }
                string str_r2 = rBase[r2].Trim();
    
                //区位码第3位 
                rnd = new Random(r2 * unchecked((int)DateTime.Now.Ticks) + i);
                int r3 = rnd.Next(10, 16);
                string str_r3 = rBase[r3].Trim();
    
                //区位码第4位 
                rnd = new Random(r3 * unchecked((int)DateTime.Now.Ticks) + i);
                int r4;
                if (r3 == 10)
                {
                    r4 = rnd.Next(1, 16);
                }
                else if (r3 == 15)
                {
                    r4 = rnd.Next(0, 15);
                }
                else
                {
                    r4 = rnd.Next(0, 16);
                }
                string str_r4 = rBase[r4].Trim();
    
                //定义两个字节变量存储产生的随机汉字区位码 
                byte byte1 = Convert.ToByte(str_r1 + str_r2, 16);
                byte byte2 = Convert.ToByte(str_r3 + str_r4, 16);
                //将两个字节变量存储在字节数组中 
                byte[] str_r = new byte[] { byte1, byte2 };
    
                //将产生的一个汉字的字节数组放入object数组中 
                bytes.SetValue(str_r, i);
            }
            return bytes;
        }
    
    
        /// <summary>
        /// 这个是使用字母,数字混合
        /// </summary>
        /// <param name="VcodeNum"></param>
        /// <returns></returns>
        private string CreateRandomCode(int codeCount)
        {
            string allChar = "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,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";
            string[] allCharArray = allChar.Split(',');
            string randomCode = "";
            int temp = -1;
            Random rand = new Random();
            for (int i = 0; i < codeCount; i++)
            {
                if (temp != -1)
                {
                    rand = new Random(i * temp * ((int)DateTime.Now.Ticks));
                }
                int t = rand.Next(61);
                if (temp == t)
                {
                    return CreateRandomCode(codeCount);
                }
                temp = t;
                randomCode += allCharArray[t];
            }
            return randomCode;
        }
        
        private void CreateImage(string checkCode)//输出为图片模式
        {
            int iwidth = (int)(checkCode.Length * 11);
            Bitmap image = new Bitmap(iwidth, 25);//输出为图片的width和heigh
            Graphics g = Graphics.FromImage(image);
            Font f = new Font("Arial", 10, FontStyle.Bold);
            Brush b = new SolidBrush(Color.White);  
            g.Clear(Color.Black);
            g.DrawString(checkCode, f, b, 5, 3);
            /* 
              在指定位置并且用指定的 Brush 和 Font 对象绘制指定的文本字符串
              checkCode要绘制的字符串;f,字符串的文本格式;
              b,文本的颜色和纹理;3,3是左上角x和y坐标
            */
            Pen blackPen = new Pen(Color.Black, 0);
            Random rand = new Random();      
            System.IO.MemoryStream ms = new System.IO.MemoryStream();
            image.Save(ms, System.Drawing.Imaging.ImageFormat.Jpeg);
            Response.ClearContent();
            Response.ContentType = "image/Jpeg";
            Response.BinaryWrite(ms.ToArray());
            g.Dispose();
            image.Dispose();
        }
    下面是一个效果图,希望可以帮到需要的同学……
          
  • 相关阅读:
    代码审计之越权及越权
    代码审计之XSS及修复
    代码审计之SQL注入及修复
    代码审计之CSRF原理及修复
    挖穿各大SRC的短信轰炸
    Kerberoasting攻击
    SPN扫描
    Windows认证 | 域认证
    Windows认证 | 网络认证
    Ceph 纠删码介绍
  • 原文地址:https://www.cnblogs.com/james1207/p/3257993.html
Copyright © 2020-2023  润新知