基于Mybatis-Plus实现Geometry字段在PostGis空间数据库中的使用

简介: 本文讲解在mybatis-plus中操作geometry空间字段,同时实现查询和插入操作​。通过geojson,结合前端可视化组件即可完成​矢量数据的空间可视化。

背景

      之前的一些个人文章介绍了空间数据库,以及Mybatis-Plus快速操作数据库组件,以及空间数据库PostGis的相关介绍。现在基于在空间数据库中已经定义了一张空间表,需要在应用程序中使用Mybatis-Plus来进行空间数据的查询、插入等常规操作。

      在OGC标准中,通常空间字段是由Geometry类型来表示。而一般编程语言中是没有这种数据类型的。以java为例,怎么操作这些数据,满足业务需求呢?跟着本文一起来学习吧。

     今天介绍基于postgis-jdbc的geometry属性的操作。

一、在pom.xml中引入postgis-jdbc相关jar包

<!-- PostgreSql 驱动包 add by wuzuhu on 2022-08-16 --><dependency><groupId>net.postgis</groupId><artifactId>postgis-jdbc</artifactId><version>2.5.0</version></dependency>

二、需要自定义Handler类来扩展字段支持。

packagecom.hngtghy.framework.handler;
importjava.sql.CallableStatement;
importjava.sql.PreparedStatement;
importjava.sql.ResultSet;
importjava.sql.SQLException;
importorg.apache.ibatis.type.BaseTypeHandler;
importorg.apache.ibatis.type.JdbcType;
importorg.apache.ibatis.type.MappedTypes;
importorg.postgis.Geometry;
importorg.postgis.PGgeometry;
@MappedTypes({String.class})
publicclassPgGeometryTypeHandlerextendsBaseTypeHandler<String> {
@OverridepublicvoidsetNonNullParameter(PreparedStatementps, inti, Stringparameter, JdbcTypejdbcType) throwsSQLException {
PGgeometrypGgeometry=newPGgeometry(parameter);
Geometrygeometry=pGgeometry.getGeometry();
geometry.setSrid(4326);
ps.setObject(i, pGgeometry);
    }
@OverridepublicStringgetNullableResult(ResultSetrs, StringcolumnName) throwsSQLException {
Stringstring=rs.getString(columnName);
returngetResult(string);
    }
@OverridepublicStringgetNullableResult(ResultSetrs, intcolumnIndex) throwsSQLException {
Stringstring=rs.getString(columnIndex);
returngetResult(string);
    }
@OverridepublicStringgetNullableResult(CallableStatementcs, intcolumnIndex) throwsSQLException {
Stringstring=cs.getString(columnIndex);
returngetResult(string);
    }
privateStringgetResult(Stringstring) throwsSQLException {
PGgeometrypGgeometry=newPGgeometry(string);
Strings=pGgeometry.toString();
returns.replace("SRID=4326;", "");
    }
}

          注意,在getResult()中关于4326坐标系的定义,可以根据需要进行废弃。这里写上为了统一投影坐标系。

三、在数据中创建表,建表语句如下:

createtable biz_point_test(  id int8 primary key,  name varchar(100),  geom geometry(Point,4326));

四、定义Mybatis-plus实体

packagecom.hngtghy.project.extend.student.domain;
importcom.baomidou.mybatisplus.annotation.TableField;
importcom.baomidou.mybatisplus.annotation.TableId;
importcom.baomidou.mybatisplus.annotation.TableName;
importcom.hngtghy.framework.handler.PgGeometryTypeHandler;
importlombok.AllArgsConstructor;
importlombok.Getter;
importlombok.NoArgsConstructor;
importlombok.Setter;
importlombok.ToString;
@TableName(value="biz_point_test",autoResultMap=true)
@NoArgsConstructor@AllArgsConstructor@Setter@Getter@ToStringpublicclassPointTest {
@TableIdprivateLongid;
privateStringname;
@TableField(typeHandler=PgGeometryTypeHandler.class)
privateStringgeom;
@TableField(exist=false)
privateStringgeomJson;
}

     提醒:1、在属性上使用@TableField(typeHandler=xxx)来指定对应的类型转换器。2、需要在实体上定义autoResultMap=true。否则配置不一定生效。

五、定义mapper查询器

packagecom.hngtghy.project.extend.student.mapper;
importorg.apache.ibatis.annotations.Param;
importorg.apache.ibatis.annotations.Select;
importcom.baomidou.mybatisplus.core.mapper.BaseMapper;
importcom.hngtghy.project.extend.student.domain.PointTest;
publicinterfacePointTestMapperextendsBaseMapper<PointTest>{
staticfinalStringFIND_GEOJSON_SQL="<script>"+"select st_asgeojson(geom) as geomJson from biz_point_test "+"where id = #{id} "+"<if test='null != name'>and p.name like concat('%', #{name}, '%')</if>"+"</script>";
@Select(FIND_GEOJSON_SQL)
PointTestfindGeoJsonById(@Param("id")Longid,@Param("name")Stringname);
}

六、定义service业务类

packagecom.hngtghy.project.extend.student.service.impl;
importjava.util.List;
importorg.springframework.beans.factory.annotation.Autowired;
importorg.springframework.stereotype.Service;
importcom.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
importcom.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
importcom.hngtghy.project.extend.student.domain.PointTest;
importcom.hngtghy.project.extend.student.mapper.PointTestMapper;
importcom.hngtghy.project.extend.student.service.IPointTestService;
@ServicepublicclassPointTestServcieImplextendsServiceImpl<PointTestMapper, PointTest>implementsIPointTestService{
@AutowiredprivatePointTestMapperpointMapper;
@OverridepublicPointTestselectById(Longid) {
returnpointMapper.selectById(id);
    }
@OverridepublicList<PointTest>selectList(PointTestpoint) {
QueryWrapper<PointTest>queryWrapper=newQueryWrapper<PointTest>();
queryWrapper.select("id,name");
returnthis.getBaseMapper().selectList(queryWrapper);
    }
@OverridepublicintinsertPointTest(PointTestpoint) {
returnpointMapper.insert(point);
    }
@OverridepublicintupdatePoint(PointTestpoint) {
returnpointMapper.updateById(point);
    }
@OverridepublicPointTestselectGeomById(Longid) {
QueryWrapper<PointTest>queryWrapper=newQueryWrapper<PointTest>();
queryWrapper.select("geom","st_asgeojson(geom) as geomJson");
queryWrapper.eq("id", id);
returnthis.getBaseMapper().selectOne(queryWrapper);
    }
@OverridepublicPointTestfindGeoJsonById(Longid) {
returnpointMapper.findGeoJsonById(id, null);
    }
}

      这里添加了一个数据库不存在的字段geomJson,会将空间属性转变成geojson字段,方便于前台的如leaflet、openlayers、cesium等组件进行展示。所以使用postgis的st_asgeojson(xxx)进行函数转换。

七、相关方法调用

//1、列表查询
List<PointTest> pointList = pointService.selectList(null);System.out.println(pointList);[PointTest(id=1559371184090423297, name=中寨居委会, geom=null, geomJson=null), PointTest(id=2, name=禾滩村, geom=null, geomJson=null), PointTest(id=1559403683801796610, name=中寨居委会, geom=null, geomJson=null)]//2、插入
PointTest point = new PointTest();point.setName("中寨居委会");point.setGeom("POINT(109.262605 27.200669)");//POINT(lng,lat) 经度,纬度
pointService.insertPointTest(point);//3、查询数据
 PointTest point = pointService.selectGeomById(1559371184090423297L); PointTest json = pointService.findGeoJsonById(1559371184090423297L);PointTest(id=null, name=null, geom=POINT(109.26260527.200669), geomJson={"type":"Point","coordinates":[109.262605,27.200669]})PointTest(id=null, name=null, geom=null, geomJson={"type":"Point","coordinates":[109.262605,27.200669]})

八、使用pgadmin可以查看到相应的点数据,如下图所示:

image.png

image.png

      总结:通过以上步骤可以实现在mybatis-plus中操作geometry空间字段,同时实现查询和插入操作。通过geojson,结合前端可视化组件即可完成矢量数据的空间可视化。希望本文可以帮到你,欢迎交流。

目录
相关文章
|
4月前
|
Java 数据库连接 数据库
mybatis查询数据,返回的对象少了一个字段
mybatis查询数据,返回的对象少了一个字段
347 8
|
15天前
|
SQL Java 数据库连接
Mybatis拦截器实现公共字段填充
通过使用MyBatis拦截器,可以实现对公共字段的自动填充,简化代码,提高开发效率。拦截器通过拦截SQL操作,在插入和更新操作时自动填充公共字段,使得开发者不再需要手动设置这些字段。本文详细介绍了实现步骤,并通过示例代码展示了具体实现方法,希望能为您的开发工作提供实用的指导和帮助。
60 13
|
27天前
|
前端开发 JavaScript 数据库
获取数据库中字段的数据作为下拉框选项
获取数据库中字段的数据作为下拉框选项
54 5
|
2月前
|
SQL Java 数据库连接
深入 MyBatis-Plus 插件:解锁高级数据库功能
Mybatis-Plus 提供了丰富的插件机制,这些插件可以帮助开发者更方便地扩展 Mybatis 的功能,提升开发效率、优化性能和实现一些常用的功能。
399 26
深入 MyBatis-Plus 插件:解锁高级数据库功能
|
2月前
|
SQL Java 数据库连接
【MyBatisPlus·最新教程】包含多个改造案例,常用注解、条件构造器、代码生成、静态工具、类型处理器、分页插件、自动填充字段
MyBatis-Plus是一个MyBatis的增强工具,在 MyBatis 的基础上只做增强不做改变,为简化开发、提高效率而生。本文讲解了最新版MP的使用教程,包含多个改造案例,常用注解、条件构造器、代码生成、静态工具、类型处理器、分页插件、自动填充字段等核心功能。
【MyBatisPlus·最新教程】包含多个改造案例,常用注解、条件构造器、代码生成、静态工具、类型处理器、分页插件、自动填充字段
|
2月前
|
SQL 安全 Java
MyBatis-Plus条件构造器:构建安全、高效的数据库查询
MyBatis-Plus 提供了一套强大的条件构造器(Wrapper),用于构建复杂的数据库查询条件。Wrapper 类允许开发者以链式调用的方式构造查询条件,无需编写繁琐的 SQL 语句,从而提高开发效率并减少 SQL 注入的风险。
53 1
MyBatis-Plus条件构造器:构建安全、高效的数据库查询
|
2月前
|
SQL Java 数据库连接
canal-starter 监听解析 storeValue 不一样,同样的sql 一个在mybatis执行 一个在数据库操作,导致解析不出正确对象
canal-starter 监听解析 storeValue 不一样,同样的sql 一个在mybatis执行 一个在数据库操作,导致解析不出正确对象
|
3月前
|
缓存 数据库 数据安全/隐私保护
Discuz! X 数据库字典详解:DZ各数据表作用及字段含义
我们使用DISCUZ做网站时,有时需要对数据表进行操作,在操作数据表之前,需要对数据表进行了解。下面是DISCUZ 数据库各数据表作用及字段含义详解,方便新手更好的了解DISCUZ数据库。
84 4
|
3月前
|
SQL Java 数据库连接
mybatis如何仅仅查询某个表的几个字段
【10月更文挑战第19天】mybatis如何仅仅查询某个表的几个字段
124 1
|
3月前
|
Java 关系型数据库 MySQL
springboot学习五:springboot整合Mybatis 连接 mysql数据库
这篇文章是关于如何使用Spring Boot整合MyBatis来连接MySQL数据库,并进行基本的增删改查操作的教程。
478 0
springboot学习五:springboot整合Mybatis 连接 mysql数据库