C#動態驗證碼


驗證碼定義:

就是將一串隨機產生的數字或符號,生成一幅圖片,圖片里加上一些干擾象素(防止OCR),由用戶肉眼識別其中的驗證碼信息,輸入表單提交網站驗證,驗證成功后才能使用某項功能

項目中,我們登錄注冊都會用到驗證碼,那為什么我們要使用驗證碼呢?

經過資料查詢以及個人理解,驗證碼的作用:防止批量進行一些操作:例如貼吧,論壇中匿名發帖,針對某特定用戶用特殊破解方式登錄嘗試,防止機器惡意注冊等


在C#中,代碼如下:
       try
              {
                Response.Cache.SetCacheability(HttpCacheability.NoCache);
                string randomcode = Globals.CreateVerifyCode(4);

                // 隨機轉動角度
                int randAngle = 45;
                int mapwidth = (int)(randomcode.Length * 33);
                // 創建圖片背景
                Bitmap map = new Bitmap(mapwidth-5, 45);
                Graphics graph = Graphics.FromImage(map);
                // 清除畫面,填充背景
                graph.Clear(Color.AliceBlue);
                // 畫一個邊框
                //graph.DrawRectangle(new Pen(Color.Gray, 0), 0, 0, map.Width-1, map.Height - 3);
                //graph.SmoothingMode = System.Drawing.Drawing2D.SmoothingMode.AntiAlias;//模式

                Random rand = new Random();

                // 背景噪點生成
                Pen blackPen = new Pen(Color.LightGray, 0);
                for (int i = 0; i < 50; i++)
                {
                    int x = rand.Next(0, map.Width);
                    int y = rand.Next(0, map.Height);
                    graph.DrawRectangle(blackPen, x, y, 1, 1);
                }

                // 驗證碼旋轉,防止機器識別
                // 拆散字符串成單字符數組
                char[] chars = randomcode.ToCharArray();

                // 文字距中
                StringFormat format = new StringFormat(StringFormatFlags.NoClip);
                format.Alignment = StringAlignment.Center;
                format.LineAlignment = StringAlignment.Center;

                // 定義顏色
                Color[] c = { Color.Black, Color.Red, Color.DarkBlue, Color.Green, Color.Brown, Color.DarkCyan, Color.Purple,Color.DarkGreen };
                // 定義字體
                //string[] font = { "Verdana", "Microsoft Sans Serif", "Comic Sans MS", "Arial","Lucida Sans Unicode", Rockwell, Batang ,Times New Roman,Bernard MT Condensed};

                for (int i = 0; i < chars.Length; i++)
                {
                    int cindex = rand.Next(7);
                    int findex = rand.Next(4);

                    // 字體樣式(參數2為字體大小)
                    Font f = new System.Drawing.Font("Microsoft Sans Serif", 17, System.Drawing.FontStyle.Bold);
                    Brush b = new System.Drawing.SolidBrush(c[cindex]);

                    Point dot = new Point(23, 15);
                    // 測試X坐標顯示間距的
                    //graph.DrawString(dot.X.ToString(),fontstyle,new SolidBrush(Color.Black),10,150);
                    // 轉動的度數
                    float angle = rand.Next(-randAngle, randAngle);

                    // 移動光標到指定位置
                    graph.TranslateTransform(dot.X, dot.Y);
                    graph.RotateTransform(angle);
                    graph.DrawString(chars[i].ToString(), f, b, 1, 10, format);
                    //graph.DrawString(chars[i].ToString(),fontstyle,new SolidBrush(Color.Blue),1,1,format);
                    // 轉回去
                    graph.RotateTransform(-angle);
                    // 移動光標到指定位置
                    graph.TranslateTransform(3, -dot.Y);
                }
                // 標准隨機碼
                //graph.DrawString(randomcode,fontstyle,new SolidBrush(Color.Blue),2,2); 

                // 生成圖片
                System.IO.MemoryStream ms = new System.IO.MemoryStream();
                map.Save(ms, System.Drawing.Imaging.ImageFormat.Gif);

                Response.ClearContent();
                Response.ContentType = "image/gif";
                Response.BinaryWrite(ms.ToArray());

                graph.Dispose();
                map.Dispose();

        }
        catch
        {
        }

 

或者我們還有更全的方法實現:

namespace DotNet.Utilities  
{  
    /// <summary>  
    /// 驗證碼類  
    /// </summary>  
    public class Rand  
    {  
        #region 生成隨機數字  
        /// <summary>  
        /// 生成隨機數字  
        /// </summary>  
        /// <param name="length">生成長度</param>  
        public static string Number(int Length)  
        {  
            return Number(Length, false);  
        }  
  
        /// <summary>  
        /// 生成隨機數字  
        /// </summary>  
        /// <param name="Length">生成長度</param>  
        /// <param name="Sleep">是否要在生成前將當前線程阻止以避免重復</param>  
        public static string Number(int Length, bool Sleep)  
        {  
            if (Sleep) System.Threading.Thread.Sleep(3);  
            string result = "";  
            System.Random random = new Random();  
            for (int i = 0; i < Length; i++)  
            {  
                result += random.Next(10).ToString();  
            }  
            return result;  
        }  
        #endregion  
 
        #region 生成隨機字母與數字  
        /// <summary>  
        /// 生成隨機字母與數字  
        /// </summary>  
        /// <param name="IntStr">生成長度</param>  
        public static string Str(int Length)  
        {  
            return Str(Length, false);  
        }  
  
        /// <summary>  
        /// 生成隨機字母與數字  
        /// </summary>  
        /// <param name="Length">生成長度</param>  
        /// <param name="Sleep">是否要在生成前將當前線程阻止以避免重復</param>  
        public static string Str(int Length, bool Sleep)  
        {  
            if (Sleep) System.Threading.Thread.Sleep(3);  
            char[] Pattern = new char[] { '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' };  
            string result = "";  
            int n = Pattern.Length;  
            System.Random random = new Random(~unchecked((int)DateTime.Now.Ticks));  
            for (int i = 0; i < Length; i++)  
            {  
                int rnd = random.Next(0, n);  
                result += Pattern[rnd];  
            }  
            return result;  
        }  
        #endregion  
 
        #region 生成隨機純字母隨機數  
        /// <summary>  
        /// 生成隨機純字母隨機數  
        /// </summary>  
        /// <param name="IntStr">生成長度</param>  
        public static string Str_char(int Length)  
        {  
            return Str_char(Length, false);  
        }  
  
        /// <summary>  
        /// 生成隨機純字母隨機數  
        /// </summary>  
        /// <param name="Length">生成長度</param>  
        /// <param name="Sleep">是否要在生成前將當前線程阻止以避免重復</param>  
        public static string Str_char(int Length, bool Sleep)  
        {  
            if (Sleep) System.Threading.Thread.Sleep(3);  
            char[] Pattern = new char[] { '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 result = "";  
            int n = Pattern.Length;  
            System.Random random = new Random(~unchecked((int)DateTime.Now.Ticks));  
            for (int i = 0; i < Length; i++)  
            {  
                int rnd = random.Next(0, n);  
                result += Pattern[rnd];  
            }  
            return result;  
        }  
        #endregion  
    }  
  
    /// <summary>  
    /// 驗證圖片類  
    /// </summary>  
    public class YZMHelper  
    {  
        #region 私有字段  
        private string text;  
        private Bitmap image;  
        private int letterCount = 4;   //驗證碼位數  
        private int letterWidth = 16;  //單個字體的寬度范圍  
        private int letterHeight = 20; //單個字體的高度范圍  
        private static byte[] randb = new byte[4];  
        private static RNGCryptoServiceProvider rand = new RNGCryptoServiceProvider();  
        private Font[] fonts =   
    {  
       new Font(new FontFamily("Times New Roman"),10 +Next(1),System.Drawing.FontStyle.Regular),  
       new Font(new FontFamily("Georgia"), 10 + Next(1),System.Drawing.FontStyle.Regular),  
       new Font(new FontFamily("Arial"), 10 + Next(1),System.Drawing.FontStyle.Regular),  
       new Font(new FontFamily("Comic Sans MS"), 10 + Next(1),System.Drawing.FontStyle.Regular)  
    };  
        #endregion  
 
        #region 公有屬性  
        /// <summary>  
        /// 驗證碼  
        /// </summary>  
        public string Text  
        {  
            get { return this.text; }  
        }  
  
        /// <summary>  
        /// 驗證碼圖片  
        /// </summary>  
        public Bitmap Image  
        {  
            get { return this.image; }  
        }  
        #endregion  
 
        #region 構造函數  
        public YZMHelper()  
        {  
            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";  
            this.text = Rand.Number(4);  
            CreateImage();  
        }  
        #endregion  
 
        #region 私有方法  
        /// <summary>  
        /// 獲得下一個隨機數  
        /// </summary>  
        /// <param name="max">最大值</param>  
        private static int Next(int max)  
        {  
            rand.GetBytes(randb);  
            int value = BitConverter.ToInt32(randb, 0);  
            value = value % (max + 1);  
            if (value < 0) value = -value;  
            return value;  
        }  
  
        /// <summary>  
        /// 獲得下一個隨機數  
        /// </summary>  
        /// <param name="min">最小值</param>  
        /// <param name="max">最大值</param>  
        private static int Next(int min, int max)  
        {  
            int value = Next(max - min) + min;  
            return value;  
        }  
        #endregion  
 
        #region 公共方法  
        /// <summary>  
        /// 繪制驗證碼  
        /// </summary>  
        public void CreateImage()  
        {  
            int int_ImageWidth = this.text.Length * letterWidth;  
            Bitmap image = new Bitmap(int_ImageWidth, letterHeight);  
            Graphics g = Graphics.FromImage(image);  
            g.Clear(Color.White);  
            for (int i = 0; i < 2; i++)  
            {  
                int x1 = Next(image.Width - 1);  
                int x2 = Next(image.Width - 1);  
                int y1 = Next(image.Height - 1);  
                int y2 = Next(image.Height - 1);  
                g.DrawLine(new Pen(Color.Silver), x1, y1, x2, y2);  
            }  
            int _x = -12, _y = 0;  
            for (int int_index = 0; int_index < this.text.Length; int_index++)  
            {  
                _x += Next(12, 16);  
                _y = Next(-2, 2);  
                string str_char = this.text.Substring(int_index, 1);  
                str_char = 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[Next(fonts.Length - 1)], newBrush, thePos);  
            }  
            for (int i = 0; i < 10; i++)  
            {  
                int x = Next(image.Width - 1);  
                int y = Next(image.Height - 1);  
                image.SetPixel(x, y, Color.FromArgb(Next(0, 255), Next(0, 255), Next(0, 255)));  
            }  
            image = TwistImage(image, true, Next(1, 3), Next(4, 6));  
            g.DrawRectangle(new Pen(Color.LightGray, 1), 0, 0, int_ImageWidth - 1, (letterHeight - 1));  
            this.image = image;  
        }  
  
        /// <summary>  
        /// 字體隨機顏色  
        /// </summary>  
        public Color GetRandomColor()  
        {  
            Random RandomNum_First = new Random((int)DateTime.Now.Ticks);  
            System.Threading.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="nMultValue">波形的幅度倍數,越大扭曲的程度越高,一般為3</param>  
        /// <param name="dPhase">波形的起始相位,取值區間[0-2*PI)</param>  
        public System.Drawing.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 = 0;  
                    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  
    }  
}  

我們將生成的驗證碼中的字符串存如session或者cookies中,在onclick事件中獲取session或者cookies值與用戶輸入的值進行匹配即可。

 /// <summary>
    ///創建驗證碼
    /// </summary>
    public class CreateIdentifyingCode
    {
        /// <summary>
        /// 生成隨機字符串
        /// </summary>
        /// <param name="codeCount"></param>
        /// <returns></returns>
        public string CreateRandomCode(int codeCount)
        {
            string randomCode = "";
            try
            {
                string allChar = "2,3,4,5,6,7,H,J,K,L,M,N,X,Y,Z,a,b,c,d,e,f,g,h,i,j,k,m,n,p,q,P,Q,R,S,T,U,V,W,r,s,t,u,v,8,9,A,B,C,D,E,F,G,w,x,y,z";
                //按 ,號拆分字符串
                string[] allCharArray = allChar.Split(',');
                
                int temp = -1;
                Random rand = new Random();
                for (int i = 0; i < codeCount; i++)
                {
                    if (temp > 0)
                    {
                        rand = new Random(1 * temp * ((int)DateTime.Now.Ticks));
                    }
                    int t = rand.Next(35);
                    if (temp == t)
                    {
                        return CreateRandomCode(codeCount);
                    }
                    temp = t;
                    randomCode += allCharArray[t];
                }
            }
            catch (Exception)
            {
                
           
            }
          
            return randomCode;
        }
        /// <summary>
        /// 創建驗證碼圖片,並保存
        /// </summary>
        /// <param name="validateCode"></param>
        /// <returns></returns>
        public void SaveValidateGraphic(string validateCode, string path)
        {

            byte[] buffer = CreateValidateGraphic(validateCode);
            MemoryStream ms2 = new MemoryStream(buffer, 0, buffer.Length);
            ms2.Seek(0, SeekOrigin.Begin);
            System.Drawing.Image image2 = System.Drawing.Image.FromStream(ms2);
            image2.Save(path + "x.jpg", ImageFormat.Jpeg);
        }
        /// <summary>
        /// 創建驗證碼圖片
        /// </summary>
        /// <param name="validateCode"></param>
        /// <returns></returns>
        public byte[] CreateValidateGraphic(string validateCode)
        {
            //設置圖片的寬度與高度
            Bitmap image = new Bitmap((int)Math.Ceiling(validateCode.Length * 16.0), 27);
            Graphics g = Graphics.FromImage(image);
            try
            {
                //生成隨機生成器
                Random random = new Random();
                //清空圖片背景色
                g.Clear(Color.White);
                //圖片的干擾線
                for (int i = 0; i < 50; i++)
                {
                    int x1 = random.Next(image.Width);
                    int x2 = random.Next(image.Width);
                    int y1 = random.Next(image.Width);
                    int y2 = random.Next(image.Width);
                    g.DrawLine(new Pen(Color.Silver), x1, y1, x2, y2);

                }
                Font font = new Font("Arial", 13, (FontStyle.Bold | FontStyle.Italic));
                LinearGradientBrush brush = new LinearGradientBrush(new Rectangle(0, 0, image.Width, image.Height), Color.Blue, Color.DarkRed, 1.2f, true);
                g.DrawString(validateCode, font, brush, 3, 2);
                //畫圖片的前景干擾點
                for (int i = 0; i < 100; i++)
                {
                    int x = random.Next(image.Width);
                    int y = random.Next(image.Height);
                    image.SetPixel(x, y, Color.FromArgb(random.Next()));
                }
                //畫圖片的干擾線
                g.DrawRectangle(new Pen(Color.Silver), 0, 0, image.Width - 1, image.Height - 1);
                //保存圖片數據
                MemoryStream stream = new MemoryStream();
                image.Save(stream, ImageFormat.Jpeg);
                //輸出圖片流
                return stream.ToArray();
            }
            finally
            {
                g.Dispose();
                image.Dispose();
            }

        }
    }

 

很簡單的實用的驗證碼就介紹到這里,有什么不足的還請大家多多指導。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM