公司的项目需要从SpringMVC迁移到SpringBoot2.0,本人用了三天的时间才基本完成迁移,今天就来大体的做一下总结
SpringBoot2.0将HikariCP替换原来的Tomcat作为默认的数据库连接池(众心所向)。
下面就说一下在配置中我们需要做的变化
原来我们在配置读写分离的数据库,是这样配置的
spring.datasource.readwrite.url=jdbc:mysql://127.0.0.1:3306/bookSystem?characterEncoding=utf-8&useSSL=false
spring.datasource.readwrite.username=root
spring.datasource.readwrite.password=123456
spring.datasource.readwrite.driver-class-name=com.mysql.jdbc.Driver
如果升级后还保持原有配置会出现错误
HikariPool-1 - jdbcUrl is required with driverClassName
而在升级以后我们需要如何配置呢?
spring.datasource.readwrite.jdbc-url=jdbc:mysql://127.0.0.1:3306/bookSystem?characterEncoding=utf-8&useSSL=false
spring.datasource.readwrite.username=root
spring.datasource.readwrite.password=123456
spring.datasource.readwrite.driver-class-name=com.mysql.jdbc.Driver
可以看出url前面加上了jdbc
当然既然是使用了读写分离的数据库,光做这些是不够的,需要进行手动配置
@Bean
// 设置为首选的数据源
@Primary
// 读取配置
@ConfigurationProperties(prefix="spring.datasource.readwrite")
public DataSource dataSource() {
return DataSourceBuilder.create().build();
}
也许有的朋友还不知道配置文件是如何读取到配置类的我们就简单说一下
可能上面说的有点抽象,下面通过一个实例来进行进一步的解释
application.yml写了这样几行配置
props:
map:
key: 123
key1: 456
test: 123456
读取类
@Component
@Data
@ConfigurationProperties(prefix = "props")
public class Props {
private Map map = new HashMap<>();
private String test;
}
可以发现我们先配置一个前缀,让配置类找到props,然后通过属性与配置的一一对应进行匹配,现在明白了如何配置,我们就来看一下HikariConfig
private String driverClassName;
private String jdbcUrl;
我们可以从源码中看到这两个属性,这也就是我们要设置jdbc-url的原因
故事到这里只是刚刚开始,请大家耐心去看
springboot2默认需要4.0以上的gradle了,所以我们修改一下gradle-wrapper.properties
distributionUrl=https\://services.gradle.org/distributions/gradle-4.1-all.zip
还有一个重要的地方,gradle的依赖管理进行了升级,在gradle中加入一个插件即可
apply plugin: 'org.springframework.boot'
apply plugin: 'io.spring.dependency-management'
在打包时的命令也发生了变化,可以使用gradle bootjar或者gradle bootwar来进行打包,然后gradle bootrun运行
当然,要补充一点,在boot2.0迁移的官方文档中说,推荐我们加入
runtime(“org.springframework.boot:spring-boot-properties-migrator”)
只要将其作为依赖添加到项目中,它不仅会分析应用程序的环境并在启动时打印诊断信息,而且还会在运行时为项目临时迁移属性
ps:boot2的报错真的有点少,我遇到了多次什么报错信息都没有Hikari就自动关闭的情况
### ORM
由于我们的项目还是使用的Hibernate,所以起初想着平滑迁移,便没有改变,但是发现在Hibernate5.2.1以上已经不推荐Criteria,这代表着正在逐渐向JPA标准化进行过度,所以下面给出两种替换方式
demo
// 传入Pageable对象和quizId,返回一个Page对象
public interface QuizRepository extends JpaRepository<QuizEntity, Integer> {
Page findByQuizId(long quizId, Pageable pageable);
}
Pageable
PageRequest pageRequest = PageRequest.of(page - 1, size, Sort.by(Sort.Direction.DESC, "createTime"));
demo
@Repository
public class QuizDao {
// 注入EntityManager
@Resource
private EntityManager entityManager;
public Pair> search(String keyword, int page, int size) {
// 创建构造器
CriteriaBuilder builder = entityManager.getCriteriaBuilder();
// 设置语句查询对应的实体
CriteriaQuery criteria = builder.createQuery(QuizEntity.class);
// 设置from的来源
Root root = criteria.from(QuizEntity.class);
// 设置查询的条件
criteria.where(builder.ge(root.get("status"), 0));
// 设置排序的属性
criteria.orderBy(builder.asc(root.get("createTime")));
TypedQuery query = entityManager.createQuery(criteria);
// 获取总数据量
long total = query.getResultList().size();
// 设置第几页,和每页的数据
query.setFirstResult((page - 1) * size);
query.setMaxResults(size);
List resultList = query.getResultList();
return new Pair<>(total, resultList);
}
}
可以发现一种更加方便快捷,一种更加灵活,大家可以自行选型,但当我使用JPA时,遇到了问题。
在我使用UPDATE时发生了error,最后发现需要进行事务和标注
demo
@Transactional(rollbackFor = Exception.class)
public interface QuizRepository extends JpaRepository<QuizEntity, Integer> {
@Modifying
@Query("update tr_quiz q set q.readState=true where q.quizId = ?1 and q.lessonId = ?2 and q.status >= 0")
void readAll(long quizId, long lessonId);
}
@Transactional和@Modifying注解大家一定不要忘记。
lombok相信大家基本都用过,就是可以通过注解来生成构造函数,getset方法等的包,而在boot2中引入最新版时,遇到了一些问题
通过查看官方文档,发现了下面这句话
BREAKING CHANGE: lombok config key lombok.addJavaxGeneratedAnnotation now defaults to false instead of true. Oracle broke this annotation with the release of JDK9, necessitating this breaking change.
lombok在最新版本中默认lombok.addJavaxGeneratedAnnotation为false
这导致了通过http请求获取数据进行转化时的失败,需要我们手动配置一下,所以我选择了降级到1.16.18省去配置的麻烦
如果使用的client为Jedis,那恭喜你,你又需要做转变了,因为boot2.0中默认为lettuce,我们需要修改一下gradle的配置
compile('org.springframework.boot:spring-boot-starter-data-redis') {
exclude module: 'lettuce-core'
}
compile('redis.clients:jedis')
我们集群中的Cassandra版本比较老,所以不能使用
compile(‘org.springframework.boot:spring-boot-starter-cassandra’)
需要使用
compile(‘com.datastax.cassandra:cassandra-driver-core:2.1.7.1’) compile(‘com.datastax.cassandra:cassandra-driver-mapping:2.1.7.1’)
但是,引入包后一直发生错误,又是Hikari自动停止,后来仔细观察了包的依赖关系(使用./gradlew dependencyInsight –dependency cassandra-driver-core可以分析)
发现在mapping中包含了core,于是去掉core,果然项目跑起来了- -,很激动。
最后再来说一下kafka在boot中的基本使用,今天直接少最简单的一种单线程消费
我们只需要创建两个类
@Slf4j
@Component
public class Consumer {
@KafkaListener(topics = {"test"})
public void process(ConsumerRecord record) {
String topic = record.topic();
String key = record.key().toString();
String message = record.value().toString();
}
}
@Slf4j
@Component
public class Producer {
@Resource
private KafkaTemplate kafkaTemplate;
public void send(String topic, String message) {
log.info("send message: topic: " + topic + " message: " + message);
kafkaTemplate.send(topic, message);
}
public void send(String topic, String key, String message) {
log.info("send message: topic: " + topic + " key: " + key + " message: " + message);
kafkaTemplate.send(topic, key, message);
}
}
怎么样?是不是很简单,但一定不要忘了在application.properties里配置一下,下面给出基本的配置
#kafka
#producer
// bootstrap-servers代替原来的broker.list
spring.kafka.bootstrap-servers=localhost:9092
// 生产者key的序列化方式
spring.kafka.producer.key-serializer=org.apache.kafka.common.serialization.StringSerializer
// 生产者value的序列化方式
spring.kafka.producer.value-serializer=org.apache.kafka.common.serialization.StringSerializer
#consumer
spring.kafka.consumer.group-id=test_group
spring.kafka.consumer.enable-auto-commit=true
// 消费者key的反序列化方式
spring.kafka.consumer.key-deserializer=org.apache.kafka.common.serialization.StringDeserializer
// 消费者value的反序列化方式
spring.kafka.consumer.value-deserializer=org.apache.kafka.common.serialization.StringDeserializer
最后,当调试时不要忘记在application。properties中设置
debug=true
打开debug可以看到更清晰的调试信息。
吃水不忘挖井人,附上boot2.0的官方迁移文档
官方迁移文档
最激动人心的不是站在高处时的耀眼,而是无人问津时的默默付出