- 浏览: 987709 次
文章分类
- 全部博客 (428)
- Hadoop (2)
- HBase (1)
- ELK (1)
- ActiveMQ (13)
- Kafka (5)
- Redis (14)
- Dubbo (1)
- Memcached (5)
- Netty (56)
- Mina (34)
- NIO (51)
- JUC (53)
- Spring (13)
- Mybatis (17)
- MySQL (21)
- JDBC (12)
- C3P0 (5)
- Tomcat (13)
- SLF4J-log4j (9)
- P6Spy (4)
- Quartz (12)
- Zabbix (7)
- JAVA (9)
- Linux (15)
- HTML (9)
- Lucene (0)
- JS (2)
- WebService (1)
- Maven (4)
- Oracle&MSSQL (14)
- iText (11)
- Development Tools (8)
- UTILS (4)
- LIFE (8)
最新评论
-
Donald_Draper:
Donald_Draper 写道刘落落cici 写道能给我发一 ...
DatagramChannelImpl 解析三(多播) -
Donald_Draper:
刘落落cici 写道能给我发一份这个类的源码吗Datagram ...
DatagramChannelImpl 解析三(多播) -
lyfyouyun:
请问楼主,执行消息发送的时候,报错:Transport sch ...
ActiveMQ连接工厂、连接详解 -
ezlhq:
关于 PollArrayWrapper 状态含义猜测:参考 S ...
WindowsSelectorImpl解析一(FdMap,PollArrayWrapper) -
flyfeifei66:
打算使用xmemcache作为memcache的客户端,由于x ...
Memcached分布式客户端(Xmemcached)
JDBC驱动初始化-Mysql:http://donald-draper.iteye.com/blog/2342010
JDBC连接的获取:http://donald-draper.iteye.com/blog/2342011
Mysql负载均衡连接的获取:http://donald-draper.iteye.com/blog/2342089
Mysql主从复制读写分离连接的获取:http://donald-draper.iteye.com/blog/2342108
ConnectionImp创建MysqlIO :http://donald-draper.iteye.com/blog/2342959
Mysql预编译SQL:http://donald-draper.iteye.com/blog/2342960
MysqlSQL PreparedStatement的查询:http://donald-draper.iteye.com/blog/2343083
MySQL ServerPreparedStatement查询:http://donald-draper.iteye.com/blog/2343124
用hibernate和mybatis做持久层时候,屏蔽jdbc的东西,现在有时间,我们来看一下jdbc底层东西。
首先创建mysql测试实例如下;
测试实例:
下面我们来看一下,从驱动加载,获取连接,及获取PreparedStatement,执行查询,结果集的处理的整个过程;
先看加载驱动
从这一句开始:
//Class
来看一下的mysql的Driver类
//com.mysql.jdbc.Driver
//DriverManager
从DriverManager的初始化过程可以看出,Driver的加载是委托给DriverService,而DriverService是通过
Service去加载,我们来看一下Service
//ServiceConfigurationError
至此,driver的加载结束,我们来回顾一下整个过程,Class加载com.mysql.jdbc.Driver,com.mysql.jdbc.Driver通过静态语句块,将Driver注册到DriverManager,DriverManager首先检查有没有初始化,没有则加载驱动类,整个驱动加载过程是通过DriverService,而DriverService则委托给Service,Service加载驱动是返回的是,所有java.sql.Driver实现的Iterator,Iterator为LazyIterator,
DriverService通过遍历LazyIterator,加载驱动所有驱动;最后将Driver包装成DriverInfo,添加到writeDrivers集合中,并clone到readDrivers。
JDBC连接的获取:http://donald-draper.iteye.com/blog/2342011
Mysql负载均衡连接的获取:http://donald-draper.iteye.com/blog/2342089
Mysql主从复制读写分离连接的获取:http://donald-draper.iteye.com/blog/2342108
ConnectionImp创建MysqlIO :http://donald-draper.iteye.com/blog/2342959
Mysql预编译SQL:http://donald-draper.iteye.com/blog/2342960
MysqlSQL PreparedStatement的查询:http://donald-draper.iteye.com/blog/2343083
MySQL ServerPreparedStatement查询:http://donald-draper.iteye.com/blog/2343124
用hibernate和mybatis做持久层时候,屏蔽jdbc的东西,现在有时间,我们来看一下jdbc底层东西。
首先创建mysql测试实例如下;
测试实例:
import java.sql.Connection; import java.sql.DriverManager; import java.sql.PreparedStatement; import java.sql.ResultSet; import java.sql.Statement; import java.sql.Timestamp; import java.util.Date; public class testMysqlX { public static void main(String[] args){ testMysqlConnection(); } @SuppressWarnings("deprecation") public static void testMysqlConnection() { Connection con = null;// 创建一个数据库连接 PreparedStatement pre = null;// 创建预编译语句对象,一般都是用这个而不用Statement ResultSet result = null;// 创建一个结果集对象 try { Class.forName("com.mysql.jdbc.Driver");// 加载驱动程序 System.out.println("开始尝试连接数据库!"); String url = "jdbc:mysql://localhost:3306/test?useUnicode=true&characterEncoding=utf8&characterSetResults=utf8"; String user = "root";// 用户名,系统默认的账户名 String password = "123456";// 你安装时选设置的密码 long startTime = System.currentTimeMillis(); con = DriverManager.getConnection(url, user, password);// 获取连接 PreparedStatement ps = con.prepareStatement("select count(*) from ?"); ps.setString(1, "user"); result = ps.executeQuery(); //result 初始游标为head,移动next到第一个记录 while(result.next()){ int sum = result.getInt(1); System.out.println("============sum:"+sum); } long endTime = System.currentTimeMillis(); System.out.println("============time:"+ (endTime-startTime)); System.out.println("============hashCode:"+ con.hashCode()); if(!con.isClosed()){ System.out.println("============连接成功!"); } } catch (Exception e) { System.out.println("=============连接失败:"+e.getMessage()); e.printStackTrace(); } finally { try { // 逐一将上面的几个对象关闭,因为不关闭的话会影响性能、并且占用资源 // 注意关闭的顺序,最后使用的最先关闭 if (result != null) result.close(); if (pre != null) pre.close(); if (con != null) con.close(); System.out.println("数据库连接已关闭!"); } catch (Exception e) { e.printStackTrace(); } } } }
下面我们来看一下,从驱动加载,获取连接,及获取PreparedStatement,执行查询,结果集的处理的整个过程;
先看加载驱动
从这一句开始:
Class.forName("com.mysql.jdbc.Driver");
//Class
public final class Class<T> implements java.io.Serializable, java.lang.reflect.GenericDeclaration, java.lang.reflect.Type, java.lang.reflect.AnnotatedElement { private static final int ANNOTATION= 0x00002000; private static final int ENUM = 0x00004000; private static final int SYNTHETIC = 0x00001000; private static native void registerNatives(); static { registerNatives(); } //根据类名加载对应class public static Class<?> forName(String className) throws ClassNotFoundException { return forName0(className, true, ClassLoader.getCallerClassLoader()); } /** Called after security checks have been made. */ private static native Class forName0(String name, boolean initialize, ClassLoader loader) throws ClassNotFoundException; }
来看一下的mysql的Driver类
//com.mysql.jdbc.Driver
public class Driver extends NonRegisteringDriver implements java.sql.Driver { public Driver() throws SQLException { } static { try { //将驱动注册到驱动管理器 DriverManager.registerDriver(new Driver()); } catch(SQLException E) { throw new RuntimeException("Can't register driver!"); } } }
//DriverManager
public class DriverManager { /* write copy of the drivers vector */ private static java.util.Vector writeDrivers = new java.util.Vector(); /* write copy of the drivers vector */ private static java.util.Vector readDrivers = new java.util.Vector(); private static int loginTimeout = 0; private static java.io.PrintWriter logWriter = null; private static java.io.PrintStream logStream = null; private static boolean initialized = false; //注册驱动 public static synchronized void registerDriver(java.sql.Driver driver) throws SQLException { if (!initialized) { initialize(); } //初始化驱动信息 DriverInfo di = new DriverInfo(); di.driver = driver; di.driverClass = driver.getClass(); di.driverClassName = di.driverClass.getName(); //将driver信息添加到writeDrivers集合中 writeDrivers.addElement(di); println("registerDriver: " + di); //克隆writeDrivers到readDrivers readDrivers = (java.util.Vector) writeDrivers.clone(); } //初始化 static void initialize() { if (initialized) { return; } initialized = true; //加载驱动,初始化 loadInitialDrivers(); println("JDBC DriverManager initialized"); } //加载驱动 private static void loadInitialDrivers() { String drivers; try { drivers = (String) java.security.AccessController.doPrivileged( new sun.security.action.GetPropertyAction("jdbc.drivers")); } catch (Exception ex) { drivers = null; } //如果driver是放在jar包中,通过类型加载器加载,所有Driver,每个java.sql.Driver.class是一个Service DriverService ds = new DriverService(); //获取所有驱动的访问权限 java.security.AccessController.doPrivileged(ds); println("DriverManager.initialize: jdbc.drivers = " + drivers); if (drivers == null) { return; } while (drivers.length() != 0) { int x = drivers.indexOf(':'); String driver; if (x < 0) { driver = drivers; drivers = ""; } else { driver = drivers.substring(0, x); drivers = drivers.substring(x+1); } if (driver.length() == 0) { continue; } try { println("DriverManager.Initialize: loading " + driver); //加载驱动 Class.forName(driver, true, ClassLoader.getSystemClassLoader()); } catch (Exception ex) { println("DriverManager.Initialize: load failed: " + ex); } } } class DriverService implements java.security.PrivilegedAction { Iterator ps = null; public DriverService() {}; public Object run() { // uncomment the followin line before mustang integration // Service s = Service.lookup(java.sql.Driver.class); // ps = s.iterator(); //通过Service加载所有Driver ps = Service.providers(java.sql.Driver.class); /* Load these drivers, so that they can be instantiated. * It may be the case that the driver class may not be there * i.e. there may be a packaged driver with the service class * as implementation of java.sql.Driver but the actual class * may be missing. In that case a sun.misc.ServiceConfigurationError * will be thrown at runtime by the VM trying to locate * and load the service. * * Adding a try catch block to catch those runtime errors * if driver not available in classpath but it's * packaged as service and that service is there in classpath. */ try { while (ps.hasNext()) { //初始化Driver ps.next(); } // end while } catch(Throwable t) { // Do nothing } return null; } //end run } //end DriverService //驱动信息类 class DriverInfo { Driver driver; Class driverClass; String driverClassName; public String toString() { return ("driver[className=" + driverClassName + "," + driver + "]"); } }
从DriverManager的初始化过程可以看出,Driver的加载是委托给DriverService,而DriverService是通过
Service去加载,我们来看一下Service
public final class Service { //加载驱动,委托给providers(Class class1, ClassLoader classloader) public static Iterator providers(Class class1) throws ServiceConfigurationError { ClassLoader classloader = Thread.currentThread().getContextClassLoader(); return providers(class1, classloader); } public static Iterator providers(Class class1, ClassLoader classloader) throws ServiceConfigurationError { //实际上返回的为一个Iterater,LazyIterator为Service的静态内部类 return new LazyIterator(class1, classloader); } //LazyIterator,懒加载驱动集合 private static class LazyIterator implements Iterator { public boolean hasNext() throws ServiceConfigurationError { if(nextName != null) return true; if(configs == null) try { String s = (new StringBuilder()).append("META-INF/services/").append(service.getName()).toString(); if(loader == null) configs = ClassLoader.getSystemResources(s); else configs = loader.getResources(s); } catch(IOException ioexception) { Service.fail(service, (new StringBuilder()).append(": ").append(ioexception).toString()); } for(; pending == null || !pending.hasNext(); pending = Service.parse(service, (URL)configs.nextElement(), returned)) if(!configs.hasMoreElements()) return false; nextName = (String)pending.next(); return true; } //next函数在DriverService,有调用,通过遍历LazyIterator,加载驱动 public Object next() throws ServiceConfigurationError { String s; Class class1; if(!hasNext()) throw new NoSuchElementException(); s = nextName; nextName = null; class1 = null; try { class1 = Class.forName(s, false, loader); } catch(ClassNotFoundException classnotfoundexception) { Service.fail(service, (new StringBuilder()).append("Provider ").append(s).append(" not found").toString()); } if(!service.isAssignableFrom(class1)) Service.fail(service, (new StringBuilder()).append("Provider ").append(s).append(" not a subtype").toString()); return service.cast(class1.newInstance()); Throwable throwable; throwable; Service.fail(service, (new StringBuilder()).append("Provider ").append(s).append(" could not be instantiated: ").append(throwable).toString(), throwable); return null; } public void remove() { throw new UnsupportedOperationException(); } Class service; ClassLoader loader; Enumeration configs; Iterator pending; Set returned; String nextName; private LazyIterator(Class class1, ClassLoader classloader) { configs = null; pending = null; returned = new TreeSet(); nextName = null; service = class1; loader = classloader; } private static void fail(Class class1, String s, Throwable throwable) throws ServiceConfigurationError { ServiceConfigurationError serviceconfigurationerror = new ServiceConfigurationError((new StringBuilder()).append(class1.getName()).append(": ").append(s).toString()); serviceconfigurationerror.initCause(throwable); throw serviceconfigurationerror; } //驱动加载失败的处理,抛出异常 private static void fail(Class class1, String s) throws ServiceConfigurationError { throw new ServiceConfigurationError((new StringBuilder()).append(class1.getName()).append(": ").append(s).toString()); } private static void fail(Class class1, URL url, int i, String s) throws ServiceConfigurationError { fail(class1, (new StringBuilder()).append(url).append(":").append(i).append(": ").append(s).toString()); } }
//ServiceConfigurationError
public class ServiceConfigurationError extends Error { public ServiceConfigurationError(String s) { super(s); } public ServiceConfigurationError(Throwable throwable) { super(throwable); } }
至此,driver的加载结束,我们来回顾一下整个过程,Class加载com.mysql.jdbc.Driver,com.mysql.jdbc.Driver通过静态语句块,将Driver注册到DriverManager,DriverManager首先检查有没有初始化,没有则加载驱动类,整个驱动加载过程是通过DriverService,而DriverService则委托给Service,Service加载驱动是返回的是,所有java.sql.Driver实现的Iterator,Iterator为LazyIterator,
DriverService通过遍历LazyIterator,加载驱动所有驱动;最后将Driver包装成DriverInfo,添加到writeDrivers集合中,并clone到readDrivers。
发表评论
-
Mysql PreparedStatement 批处理
2016-12-06 18:09 1390JDBC驱动初始化-Mysql:http://donald-d ... -
MySQL ServerPreparedStatement查询
2016-12-06 14:42 1303JDBC驱动初始化-Mysql:http://donald-d ... -
MysqlSQL PreparedStatement的查询
2016-12-06 11:40 2041JDBC驱动初始化-Mysql:http://donald-d ... -
Mysql预编译SQL
2016-12-05 19:06 1174JDBC驱动初始化-Mysql:http://donald-d ... -
ConnectionImp创建MysqlIO
2016-12-05 19:01 1033JDBC驱动初始化-Mysql:http://donald-d ... -
Mysql主从复制读写分离连接的获取
2016-12-01 08:43 1178JDBC驱动初始化-Mysql:http://donald-d ... -
Mysql负载均衡连接的获取
2016-12-01 08:38 1342Java动态代理:http://www.cnblogs.com ... -
JDBC连接的获取
2016-11-30 12:44 2939JDBC驱动初始化-Mysql:http://donald-d ... -
mysql 存储过程
2016-07-17 14:49 907存储过程基础:http://sishuok.com/forum ... -
java.sql.Date,java.util.Date,java.sql.Timestamp的区别
2016-07-17 11:50 885java.sql.Date,jdbc映射数据库中的date类型 ... -
JDBC PreparedStatement 的用法
2016-07-15 17:27 901import java.sql.Connection; im ...
相关推荐
在给定的文件列表中,虽然没有明确的 JDBC 驱动 jar 包,但在实际应用中,你需要下载并添加 `mysql-connector-java-x.x.x.jar` 到你的 Java 项目的类路径中,这里的 x.x.x 是具体的版本号。 安装步骤通常包括以下几...
1. **添加驱动依赖**:`mysql-connector-java-5.0.7-bin.jar` 文件是MySQL JDBC驱动的二进制版本,它包含了所有必要的类和库,使得Java程序能够连接到MySQL数据库。在Java项目中,通常将这个JAR文件添加到项目的类...
1. 引入JDBC驱动的依赖:在项目中,你需要将驱动的jar文件(如`mysql-connector-java-x.x.x.jar`)添加到类路径中。对于现代构建系统(如Maven或Gradle),可以在相应的配置文件中声明依赖。 2. 加载驱动:使用`...
5. 配置文件:如web.xml,配置Servlet的映射和初始化参数,包括JDBC连接池的配置。 6. HTML/JavaScript页面:前端页面,可能包含分页链接,触发Ajax请求,与Servlet进行交互。 这个项目对初学者和开发者来说很有...
MySQL的jdbc驱动jar,如"mysql-connector-java-5.1.35",是连接Java应用程序与MySQL数据库的关键组件。这个驱动程序遵循Java Database Connectivity (JDBC) API规范,使得Java开发者能够通过编写Java代码来执行SQL...
- **静态初始化块**:对于JDBC驱动来说,通常在驱动类中包含一个静态初始化块。在类被加载时,静态初始化块会被执行。在这个例子中,`JdbcOdbcDriver`类可能包含了必要的静态初始化逻辑,比如注册驱动到`...
这个方法会加载并初始化指定的类,确保JDBC驱动已准备就绪。 3. **建立连接**:注册完驱动后,你可以使用`java.sql.DriverManager`类的`getConnection()`方法创建到MySQL数据库的连接。需要提供数据库URL、用户名...
这通常通过`Class.forName()`方法完成,比如`Class.forName("com.mysql.jdbc.Driver")`,这会加载并初始化对应的驱动类。 5. 数据库连接:连接MySQL数据库使用`java.sql.DriverManager.getConnection()`方法,传入...
`mysql-connector-java`是MySQL官方提供的JDBC驱动,符合JDBC规范,使得Java程序可以无缝连接到MySQL数据库。 2. **版本兼容性**:`mysql-connector-java-5.4.40.jar`是特定版本的驱动,5.4系列适用于MySQL 5.x到8....
连接池预先初始化并维护一定数量的数据库连接,当需要连接时,可以从池中获取,用完后归还,而不是每次建立和关闭新的连接。这样可以减少连接创建和销毁的开销,提高系统性能。 总结来说,MySQL JDBC驱动包和连接...
1. **mysql-connector-java-5.1.8-bin.jar**:这是MySQL官方提供的JDBC驱动程序,版本为5.1.8。这个JAR文件包含了所有必要的类和资源,使得Java应用程序能够连接到MySQL服务器并执行SQL命令。例如,你可以使用`...
3. 添加JDBC驱动:将`mysql-connector-java-5.1.47-bin.jar`复制到DolphinScheduler的lib目录下,使其成为系统的类路径一部分。 4. 初始化数据库:使用DolphinScheduler提供的SQL脚本对数据库进行初始化,创建必要...
对于JDBC驱动来说,这个初始化块通常会包含对`java.sql.Driver`接口的实现类的实例化。例如,如果我们使用MySQL的JDBC驱动,代码可能如下所示: ```java try { Class.forName("com.mysql.jdbc.Driver"); } catch ...
1. **加载驱动**:首先,我们需要在程序中加载MySQL的JDBC驱动。这通常通过`Class.forName()`方法完成,指定驱动类名,例如`com.mysql.cj.jdbc.Driver`。 2. **创建连接**:加载驱动后,使用`DriverManager.get...
这会触发驱动的静态初始化,从而注册到JDBC驱动管理器。 - 从Java 6开始,可以使用服务提供者接口(SPI)自动加载驱动,无需显式调用`Class.forName()`。 3. **建立数据库连接** - 使用`DriverManager.get...
`mysql-sharding-jdbc` 是一个用于 MySQL 数据库分片的 Java 框架,它提供了 JDBC 驱动接口,使得开发者可以方便地在应用层进行分库分表操作。 一、MySQL 分库分表原理 1. 分库:将一个大型数据库拆分为多个小型...
使用时,开发者需要按照特定的配置方式初始化连接池服务,并通过服务接口获取和释放数据库连接。 总的来说,`oracle-mysql-sqlservice jar驱动` 是一个集成多种数据库驱动的服务,它简化了在Java项目中同时处理...
这一步会确保驱动类被初始化。 2. **建立连接**:通过`DriverManager.getConnection()`方法建立到数据库的连接,需要提供URL、用户名和密码,如:`Connection conn = DriverManager.getConnection("jdbc:mysql://...
MySQL的JDBC驱动通常称为`mysql-connector-java.jar`,它允许Java应用程序与MySQL数据库进行通信。该驱动支持多种认证方式,如标准的用户名/密码认证、SSL加密连接以及使用Connector/J的MySQL Native Password认证...
JDBC驱动程序是Java平台上的一个接口,它定义了如何在Java代码中创建、提交和管理SQL语句。`mysql-connector-java-8.0.20.jar`是MySQL Connector/J的特定版本,它实现了这些JDBC接口,使得Java开发者可以方便地在...