#include "stdafx.h"
#include <stdio.h>
#include <malloc.h>
#include "string.h"FILE *fpi; /*打开QQWry.dat的文件指针*/
#define QQWRY "QQWry.dat"
#define REDIRECT_MODE_1 0x01
#define REDIRECT_MODE_2 0x02
#define MAXBUF 255/*unsigned long getValue( 获取文件中指定的16进制串的值,并返回
FILE *fp, 指定文件指针
unsigned long start, 指定文件偏移量
int length) 获取的16进制字符个数/长度
*/
unsigned long getValue(FILE *fp, unsigned long start, int length=1)
{
unsigned long variable=0;
long i;
long *val= new long[length]; fseek(fp,start,SEEK_SET);
for(i=0;i<length;i++)
{
/*过滤高位,一次读取一个字符*/
val[i]=fgetc(fp)&0x000000FF;
}
for(i=length-1;i>=0;i--)
{
/*因为读取多个16进制字符,叠加*/
variable=variable*0x100+val[i];
}
delete []val;
return variable;
};/*void getHead( 读取索引部分的范围(在文件头中,最先的2个8位16进制)
FILE *fp, 指定文件指针
unsigned long *start, 文件偏移量,索引的起止位置
unsigned long *end) 文件偏移量,索引的结束位置
*/
void getHead(FILE *fp,unsigned long *start,unsigned long *end)
{
/*索引的起止位置的文件偏移量,存储在文件头中的前8个16进制中
设置偏移量为0,读取4个字符*/
*start=getValue(fp,0L,4);
/*索引的结束位置的文件偏移量,存储在文件头中的第8个到第15个的16进制中
设置偏移量为4个字符,再读取4个字符*/
*end=getValue(fp,4L,4);
};/*判断一个字符是否为数字字符,
如果是,返回0
如果不是,返回1*/
int beNumber(char c)
{
if(c>='0'&&c<='9')
return 0;
else
return 1;
};/*函数的参数是一个存储着IP地址的字符串首地址
返回该IP的16进制代码
如果输入的IP地址有错误,函数将返回0*/
unsigned long getIP(char *ip_addr)
{
unsigned long ip=0;
int i,j=0;
/*依次读取字符串中的各个字符*/
for(i=0;i<strlen(ip_addr);i++)
{
/*如果是IP地址间隔的‘.’符号
把当前读取到的IP字段的值,存入ip变量中
(注意,ip为叠加时,乘以16进制的0x100)
并清除临时变量的值*/
if(*(ip_addr+i)=='.')
{
ip=ip*0x100+j;
j=0;
}
/*往临时变量中写入当前读取到的IP字段中的字符值
叠加乘以10,因为输入的IP地址是10进制*/
else
{
/*判断,如果输入的IP地址不规范,不是10进制字符
函数将返回0*/
if(beNumber(*(ip_addr+i))==0)
j=j*10+*(ip_addr+i)-'0';
else
return 0;
}
}
/*IP字段有4个,但是‘.’只有3个,叠加第四个字段值*/
ip=ip*0x100+j;
return ip;
};/*unsigned long searchIP( 搜索指定IP在索引区的位置,采用二分查找法;
返回IP在索引区域的文件偏移量
一条索引记录的结果是,前4个16进制表示起始IP地址
后面3个16进制,表示该起始IP在IP信息段中的位置,文件偏移量
FILE *fp,
unsigned long index_start, 索引起始位置的文件偏移量
unsigned long index_end, 索引结束位置的文件偏移量
unsigned long ip) 关键字,要索引的IP
*/
unsigned long searchIP(FILE *fp, unsigned long index_start, unsigned long index_end, unsigned long ip)
{
unsigned long index_current,index_top,index_bottom;
unsigned long record;
index_bottom=index_start;
index_top=index_end;
/*此处的7,是因为一条索引记录的长度是7*/
index_current=((index_top-index_bottom)/7/2)*7+index_bottom;
/*二分查找法*/
do{
record=getValue(fp,index_current,4);
if(record>ip)
{
index_top=index_current;
index_current=((index_top-index_bottom)/14)*7+index_bottom;
}
else
{
index_bottom=index_current;
index_current=((index_top-index_bottom)/14)*7+index_bottom;
}
}while(index_bottom<index_current);
/*返回关键字IP在索引区域的文件偏移量*/
return index_current;
};/*int getString( 获取文件中指定的字符串,返回字符串长度
FILE *fp, 指定文件指针
unsigned long start, 指定文件偏移量
char **string) 用来存放将读取字符串的字符串空间的首地址
*/
int getString(FILE *fp, unsigned long start, char **string)
{
unsigned long i=0;
char val;
fseek(fp,start,SEEK_SET);
/*读取字符串,直到遇到0x00为止*/
do
{
val=fgetc(fp);
/*依次放入用来存储的字符串空间中*/
*(*string+i)=val;
i++;
}while(val!=0x00);
/*返回字符串长度*/
return i;
};
#include <stdio.h>
#include <malloc.h>
#include "string.h"FILE *fpi; /*打开QQWry.dat的文件指针*/
#define QQWRY "QQWry.dat"
#define REDIRECT_MODE_1 0x01
#define REDIRECT_MODE_2 0x02
#define MAXBUF 255/*unsigned long getValue( 获取文件中指定的16进制串的值,并返回
FILE *fp, 指定文件指针
unsigned long start, 指定文件偏移量
int length) 获取的16进制字符个数/长度
*/
unsigned long getValue(FILE *fp, unsigned long start, int length=1)
{
unsigned long variable=0;
long i;
long *val= new long[length]; fseek(fp,start,SEEK_SET);
for(i=0;i<length;i++)
{
/*过滤高位,一次读取一个字符*/
val[i]=fgetc(fp)&0x000000FF;
}
for(i=length-1;i>=0;i--)
{
/*因为读取多个16进制字符,叠加*/
variable=variable*0x100+val[i];
}
delete []val;
return variable;
};/*void getHead( 读取索引部分的范围(在文件头中,最先的2个8位16进制)
FILE *fp, 指定文件指针
unsigned long *start, 文件偏移量,索引的起止位置
unsigned long *end) 文件偏移量,索引的结束位置
*/
void getHead(FILE *fp,unsigned long *start,unsigned long *end)
{
/*索引的起止位置的文件偏移量,存储在文件头中的前8个16进制中
设置偏移量为0,读取4个字符*/
*start=getValue(fp,0L,4);
/*索引的结束位置的文件偏移量,存储在文件头中的第8个到第15个的16进制中
设置偏移量为4个字符,再读取4个字符*/
*end=getValue(fp,4L,4);
};/*判断一个字符是否为数字字符,
如果是,返回0
如果不是,返回1*/
int beNumber(char c)
{
if(c>='0'&&c<='9')
return 0;
else
return 1;
};/*函数的参数是一个存储着IP地址的字符串首地址
返回该IP的16进制代码
如果输入的IP地址有错误,函数将返回0*/
unsigned long getIP(char *ip_addr)
{
unsigned long ip=0;
int i,j=0;
/*依次读取字符串中的各个字符*/
for(i=0;i<strlen(ip_addr);i++)
{
/*如果是IP地址间隔的‘.’符号
把当前读取到的IP字段的值,存入ip变量中
(注意,ip为叠加时,乘以16进制的0x100)
并清除临时变量的值*/
if(*(ip_addr+i)=='.')
{
ip=ip*0x100+j;
j=0;
}
/*往临时变量中写入当前读取到的IP字段中的字符值
叠加乘以10,因为输入的IP地址是10进制*/
else
{
/*判断,如果输入的IP地址不规范,不是10进制字符
函数将返回0*/
if(beNumber(*(ip_addr+i))==0)
j=j*10+*(ip_addr+i)-'0';
else
return 0;
}
}
/*IP字段有4个,但是‘.’只有3个,叠加第四个字段值*/
ip=ip*0x100+j;
return ip;
};/*unsigned long searchIP( 搜索指定IP在索引区的位置,采用二分查找法;
返回IP在索引区域的文件偏移量
一条索引记录的结果是,前4个16进制表示起始IP地址
后面3个16进制,表示该起始IP在IP信息段中的位置,文件偏移量
FILE *fp,
unsigned long index_start, 索引起始位置的文件偏移量
unsigned long index_end, 索引结束位置的文件偏移量
unsigned long ip) 关键字,要索引的IP
*/
unsigned long searchIP(FILE *fp, unsigned long index_start, unsigned long index_end, unsigned long ip)
{
unsigned long index_current,index_top,index_bottom;
unsigned long record;
index_bottom=index_start;
index_top=index_end;
/*此处的7,是因为一条索引记录的长度是7*/
index_current=((index_top-index_bottom)/7/2)*7+index_bottom;
/*二分查找法*/
do{
record=getValue(fp,index_current,4);
if(record>ip)
{
index_top=index_current;
index_current=((index_top-index_bottom)/14)*7+index_bottom;
}
else
{
index_bottom=index_current;
index_current=((index_top-index_bottom)/14)*7+index_bottom;
}
}while(index_bottom<index_current);
/*返回关键字IP在索引区域的文件偏移量*/
return index_current;
};/*int getString( 获取文件中指定的字符串,返回字符串长度
FILE *fp, 指定文件指针
unsigned long start, 指定文件偏移量
char **string) 用来存放将读取字符串的字符串空间的首地址
*/
int getString(FILE *fp, unsigned long start, char **string)
{
unsigned long i=0;
char val;
fseek(fp,start,SEEK_SET);
/*读取字符串,直到遇到0x00为止*/
do
{
val=fgetc(fp);
/*依次放入用来存储的字符串空间中*/
*(*string+i)=val;
i++;
}while(val!=0x00);
/*返回字符串长度*/
return i;
};
FILE *fp, 指定文件指针
unsigned long start, 指定IP在索引中的文件偏移量
char **country, 用来存放国家位置的字符串空间的首地址
char **location) 用来存放地域位置的字符串空间的首地址
*/
void getAddress(FILE *fp, unsigned long start, char **country, char **location)
{
unsigned long redirect_address,counrty_address,location_address;
char val;
start+=4;
fseek(fp,start,SEEK_SET);
/*读取首地址的值*/
val=(fgetc(fp)&0x000000FF); if(val==REDIRECT_MODE_1)
{
/*重定向1类型的*/
redirect_address=getValue(fp,start+1,3);
fseek(fp,redirect_address,SEEK_SET);
/*混合类型,重定向1类型进入后遇到重定向2类型
读取重定向后的内容,并设置地域位置的文件偏移量*/
if((fgetc(fp)&0x000000FF)==REDIRECT_MODE_2)
{
counrty_address=getValue(fp,redirect_address+1,3);
location_address=redirect_address+4;
getString(fp,counrty_address,country);
}
/*读取重定向1后的内容,并设置地域位置的文件偏移量*/
else
{
counrty_address=redirect_address;
location_address=redirect_address+getString(fp,counrty_address,country);
}
}
/*重定向2类型的*/
else if(val==REDIRECT_MODE_2)
{
counrty_address=getValue(fp,start+1,3);
location_address=start+4;
getString(fp,counrty_address,country);
}
else
{
counrty_address=start;
location_address=counrty_address+getString(fp,counrty_address,country);
}
/*读取地域位置*/
fseek(fp,location_address,SEEK_SET);
if((fgetc(fp)&0x000000FF)==REDIRECT_MODE_2||(fgetc(fp)&0x000000FF)==REDIRECT_MODE_1)
{
location_address=getValue(fp,location_address+1,3);
}
getString(fp,location_address,location); return;
};/*打开QQWry.dat文件指针*/
extern "C" int __stdcall OpenQQwry()
{
/*打开QQWry.dat文件*/
if((fpi=fopen(QQWRY,"rb"))==NULL)
{
return 0;
}
return 1;
}/*关闭QQWry.dat文件指针*/
extern "C" int __stdcall CloseQQwry()
{
/*关闭文件指针*/
if(fpi==NULL)
{
return -1;
}
return fclose(fpi);
}/*函数的参数是一个字符串IP地址
返回该IP的国家地域位置*/
extern "C" char * __stdcall IPtoAddr(char *ip_addr)
{
unsigned long index_start; /*索引部分的起始位置的文件偏移量*/
unsigned long index_end; /*索引部分的结束位置的文件偏移量*/
unsigned long current; /*待搜索IP地址的索引条目的文件偏移量*/
char *country; /*国家位置*/
char *location; /*地域位置*/
country=(char*)malloc(MAXBUF);
location=(char*)malloc(MAXBUF);
/*打开QQWry.dat文件*/
if(fpi==NULL)
{
return "没找到纯真数据库";
}
else
/*显示QQWry.dat文件信息*/
getHead(fpi,&index_start,&index_end);
/*搜索IP*/
unsigned long ip;
ip=getIP(ip_addr);
if(ip==0)
{
return "输入的地址有错误";
}
/*搜索IP在索引区域的条目的偏移量*/
current=searchIP(fpi,index_start,index_end,ip);
/*获取该IP对因的国家地址和地域地址*/
getAddress(fpi,getValue(fpi,current+4,3),&country,&location);
char ret[1024];
sprintf(ret, "%s %s", country, location);
/*关闭文件指针,释放变量空间,结束程序*/
free(country);
free(location);
return (char *)&ret;
}
这是我写的静态库... 编译一切正常.. 但有个小BUG,
就是在你写入大量的代码以后,加入此库会查询IP..
unsigned long getValue(FILE *fp, unsigned long start, int length=1)
在这个函数里面的..
val[i]=fgetc(fp)&0x000000FF;
variable=variable*0x100+val[i];
这2句话会导致程序错误..! 错误原因不明..!
我估计是 fopen, fseek , fgetc 这个几个函数导致的.!
我求助高手解决这个问题..! 请修改后复全部完整无BUG代码张贴楼下.谢谢大家!
就是在你写入大量的代码以后,加入此库会查询IP..
unsigned long getValue(FILE *fp, unsigned long start, int length=1)
在这个函数里面的..
val[i]=fgetc(fp)&0x000000FF;
variable=variable*0x100+val[i];
这2句话会导致程序错误..! 错误原因不明..!
我估计是 fopen, fseek , fgetc 这个几个函数导致的.!
我求助高手解决这个问题..! 请修改后复全部完整无BUG代码张贴楼下.谢谢大家!
unsigned long getValue(FILE *fp, unsigned long start, int length=1)
{
if(length<1) return 0xFFFFFFFF;
......
}
实际的代码中应该做判断
比如unsigned long getValue(FILE *fp, unsigned long start, int length=1)
{
ASSERT(fp);//文件句柄有效
ASSERT(start >= 0);//起始地址有效
ASSERT(length > 0);//长度有效
unsigned long variable=0;
long i;
long *val= new long[length];
ASSERT(val);//分配内存成功 int ret = fseek(fp,start,SEEK_SET);
ASSERT(!ret);//文件指针移动正确 for(i=0;i <length;i++)
{
/*过滤高位,一次读取一个字符*/
int getc = fgetc(fp);
ASSERT(getc != EOF);//非文件结束 val[i]=get&0x000000FF;
}
for(i=length-1;i>=0;i--)
{
/*因为读取多个16进制字符,叠加*/
variable=variable*0x100+val[i];
}
delete []val;
return variable;
};
#include <stdio.h>
#include <assert.h>
#include <stdlib.h>
#include <malloc.h>
#include "string.h"
啊 怎么还有这样的错误啊.
{
/*过滤高位,一次读取一个字符*/
if(feof(fp) == 0)
val[i]=fgetc(fp)&0x000000FF;
}
fseek(fp,start,SEEK_SET);
是这个的错误./!
unsigned long getValue(FILE *fp, unsigned long start, int length=1)
{
unsigned long variable=0;
long i;
long *val= new long[length]; fseek(fp,start,SEEK_SET);
for(i=0;i <length;i++)
{
/*过滤高位,一次读取一个字符*/
val[i]=fgetc(fp)&0x000000FF;
}
for(i=length-1;i>=0;i--)
{
/*因为读取多个16进制字符,叠加*/
variable=variable*0x100+val[i];
}
delete []val;
return variable;
};