.NET CORE下读取IP纯真库

用以前放在牛腩公用类库里的IPSearch就行了,编译没有错误,不过直接用发现中文乱码,一通乱改才发现问题,看最后的ReadString方法 ,

 

 

using System;

using System.Collections.Generic;

using System.Text;

using System.IO;

namespace Niunan.WebAssign.Util

{

///<summary>

/// 提供从纯真IP数据库搜索IP信息的方法;

/// 感谢LumaQQ提供纯真IP数据库格式文档;

/// ----HeDaode 2007-12-28 四川教育学院

///</summary>

internal class IPSearch

{

FileStream ipFile;

long ip;

string ipfilePath;

 

///<summary>

/// 构造函数

///</summary>

///<param name="ipfilePath">纯真IP数据库路径</param>

public IPSearch(string ipfilePath)

{

this.ipfilePath = ipfilePath;

}

 

///<summary>

/// 地理位置,包括国家和地区

///</summary>

public struct IPLocation

{

public string country, area;

}

///<summary>

/// 获取指定IP所在地理位置

///</summary>

///<param name="strIP">要查询的IP地址</param>

///<returns></returns>

public IPLocation GetIPLocation(string strIP)

{

ip = IPToLong(strIP);

ipFile = new FileStream(ipfilePath, FileMode.Open, FileAccess.Read);

long[] ipArray = BlockToArray(ReadIPBlock());

long offset = SearchIP(ipArray, 0, ipArray.Length - 1) * 7 + 4;

ipFile.Position += offset;//跳过起始IP

ipFile.Position = ReadLongX(3) + 4;//跳过结束IP

 

IPLocation loc = new IPLocation();

int flag = ipFile.ReadByte();//读取标志

if (flag == 1)//表示国家和地区被转向

{

ipFile.Position = ReadLongX(3);

flag = ipFile.ReadByte();//再读标志

}

long countryOffset = ipFile.Position;

loc.country = ReadString(flag);

 

if (flag == 2)

{

ipFile.Position = countryOffset + 3;

}

flag = ipFile.ReadByte();

loc.area = ReadString(flag);

 

ipFile.Close();

ipFile = null;

return loc;

}

///<summary>

/// 将字符串形式的IP转换位long

///</summary>

///<param name="strIP"></param>

///<returns></returns>

public long IPToLong(string strIP)

{

byte[] ip_bytes = new byte[8];

string[] strArr = strIP.Split(new char[] { '.' });

for (int i = 0; i < 4; i++)

{

ip_bytes[i] = byte.Parse(strArr[3 - i]);

}

return BitConverter.ToInt64(ip_bytes, 0);

}

///<summary>

/// 将索引区字节块中的起始IP转换成Long数组

///</summary>

///<param name="ipBlock"></param>

long[] BlockToArray(byte[] ipBlock)

{

long[] ipArray = new long[ipBlock.Length / 7];

int ipIndex = 0;

byte[] temp = new byte[8];

for (int i = 0; i < ipBlock.Length; i += 7)

{

Array.Copy(ipBlock, i, temp, 0, 4);

ipArray[ipIndex] = BitConverter.ToInt64(temp, 0);

ipIndex++;

}

return ipArray;

}

///<summary>

/// 从IP数组中搜索指定IP并返回其索引

///</summary>

///<param name="ipArray">IP数组</param>

///<param name="start">指定搜索的起始位置</param>

///<param name="end">指定搜索的结束位置</param>

///<returns></returns>

int SearchIP(long[] ipArray, int start, int end)

{

int middle = (start + end) / 2;

if (middle == start)

return middle;

else if (ip < ipArray[middle])

return SearchIP(ipArray, start, middle);

else

return SearchIP(ipArray, middle, end);

}

///<summary>

/// 读取IP文件中索引区块

///</summary>

///<returns></returns>

byte[] ReadIPBlock()

{

long startPosition = ReadLongX(4);

long endPosition = ReadLongX(4);

long count = (endPosition - startPosition) / 7 + 1;//总记录数

ipFile.Position = startPosition;

byte[] ipBlock = new byte[count * 7];

ipFile.Read(ipBlock, 0, ipBlock.Length);

ipFile.Position = startPosition;

return ipBlock;

}

///<summary>

/// 从IP文件中读取指定字节并转换位long

///</summary>

///<param name="bytesCount">需要转换的字节数,主意不要超过8字节</param>

///<returns></returns>

long ReadLongX(int bytesCount)

{

byte[] _bytes = new byte[8];

ipFile.Read(_bytes, 0, bytesCount);

return BitConverter.ToInt64(_bytes, 0);

}

///<summary>

/// 从IP文件中读取字符串

///</summary>

///<param name="flag">转向标志</param>

///<returns></returns>

string ReadString(int flag)

{

if (flag == 1 || flag == 2)//转向标志

ipFile.Position = ReadLongX(3);

else

ipFile.Position -= 1;

 

List<byte> list = new List<byte>();

byte b = (byte)ipFile.ReadByte();

while (b > 0)

{

list.Add(b);

b = (byte)ipFile.ReadByte();

}

 

return Encoding.GetEncoding("GB2312").GetString(list.ToArray());//.net core下用这个,要不然会出中文乱码

//return Encoding.Default.GetString(list.ToArray()); //.netframework下用这个

}

}

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
本模块代码是针对在 2011 年在 CSDN 论坛个发布的“最新 NET 读取纯真IP数据代码(C#)”源码,做了一次升级,这次升级不是简单的修补,是本人精心的重写,现在只需要 5 分哦,您值得拥有!该源代码不同于网上的代码,网上代码基本可分为两大类,第一类直接使用文件流,通过移动文件流指针(即更改 Stream.Position 属性值)搜索 IP 地址对应的信息,此类代码问题是:其一移动文件指针效率是比较低的(给 Position 赋值),多线程并发时,会重复打开多个文件效率更加底下;第二类是把文件直接加载内存中,通过这种缓冲,速度是提升了,但并没有为多线程环境优化,多线程并发时(如:Web 中每位访客,都是一根线程),意味会重复的读取文件,重复的创建缓存,浪费内存空间。 该源代码特点是考虑到了多线程应用环境(如:Web 每个会话,都是一根线程),设计了缓存对象 QQWryCache 用于管理缓存,用 QQCacheStream 流读取缓存数据。在多线程应用环境中,假设 10 根线程访问同一个纯真 IP 数据时,只会开辟 1 份缓存,给多根线程共享,避免了不必要的内存浪费。 注1:本模块代码,保证所有静态方法都是线程安全的,但不保证所有实例方法都是线程安全的。 注2:每根线程访问缓存时,请通过 QQWryCache.GetCache 静态方法获取缓存对象。 注3:多根线程获取到的缓存对象,通常都是同一对象,该对象已经考虑了线程同步,不必担心线程安全问题。 /* >>> 使用完全缓存(缓存整个文件,约 8.8MB),调用方法如下: */ QQWryCache cache = QQWryCache.GetCache("qqwry.dat", true); Stream stream = cache.GetCacheStream(); QQWrySearcher searcher = new QQwryScanner(stream); QQWryLocation location = searcher.Query("IP 地址"); Console.WritleLine("Country = {0}, Location = {1}", location.Country, location.Location); /* 完全缓冲, * 缓存一旦初始化完毕,就会自动关闭文件, * 所以不再依赖于文件,因此可以不用关闭缓冲流, * 下面调用 Close 方法,其实没有实际意义,但也不会引发异常。 */ stream.Close(); /* >>> 使用索引缓存(仅缓存索引部分,约 3MB),调用方法如下: <<>> 直接使用文件流(不使用缓存),调用方法如下: <<>> 遍历 IP 数据。 <<< */ QQWryCache cache = QQWryCache.GetCache("qqwry.dat", true); Stream stream = cache.GetCacheStream(); QQWrySearcher searcher = new QQWrySearcher(stream); // 用 for 循环遍历 for(int i = 0; i < searcher.Count; i++) { QQWryIpLocation item = searcher[i]; Console.WritleLine("Country = {0}, Location = {1}", location.Country, location.Location); } // 用 foreach 循环遍历 foreach(QQWryIpLocation item in searcher) { QQWryIpLocation item = searcher[i]; Console.WritleLine("Country = {0}, Location = {1}", location.Country, location.Location); }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值