前言 最近接手同事做了一半的项目,其中有个需求是元数据入库需要存储空间数据,支持空间查询。SpringBoot + MyBatisPlus + MySQL, 前端使用的是vue2。PostgreSql, 它有着强大空间数据处理能力,且项目中使用也很方便。MySQL不支持空间数据,没想到一查资料居然是支持的,于是开始了连续几个小时的踩坑,终于搞定,整理成完成路线供大家参考。
1. MySQL空间数据 MySQL为空间数据存储及处理提供了专用的类型geometry(支持所有的空间结构),还有有细分类型Point, LineString,
2. GeoJson介绍 GeoJSON是一种对各种地理数据结构进行编码的格式。GeoJSON对象可以表示几何、特征或者特征集合。GeoJSON支持下面几何类型:
点、线、面、多点、多线、多面和几何集合。
GeoJSON里的特征包含一个几何对象和其他属性,特征集合表示一系列特征,一个完整的GeoJSON数据结构总是一个(JSON术语里的)对象。
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 {"type" : "FeatureCollection" ,"features" : [{"type" : "Feature" ,"geometry" : {"type" : "Point" ,"coordinates" : [102.0 , 0.5 ]"properties" : {"prop0" : "value0" "type" : "Feature" ,"geometry" : {"type" : "LineString" ,"coordinates" : [[102.0 , 0.0 ], [103.0 , 1.0 ], [104.0 , 0.0 ], [105.0 , 1.0 ]]"properties" : {"prop0" : "value0" ,"prop1" : 0.0 "type" : "Feature" ,"geometry" : {"type" : "Polygon" ,"coordinates" : [[100.0 , 0.0 ], [101.0 , 0.0 ], [101.0 , 1.0 ], [100.0 , 1.0 ], [100.0 , 0.0 ]]"properties" : {"prop0" : "value0" ,"prop1" : {"this" : "that" 
3.MySql格式化空间数据类型(geometry相互转换geojson) MySQL提供了几个空间函数用来解析及格式化空间数据,geojson是gis空间数据展示的标准格式,前端地图框架及后端空间分析相关框架都会支持geojson格式。
转换 
空间函数 
 
 
geojson -> geometry 
ST_GeomFromGeoJSON 
 
geometry -> geojson 
ST_ASGEOJSON 
 
geometry(字符串) -> geometry 
ST_GEOMFROMTEXT 
 
1 select id,point_name,ST_ASGEOJSON(point_geom) as geojson from meteorological_point where id = 1
1 insert into meteorological_point(point_name, point_geom) values("新帅集团监测点", ST_GEOMFROMTEXT("POINT(117.420671499 40.194914201)"))
4. SpringBoot + MyBatisPlus + MySQL 集成空间数据实战 在我们的Java项目中操作空间数据一般有两种方式:
使用上面的ST_ASGEOJSON,ST_GEOMFROMTEXT等方法用原生sql直接操作数据; 
使用MyBatisPlus中的typeHandler在切面层做类型转换; 
 
因为我们的项目已经集成了MyBatisPlus且集成度较高,而且切面处理更为优雅便捷,所以这里使用第二种。
4.1 实现GeometryTypeHandler.class工具类 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 package  org.jeecg.config.mybatis;import  com.vividsolutions.jts.geom.Geometry;import  com.vividsolutions.jts.geom.GeometryFactory;import  com.vividsolutions.jts.geom.PrecisionModel;import  com.vividsolutions.jts.io.*;import  org.apache.ibatis.type.BaseTypeHandler;import  org.apache.ibatis.type.JdbcType;import  org.apache.ibatis.type.MappedJdbcTypes;import  org.apache.ibatis.type.MappedTypes;import  java.io.ByteArrayOutputStream;import  java.io.InputStream;import  java.sql.CallableStatement;import  java.sql.PreparedStatement;import  java.sql.ResultSet;import  java.sql.SQLException;@MappedTypes({String.class}) @MappedJdbcTypes({JdbcType.OTHER}) public  class  GeometryTypeHandler  extends  BaseTypeHandler <String> {@Override public  void  setNonNullParameter (PreparedStatement preparedStatement, int  i, String s, JdbcType jdbcType)  throws  SQLException {Geometry  geo  =  null ;try {new  WKTReader (new  GeometryFactory (new  PrecisionModel ())).read(s);byte [] geometryBytes = new  WKBWriter (2 , ByteOrderValues.LITTLE_ENDIAN, false ).write(geo);byte [] wkb = new  byte [geometryBytes.length+4 ];0 ] = wkb[1 ] = wkb[2 ] = wkb[3 ] = 0 ;0 , wkb, 4 , geometryBytes.length);catch  (ParseException e){@Override public  String getNullableResult (ResultSet resultSet, String s) {try (InputStream  inputStream  =  resultSet.getBinaryStream(s)){Geometry  geo  =  getGeometryFromInputStream(inputStream);if (geo != null ){return  geo.toString();catch (Exception e){return  null ;@Override public  String getNullableResult (ResultSet resultSet, int  i) {try (InputStream  inputStream  =  resultSet.getBinaryStream(i)){Geometry  geo  =  getGeometryFromInputStream(inputStream);if (geo != null ){return  geo.toString();catch (Exception e){return  null ;@Override public  String getNullableResult (CallableStatement callableStatement, int  i)  throws  SQLException {return  "" ;private   Geometry getGeometryFromInputStream (InputStream inputStream)  throws  Exception {Geometry  dbGeometry  =  null ;if  (inputStream != null ) {byte [] buffer = new  byte [255 ];int  bytesRead  =  0 ;ByteArrayOutputStream  baos  =  new  ByteArrayOutputStream ();while  ((bytesRead = inputStream.read(buffer)) != -1 ) {0 , bytesRead);byte [] geometryAsBytes = baos.toByteArray();if  (geometryAsBytes.length < 5 ) {byte [] sridBytes = new  byte [4 ];0 , sridBytes, 0 , 4 );boolean  bigEndian  =  (geometryAsBytes[4 ] == 0x00 );int  srid  =  0 ;if  (bigEndian) {for  (int  i  =  0 ; i < sridBytes.length; i++) {8 ) + (sridBytes[i] & 0xff );else  {for  (int  i  =  0 ; i < sridBytes.length; i++) {0xff ) << (8  * i);WKBReader  wkbReader  =  new  WKBReader ();byte [] wkb = new  byte [geometryAsBytes.length - 4 ];4 , wkb, 0 , wkb.length);return  dbGeometry;
4.2 实体类字段上加类型转换注解 1 2 3 @ApiModelProperty(value = "空间点位") @TableField(typeHandler = GeometryTypeHandler.class) private  String geomPoint;
此处有坑,还需在实体类加autoResultMap = true注解
1 2 3 4 @TableName(value = "f_metadata",autoResultMap = true) public  class  Metadata  extends  JeecgEntity  {
由于能查到的资料很有限,这里卡了半天,中间还尝试了MySQL方言 + Geometry格式化注解的方式,失败了
 
至此,我们的后台服务已经可以支持基本的增加、分页查询操作,空间数据可以完成自动转换。
4.3 MyBatisPlus的QueryWrapper构造空间查询 我们在项目开发中除了基本的分页查询,还要用到区域检索,而前面提到MySQL提供了大量的空间函数去支持空间查询。
MBRContains(A,B) –> A包含B 
MBRWithin(A,B) –> A在B中 
 
文末附MySQL空间处理函数和方法 
所以,我们可以利用相关函数去做区域查询,如:
1 select * from f_metadata where MBRContains (ST_GeomFromText('Polygon((89.76 41.31,125.36 44.56,117.07 23.29,92.33 23.39,89.76 41.31))' ) ,geom_point)
该sql语句可以检索到geom_point在此闭合区域内的所有数据。
现在我们需要考虑下QueryWrapper有没有MBRContains方法,查了下果然没有,那么怎么用QueryWrapper来构造空间查询呢?
通过查看QueryWrapper的所有内部方法,我发现了exist函数:
 
1 2 3 default  Children exists (String existsSql)  {return  this .exists(true , existsSql);
它可以传入原生sql语句,然后先执行外层查询,再用结果去匹配是否存在于exists内,如果为true,则作为结果返回。
1 select * from f_metadata where EXISTS (select* from (select * from f_metadata where Contains(ST_GeomFromText('Polygon((89.76 41.31,125.36 44.56,117.07 23.29,92.33 23.39,89.76 41.31))' ) ,geom_point)) as b where f_metadata.id = b.id  );
后台代码中的QueryWrapper构造:
1 2 3 4 5 6 7 if  (ObjectUtil.isNotEmpty(metadata.getGeomPoint())) {"select * from (select * from f_metadata where Contains(ST_GeomFromText('"  + metadata.getGeomPoint() + "'),geom_point)) as b where f_metadata.id = b.id" );return  queryWrapper;
搞定!
5. 附:MySQL空间数据处理函数和方法 空间查询函数 
包含相关
覆盖相关
相交相关
接触
重叠
相同
 
空间数据相关方法 
点独有
获取x或y
凸包
返回矩形
线独有
线是否闭合 
线中点数量 
线中第n个点 
线长度 
生成矩形 
 
面积
面的内外边界 
 
集合
交集 
异或 
并集 
质心 
距离 
不同 
抽稀 
 
缓冲区
 
5.7.7后可以添加策略影响缓冲区的计算,设置的语句是ST_Buffer_Strategy()
point策略
join策略
end策略
举例生成缓冲区
相交
重叠
接触
包含
验证数据是否合法
geo对象返回格式
 
注意:每个方法前的MBR、ST可要可不要,在mysql5.7.6之后,不带MBR、ST的方法开始弃用