时间:2021-07-01 10:21:17 帮助过:2人阅读
想写这类文章一经很久了,一直没时间,其实也不是真的没时间,只是感觉太累了,而且自己学的不够深,就一直搁浅。算下来,从真的开始学习写代码到现在已经四年了。
到了新公司这半年,确实每天都跟打了鸡血一样(每晚11点30分左右下班,回家还要看会书,早上七点起床到公司),没感觉到累,只是感觉压力太大,现在算是适应了,对Java算一个全新的认识,排名第一的语言总是有它的独特之处,强大的社区支持,Java EE其实真的博大精深,虽然大家都不采用,但是设计理念真的很先进,不过稍微重量级一点,而且强依赖了Java EE容器,预期说容器,我觉得不如说框架。强大归强大,大家都不领情,随着spring等轻量级工具(说工具,其实是spring的设计理念都是非入侵式的,然而我个人感觉整个spring 的建设都是围绕着Java EE,并试图简化Java EE)。
说了这么多,并不是觉得自己牛逼,牛逼的人大有人在,我在公司现在估计连螺丝钉都算不上…讲这些只是想说说自己的经历以及对自己学习过程中的一些想法的纪录。
进入正题,瞎扯了这么多,哈哈哈~~
目前广泛使用的应该还是mysql,nosql。对于NoSQL目前我还没有太多的认识,长路漫漫,慢慢学习中…
在Java中,JDBC应该是被大家摒弃的,因为如果真的采用JDBC去写代码,太多模版代码,有点洁癖的程序猿估计都不想看到自己的代码冗余,会想方设法的去减少模版代码,这个时候就有太多的工具了,意图减少此类模板代码(其实主要的模版代码是建立connection,组装sql,获取结果集,组装POJO,顺便说明一下,有人主张划分VO,DTO,POJO,经过我自己写代码,有好处有不好的地方,看自己的偏好吧),spring jdbctemplate(技术应该已经过时)、hibernate、mybatis、JSR-220、spring data…东西太多了,我知道的太有限,但是估计大家都知道,这些框架,或者工具的简历都离不开JDBC,由此可见JDBC其实还是很重要的,虽然这些工具简化了数据库的操作,但是如果想知道为什么这么设计,了解JDBC你就知道了~~~
这个问题其实我在网上查了很久,翻了一下我的第一条查询纪录,大概是四月份,在两个月之前我就开始查了, 一直没找到合适的解释,不过我还是想要说说我自己的看法.
对于以上内容,估计大部分公司都会做出自己的选择,我也在网上咨询了很多扣扣群,问了群里的大神们自己对数据库框架的选型,其实貌似也没告诉我为什么。一下就是我对框架的看法:
对于这么多框架,其实主要使用的就是hibernate、mybatis。spring data的使用应该还比较少。我问了我很多同事,大家都说hibernate太重,其实功能强大为什么不重呢…mybatis大家可以定制sql,扪心自问,我们有多少数据库查询需要这么care数据库性能。如果我来制定数据库技术方案,我肯定优先使用spring data + JPA + hibernate,实在不行再采用spring + mybatis,原因如下:我自己做事情比较喜欢标准化,因为JPA是标准,而hibernate则很好的支持了JPA,spring data有效的减少了模版代码的产生,加上spring data 可以与批处理框架结合使用,因此我觉得使用它真的是不二之选
大部分网上的资料对JDBC的介绍过于简单,只提及了JDBC都不算主要功能的功能。在我看来,JDBC分为以下几部分:
为了获取链接,我们必须要有驱动,驱动其实就是数据库厂商为了适配JDBC而对自己的数据库客户端做的初始化而已,在JDBC 4.0之前我们需要这样初始化:
Class.forName("com.mysql.jdbc.Driver");
在JDBC 4.0以后我们将不在需要这行代码。
按照oracle的划分,数据库驱动分为四大类:
与其说oracle划分了四种类型,不如说oracle提供了四种方案,让我们可以选择。
对于数据库与业务之间,Connection是我们的主体部分,大部分异常处理其实都是这货给的。这里不得不说一句,Golang的多返回值是多么的好用呀,让异常都消失吧。在JDBC中获取链接的方式有两种,分别如下:
没什么好说的,我直接贴代码了~
private static final String DB_URL = "jdbc:mysql://localhost:3308/database";
private static final String DB_USERNAME = "admin";
private static final String DB_PASSWORD = "admin";
public Connection getConnection() throws SQLException{
Connection conn = null;
Properties connectionProps = new Properties();
connectionProps.put("user", DB_USERNAME);
connectionProps.put("password", DB_PASSWORD);
conn = DriverManager.getConnection(DB_URL, connectionProps);
return conn;
}
DataSource接口由各个驱动器厂商实现,oracle把他们划分如下:
oracle 规定,一个驱动至少实现一个基本的DataSource,在学习数据库的时候我相信大家肯定见过BaseicDataSouce这个东西,当时我看到的时候我也不明白,不过现在明白了,哈哈哈,在使用DataSource中其实比DriverManager更复杂,但是更实.
BasicDatabaseSource ds = BasicDatabaseSource();
ds.setServerName("server");
ds.setDatabaseName("database");
ds.setDescription("description");
Context ctx = new InitialContext();
ctx.bind("jdbc/sty", ds);
在学习之前,觉得JNDI到底是什么,其实花点时间,就知道了,道理很简单,反正Java EE容器做了。
Context ctx = new InitialContext();
DataSource ds = (DataSource)ctx.lookup("jdbc/sty");
Connection conn = ds.getConnection("server", "database");
使用DatabaseSource 解决了数据库url的硬编码,当然有很多方式可以解除硬编码,除此之外,DatabaseSource一般会和ConnectionPoolDataSrouce配合食用。类似的,Datasource还和XADataSource一起使用,实现分布式事务。
部署Pooled DataSource比Basic DataSource稍微复杂一些(分布式也是类似),不过都是模版话的配置,学习成本并不高。
之前提到过,Polled DataSource与基本的DataSource是配合使用的,意味着我们需要部署两个类。下面假设我们有两个类,第一个类是我们的池实现类:
class PooledDataSource implements javax.sql.ConnectionPoolDataSource{
// implements here
}
第二个类如下:
class FastDataSource implements javax.sql.DataSource{
// adapt ConnectionPoolDataSource here
}
为什么采用这样的方式,其实去看看JDBC的接口就可以了。
下面就是使用它们的时候了,首先我们注册我们的连接池类:
PooledDataSource pds = new PooledDataSource();
pds.setServerName("pds");
pds.setDatabaseName("database");
pds.setPortNumber(3800);
pds.setDescription("Polled DataSource");
Context ctx = new InitialContext();
ctx.bind("jdbc/pool/datasource", pds);
现在设置我们的另一个类:
FastDataSource fds = new FastDataSource();
fds.setDescription("produces pooled connections to COFFEEBREAK");
fds.setDataSourceName("jdbc/pool/fastCoffeeDB");
Context ctx = new InitialContext();
ctx.bind("jdbc/fastDatasource", fds);
使用数据源:
ctx = new InitialContext();
ds = (DataSource)ctx.lookup("jdbc/fastCoffeeDB");
与池比较类似,我们有两个类:
class XATransactionalDataSource implements javax.sql.XADataSource{
// implements here
}
class TransactionalDataSource implements javax.sql.DataSource{
// adapt here
}
部署如下:
XATransactionalDataSource xads = new XATransactionalDataSource();
xads.setServerName("xads");
xads.setDatabaseName("database");
xads.setPortNumber(9040);
xads.setDescription("Distributed transactions DataSource");
Context ctx = new InitialContext();
ctx.bind("jdbc/xa/xads", xads);
TransactionalDataSource tds = new TransactionalDataSource();
tds.setDescription("suibianla");
tds.setDataSourceName("jdbc/xa/xads");
Context ctx = new InitialContext();
ctx.bind("jdbc/ds", ds);
使用如下:
Context ctx = new InitialContext();
DataSource ds = (DataSource)ctx.lookup("jdbc/ds");
Connection con = ds.getConnection();
按照API的划分,Statement分为三大类:
这部分比较简单,再次就不在介绍了,以后有时间的话再来完善。
ResultSet 按照功能被划分为三类,这里直接使用它们的定义介绍了:
按照是否能修改的层数据库划分为两类:
读取ResultSet的时候有有一个概念需要介绍,那就是Cursor,翻译成光标感觉不太好,下文就直接沿用Cursor这个单词,当调用Collection.commit的时候ResultSet对象会自动关闭,有时候我们的需求可能并不是这样,这时候我们需要设置另一个属性,ResultSet提供了这个属性,描述如下:
事务的主要功能是把数据库操作划分成一个单元,这个单元可以看作是原子的,其中事务也有不同的等级划分,这里直接摘抄一个表格:
事务隔离等级 | 是否会话 | 读取的脏数据 | 幂等读取 | 幻觉的读取 |
---|---|---|---|---|
TRANSACTION_NONE | 不适用 | 不适用 | 不适用 | 不适用 |
TRANSACTION_READ_COMMITTED | 支持 | 不允许 | 允许 | 允许 |
TRANSACTION_READ_UNCOMMITTED | 支持 | 允许 | 允许 | 允许 |
TRANSACTION_REPEATABLE_READ | 支持 | 不允许 | 不允许 | 允许 |
TRANSACTION_SERIALIZABLE | 支持 | 不允许 | 不允许 | 不允许 |
Java学习之JDBC
标签: