C#讀取QQ純真IP數(shù)據(jù)庫(kù)QQWry.Dat的代碼
更新時(shí)間:2007年03月09日 00:00:00 作者:
QQ純真IP庫(kù)算是IP地址收集較為全的一個(gè)IP庫(kù),對(duì)于IP查詢(xún)來(lái)說(shuō)這個(gè)是不錯(cuò)的選擇。下面是如何查詢(xún)純真IP庫(kù)的一個(gè)類(lèi),使用C#代碼。
純真版QQ IP數(shù)據(jù)庫(kù)
復(fù)制代碼 代碼如下:
using System;
using System.Collections.Generic;
using System.Text;
using System.IO;
using System.Web;
using System.Configuration;
namespace BLL
{
public class IPLocationSearch
{
private static readonly QQWry qq = new QQWry(ConfigurationManager.AppSettings["ip"] + "qqwry.dat");
public static IPLocation GetIPLocation(string ip)
{
return qq.SearchIPLocation(ip);
}
}
/*
使用方法:
例子:
BDQQ.Data.QQWry qq=new BDQQ.Data.QQWry("d:\\QQWry.Dat");
BDQQ.Data.IPLocation ip=qq.SearchIPLocation("127.0.0.1");//這里添寫(xiě)IP地址
Console.WriteLine(ip.country);//國(guó)家
Console.WriteLine(ip.area);//地區(qū)
*/
//以下是類(lèi)文件
//根據(jù)LumaQQ改寫(xiě)而成.
/**/
///<summary>
/// QQWry 的摘要說(shuō)明。
///</summary>
public class QQWry
{
//第一種模式
#region 第一種模式
/**/
///<summary>
///第一種模式
///</summary>
#endregion
private const byte REDIRECT_MODE_1 = 0x01;
//第二種模式
#region 第二種模式
/**/
///<summary>
///第二種模式
///</summary>
#endregion
private const byte REDIRECT_MODE_2 = 0x02;
//每條記錄長(zhǎng)度
#region 每條記錄長(zhǎng)度
/**/
///<summary>
///每條記錄長(zhǎng)度
///</summary>
#endregion
private const int IP_RECORD_LENGTH = 7;
//數(shù)據(jù)庫(kù)文件
#region 數(shù)據(jù)庫(kù)文件
/**/
///<summary>
///文件對(duì)象
///</summary>
#endregion
private FileStream ipFile;
private const string unCountry = "未知國(guó)家";
private const string unArea = "未知地區(qū)";
//索引開(kāi)始位置
#region 索引開(kāi)始位置
/**/
///<summary>
///索引開(kāi)始位置
///</summary>
#endregion
private long ipBegin;
//索引結(jié)束位置
#region 索引結(jié)束位置
/**/
///<summary>
///索引結(jié)束位置
///</summary>
#endregion
private long ipEnd;
//IP地址對(duì)象
#region IP地址對(duì)象
/**/
///<summary>
/// IP對(duì)象
///</summary>
#endregion
private IPLocation loc;
//存儲(chǔ)文本內(nèi)容
#region 存儲(chǔ)文本內(nèi)容
/**/
///<summary>
///存儲(chǔ)文本內(nèi)容
///</summary>
#endregion
private byte[] buf;
//存儲(chǔ)3字節(jié)
#region 存儲(chǔ)3字節(jié)
/**/
///<summary>
///存儲(chǔ)3字節(jié)
///</summary>
#endregion
private byte[] b3;
//存儲(chǔ)4字節(jié)
#region 存儲(chǔ)4字節(jié)
/**/
///<summary>
///存儲(chǔ)4字節(jié)IP地址
///</summary>
#endregion
private byte[] b4;
//構(gòu)造函數(shù)
#region 構(gòu)造函數(shù)
/**/
///<summary>
///構(gòu)造函數(shù)
///</summary>
///<param name="ipfile">IP數(shù)據(jù)庫(kù)文件絕對(duì)路徑</param>
#endregion
public QQWry(string ipfile)
{
buf = new byte[100];
b3 = new byte[3];
b4 = new byte[4];
try
{
ipFile = new FileStream(ipfile, FileMode.Open);
}
catch (Exception ex)
{
throw new Exception(ex.Message);
}
ipBegin = readLong4(0);
ipEnd = readLong4(4);
loc = new IPLocation();
}
//根據(jù)IP地址搜索
#region 根據(jù)IP地址搜索
/**/
///<summary>
///搜索IP地址搜索
///</summary>
///<param name="ip"></param>
///<returns></returns>
#endregion
public IPLocation SearchIPLocation(string ip)
{
//將字符IP轉(zhuǎn)換為字節(jié)
string[] ipSp = ip.Split('.');
if (ipSp.Length != 4)
{
throw new ArgumentOutOfRangeException("不是合法的IP地址!");
}
byte[] IP = new byte[4];
for (int i = 0; i < IP.Length; i++)
{
IP[i] = (byte)(Int32.Parse(ipSp[i]) & 0xFF);
}
IPLocation local = null;
long offset = locateIP(IP);
if (offset != -1)
{
local = getIPLocation(offset);
}
if (local == null)
{
local = new IPLocation();
local.area = unArea;
local.country = unCountry;
}
return local;
}
//取得具體信息
#region 取得具體信息
/**/
///<summary>
///取得具體信息
///</summary>
///<param name="offset"></param>
///<returns></returns>
#endregion
private IPLocation getIPLocation(long offset)
{
ipFile.Position = offset + 4;
//讀取第一個(gè)字節(jié)判斷是否是標(biāo)志字節(jié)
byte one = (byte)ipFile.ReadByte();
if (one == REDIRECT_MODE_1)
{
//第一種模式
//讀取國(guó)家偏移
long countryOffset = readLong3();
//轉(zhuǎn)至偏移處
ipFile.Position = countryOffset;
//再次檢查標(biāo)志字節(jié)
byte b = (byte)ipFile.ReadByte();
if (b == REDIRECT_MODE_2)
{
loc.country = readString(readLong3());
ipFile.Position = countryOffset + 4;
}
else
loc.country = readString(countryOffset);
//讀取地區(qū)標(biāo)志
loc.area = readArea(ipFile.Position);
}
else if (one == REDIRECT_MODE_2)
{
//第二種模式
loc.country = readString(readLong3());
loc.area = readArea(offset + 8);
}
else
{
//普通模式
loc.country = readString(--ipFile.Position);
loc.area = readString(ipFile.Position);
}
return loc;
}
//取得地區(qū)信息
#region 取得地區(qū)信息
/**/
///<summary>
///讀取地區(qū)名稱(chēng)
///</summary>
///<param name="offset"></param>
///<returns></returns>
#endregion
private string readArea(long offset)
{
ipFile.Position = offset;
byte one = (byte)ipFile.ReadByte();
if (one == REDIRECT_MODE_1 || one == REDIRECT_MODE_2)
{
long areaOffset = readLong3(offset + 1);
if (areaOffset == 0)
return unArea;
else
{
return readString(areaOffset);
}
}
else
{
return readString(offset);
}
}
//讀取字符串
#region 讀取字符串
/**/
///<summary>
///讀取字符串
///</summary>
///<param name="offset"></param>
///<returns></returns>
#endregion
private string readString(long offset)
{
ipFile.Position = offset;
int i = 0;
for (i = 0, buf[i] = (byte)ipFile.ReadByte(); buf[i] != (byte)(0); buf[++i] = (byte)ipFile.ReadByte()) ;
if (i > 0)
return Encoding.Default.GetString(buf, 0, i);
else
return "";
}
//查找IP地址所在的絕對(duì)偏移量
#region 查找IP地址所在的絕對(duì)偏移量
/**/
///<summary>
///查找IP地址所在的絕對(duì)偏移量
///</summary>
///<param name="ip"></param>
///<returns></returns>
#endregion
private long locateIP(byte[] ip)
{
long m = 0;
int r;
//比較第一個(gè)IP項(xiàng)
readIP(ipBegin, b4);
r = compareIP(ip, b4);
if (r == 0)
return ipBegin;
else if (r < 0)
return -1;
//開(kāi)始二分搜索
for (long i = ipBegin, j = ipEnd; i < j; )
{
m = this.getMiddleOffset(i, j);
readIP(m, b4);
r = compareIP(ip, b4);
if (r > 0)
i = m;
else if (r < 0)
{
if (m == j)
{
j -= IP_RECORD_LENGTH;
m = j;
}
else
{
j = m;
}
}
else
return readLong3(m + 4);
}
m = readLong3(m + 4);
readIP(m, b4);
r = compareIP(ip, b4);
if (r <= 0)
return m;
else
return -1;
}
//讀出4字節(jié)的IP地址
#region 讀出4字節(jié)的IP地址
/**/
///<summary>
///從當(dāng)前位置讀取四字節(jié),此四字節(jié)是IP地址
///</summary>
///<param name="offset"></param>
///<param name="ip"></param>
#endregion
private void readIP(long offset, byte[] ip)
{
ipFile.Position = offset;
ipFile.Read(ip, 0, ip.Length);
byte tmp = ip[0];
ip[0] = ip[3];
ip[3] = tmp;
tmp = ip[1];
ip[1] = ip[2];
ip[2] = tmp;
}
//比較IP地址是否相同
#region 比較IP地址是否相同
/**/
///<summary>
///比較IP地址是否相同
///</summary>
///<param name="ip"></param>
///<param name="beginIP"></param>
///<returns>0:相等,1:ip大于beginIP,-1:小于</returns>
#endregion
private int compareIP(byte[] ip, byte[] beginIP)
{
for (int i = 0; i < 4; i++)
{
int r = compareByte(ip[i], beginIP[i]);
if (r != 0)
return r;
}
return 0;
}
//比較兩個(gè)字節(jié)是否相等
#region 比較兩個(gè)字節(jié)是否相等
/**/
///<summary>
///比較兩個(gè)字節(jié)是否相等
///</summary>
///<param name="bsrc"></param>
///<param name="bdst"></param>
///<returns></returns>
#endregion
private int compareByte(byte bsrc, byte bdst)
{
if ((bsrc & 0xFF) > (bdst & 0xFF))
return 1;
else if ((bsrc ^ bdst) == 0)
return 0;
else
return -1;
}
//根據(jù)當(dāng)前位置讀取4字節(jié)
#region 根據(jù)當(dāng)前位置讀取4字節(jié)
/**/
///<summary>
///從當(dāng)前位置讀取4字節(jié),轉(zhuǎn)換為長(zhǎng)整型
///</summary>
///<param name="offset"></param>
///<returns></returns>
#endregion
private long readLong4(long offset)
{
long ret = 0;
ipFile.Position = offset;
ret |= (ipFile.ReadByte() & 0xFF);
ret |= ((ipFile.ReadByte() << 8) & 0xFF00);
ret |= ((ipFile.ReadByte() << 16) & 0xFF0000);
ret |= ((ipFile.ReadByte() << 24) & 0xFF000000);
return ret;
}
//根據(jù)當(dāng)前位置,讀取3字節(jié)
#region 根據(jù)當(dāng)前位置,讀取3字節(jié)
/**/
///<summary>
///根據(jù)當(dāng)前位置,讀取3字節(jié)
///</summary>
///<param name="offset"></param>
///<returns></returns>
#endregion
private long readLong3(long offset)
{
long ret = 0;
ipFile.Position = offset;
ret |= (ipFile.ReadByte() & 0xFF);
ret |= ((ipFile.ReadByte() << 8) & 0xFF00);
ret |= ((ipFile.ReadByte() << 16) & 0xFF0000);
return ret;
}
//從當(dāng)前位置讀取3字節(jié)
#region 從當(dāng)前位置讀取3字節(jié)
/**/
///<summary>
///從當(dāng)前位置讀取3字節(jié)
///</summary>
///<returns></returns>
#endregion
private long readLong3()
{
long ret = 0;
ret |= (ipFile.ReadByte() & 0xFF);
ret |= ((ipFile.ReadByte() << 8) & 0xFF00);
ret |= ((ipFile.ReadByte() << 16) & 0xFF0000);
return ret;
}
//取得begin和end之間的偏移量
#region 取得begin和end之間的偏移量
/**/
///<summary>
///取得begin和end中間的偏移
///</summary>
///<param name="begin"></param>
///<param name="end"></param>
///<returns></returns>
#endregion
private long getMiddleOffset(long begin, long end)
{
long records = (end - begin) / IP_RECORD_LENGTH;
records >>= 1;
if (records == 0)
records = 1;
return begin + records * IP_RECORD_LENGTH;
}
} //class QQWry
public class IPLocation
{
public String country;
public String area;
public IPLocation()
{
country = area = "";
}
public IPLocation getCopy()
{
IPLocation ret = new IPLocation();
ret.country = country;
ret.area = area;
return ret;
}
}
}
相關(guān)文章
unity通過(guò)Mesh網(wǎng)格繪制圖形球體
這篇文章主要為大家詳細(xì)介紹了unity通過(guò)Mesh網(wǎng)格繪制圖形球體,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-11-11
C#使用yield關(guān)鍵字讓自定義集合實(shí)現(xiàn)foreach遍歷的方法
這篇文章主要介紹了C#使用yield關(guān)鍵字讓自定義集合實(shí)現(xiàn)foreach遍歷的方法,需要的朋友可以參考下2014-08-08
C#實(shí)現(xiàn)數(shù)組元素的數(shù)據(jù)類(lèi)型轉(zhuǎn)換方法詳解
這篇文章主要為大家介紹了C#中一維數(shù)組如何快速實(shí)現(xiàn)數(shù)組元素的數(shù)據(jù)類(lèi)型的轉(zhuǎn)換,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解一下2022-04-04
C#監(jiān)控文件夾并自動(dòng)給圖片文件打水印的方法
這篇文章主要介紹了C#監(jiān)控文件夾并自動(dòng)給圖片文件打水印的方法,涉及C#針對(duì)文件夾及圖片操作的相關(guān)技巧,非常具有實(shí)用價(jià)值,需要的朋友可以參考下2015-05-05
Silverlight將圖片轉(zhuǎn)換為byte的實(shí)現(xiàn)代碼
這篇文章主要介紹了Silverlight將圖片轉(zhuǎn)換為byte的實(shí)現(xiàn)代碼,需要的朋友可以參考下2015-11-11
C# HTTP認(rèn)證方式詳解與代碼實(shí)現(xiàn)
在C#中,HTTP認(rèn)證是客戶(hù)端與服務(wù)器之間進(jìn)行身份驗(yàn)證的一種機(jī)制,常見(jiàn)的HTTP認(rèn)證方式包括:Basic認(rèn)證、Digest認(rèn)證、OAuth、Bearer Token等,下面我們將從工作原理、優(yōu)缺點(diǎn)對(duì)比、代碼實(shí)現(xiàn)、案例實(shí)戰(zhàn)四個(gè)方面詳細(xì)介紹這些認(rèn)證方式,需要的朋友可以參考下2025-03-03
WPF自定義實(shí)現(xiàn)雷達(dá)圖控件的示例詳解
雷達(dá)圖用于表示不同內(nèi)容的占比關(guān)系,在項(xiàng)目中有廣泛的應(yīng)用,但是目前未曾有封裝良好的雷達(dá)圖控件,所以本文分享了如何封裝一個(gè)通用的雷達(dá)圖控件,希望對(duì)大家有所幫助2023-08-08
C#使用Lazy<T>實(shí)現(xiàn)對(duì)客戶(hù)訂單的延遲加載
這篇文章介紹了C#使用Lazy<T>實(shí)現(xiàn)對(duì)客戶(hù)訂單延遲加載的方法,文中通過(guò)示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-08-08

