C#讀取QQ純真IP資料庫QQWry.Dat的代碼

來源:互聯網
上載者:User

純真版QQ IP資料庫

複製代碼 代碼如下: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");//這裡添寫IP地址
Console.WriteLine(ip.country);//國家
Console.WriteLine(ip.area);//地區
*/

//以下是類檔案
//根據LumaQQ改寫而成.

/**/
///<summary>
/// QQWry 的摘要說明。
///</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;

//每條記錄長度
#region 每條記錄長度
/**/
///<summary>
///每條記錄長度
///</summary>
#endregion
private const int IP_RECORD_LENGTH = 7;

//資料庫檔案
#region 資料庫檔案
/**/
///<summary>
///檔案對象
///</summary>
#endregion
private FileStream ipFile;

private const string unCountry = "未知國家";
private const string unArea = "未知地區";

//索引開始位置
#region 索引開始位置
/**/
///<summary>
///索引開始位置
///</summary>
#endregion
private long ipBegin;

//索引結束位置
#region 索引結束位置
/**/
///<summary>
///索引結束位置
///</summary>
#endregion
private long ipEnd;

//IP地址對象
#region IP地址對象
/**/
///<summary>
/// IP對象
///</summary>
#endregion
private IPLocation loc;

//儲存常值內容
#region 儲存常值內容
/**/
///<summary>
///儲存常值內容
///</summary>
#endregion
private byte[] buf;

//儲存3位元組
#region 儲存3位元組
/**/
///<summary>
///儲存3位元組
///</summary>
#endregion
private byte[] b3;

//儲存4位元組
#region 儲存4位元組
/**/
///<summary>
///儲存4位元組IP地址
///</summary>
#endregion
private byte[] b4;

//建構函式
#region 建構函式
/**/
///<summary>
///建構函式
///</summary>
///<param name="ipfile">IP資料庫檔案絕對路徑</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();
}

//根據IP地址搜尋
#region 根據IP地址搜尋
/**/
///<summary>
///搜尋IP地址搜尋
///</summary>
///<param name="ip"></param>
///<returns></returns>
#endregion
public IPLocation SearchIPLocation(string ip)
{
//將字元IP轉換為位元組
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;
//讀取第一個位元組判斷是否是標誌位元組
byte one = (byte)ipFile.ReadByte();
if (one == REDIRECT_MODE_1)
{
//第一種模式
//讀取國家位移
long countryOffset = readLong3();
//轉至位移處
ipFile.Position = countryOffset;
//再次檢查標誌位元組
byte b = (byte)ipFile.ReadByte();
if (b == REDIRECT_MODE_2)
{
loc.country = readString(readLong3());
ipFile.Position = countryOffset + 4;
}
else
loc.country = readString(countryOffset);

//讀取地區標誌
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;
}

//取得地區資訊
#region 取得地區資訊
/**/
///<summary>
///讀取地區名稱
///</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地址所在的絕對位移量
#region 尋找IP地址所在的絕對位移量
/**/
///<summary>
///尋找IP地址所在的絕對位移量
///</summary>
///<param name="ip"></param>
///<returns></returns>
#endregion
private long locateIP(byte[] ip)
{
long m = 0;
int r;

//比較第一個IP項
readIP(ipBegin, b4);
r = compareIP(ip, b4);
if (r == 0)
return ipBegin;
else if (r < 0)
return -1;
//開始二分搜尋
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位元組的IP地址
#region 讀出4位元組的IP地址
/**/
///<summary>
///從當前位置讀取四位元組,此四位元組是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;
}

//比較兩個位元組是否相等
#region 比較兩個位元組是否相等
/**/
///<summary>
///比較兩個位元組是否相等
///</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;
}

//根據當前位置讀取4位元組
#region 根據當前位置讀取4位元組
/**/
///<summary>
///從當前位置讀取4位元組,轉換為長整型
///</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;
}

//根據當前位置,讀取3位元組
#region 根據當前位置,讀取3位元組
/**/
///<summary>
///根據當前位置,讀取3位元組
///</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;
}

//從當前位置讀取3位元組
#region 從當前位置讀取3位元組
/**/
///<summary>
///從當前位置讀取3位元組
///</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;
}
}
}

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.