用PHP+纯真数据库获得访客的真实地理位置

现在大部分网站都会在地域上给用户不同的体验,更符合用户的习惯等等。于是我们在写网站的时候就需要获得用户的IP地址从而获得用户的真实地理位置。程序是网站找到封装好的类。不过很多新手可能不会使用类。这里稍微讲个例子。

之前请先自行下载DAT格式的IP库文件,百度搜“纯真IP库”即可。

首先是类文件 cz88net.php

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
<?php 
 
 
/**  
* IP 地理位置查询类
*
* @author apieye
* @version 1.0
* @copyright 2012 apieye.com
*/  
class ip_area {
 
 
	/**
	* 
	*
	* @var resource
	*/
	private $fp;//IP库文件指针 resource
	private $firstip;//第一条IP记录的偏移地址 int
	private $lastip;//最后一条IP记录的偏移地址 int
	private $totalip;//IP记录的总条数(不包含版本信息记录) int
 
 
	/**
	* 构造函数,打开 QQWry.Dat 文件并初始化类中的信息
	*
	* @param string $filename
	* @return IpLocation
	*/
	public function __construct($filename = "cz88ip.dat") {//////////////////////////////////////////////这里注意,根据你的数据库存放位置不同,把这个数据库进行引入。
		$this->fp = 0;
		if (($this->fp = fopen($filename, 'rb')) !== false) {
			$this->firstip = $this->getlong();
			$this->lastip = $this->getlong();
			$this->totalip = ($this->lastip - $this->firstip) / 7;//注册析构函数,使其在程序执行结束时执行
			register_shutdown_function(array(&$this, '__destruct'));
		}
	}
 
	/**
	* 析构函数,用于在页面执行结束后自动关闭打开的文件。
	*
	*/
	public function __destruct() {
		if ($this->fp) {
			fclose($this->fp);
		}
		$this->fp = 0;
	}
 
	/**
	* 返回读取的长整型数
	*
	* @access private
	* @return int
	*/
	private function getlong() {//将读取的little-endian编码的4个字节转化为长整型数
	$result = unpack('Vlong', fread($this->fp, 4));
	return $result['long'];
	}
 
	/**
	* 返回读取的3个字节的长整型数
	*
	* @access private
	* @return int
	*/
	private function getlong3() {//将读取的little-endian编码的3个字节转化为长整型数
	$result = unpack('Vlong', fread($this->fp, 3).chr(0));
	return $result['long'];
	}
 
	/**
	* 返回压缩后可进行比较的IP地址
	*
	* @access private
	* @param string $ip
	* @return string
	*/
	private function packip($ip) {// 将IP地址转化为长整型数,如果在PHP5中,IP地址错误,则返回False,// 这时intval将Flase转化为整数-1,之后压缩成big-endian编码的字符串
	return pack('N', intval(ip2long($ip)));
	}
 
	/**
	* 返回读取的字符串
	*
	* @access private
	* @param string $data
	* @return string
	*/
	private function getstring($data = "") {
	$char = fread($this->fp, 1);
	while (ord($char) > 0) {// 字符串按照C格式保存,以\0结束
	$data .= $char;// 将读取的字符连接到给定字符串之后
	$char = fread($this->fp, 1);
	}
	return $data;
	}
 
	/**
	* 返回地区信息
	*
	* @access private
	* @return string
	*/
	private function getarea() {
		$byte = fread($this->fp, 1);// 标志字节
		switch (ord($byte)) {
		case 0:// 没有区域信息
			$area = "";
			break;
		case 1:
		case 2:// 标志字节为1或2,表示区域信息被重定向
			fseek($this->fp, $this->getlong3());
			$area = $this->getstring();
			break;
		default:// 否则,表示区域信息没有被重定向
			$area = $this->getstring($byte);
			break;
		}
		return $area;
	}
 
	/**
	* 根据所给 IP 地址或域名返回所在地区信息
	*
	* @access public
	* @param string $ip
	* @return array
	*/
	public function get($ip) {
		if (!$this->fp) return null;// 如果数据文件没有被正确打开,则直接返回空
		$location['ip'] = gethostbyname($ip);   // 将输入的域名转化为IP地址
		$ip = $this->packip($location['ip']);   // 将输入的IP地址转化为可比较的IP地址
		// 不合法的IP地址会被转化为255.255.255.255// 对分搜索
		$l = 0;// 搜索的下边界
		$u = $this->totalip;// 搜索的上边界
		$findip = $this->lastip;// 如果没有找到就返回最后一条IP记录(QQWry.Dat的版本信息)
		while ($l <= $u) {// 当上边界小于下边界时,查找失败
			$i = floor(($l + $u) / 2); // 计算近似中间记录
			fseek($this->fp, $this->firstip + $i * 7);
			$beginip = strrev(fread($this->fp, 4));// 获取中间记录的开始IP地址// strrev函数在这里的作用是将little-endian的压缩IP地址转化为big-endian的格式// 以便用于比较,后面相同。
			if ($ip < $beginip) {// 用户的IP小于中间记录的开始IP地址时
				$u = $i - 1;// 将搜索的上边界修改为中间记录减一
			}else{
				fseek($this->fp, $this->getlong3());
				$endip = strrev(fread($this->fp, 4));   // 获取中间记录的结束IP地址
				if ($ip > $endip) {// 用户的IP大于中间记录的结束IP地址时
					$l = $i + 1;// 将搜索的下边界修改为中间记录加一
				}else{// 用户的IP在中间记录的IP范围内时
					$findip = $this->firstip + $i * 7;
					break;// 则表示找到结果,退出循环
				}
			}
		}//获取查找到的IP地理位置信息
		fseek($this->fp, $findip);
		$location['beginip'] = long2ip($this->getlong());   // 用户IP所在范围的开始地址
		$offset = $this->getlong3();
		fseek($this->fp, $offset);
		$location['endip'] = long2ip($this->getlong());// 用户IP所在范围的结束地址
		$byte = fread($this->fp, 1);// 标志字节
		switch (ord($byte)) {
		case 1:// 标志字节为1,表示国家和区域信息都被同时重定向
			$countryOffset = $this->getlong3();// 重定向地址
			fseek($this->fp, $countryOffset);
			$byte = fread($this->fp, 1);// 标志字节
			switch (ord($byte)) {
			case 2:// 标志字节为2,表示国家信息又被重定向
				fseek($this->fp, $this->getlong3());
				$location['country'] = $this->getstring();
				fseek($this->fp, $countryOffset + 4);
				$location['area'] = $this->getarea();
				break;
			default:// 否则,表示国家信息没有被重定向
				$location['country'] = $this->getstring($byte);
				$location['area'] = $this->getarea();
				break;
			}
			break;
		case 2:// 标志字节为2,表示国家信息被重定向
			fseek($this->fp, $this->getlong3());
			$location['country'] = $this->getstring();
			fseek($this->fp, $offset + 8);
			$location['area'] = $this->getarea();
			break;
		default:// 否则,表示国家信息没有被重定向
			$location['country'] = $this->getstring($byte);
			$location['area'] = $this->getarea();
			break;
		}
		if ($location['country'] == " CZ88.NET") { // CZ88.NET表示没有有效信息
			$location['country'] = "未知";
		}
		if ($location['area'] == " CZ88.NET") {
			$location['area'] = "";
		}
		$location['country']=iconv('gbk', 'utf-8', $location['country']);
		$location['area']=iconv('gbk', 'utf-8', $location['area']);
		return $location;
	}  
 
	public function getip(){//-------------------------------------------获得用户IP;
	if(isset($_SERVER)){
		if (isset($_SERVER["HTTP_X_FORWARDED_FOR"])){
			$realip=$_SERVER["HTTP_X_FORWARDED_FOR"];
		}elseif(isset($_SERVER["HTTP_CLIENT_IP"])){
			$realip=$_SERVER["HTTP_CLIENT_IP"];
		}else{
			$realip=$_SERVER["REMOTE_ADDR"];
		}
	}else{
		if(getenv("HTTP_X_FORWARDED_FOR")){
			$realip=getenv("HTTP_X_FORWARDED_FOR");
		}elseif(getenv("HTTP_CLIENT_IP")){
			$realip=getenv("HTTP_CLIENT_IP");
		}else{
			$realip=getenv("REMOTE_ADDR");
		}
	}
	return $realip;
}
}  
?>

我在里面加了个方法getip(),可以获得用户的真实IP。

下面是一个例子,简单的写了如何去使用这个封装好的类。

1
2
3
4
5
6
7
<?php
	header("Content-Type: text/html; charset=UTF-8");
	include ("cz88ip.php");
	$fip = new ip_area();
	$ip = $fip->get("8.8.8.8");
	echo $ip['country']."  ".$ip['area'];
?>

由于这个数据库字段设置问题,country 是国外的话指国家,国内的话包含了省份和城市,有特殊需要的还需自行截取。area 是IP的机构名字,如谷歌DNS服务器,电信,网通之类的。可以自己写几个IP进去试试就知道了。

注意:该类文件默认写的为GB2312,输出UTF-8。
如果想GB2312编码文件输出GB2312就注释掉198,199行。
如果想UTF-8编码文件输出UTF-8就把文件保存为UTF-8并注释掉198,199行。
总之198,199行有点多余。有空再改下,并提供几种编码的版本下载。

本文固定链接: https://www.apieye.com/191.html | APIEYE

该日志由 Jazzy 于2012年09月11日发表在 PHP, web基础 分类下, 你可以发表评论,并在保留原文地址及作者的情况下引用到你的网站或博客。
原创文章转载请注明: 用PHP+纯真数据库获得访客的真实地理位置 | APIEYE
关键字: ,

用PHP+纯真数据库获得访客的真实地理位置:等您坐沙发呢!

发表评论

您必须 [ 登录 ] 才能发表留言!