12306動態(tài)驗證碼啟發(fā)之ASP.NET實現(xiàn)動態(tài)GIF驗證碼(附源碼)
12306網(wǎng)站推出“彩色動態(tài)驗證碼機制”,新版驗證碼不但經(jīng)常出現(xiàn)字符疊壓,還不停抖動,不少人大呼“看不清”,稱“那個驗證碼,是畢加索的抽象畫么!”鐵總客服則表示:為了能正常購票只能這樣。而多家搶票軟件接近“報廢”,引發(fā)不少網(wǎng)友不滿的吐槽稱“太抽象太藝術了”。
以前做項目有時候也會用到驗證碼,但基本都是靜態(tài)的,這次也想湊湊12306的熱鬧。閑言少續(xù),切入正題,先上代碼。
實現(xiàn)方法:
public void ShowCode()
{
//對象實例化
Validate GifValidate = new Validate();
#region 對驗證碼進行設置(不進行設置時,將以默認值生成)
//驗證碼位數(shù),不小于4位
GifValidate.ValidateCodeCount = 4;
//驗證碼字體型號(默認13)
GifValidate.ValidateCodeSize = 13;
//驗證碼圖片高度,高度越大,字符的上下偏移量就越明顯
GifValidate.ImageHeight = 23;
//驗證碼字符及線條顏色(需要參考顏色類)
GifValidate.DrawColor = System.Drawing.Color.BlueViolet;
//驗證碼字體(需要填寫服務器安裝的字體)
GifValidate.ValidateCodeFont = "Arial";
//驗證碼字符是否消除鋸齒
GifValidate.FontTextRenderingHint = false;
//定義驗證碼中所有的字符(","分離),似乎暫時不支持中文
GifValidate.AllChar = "1,2,3,4,5,6,7,8,9,0,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";
#endregion
//輸出圖像(Session名稱)
GifValidate.OutPutValidate("GetCode");
}
調(diào)用主要方法:
public class Validate
{
public string AllChar = "1,2,3,4,5,6,7,8,9,0,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";
public Color DrawColor = Color.BlueViolet;
public bool FontTextRenderingHint = false;
public int ImageHeight = 0x17;
private byte TrueValidateCodeCount = 4;
protected string ValidateCode = "";
public string ValidateCodeFont = "Arial";
public float ValidateCodeSize = 13f;
private void CreateImageBmp(out Bitmap ImageFrame)
{
char[] chArray = this.ValidateCode.ToCharArray(0, this.ValidateCodeCount);
int width = (int) (((this.TrueValidateCodeCount * this.ValidateCodeSize) * 1.3) + 4.0);
ImageFrame = new Bitmap(width, this.ImageHeight);
Graphics graphics = Graphics.FromImage(ImageFrame);
graphics.Clear(Color.White);
Font font = new Font(this.ValidateCodeFont, this.ValidateCodeSize, FontStyle.Bold);
Brush brush = new SolidBrush(this.DrawColor);
int maxValue = (int) Math.Max((float) ((this.ImageHeight - this.ValidateCodeSize) - 3f), (float) 2f);
Random random = new Random();
for (int i = 0; i < this.TrueValidateCodeCount; i++)
{
int[] numArray = new int[] { (((int) (i * this.ValidateCodeSize)) + random.Next(1)) + 3, random.Next(maxValue) };
Point point = new Point(numArray[0], numArray[1]);
if (this.FontTextRenderingHint)
{
graphics.TextRenderingHint = TextRenderingHint.SingleBitPerPixel;
}
else
{
graphics.TextRenderingHint = TextRenderingHint.AntiAlias;
}
graphics.DrawString(chArray[i].ToString(), font, brush, (PointF) point);
}
graphics.Dispose();
}
private void CreateImageGif()
{
AnimatedGifEncoder encoder = new AnimatedGifEncoder();
MemoryStream stream = new MemoryStream();
encoder.Start();
encoder.SetDelay(5);
encoder.SetRepeat(0);
for (int i = 0; i < 10; i++)
{
Bitmap bitmap;
this.CreateImageBmp(out bitmap);
this.DisposeImageBmp(ref bitmap);
bitmap.Save(stream, ImageFormat.Png);
encoder.AddFrame(Image.FromStream(stream));
stream = new MemoryStream();
}
encoder.OutPut(ref stream);
HttpContext.Current.Response.ClearContent();
HttpContext.Current.Response.ContentType = "image/Gif";
HttpContext.Current.Response.BinaryWrite(stream.ToArray());
stream.Close();
stream.Dispose();
}
private void CreateValidate()
{
this.ValidateCode = "";
string[] strArray = this.AllChar.Split(new char[] { ',' });
int index = -1;
Random random = new Random();
for (int i = 0; i < this.ValidateCodeCount; i++)
{
if (index != -1)
{
random = new Random((i * index) * ((int) DateTime.Now.Ticks));
}
int num3 = random.Next(0x23);
if (index == num3)
{
this.CreateValidate();
}
index = num3;
this.ValidateCode = this.ValidateCode + strArray[index];
}
if (this.ValidateCode.Length > this.TrueValidateCodeCount)
{
this.ValidateCode = this.ValidateCode.Remove(this.TrueValidateCodeCount);
}
}
private void DisposeImageBmp(ref Bitmap ImageFrame)
{
Graphics graphics = Graphics.FromImage(ImageFrame);
Pen pen = new Pen(this.DrawColor, 1f);
Random random = new Random();
Point[] pointArray = new Point[2];
for (int i = 0; i < 15; i++)
{
pointArray[0] = new Point(random.Next(ImageFrame.Width), random.Next(ImageFrame.Height));
pointArray[1] = new Point(random.Next(ImageFrame.Width), random.Next(ImageFrame.Height));
graphics.DrawLine(pen, pointArray[0], pointArray[1]);
}
graphics.Dispose();
}
public void OutPutValidate(string ValidateCodeSession)
{
this.CreateValidate();
this.CreateImageGif();
HttpContext.Current.Session[ValidateCodeSession] = this.ValidateCode;
}
public byte ValidateCodeCount
{
get
{
return this.TrueValidateCodeCount;
}
set
{
if (value > 4)
{
this.TrueValidateCodeCount = value;
}
}
}
}
驗證碼效果:
-----下載源碼-----
以上就是實現(xiàn)ASP.NET的全部過程,還附有源碼,希望可以幫到大家更好地了解ASP.NET驗證碼的生成方法。
相關文章
asp.net 分頁sql語句(結合aspnetpager)
一直用的是存儲過程分頁,小項目一般不寫存儲過程,就需要直接寫分頁sql語句。2009-01-01
ASP.NET Core根據(jù)環(huán)境變量支持多個 appsettings.json配置文件
這篇文章主要介紹了ASP.NET Core根據(jù)環(huán)境變量支持多個 appsettings.json配置文件,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-08-08
ASP.NET MVC3關于生成純靜態(tài)后如何不再走路由直接訪問靜態(tài)頁面
高訪問量類型的電子商務網(wǎng)站,需要將一些不是經(jīng)常變化的頁面生成靜態(tài)頁面,然后普通用戶就可以直接訪問這些靜態(tài)頁面而不用再訪問需要連接數(shù)據(jù)庫的動態(tài)頁面。那么ASP.NET MVC3中如何做到這一點呢2011-12-12
發(fā)布一個基于TokyoTyrant的C#客戶端開源項目
目前在網(wǎng)上關于TokyoCabinet(以下簡稱TC)和TokyoTyrant(以下簡稱TT)的資料已相對豐富了,但在.NET平臺上的客戶端軟件卻相對匱乏,因為做Discuz!NT企業(yè)版的關系,兩個月前開始接觸TC和TT,開始寫相關的客戶端代碼。2010-07-07

