百度地图聚合折腾

被客户爸爸折腾了好久,最后终于做出基本满意的“热力图”

起初做的热力图,但是来回反复测试百度地图热力图和echars热力图+百度地图,终究不满足客户爸爸,原因很简单某些地区放大后,为什么啥也没有了,为什么?为什么?mmmp不知当讲不当讲,热力图放大数据稀疏的地方当然不明显,遂开始改热力图各种配置,渐变色,透明度,无奈怎么也不满足。最后技术总监说干脆用某某项目的聚合算了,接着就进入了正题百度地图–聚合折腾之路。

百度地图的聚合在文档里不太好找,在百度地图开源库里才能找到

#####贴一下官方的简单说明
|类 | 描述 |
| - | - |
|BMapLib.MarkerClusterer(map, options)|MarkerClusterer|

方法
方法 返回值 描述
addMarker(marker) None 添加一个聚合的标记。
addMarkers(markers) None 添加要聚合的标记数组。
clearMarkers() None 从地图上彻底清除所有的标记
getClustersCount() Number 获取聚合的总数量。
getGridSize() Number 获取网格大小
getMap() Map 获取聚合的Map实例。
getMarkers() Array 获取所有的标记数组。
getMaxZoom() Number 获取聚合的最大缩放级别。
getMinClusterSize() Number 获取单个聚合的最小数量。
getStyles() Array 获取聚合的样式风格集合
isAverageCenter() Boolean 获取单个聚合的落脚点是否是聚合内所有标记的平均中心。
removeMarker(marker) Boolean 删除单个标记
removeMarkers(markers) Boolean 删除一组标记
setGridSize(size) None 设置网格大小
setMaxZoom(maxZoom) None 设置聚合的最大缩放级别
setMinClusterSize(size) None 设置单个聚合的最小数量。
setStyles(styles) None 设置聚合的样式风格集合

事实上要不是逼急一般人不会来这里看这些文档,百度地图的demo基本上就够用。
由于数据量巨大,本项目的就有十三万数据,一股脑显示在地图上之后来回缩放,发现….卡,卡到没朋友,还有就是这十三万数据从服务器传送到前端耗时十多秒赶上网速不好就得挂在半路500错误码。
无奈之下的继续优化,终于在苦苦搜索之后找到了网上大神的优化代码 (传送门),不得不说效率确实大增,但是,优化后依然扛不住13万的数据。将来的数据可不止13万。进一步优化。
结合网上的仿照链家等地图找房例子(传送门),规划这样的方案:
1.进行区域划分,省、市、区对应不同的地图等级显示不同等级的数据,这样起码就解决的全国数据时有十三万的尴尬,这样折合下来全国数据也就30多条,效率不言而喻的高!
2.省、市、区级别之后进行自动聚合,就是百度地图真正的聚合。
3.为了避免数据多余营销效率,每次请求只请求可视区域内的数据。

然后是甩锅到本渣渣这里,开始自我猜想:
1.地图要做到拖动请求,缩放请求
2.每次请求都要知道现在的地图缩放级别是多少,以便判断当前是省、市、区
3.每次请求还要有地图可视的区域经纬度
4.省市区要用地图的自定义覆盖物,Label是个不错的选择
5.请求数据地图级别省市区的时候,数据要包含行政区域名称,人数,经纬度
基本上是这样,然后开始搞

1
2
3
4
5
6
7
8
9
获取地图缩放级别
var zoom = map.getZoom(); // 直接返回数字1-20吧好像

获取可视区域的经纬度
getBounds()
用法
var bs = map.getBounds(); // 获取可视区域
var bssw = bs.getSouthWest(); // bssw.lng, bssw.lat
var bsne = bs.getNorthEast(); // bsne.lng, bsne.lat
添加文本标注label

Label(content: String, opts: LabelOptions)
创建一个文本标注实例。point参数指定了文本标注所在的地理位置
|方法|返回值| 描述|
|setStyle(styles: Object)|none|设置文本标注样式,该样式将作用于文本标注的容器元素上。其中styles为JavaScript对象常量,比如: setStyle({ color : “red”, fontSize : “12px” }) 注意:如果css的属性名中包含连字符,需要将连字符去掉并将其后的字母进行大写处理,例如:背景色属性要写成:backgroundColor|
|setContent(content: String)|none|设置文本标注的内容。支持HTML|
|setPosition(position: Point)|none|设置文本标注坐标。仅当通过Map.addOverlay()方法添加的文本标注有效|

然后就是代码说事儿

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
var map = new BMap.Map("echars_heatmap", {minZoom:5});
var BMapLib = window.BMapLib = BMapLib || {};
var Page = {
init: function() {
this.initMap();
this.echarsHeatmap(this.data.heatmapDatas);
},
data:{
timerMapGetdata:null,
zoom: 5,
condition:{
zoom: 5,
level: 1,
},
markerClusterer: null,
heatmapDatas: {
centerPoint: null,
points: null,
lengthMax: 0
},
},
// 初始化地图触发事件
initMap:function(){
var _this = this;
var point = new BMap.Point(116.403981, 39.91571);

// 设置中心点坐标和地图级别 5全国 10省
map.centerAndZoom(point, 5);

// 允许滚轮缩放
map.enableScrollWheelZoom();

// 获取地图的当前缩放级别
_this.data.zoom = map.getZoom();

// 地图缩放事件,每次缩放都要重新触发获取视野内新数据
map.addEventListener('zoomend', function(){
console.log('缩放事件:');
// 事件触发优化,防止短时间内多次触发事件频繁请求数据,造成服务器压力,同时也为了用户体验
clearTimeout(_this.data.timerMapGetdata);
_this.data.timerMapGetdata = null;
_this.data.timerMapGetdata = setTimeout(function() {
_this.getHeatData();
}, 300);
});

// 地图拖动事件,每次拖动地图,地图的可视区域都会变化,都要重新触发获取视野内新数据
map.addEventListener("moveend", function(){
console.log('拖动事件:');
// 事件触发优化,防止短时间内多次触发事件频繁请求数据,造成服务器压力,同时也为了用户体验
clearTimeout(_this.data.timerMapGetdata);
_this.data.timerMapGetdata = null;
_this.data.timerMapGetdata = setTimeout(function() {
_this.getHeatData();
}, 300);
});
},
getHeatData: function() {
var _this = this;
// layer弹窗插件,加载弹窗
layer.load(1, {
shade: [0.1, '#fff'],
});
//获取可视区域
var bs = map.getBounds();
var bssw = bs.getSouthWest();
var bsne = bs.getNorthEast();

// 获取地图缩放级别,并且划分请求级别
var zoom = map.getZoom();
var level = (zoom < 8) ? 1 : ((zoom >= 8 && zoom <= 11) ? 2 : ((zoom > 11 && zoom < 15) ? 3 : 4));
_this.data.level = level;
_this.data.condition['level'] = level;
_this.data.condition['left'] = bssw.lng;
_this.data.condition['top'] = bssw.lat;
_this.data.condition['right'] = bsne.lng;
_this.data.condition['bottom'] = bsne.lat;
$.ajax({
url: '',
data: {
condition: _this.data.condition
},
type: 'POST',
success: function(res) {
layer.closeAll();

// 重新渲染地图
_this.data.heatmapDatas.points = res.data;
_this.data.heatmapDatas.lengthMax = res.max;
_this.echarsHeatmap(_this.data.heatmapDatas);
}
});
},

// 热力图
echarsHeatmap: function(obj) {
var _this = this;
var points = obj.points ? obj.points : [];
_this.data.zoom = map.getZoom();
var zoom = Number(_this.data.zoom);
var level = _this.data.level;
// 清除覆盖物
map.clearOverlays();
if(_this.markerClusterer) {
_this.markerClusterer.clearMarkers();
}
// 1.当请求级别是4的时候,百度地图自动集合;
// 2.当请求级别小于4,使用后台做聚合查询,前端label展示。
if (level != 4) {
for (var i = 0; i < points.length; i++) {
_this.addLabel(points[i]);
}
} else {
var markers = [];
var pt = null;
for (var i = 0; i < points.length; i++) {
pt = new BMap.Point(points[i].lng, points[i].lat);
var maker = new BMap.Marker(pt);
maker.setTitle(points[i].macs);
markers.push(maker);
}
//最简单的用法,生成一个marker数组,然后调用markerClusterer类即可。
var tt1 = new Date().getTime();
_this.markerClusterer = new BMapLib.MarkerClusterer(map, {markers:markers});
var tt2 = new Date().getTime();
}

},
// 添加覆盖物
addLabel: function(pointObj) {
var zoom = Number(this.data.zoom);

var point = new BMap.Point(pointObj.longitude, pointObj.latitude);
var label = new BMap.Label();
label.setStyle({
maxWidth: 'none',
color:"#fff", //颜色
fontSize:"12px", //字号
border:"0", //边
borderRadius: '50%',
height:"60px", //高度
width:"60px", //宽
textAlign:"center", //文字水平居中显示
background: "#F99D32",
cursor:"pointer",
position: "absolute",
left: "-60px",
top: "-60px"
});

var content = '<p style="text-align:center;margin-top:16px;">'+ pointObj.name +'</p>';
content += '<p style="text-align:center;">'+ pointObj.macs +'</p>';
label.setContent(content);
label.setPosition(point);
label.addEventListener("click", function() { //拖动事件
map.centerAndZoom(point, zoom+2)
});
map.addOverlay(label);
},
};

后台用的php,这里贴上sql说明一下问题就行

当请求级别是小于4的时候,sql如下:

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
/*
接受参数:
conditioin:{
level: 1,
left: 110.329288,
right: 119.895882,
top:31.356779,
bottom: 36.253414
}
简要说明:
当请求级别小于4的时候,后台数据要做聚合处理,从user表查询出所有人数,通过‘order by city’做分组聚合处理(根据请求级别 省1:state,市2:city,区3:country),然后关联address表(tf_address_region表存储中国所有的省市区对应的名字、经纬度,具体表结构和sql文件在最下边展示与下载)查询出每个地址对应的经纬度
*/
SELECT A.NAME,
A.LEVEL,
A.longitude,
A.latitude,
b.member_skey
FROM
tf_address_region A,
(
SELECT
city, /*省1:state,市2:city,区3:country*/
COUNT ( 1 ) member_skey
FROM
td_user
WHERE
(
AND ( latitude <> '' AND longitude IS NOT NULL )
AND ( latitude BETWEEN 31.356779 AND 36.253414 AND longitude BETWEEN 110.329288 AND 119.895882 )
)
GROUP BY
city /*省1:state,市2:city,区3:country*/
) b
WHERE
( b.city = A.NAME ) /*省1:state,市2:city,区3:country*/

当请求级别是4的时候,sql如下:

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
/*
接受参数:
conditioin:{
level: 4,
left: 113.622692
right: 113.697431
top:34.739694,
bottom: 34.775282
}
简要说明:
当请求级别等于4的时候,后台数据不要做聚合处理,直接从user表查询出所有人数,聚合工作直接给百度地图处理
*/
SELECT
longitude lng,
latitude lat,
COUNT ( 1 ) member_skey
FROM
td_user
WHERE
(
AND ( latitude <> '' AND longitude IS NOT NULL )
AND ( latitude BETWEEN 34.739694 AND 34.775282 AND longitude BETWEEN 113.622692 AND 113.697431 )
)
GROUP BY
latitude,
longitude
效果图

map1
map2
map3
map4

附加address表的说明:

表结构:

名称 类型 备注
id bigint(11) 表id
name varchar(32) 地区名称
level tinyint(4) 地区等级 分省市县区
pid bigint(10) 父id
longitude float 百度坐标,经度
latitude float 百度坐标,纬度

下载地址