禛陌 2019-08-15 12:20:10 6752 收藏 7
分类专栏: 技术相关 文章标签: Springboot+MyBatis-plus+postgresSQL
版权
MyBatis-plus 请参看https://mp.baomidou.com/,之前也有写过一个较简单的入门级文章,可以参看
在整合的过程中遇到了一些坑,来扒一扒。
(1) 首先在利用MyBatis-plus-generator 的AutoGenerator类生成代码阶段,死活不生成相应的类。换成MYSQL也是OK的, Oracle也是没有问题。写了一个PostresSQL的直连操作也OK,这说明驱动和连接地址是OK的,问题出在生成过程中。跟入源码,发现在生成时,postgres如果不指定命名空间,会自动加一个命名空间为public。我给数据库分了命名空间,所以,是这个原因,无法找到相应的表,所以一直不能生成相应的类,所以需要给dataSourceConfig.setSchemaName设置指定的命名空间.... 源码在 com.baomidou.mybatisplus.generator.config.builder.ConfigBuilder.getTablesInfo方法中如下图所示:
(2) 整合后又报了个破错:Caused by: java.sql.SQLFeatureNotSupportedException: 这个 org.postgresql.jdbc.PgConnection.createClob() 方法尚未被实作。网上找了一个配制,制完可以解决这个问题
spring.jpa.properties.hibernate.jdbc.lob.non_contextual_creation=true
(3) 类生成了,但发现生成的相应MAPP无法被扫描到,手动在相应的mapper中加了@Mapper注解,这可累死了,所以去掉注解,直接在springboot启动类中配置@MapperScan(mapper的相应名表)。
(4) 写了个测试用例 ,死活告诉我说表名不存在,实际上是没有加上命名空间,即表名前缀,所以死活报错过不去。接着跟源码吧,看了生成的地方,没有办法在生成mapper时把这个表名前缀加上。那只能想mybaits是如何工作的了。假装么想一下,也应该是把相应的表名和映射的类放在一个类似map的对象中,根据类名的表名之类 ,然后在拼装SQL,即然map生成了,那如何在拼时加上呢。嗯,得去查查springboot整合mybaits的配置,有没有加命名空间的,果然,搜到了加前缀的配制:
mybatis-plus:
global-config:
db-config:
table-prefix: platform
以上是在做整合、生成代码、测试过程中遇到的坑,下面来说说整合过程~~~~~~~~~~~~~~~~~~~~
org.projectlombok
lombok
1.16.16
org.postgresql
postgresql
42.2.5.jre7
com.baomidou
mybatis-plus-boot-starter
3.1.2
org.javassist
javassist
org.javassist
javassist
3.21.0-GA
org.freemarker
freemarker
# springboot整合postgres连接配制
spring:
datasource:
url: jdbc:postgresql://192.168.1.84:5432/supervison_platform
username: supervison
password: 123
driver-class-name: org.postgresql.Driver
#解决整合后报的一个错:
jpa:
properties:
hibernate:
jdbc:
lob:
non_contextual_creation: true
#mybatis-plus整合,加了表前缀的全局配制,加了类中与表中驼峰的映射,不加会查不出数据,因无法做表与类字段映射
mybatis-plus:
global-config:
db-config:
table-prefix: platform.
mapper-locations: classpath*:mapper/*Mapper.xml
type-aliases-package: com.zhanglu.test.supervison.api.dao.entity
configuration:
map-underscore-to-camel-case: true
package com.zhanglu.test.supervison.api.util;
import com.baomidou.mybatisplus.annotation.DbType;
import com.baomidou.mybatisplus.core.exceptions.MybatisPlusException;
import com.baomidou.mybatisplus.core.toolkit.StringUtils;
import com.baomidou.mybatisplus.generator.AutoGenerator;
import com.baomidou.mybatisplus.generator.InjectionConfig;
import com.baomidou.mybatisplus.generator.config.*;
import com.baomidou.mybatisplus.generator.config.rules.NamingStrategy;
import com.baomidou.mybatisplus.generator.engine.FreemarkerTemplateEngine;
import java.io.File;
import java.util.HashMap;
import java.util.Map;
import java.util.Scanner;
import static com.baomidou.mybatisplus.generator.config.rules.DateType.ONLY_DATE;
/**
* @author zhanglu
* @ClassName: CodeGeneration
* @Description: 代码生成器
* @date 2019年8月14日 下午2:55:14
*/
public class CodeGeneration {
public static String scanner(String tip) {
/*
* 查询出当前库所有表信息
* SELECT A.tablename, obj_description(relfilenode, 'pg_class') AS comments FROM pg_tables A, pg_class B WHERE A.schemaname='platform' AND A.tablename = B.relname
*/
Scanner scanner = new Scanner(System.in);
StringBuilder help = new StringBuilder();
help.append("请输入" + tip + ":");
System.out.println(help.toString());
if (scanner.hasNext()) {
String ipt = scanner.next();
if (StringUtils.isNotEmpty(ipt)) {
return ipt;
}
}
throw new MybatisPlusException("请输入正确的" + tip + "!");
}
/**
* @param args
* @Title: main
* @Description: 生成
*/
public static void main(String[] args) {
AutoGenerator mpg = new AutoGenerator();
// 全局配置
GlobalConfig gc = new GlobalConfig();
final String projectPath = System.getProperty("user.dir") + "/supervison-api";
String filePath = projectPath + "/src/main/java";
System.out.println("生成文件 的路径为:" + filePath);
gc.setOutputDir(filePath);
// gc.setOutputDir("E://code");
// gc.setFileOverride(true);
// gc.setActiveRecord(false);// 不需要ActiveRecord特性的请改为false
// gc.setEnableCache(false);// XML 二级缓存
// gc.setBaseResultMap(true);// XML ResultMap
// gc.setBaseColumnList(false);// XML columList
gc.setOpen(false);
gc.setDateType(ONLY_DATE);
gc.setAuthor("zhanglu_autoGeneration");// 作者
// 自定义文件命名,注意 %s 会自动填充表实体属性!
// gc.setControllerName("%sAction");
// gc.setServiceName("%sService");
// gc.setServiceImplName("%sServiceImpl");
// gc.setMapperName("%sMapper");
// gc.setXmlName("%sMapper");
mpg.setGlobalConfig(gc);
// 数据源配置
DataSourceConfig dsc = new DataSourceConfig();
dsc.setDriverName("org.postgresql.Driver");
dsc.setUsername("supervison");
dsc.setPassword("123");
dsc.setDbType(DbType.POSTGRE_SQL);
dsc.setUrl("jdbc:postgresql://10.212.170.84:5432/supervison_platform");
dsc.setSchemaName("platform");
mpg.setDataSource(dsc);
// 包配置
final PackageConfig pc = new PackageConfig();
// pc.setModuleName("");
pc.setParent("com.chnenergy.monitoring.supervison.api.dao");
//以下生成的类类型的map的KEY值,可以去常量类中ConstVal获得,为了省事,直接写了字符串
Map m = new HashMap();
m.put("entity_path", gc.getOutputDir() + File.separator + (pc.getParent().replaceAll("\\.", "\\" + File.separator)) + "/entity");
m.put("mapper_path", gc.getOutputDir() + File.separator + (pc.getParent().replaceAll("\\.", "\\" + File.separator)) + "/mapper");
// m.put("service_path",gc.getOutputDir() + File.separator + (pc.getParent().replaceAll("\\.", "\\" + File.separator)) +"/service");
// m.put("service_impl_path",gc.getOutputDir() + File.separator + (pc.getParent().replaceAll("\\.", "\\" + File.separator)) + "/service/impl");
// m.put(ConstVal.CONTROLLER_PATH,gc.getOutputDir() + File.separator + (pc.getParent().replaceAll("\\.", "\\" + File.separator)) + "controller");
// m.put(ConstVal.XML_PATH,gc.getOutputDir() + File.separator + (pc.getParent().replaceAll("\\.", "\\" + File.separator)) + "/xml");
pc.setPathInfo(m);
// 自定义配置
InjectionConfig cfg = new InjectionConfig() {
@Override
public void initMap() {
// to do nothing
}
};
cfg.setFileOutConfigList(null);
mpg.setCfg(cfg);
mpg.setPackageInfo(pc);
// 配置模板
TemplateConfig templateConfig = new TemplateConfig();
// 配置自定义输出模板
//指定自定义模板路径,注意不要带上.ftl/.vm, 会根据使用的模板引擎自动识别
// templateConfig.setEntity("templates/entity2.java");
// templateConfig.setService();
// templateConfig.setController();
// templateConfig.setXml(TEMPLATE_XML);
mpg.setTemplate(templateConfig);
// 策略配置
StrategyConfig strategy = new StrategyConfig();
strategy.setNaming(NamingStrategy.underline_to_camel);
strategy.setColumnNaming(NamingStrategy.underline_to_camel);
// strategy.setTablePrefix("platform");
//strategy.setSuperEntityClass("com.baomidou.ant.common.BaseEntity");
strategy.setEntityLombokModel(true);
strategy.setRestControllerStyle(true);
// 写于父类中的公共字段
//strategy.setSuperEntityColumns("id");
strategy.setInclude(scanner("表名,多个英文逗号分割").split(","));
// strategy.setControllerMappingHyphenStyle(true);
// strategy.setTablePrefix(pc.getModuleName() + "_");
mpg.setStrategy(strategy);
mpg.setTemplateEngine(new FreemarkerTemplateEngine());
// 执行生成
mpg.execute();
}
}
当然,完全可以去掉xml,没啥用,也可以生成实现类和controller类,SQL可以通过JPA方式直接写就行,eg:
@Select("SELECT * FROM ZHANGLU_OFFICIAL_NEWS N WHERE N. ID IN( SELECT G .NEWS_ID FROM ZHANGLU_OFFICIAL_GROUP_INTE G WHERE G . GROUP_ID = #{groupId}) ORDER BY CREATE_DATE")
IPage selectPageByGroupId(IPage page, @Param("groupId") String groupId);
package com.zhanglu.test.supervison.api.service.impl;
import com.zhanglu.test.supervison.api.dao.entity.PhoOrganization;
import com.zhanglu.test.supervison.api.dao.mapper.PhoOrganizationMapper;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringRunner;
@SpringBootTest
@RunWith(SpringRunner.class)
public class PhoOrganizationServiceImplTest {
@Autowired
private PhoOrganizationMapper mapper;
@Test
public void query() {
PhoOrganization organization = mapper.selectById("ac524ede04b847079cb44c9db4fde14e");
System.out.println(organization.getAddress());
}
}
以上就OK了,大半天遇到的几个小坑。整理下来,希望对大家有点帮助