(2017-07-02 银河统计)
地理编码(Geocoding)指的是将统计资料或是地址信息建立空间坐标关系的过程。地址经过地理编码后,便可在地图中通过经纬度显示各地址的位置(正向地理编码:地址 - 坐标)。反之,也可以通过经纬度解析出对应地址(逆向地理编码:坐标 - 地址)。地理编码也称为地址解析。
一、标准地址规范
当我们书写地址是,应该从大到小分级书写,地址等级可参考如下分级:
- 第一级省
- 第二级市
- 第三级区、县
- 第四级镇、乡、街道
- 第五级路、巷、行政村
- 第六级小区、自然村
- 第七级门牌、村组
- 第八级楼号、办公楼房(单位)
- 第九级单元号、房号
如,黑龙江省哈尔滨市道里区通达街138号。如果城市较大,可以省略省份。另外,地理编码一般只解析到门牌号,没必要书写楼层或房间号等细节。
二、地址解析
本文根据百度地图开发平台介绍地址和经纬度间的相互转换。
1、在网页中引用百度地图API
<!DOCTYPE html>
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
<script type="text/javascript" src="http://api.map.baidu.com/api?v=2.0&ak=您的密钥">
</script>
<title>地址解析</title>
</head>
<body></body>
</html>
使用前登陆百度地图开放平台开发者注册页(http://lbsyun.baidu.com/apiconsole/key),获得自己的密钥。
2、百度地址解析
注:在地址栏中输入国内任何地址,可获得该地址经纬度和墨卡托坐标
代码样例
<!DOCTYPE html>
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
<title>地址解析</title>
<script type="text/javascript" src="http://api.map.baidu.com/api?v=2.0&ak=您的密钥"></script>
</head>
<body>
<center>地址 <input id="myAddress" value="北京市海淀区上地10街" style="width:396px;">
<button onclick="setJWD(document.getElementById('myAddress').value)">地址解析</button>
<div style="width:500px; height:300px; border:#aaaaaa solid 1px;"></div>
经度 <input id="myJD" style="width:80px;"> 纬度 <input id="myWD" style="width:80px;">
墨卡托X <input id="myMX" style="width:80px;"> 墨卡托Y <input id="myMY" style="width:80px;">
</center>
</body>
</html>
<script type="text/javascript" charset="UTF-8">
var map = new BMap.Map("myMap");
setJWD(document.getElementById("myAddress").value);
function setJWD(addr) {
var point = new BMap.Point(116.331398,39.897445);
map.centerAndZoom(point,12);
var myGeo = new BMap.Geocoder(); // 创建地址解析器实例
var mpObj = new BMap.MercatorProjection(); // 墨卡托坐标对象
myGeo.getPoint(addr, function(point){
if (point) {
map.centerAndZoom(point, 16); //设置地图中心点和显示级别
map.addOverlay(new BMap.Marker(point)); //加载中心点图层
document.getElementById("myJD").value=point.lng; //获得经度
document.getElementById("myWD").value=point.lat; //获得纬度
var mpoint = mpObj.lngLatToPoint(new BMap.Point(point.lng, point.lat)); //经纬度转换为墨卡托坐标
document.getElementById("myMX").value=mpoint.x; //获得墨卡托横坐标
document.getElementById("myMY").value=mpoint.y; //获得墨卡托纵坐标
}}, "");
}
</script>
3、百度批量地址解析
批量地址格式文本(地址加引号并用逗号分隔)
"合肥市包河区金寨路1号(金寨路与望江西路交叉口)","合肥市庐阳区凤台路209号(凤台路与蒙城北路交叉口)","合肥市蜀山区金寨路217号(近安医附院公交车站)","合肥市蜀山区梅山路10号(近安徽饭店) ","合肥市蜀山区 长丰南路159号铜锣湾广场312室","合肥市寿春路93号钱柜星乐町KTV(逍遥津公园对面)","合肥市庐阳区长江中路177号","合肥市新站区胜利路89"
注:按格式批量添加地址,然后运行“批量地址解析”按钮 批量地址解析
批量地址经纬度坐标
代码样例
<!DOCTYPE html>
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf8" />
<meta name="viewport" content="initial-scale=1.0, user-scalable=no" />
<title>批量地址</title>
<style type="text/css">
body, html{width: 100%;height: 100%;margin:0;font-family:"微软雅黑";}
#l-map{height:300px;width:100%;}
#r-result{width:100%; font-size:14px;line-height:20px;}
</style>
<script type="text/javascript" src="http://api.map.baidu.com/api?v=2.0&ak=您的密钥"></script>
</head>
<body>
<div id="l-map"></div>
<div >
<input type="button" value="批量地址解析" onclick="bdGEO()" />
<div id="result"></div>
</div>
</body>
</html>
<script type="text/javascript">
// 百度地图API功能
var map = new BMap.Map("l-map");
map.centerAndZoom(new BMap.Point(117.269945,31.86713), 13);
map.enableScrollWheelZoom(true);
var index = 0;
var myGeo = new BMap.Geocoder();
var adds = [
"包河区金寨路1号(金寨路与望江西路交叉口)",
"庐阳区凤台路209号(凤台路与蒙城北路交叉口)",
"蜀山区金寨路217号(近安医附院公交车站)",
"蜀山区梅山路10号(近安徽饭店) ",
"蜀山区 长丰南路159号铜锣湾广场312室",
"合肥市寿春路93号钱柜星乐町KTV(逍遥津公园对面)",
"庐阳区长江中路177号",
"新站区胜利路89"
];
function bdGEO(){
var add = adds[index];
geocodeSearch(add);
index++;
}
function geocodeSearch(add){
if(index < adds.length){
setTimeout(window.bdGEO,400);
}
myGeo.getPoint(add, function(point){
if (point) {
document.getElementById("result").innerHTML += index + "、" + add + ":" + point.lng + "," + point.lat + "</br>";
var address = new BMap.Point(point.lng, point.lat);
addMarker(address,new BMap.Label(index+":"+add,{offset:new BMap.Size(20,-10)}));
}
}, "合肥市");
}
// 编写自定义函数,创建标注
function addMarker(point,label){
var marker = new BMap.Marker(point);
map.addOverlay(marker);
marker.setLabel(label);
}
</script>
注:代码摘自百度API示例文档
4、百度逆地址解析
省份 城市 区镇 街道 牌号
地址
注:可替换经纬度(如,经度126.616759、纬度45.74989),点击“逆地址解析”按钮获得明码地址
代码样例
<!DOCTYPE html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
<meta name="viewport" content="initial-scale=1.0, user-scalable=no" />
<style type="text/css">
body, html{width: 100%;height: 100%;margin:0;font-family:"微软雅黑";font-size:14px;}
#allmap {width:100%;height:500px;}
</style>
<script type="text/javascript" src="http://api.map.baidu.com/api?v=2.0&ak=您的密钥"></script>
<title>逆地址解析</title>
</head>
<body>
<div id="allmap"></div>
<p>点击地图展示详细地址</p>
</body>
</html>
<script type="text/javascript">
// 百度地图API功能
var map = new BMap.Map("allmap");
var point = new BMap.Point(116.331398,39.897445);
map.centerAndZoom(point,12);
var geoc = new BMap.Geocoder();
map.addEventListener("click", function(e){
var pt = e.point;
geoc.getLocation(pt, function(rs){
var addComp = rs.addressComponents;
alert(addComp.province + ", " + addComp.city + ", " + addComp.district + ", " + addComp.street + ", " + addComp.streetNumber);
});
});
</script>
注:代码摘自百度API示例文档
5、百度批量逆地址解析
[[116.307852,40.057031],[116.313082,40.047674],[116.328749,40.026922],[116.347571,39.988698],[116.316163,39.997753],[116.345867,39.998333],[116.403472,39.999411],[116.307901,40.05901]]
注:数组按格式批量添加经纬度,然后运行“批量反地址解析”按钮
代码样例
<!DOCTYPE html>
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=gb2312" />
<meta name="viewport" content="initial-scale=1.0, user-scalable=no" />
<title>批量反地址解析+商圈</title>
<style type="text/css">
body, html{width: 100%;height: 100%;margin:0;font-family:"微软雅黑";}
#l-map{height:300px;width:100%;}
#r-result{width:100%; font-size:14px;line-height:20px;}
</style>
<script type="text/javascript" src="http://api.map.baidu.com/api?v=2.0&ak=您的密钥"></script>
</head>
<body>
<div id="l-map"></div>
<div >
<input type="button" value="批量反地址解析+商圈" onclick="bdGEO(0)" />
<div id="result"></div>
</div>
</body>
</html>
<script type="text/javascript">
// 百度地图API功能
var map = new BMap.Map("l-map");
map.centerAndZoom(new BMap.Point(116.328749,40.026922), 13);
map.enableScrollWheelZoom(true);
var index = 0;
var myGeo = new BMap.Geocoder();
var adds = [
new BMap.Point(116.307852,40.057031),
new BMap.Point(116.313082,40.047674),
new BMap.Point(116.328749,40.026922),
new BMap.Point(116.347571,39.988698),
new BMap.Point(116.316163,39.997753),
new BMap.Point(116.345867,39.998333),
new BMap.Point(116.403472,39.999411),
new BMap.Point(116.307901,40.05901)
];
for(var i = 0; i<adds.length; i++){
var marker = new BMap.Marker(adds[i]);
map.addOverlay(marker);
marker.setLabel(new BMap.Label("我是商圈:"+(i+1),{offset:new BMap.Size(20,-10)}));
}
function bdGEO(){
var pt = adds[index];
geocodeSearch(pt);
index++;
}
function geocodeSearch(pt){
if(index < adds.length-1){
setTimeout(window.bdGEO,400);
}
myGeo.getLocation(pt, function(rs){
var addComp = rs.addressComponents;
document.getElementById("result").innerHTML += index + ". " +adds[index-1].lng + "," + adds[index-1].lat + ":" + "商圈(" + rs.business + ") 结构化数据(" + addComp.province + ", " + addComp.city + ", " + addComp.district + ", " + addComp.street + ", " + addComp.streetNumber + ")<br/><br/>";
});
}
</script>
注:代码摘自百度API示例文档
地址正、逆解析精度取决于系统数据库数据采集密度和更新速度。国内地址服务较好的为百度和高德地图。在实践中,地址解析、特别是大数据批量地址解析较为常用。凡是和位置有关的自然和社会现象,通过地址解析获得数量化指标,从而为统计量化分析、特别是空间统计分析奠定坚实基础。
本文提供的各种地址解析工具仅用于个人学习或教学研究用途,不支持大批量商业用途地址解析(由于百度地图key的限制)。