ホームページ >バックエンド開発 >PHPチュートリアル >Google Map API を PHP と組み合わせて、ログイン マップの位置情報を実装します。

Google Map API を PHP と組み合わせて、ログイン マップの位置情報を実装します。

WBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWB
WBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBオリジナル
2016-06-13 13:06:081068ブラウズ

Google Map API と PHP を組み合わせてログイン マップの位置情報を実現します

Google Map API と PHP を組み合わせてログイン位置特定を実現する主なアイデアは、ユーザーがシステムにログインするときに IP アドレスを記録し、その IP アドレスを物理アドレスに変換することです。関連する住所インデックスを通じて、最終的に Google Map API を通じて取得します。物理的な住所は Google マップ上にマークされます。

?

php プラットフォームは、ThinkPHP フレームワークによって構築されています。具体的なフローチャートは以下の通りです:


?

ここでは CoralWry データ ファイルが使用されています。以前は、Rainbow QQ やその他のサービスがこのファイルを使用して解析されていました。 IPを更新する必要があります。更新するには、以下のダウンロードパッケージがあります。このファイルはルート ディレクトリ

に配置されます

?

続けて、外部リンク呼び出しライブラリとして 2 つのファイルを TP に導入します。これらのファイルは、TP コア フォルダの Lib の下にある ORG 拡張ライブラリに直接配置できます

IpLocation クラス。この拡張クラスは、DAT ファイルから IP アドレスを抽出するために使用されます

?

class IpLocation {
var $fp;
var $firstip;  //第一条ip索引的偏移地址
var $lastip;   //最后一条ip索引的偏移地址
var $totalip;  //总ip数
//*
//构造函数,初始化一些变量
//$datfile 的值为纯真IP数据库的名子,可自行修改.
//*
function ipLocation($datfile = "CoralWry.Dat"){
  $this->fp=fopen($datfile,'rb')or die("CoralWry.Dat不存在,请去网上下载纯真IP数据库, 'CoralWry.dat' 放到当前目录下");   //二制方式打开
  $this->firstip = $this->get4b(); //第一条ip索引的绝对偏移地址
  $this->lastip = $this->get4b();  //最后一条ip索引的绝对偏移地址
  $this->totalip =($this->lastip - $this->firstip)/7 ; //ip总数 索引区是定长的7个字节,在此要除以7,
  register_shutdown_function(array($this,"closefp"));  //为了兼容php5以下版本,本类没有用析构函数,自动关闭ip库.
}
//*
//关闭ip库
//*
function closefp(){
fclose($this->fp);
}
//*
//读取4个字节并将解压成long的长模式
//*
function get4b(){
  $str=unpack("V",fread($this->fp,4));
  return $str[1];
}
//*
//读取重定向了的偏移地址
//*
function getoffset(){
  $str=unpack("V",fread($this->fp,3).chr(0));
  return $str[1];
}
//*
//读取ip的详细地址信息
//*
function getstr(){
	$str="";
  $split=fread($this->fp,1);
  while (ord($split)!=0) {
    $str .=$split;
	$split=fread($this->fp,1);
  }
  return $str;
}
//*
//将ip通过ip2long转成ipv4的互联网地址,再将他压缩成big-endian字节序
//用来和索引区内的ip地址做比较
//*
function iptoint($ip){
  return pack("N",intval(ip2long($ip)));
}
//*
//获取客户端ip地址
//注意:如果你想要把ip记录到服务器上,请在写库时先检查一下ip的数据是否安全.
//*
function getIP() {
        if (getenv('HTTP_CLIENT_IP')) {
				$ip = getenv('HTTP_CLIENT_IP'); 
		}
		elseif (getenv('HTTP_X_FORWARDED_FOR')) { //获取客户端用代理服务器访问时的真实ip 地址
				$ip = getenv('HTTP_X_FORWARDED_FOR');
		}
		elseif (getenv('HTTP_X_FORWARDED')) { 
				$ip = getenv('HTTP_X_FORWARDED');
		}
		elseif (getenv('HTTP_FORWARDED_FOR')) {
				$ip = getenv('HTTP_FORWARDED_FOR'); 
		}
		elseif (getenv('HTTP_FORWARDED')) {
				$ip = getenv('HTTP_FORWARDED');
		}
		else { 
				$ip = $_SERVER['REMOTE_ADDR'];
		}
		return $ip;
}
//*
//获取地址信息
//*
function readaddress(){
  $now_offset=ftell($this->fp); //得到当前的指针位址
  $flag=$this->getflag();
  switch (ord($flag)){
         case 0:
		     $address="";
		 break;
		 case 1:
		 case 2:
		     fseek($this->fp,$this->getoffset());
			 $address=$this->getstr();
		 break;
		 default:
		     fseek($this->fp,$now_offset);
		     $address=$this->getstr();
		 break;
  }
  return $address;
}
//*
//获取标志1或2
//用来确定地址是否重定向了.
//*
function getflag(){
  return fread($this->fp,1);
}
//*
//用二分查找法在索引区内搜索ip
//*
function searchip($ip){
  $ip=gethostbyname($ip);     //将域名转成ip
  $ip_offset["ip"]=$ip;
  $ip=$this->iptoint($ip);    //将ip转换成长整型
  $firstip=0;                 //搜索的上边界
  $lastip=$this->totalip;     //搜索的下边界
  $ipoffset=$this->lastip;    //初始化为最后一条ip地址的偏移地址
  while ($firstip fp,$this->firstip + $i * 7);    //定位指针到中间记录
	$startip=strrev(fread($this->fp,4));         //读取当前索引区内的开始ip地址,并将其little-endian的字节序转换成big-endian的字节序
	if ($ip fp,$this->getoffset());
	   $endip=strrev(fread($this->fp,4));
	   if ($ip > $endip){
	      $firstip=$i + 1;
	   }
	   else {
	      $ip_offset["offset"]=$this->firstip + $i * 7;
	      break;
	   }
	}
  }
  return $ip_offset;
}
//*
//获取ip地址详细信息
//*
function getaddress($ip){
  $ip_offset=$this->searchip($ip);  //获取ip 在索引区内的绝对编移地址
  $ipoffset=$ip_offset["offset"];
  $address["ip"]=$ip_offset["ip"];
  fseek($this->fp,$ipoffset);      //定位到索引区
  $address["startip"]=long2ip($this->get4b()); //索引区内的开始ip 地址
  $address_offset=$this->getoffset();            //获取索引区内ip在ip记录区内的偏移地址
  fseek($this->fp,$address_offset);            //定位到记录区内
  $address["endip"]=long2ip($this->get4b());   //记录区内的结束ip 地址
  $flag=$this->getflag();                      //读取标志字节
  switch (ord($flag)) {
         case 1:  //地区1地区2都重定向
		 $address_offset=$this->getoffset();   //读取重定向地址
		 fseek($this->fp,$address_offset);     //定位指针到重定向的地址
		 $flag=$this->getflag();               //读取标志字节
		 switch (ord($flag)) {
		        case 2:  //地区1又一次重定向,
				fseek($this->fp,$this->getoffset());
				$address["area1"]=$this->getstr();
				fseek($this->fp,$address_offset+4);      //跳4个字节
				$address["area2"]=$this->readaddress();  //地区2有可能重定向,有可能没有
				break;
				default: //地区1,地区2都没有重定向
				fseek($this->fp,$address_offset);        //定位指针到重定向的地址
				$address["area1"]=$this->getstr();
				$address["area2"]=$this->readaddress();
				break;
		 }
		 break;
		 case 2: //地区1重定向 地区2没有重定向
		 $address1_offset=$this->getoffset();   //读取重定向地址
		 fseek($this->fp,$address1_offset);  
		 $address["area1"]=$this->getstr();
		 fseek($this->fp,$address_offset+8);
		 $address["area2"]=$this->readaddress();
		 break;
		 default: //地区1地区2都没有重定向
		 fseek($this->fp,$address_offset+4);
		 $address["area1"]=$this->getstr();
		 $address["area2"]=$this->readaddress();
		 break;
  }
  //*过滤一些无用数据
  if (strpos($address["area1"],"CZ88.NET")!=false){
      $address["area1"]="未知";
  }
  if (strpos($address["area2"],"CZ88.NET")!=false){
      $address["area2"]=" ";
  }
  return $address;
 }

} 

?

?

mapService クラス、メソッド クラス

?

?

/**
 * Google Map Service
 * 2011.4.8
 */
import("ORG.IPA.IpLocation");
class mapService{
	
	public static function getIPaddress($ip){
		
	//返回格式
	$format = "text";//默认text,json,xml,js
	//返回编码
	$charset = "utf8"; //默认utf-8,gbk或gb2312
	$ip_l=new IpLocation();
	$address=$ip_l->getaddress($ip);
	$address["area1"] = iconv('GB2312','utf-8',$address["area1"]);
	$address["area2"] = iconv('GB2312','utf-8',$address["area2"]);
	$add=$address["area1"].$address["area2"];
	if($add=="本机地址 "){
		$add="杭州";
	}
	
	return  $add;
	}
}
?

その後、ログインしたアクション インターフェイス関数で 2 つのファイル クラスを呼び出すことができます。

?

?

	import("ORG.IPA.MapService");
        $ipaddress = get_client_ip();

	$adrInfo=MapService::getIPaddress($ipaddress);
?

?

上記の情報はデータベースに記録でき、ユーザーがマップ ページに入ると、データをデータベースからエクスポートして表示レイヤーに送信し、フロントで Google Map API を呼び出すことができます。エンドレイヤー

?

バックグラウンド制御層のコードは非常に単純です:

?

?

	public function map(){
		
		parent::islogin();
		$model = D("Topicview");
		
        $list =  $model->field('id,tid,imgid,avatar,address,create_time,topic_from,content,nickname,rootid,homepage')
        	->where("Topic.status=1 and Topic.type='first'")
        	->order("id desc")
			->find();

		//dump($list);
		$this->assign('addrList',$list);
		parent::showSiteInfo("Lab前端实验室 - Map Position");
        $this->display();
        
	}
?

?

次のステップは、フロントエンド テンプレート レイヤーの JS コードです:

?

?

if (typeof flowg == "undefined" || !flowg) {
    var flowg = {};
}
flowg.initMap = (function(){

	var htmlString = '<div style="overflow: auto;">' +
    '<div style="width:300px;overflow:hidden;" class="map-item">' +
    '<div class="map-left">' +
    '<img src="%7B:getUserAvatar(%24addrList%5B" avatar alt="Google Map API を PHP と組み合わせて、ログイン マップの位置情報を実装します。"   style="max-width:90%">' +
    '</div>
<div class="map-right">' +
    '<div class="map-content">Google Map API を PHP と組み合わせて、ログイン マップの位置情報を実装します。 : {$addrList.content}</div>' +
	'<div class="time">他在{$addrList.address}</div>' +
    '</div>' +
    '</div>';
    var geocoder;
    var map;
    var oldinfo = null;
    function initialize(){
        geocoder = new google.maps.Geocoder();
        var latlng = new google.maps.LatLng(34.016, 103.535);
        var myOptions = {
            zoom: 8,
            center: latlng,
            mapTypeId: google.maps.MapTypeId.ROADMAP
        };
        map = new google.maps.Map(document.getElementById("map_canvas"), myOptions);
        codeAddress("{$addrList.address}", htmlString);
    }
    
    function codeAddress(address, html){
        geocoder.geocode({
            'address': address
        }, function(results, status){
            if (status == google.maps.GeocoderStatus.OK) {
                map.setCenter(results[0].geometry.location);
                var marker = new google.maps.Marker({
                    map: map,
                    position: results[0].geometry.location
                });
                var contentString = html;
                var infowindow = new google.maps.InfoWindow({
                    content: contentString
                });
                infowindow.open(map, marker);
                if (oldinfo != null) {
                    oldinfo.close();
                }
                oldinfo = infowindow;
            }
            else {
                return false;
            }
        });
    }

    return initialize;
	
})();

$(flowg.initMap);
?
<p>?</p>
<p>上記は、Google マップのマーカー関数を直接呼び出して情報を表示します。</p>
<p>?</p>
<p><br><img alt="" src="http://www.daimami.com/img/2014/01/01/1332391838.png"></p>
 <div class="clear"></div>
</div>
声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。