.net 隨機(jī)生成漢字
在沒(méi)講.net如何隨機(jī)生成漢字之前先給大家講下漢字編碼組成及原理。
1、漢字編碼原理
到底怎么辦到隨機(jī)生成漢字的呢?漢字從哪里來(lái)的呢?是不是有個(gè)后臺(tái)數(shù)據(jù)表,其中存放了所需要的所有漢字,使用程序
隨機(jī)取出幾個(gè)漢字組合就行了呢?使用后臺(tái)數(shù)據(jù)庫(kù)先將所有漢字存起來(lái)使用時(shí)隨機(jī)取
出,這也是一種辦法,但是中文漢字有這么多,怎么來(lái)制作呢?其實(shí)可以不使用任何后臺(tái)數(shù)據(jù)庫(kù),使用程序就能做到這一切。
要知道如何生成漢字,就得先了解中文漢字的編碼原理。
1980年,為了使每一個(gè)漢字有一個(gè)全國(guó)統(tǒng)一的代碼,我國(guó)頒布了第一個(gè)漢字編碼的國(guó)家標(biāo)準(zhǔn):GB2312-80《信息交換用漢
字編碼字符集》基本集,簡(jiǎn)稱(chēng)GB2312,這個(gè)字符集是我國(guó)中文信息處理技術(shù)的
發(fā)展基礎(chǔ),也是國(guó)內(nèi)所有漢字系統(tǒng)的統(tǒng)一標(biāo)準(zhǔn)。到了后來(lái)又公布了國(guó)家標(biāo)準(zhǔn)GB18030-2000《信息交換用漢字編碼字符集基本
集的擴(kuò)充》,簡(jiǎn)稱(chēng)GB18030,編程時(shí)如果涉及到編碼和本地化的朋友應(yīng)該對(duì)GB18030很熟
悉。這是是我國(guó)繼GB2312-1980和GB13000-1993之后最重要的漢字編碼標(biāo)準(zhǔn),同時(shí)也是未來(lái)我國(guó)計(jì)算機(jī)系統(tǒng)必須遵循的基礎(chǔ)
性標(biāo)準(zhǔn)之一。
目前在中文WINDOWS操作系統(tǒng)中,.NET編程中默認(rèn)的的代碼頁(yè)就是GB18030簡(jiǎn)體中文。但是事實(shí)上如果生成中文漢字驗(yàn)
證碼只須要使用GB2312字符集就已經(jīng)足夠了。字符集中除了我們平時(shí)大家都認(rèn)識(shí)的漢字外
,也包含了很多我們不認(rèn)識(shí)平時(shí)也很少見(jiàn)到的漢字。如果生成中文漢字驗(yàn)證碼中有很多我們不認(rèn)識(shí)的漢字讓我們輸入,對(duì)于使
用拼音輸入法的朋友來(lái)說(shuō)可不是好事,五筆使用者還能勉強(qiáng)根據(jù)漢字的長(zhǎng)相打出來(lái),
呵呵!所以對(duì)于GB2312字符集中的漢字我們也不是全都要用。
中文漢字字符可以使用區(qū)位碼來(lái)表示,見(jiàn)其實(shí)這兩個(gè)表是同一回事,只不過(guò)一個(gè)使用十六進(jìn)制分區(qū)表示,一個(gè)使用區(qū)位所在的
數(shù)字位置表示。 例如“好”字的十六進(jìn)制區(qū)位碼是ba c3,前兩位是區(qū)域,后兩位代表位置,ba處在第26區(qū),“好”處在此
區(qū)漢字的第35位也就是c3位置,所以數(shù)字代碼就是2635。這就是GB2312漢字區(qū)位原理。根據(jù)《漢字區(qū)位碼表 》我們可以發(fā)現(xiàn)
第15區(qū)也就是AF區(qū)以前都沒(méi)有漢字,只有少量符號(hào),漢字都從第16區(qū)B0開(kāi)始,這就是
為什么GB2312字符集都是從16區(qū)開(kāi)始的。
2、.Net程序處理漢字編碼原理分析
在.Net中可以使用System.Text來(lái)處理所有語(yǔ)言的編碼。在System.Text命名空間中包含眾多編碼的類(lèi),可供進(jìn)行操作及轉(zhuǎn)換。其
中的Encoding類(lèi)就是重點(diǎn)處理漢字編碼的類(lèi)。通過(guò)在.NET文檔中查詢(xún)Encoding類(lèi)的
方法我們可以發(fā)現(xiàn)所有和文字編碼有關(guān)的都是字節(jié)數(shù)組,其中有兩個(gè)很好用的方法:
Encoding.GetBytes ()方法將指定的 String 或字符數(shù)組的全部或部分內(nèi)容編碼為字節(jié)數(shù)組
Encoding.GetString ()方法將指定字節(jié)數(shù)組解碼為字符串。
沒(méi)錯(cuò)我們可以通過(guò)這兩個(gè)方法將漢字字符編碼為字節(jié)數(shù)組,同樣知道了漢字GB2312的字節(jié)數(shù)組編碼也就可以將字節(jié)數(shù)組解碼為漢字字符。通過(guò)對(duì)“好”字進(jìn)行編碼為字節(jié)數(shù)組后
Encoding gb=System.Text.Encoding.GetEncoding("gb2312");
object[] bytes=gb.Encoding.GetBytes ("好");
發(fā)現(xiàn)得到了一個(gè)長(zhǎng)度為2的字節(jié)數(shù)組bytes,使用
string lowCode = System.Convert.ToString(bytes[0], 16); //取出元素1編碼內(nèi)容(兩位16進(jìn)制)
string hightCode = System.Convert.ToString(bytes[1], 16);//取出元素2編碼內(nèi)容(兩位16進(jìn)制)
之后發(fā)現(xiàn)字節(jié)數(shù)組bytes16進(jìn)制變碼后內(nèi)容竟然是{ba,c3},剛好是“好”字的十六進(jìn)制區(qū)位碼(見(jiàn)區(qū)
位碼表)。
因此我們就可以隨機(jī)生成一個(gè)長(zhǎng)度為2的十六進(jìn)制字節(jié)數(shù)組,使用GetString ()方法對(duì)其進(jìn)行解碼就可以得到漢字字符了。不
過(guò)對(duì)于生成中文漢字驗(yàn)證碼來(lái)說(shuō),因?yàn)榈?5區(qū)也就是AF區(qū)以前都沒(méi)有漢字,只有
少量符號(hào),漢字都從第16區(qū)B0開(kāi)始,并且從區(qū)位D7開(kāi)始以后的漢字都是和很難見(jiàn)到的繁雜漢字,所以這些都要排出掉。所以隨
機(jī)生成的漢字十六進(jìn)制區(qū)位碼第1位范圍在B、C、D之間,如果第1位是D的話,第2位
區(qū)位碼就不能是7以后的十六進(jìn)制數(shù)。在來(lái)看看區(qū)位碼表發(fā)現(xiàn)每區(qū)的第一個(gè)位置和最后一個(gè)位置都是空的,沒(méi)有漢字,因此隨機(jī)
生成的區(qū)位碼第3位如果是A的話,第4位就不能是0;第3位如果是F的話,第4位就
不能是F。
好了,知道了原理,隨機(jī)生成中文漢字的程序也就出來(lái)了,以下就是生成4個(gè)隨機(jī)漢字的C#控制臺(tái)代碼:
/// <summary>
/// 隨機(jī)生成漢字
/// </summary>
/// <param name="strlength">長(zhǎng)度(4位)</param>
/// <returns></returns>
public string CreateCode(int strlength)
{
//定義一個(gè)字符串?dāng)?shù)組儲(chǔ)存漢字編碼的組成元素
string[] r = new String[16] { "0", "1", "2", "3", "4", "5", "6", "7", "8", "9", "a", "b", "c", "d", "e", "f" };
Random rnd = new Random();
//定義一個(gè)object數(shù)組用來(lái)
object[] bytes = new object[strlength];
/**/
/*每循環(huán)一次產(chǎn)生一個(gè)含兩個(gè)元素的十六進(jìn)制字節(jié)數(shù)組,并將其放入bject數(shù)組中
每個(gè)漢字有四個(gè)區(qū)位碼組成
區(qū)位碼第1位和區(qū)位碼第2位作為字節(jié)數(shù)組第一個(gè)元素
區(qū)位碼第3位和區(qū)位碼第4位作為字節(jié)數(shù)組第二個(gè)元素
*/
for (int i = 0; i < strlength; i++)
{
//區(qū)位碼第1位
int r1 = rnd.Next(11, 14);
string str_r1 = r[r1].Trim();
//區(qū)位碼第2位
rnd = new Random(r1 * unchecked((int)DateTime.Now.Ticks) + i);//更換隨機(jī)數(shù)發(fā)生器的種子避免產(chǎn)生重復(fù)值
int r2;
if (r1 == 13)
r2 = rnd.Next(0, 7);
else
r2 = rnd.Next(0, 16);
string str_r2 = r[r2].Trim();
//區(qū)位碼第3位
rnd = new Random(r2 * unchecked((int)DateTime.Now.Ticks) + i);
int r3 = rnd.Next(10, 16);
string str_r3 = r[r3].Trim();
//區(qū)位碼第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 = r[r4].Trim();
//定義兩個(gè)字節(jié)變量存儲(chǔ)產(chǎn)生的隨機(jī)漢字區(qū)位碼
byte byte1 = Convert.ToByte(str_r1 + str_r2, 16);
byte byte2 = Convert.ToByte(str_r3 + str_r4, 16);
//將兩個(gè)字節(jié)變量存儲(chǔ)在字節(jié)數(shù)組中
byte[] str_r = new byte[] { byte1, byte2 };
//將產(chǎn)生的一個(gè)漢字的字節(jié)數(shù)組放入object數(shù)組中
bytes.SetValue(str_r, i);
}
//獲取GB2312編碼頁(yè)(表)
Encoding gb = Encoding.GetEncoding("gb2312");
//根據(jù)漢字編碼的字節(jié)數(shù)組解碼出中文漢字
string str1 = gb.GetString((byte[])Convert.ChangeType(bytes[0], typeof(byte[])));
string str2 = gb.GetString((byte[])Convert.ChangeType(bytes[1], typeof(byte[])));
string str3 = gb.GetString((byte[])Convert.ChangeType(bytes[2], typeof(byte[])));
string str4 = gb.GetString((byte[])Convert.ChangeType(bytes[3], typeof(byte[])));
string txt = str1 + str2 + str3 + str4;
return txt;
}
以上代碼實(shí)現(xiàn)隨機(jī)生成漢字,以上有一點(diǎn)需要說(shuō)明代碼在中文版windows下才能運(yùn)行,因?yàn)閹в術(shù)b的字符集,如果你的電腦是其他語(yǔ)言的操作系統(tǒng),需要安裝GB字符集。
相關(guān)文章
C#實(shí)現(xiàn)提取Word中插入的多媒體文件(視頻,音頻)
在Word中可將文件通過(guò)OLE對(duì)象嵌入的方式插入到文檔,包括Word、excel、PDF、PPT、圖片、宏文件、文件包等在內(nèi)的多種文件類(lèi)型。本文將利用C#實(shí)現(xiàn)提取插入在Word文件中的這些多媒體文件,感興趣的可以了解一下2022-02-02C#實(shí)現(xiàn)驗(yàn)證身份證是否合法的方法
這篇文章主要介紹了C#實(shí)現(xiàn)驗(yàn)證身份證是否合法的方法,實(shí)例分析了通過(guò)自定義函數(shù)實(shí)現(xiàn)針對(duì)身份證合法性驗(yàn)證的技巧,非常具有實(shí)用價(jià)值,需要的朋友可以參考下2015-03-03c#實(shí)現(xiàn)獲取字符串陣列中元素最長(zhǎng)或最短的長(zhǎng)度
下面小編就為大家分享一篇c#實(shí)現(xiàn)獲取字符串陣列中元素最長(zhǎng)或最短的長(zhǎng)度方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助2017-12-12WPF使用DrawingContext實(shí)現(xiàn)二維繪圖
這篇文章介紹了WPF使用DrawingContext實(shí)現(xiàn)二維繪圖的方法,對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-06-06C#中DateTime.Compare()比較時(shí)間大小
本文主要介紹了C#中DateTime.Compare()比較時(shí)間大小,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-04-04