
纠正下你的说法一个类是不会无缘无故运行的,所以谈不上占用内存,更没有回收你问的应该是使用一个类创建了一个对象,如何回收该对象所占的内存空间
例如有一个Person类,我们创建他的一个对象,一般是如下:
Person mPerson = new Person();
这个时候情况是,在栈内存中标记了一个名字叫mPerson的空间,它存储了一个关于某块内存区域位置的信息,说白了,就是你使用new Person();创建出来的对象的位置,我们称mPerson为Person对象的一个引用。这个对象位于堆内存,它是有默认值的,必须占内存一部分空间,想使用的时候,可以通过mPerson找到它。当你把mPerson赋值为null后,你那个对象就没有引用了,这个时候,你的JVM会在适当的时候自动去回收掉Person的对象。
JDBC连接数据库
•创建一个以JDBC连接数据库的程序,包含7个步骤:
1、加载JDBC驱动程序:
在连接数据库之前,首先要加载想要连接的数据库的驱动到JVM(Java虚拟机),
这通过javalangClass类的静态方法forName(String className)实现。
例如:
try{
//加载MySql的驱动类
ClassforName("commysqljdbcDriver") ;
}catch(ClassNotFoundException e){
Systemoutprintln("找不到驱动程序类 ,加载驱动失败!");
eprintStackTrace() ;
}
成功加载后,会将Driver类的实例注册到DriverManager类中。
2、提供JDBC连接的URL
•连接URL定义了连接数据库时的协议、子协议、数据源标识。
•书写形式:协议:子协议:数据源标识
协议:在JDBC中总是以jdbc开始 子协议:是桥连接的驱动程序或是数据库管理系统名称。
数据源标识:标记找到数据库来源的地址与连接端口。
例如:
(MySql的连接URL)
jdbc:mysql: //localhost:3306/testuseUnicode=true&characterEncoding=gbk ;
useUnicode=true:
表示使用Unicode字符集。如果characterEncoding设置为 gb2312或GBK,本参数必须设置为true 。characterEncoding=gbk:字符编码方式。
3、创建数据库的连接
•要连接数据库,需要向javasqlDriverManager请求并获得Connection对象, 该对象就代表一个数据库的连接。
•使用DriverManager的getConnectin(String url , String username , String password )方法传入指定的欲连接的数据库的路径、数据库的用户名和 密码来获得。
例如: //连接MySql数据库,用户名和密码都是root
String url = "jdbc:mysql://localhost:3306/test" ;
String username = "root" ;
String password = "root" ;
try{
Connection con = DriverManagergetConnection(url , username , password ) ;
}catch(SQLException se){
Systemoutprintln("数据库连接失败!");
seprintStackTrace() ;
}
4、创建一个Statement
•要执行SQL语句,必须获得javasqlStatement实例,Statement实例分为以下3 种类型:
1、执行静态SQL语句。通常通过Statement实例实现。
2、执行动态SQL语句。通常通过PreparedStatement实例实现。
3、执行数据库存储过程。通常通过CallableStatement实例实现。
具体的实现方式:
Statement stmt = concreateStatement() ; PreparedStatement pstmt = conprepareStatement(sql) ; CallableStatement cstmt = conprepareCall("{CALL demoSp( , )}") ;
5、执行SQL语句
Statement接口提供了三种执行SQL语句的方法:executeQuery 、executeUpdate 和execute
1、ResultSet executeQuery(String sqlString):执行查询数据库的SQL语句 ,返回一个结果集(ResultSet)对象。
2、int executeUpdate(String sqlString):用于执行INSERT、UPDATE或 DELETE语句以及SQL DDL语句,如:CREATE TABLE和DROP TABLE等
3、execute(sqlString):用于执行返回多个结果集、多个更新计数或二者组合的 语句。 具体实现的代码:
ResultSet rs = stmtexecuteQuery("SELECT FROM ") ; int rows = stmtexecuteUpdate("INSERT INTO ") ; boolean flag = stmtexecute(String sql) ;
6、处理结果 两种情况:
1、执行更新返回的是本次 *** 作影响到的记录数。
2、执行查询返回的结果是一个ResultSet对象。
• ResultSet包含符合SQL语句中条件的所有行,并且它通过一套get方法提供了对这些 行中数据的访问。
• 使用结果集(ResultSet)对象的访问方法获取数据:
while(rsnext()){
String name = rsgetString("name") ;
String pass = rsgetString(1) ; // 此方法比较高效
}
(列是从左到右编号的,并且从列1开始)
7、关闭JDBC对象
*** 作完成以后要把所有使用的JDBC对象全都关闭,以释放JDBC资源,关闭顺序和声 明顺序相反:
1、关闭记录集
2、关闭声明
3、关闭连接对象
if(rs != null){ // 关闭记录集
try{
rsclose() ;
}catch(SQLException e){
eprintStackTrace() ;
}
}
if(stmt != null){ // 关闭声明
try{
stmtclose() ;
}catch(SQLException e){
eprintStackTrace() ;
}
}
if(conn != null){ // 关闭连接对象
try{
connclose() ;
}catch(SQLException e){
eprintStackTrace() ;
}
}
;我们在深入Java核心系列文章中给大家讲过JVM中的栈和局部变量 在做Java开发的时候常用的JVM内存管理有两种 一种是堆内存 一种是栈内存 堆内存主要用来存储程序在运行时创建或实例化的对象与变量 例如 我们通过new MyClass()创建的类MyClass的对象 而栈内存则是用来存储程序代码中声明为静态(或非静态)的方法 下面我给大家举个例子 就拿上面的例子来说 放在栈内存中的有 main makeThings 放在堆内存中有 Test list object JVM中对象的生命周期大致可以分为 个阶段 创建阶段 应用阶段 不可视阶段 不可到达阶段 可收集阶段 终结阶段与释放阶段 创建阶段 ( )为对象分配存储空间 ( )开始构造对象 ( )递归调用其超类的构造方法 ( )进行对象实力初始化与变量初始化 ( )执行构造方法体 还有就是你在创建对象的时候需要注意的地方 ( )避免在循环体中创建对象 即使该对象占用内存空间不大 ( )尽量及时使对象符合垃圾回收标准 ( )不要采用过深的继承层次 ( )访问本地变量优于访问类中的变量 应用阶段 在应用阶段涉及到 个引用 ( )强引用 是指JVM内存管理器从根引用集合出发遍寻堆中所有到达对象的路径 ( )软引用 是具有较强的引用功能 只有当内存不够的时候 才回收这类内存 因此内存足够的时候 不会被回收 ( )弱引用 弱引用与软引用对象的最大不同在于 GC在进行回收时 需要通过算法检查是否回收软引用对象 而对于弱引用来说 GC总是进行回收 ( )虚引用 主要用于辅助finalize函数的使用 虚引用主要适用于以某种比Java终结机制更灵活的方式调度pre mortem清除 *** 作
不可视阶段 先看一段代码 如果一个对象已使用完了 应该主动将其设置为null 可以在上面的代码行obj doSomething();下添加代码行obj=null;这样一行代码强制将obj对象置为空值 这样做的意义就是帮助JVM及时的发现这个垃圾对象 并且可以及时的回收该对象占用的系统资源 不可到达阶段 处于不可到达阶段的对象 在虚拟机所管理的对象引用根集合中再也找不到直接或间接的强引用 这些对象通常是指多有线程栈中的临时变量 所有已装载的类的静态变量或者对本地代码接口(JNI)引用 可收集阶段 终结阶段与释放阶段 当对象处于这个阶段的时候 可能处于下面三种情况 ( )垃圾回收器发现该对象已经不可到达 ( )finalize方法已经被执行 ( )对象空间已被重用 当对象处于上面三种清空的时候 虚拟机就可以直接将该对象回收了 析构方法finalize 前面我们说了JVM的垃圾回收机制和JVM中对象的生命周期 今天给大家讲个方法 叫做析构方法finalize 我想搞过C++的人都知道 而且是内存管理技术中相当重要的一部分 但是 在Java中好像没有这个概念 这是因为 理论上JVM负责对象的析构(销毁与回收)工作 finalize是Object类中的一个方法 并且是protected 由于所有的类都继承了Object对象 因此 就都隐式的继承了改方法 不过可以重写这个方法 如果重写此方法 最后一句必须写上super finalize()语句 因为finalize方法没有自动实现递归调用 那我们在什么时候要重写它呢?当有一些不容易控制并且非常重要的资源时 要放到finalize方法中 例如 一些I/O的 *** 作 数据的连接等等 这些资源的释放对整个应用程序是非常关键的 我先让大家看一段代码 finalize方法最终是由JVM中的垃圾回收器调用的 由于垃圾回收器调用finalize的时间是不确定或者不及时的 调用时机对我们来说是不可控的 因此我们可以在自己的类中声明一个destory()方法 在这个方法中添加释放系统资源的处理代码 但是还是建议你将对destroy()方法的调用放入当前类的finalize()方法体中 因为这样做更保险 更安全
静态变量 我们知道类中的静态变量在程序运行期间 其内存空间对所有该类的对象实例而言是共享的 为了节省系统内存开销 共享资源 应该将一些变量声明为静态变量 通过下面的例子 你就会发现有什么不同 代码一 代码二 我想大家应该发现上面那两个类的区别了吧!
代码一会在内存中保存 个weeks的副本 而代码二则在内存中保存 个weeks的副本 然后共享该副本 这样的话就不会造成内存的浪费 虽然静态的变量能节约大量的内存 但是并不是所有的地方都适合用 建议大家在下列条件都符合的情况下 尽量用静态变量 ( )变量所包含的对象体积较大 占用内存较多 ( )变量所包含的对象生命周期较长 ( )变量所包含的对象数据稳定 ( )该类的对象实例有对该变量所包含的对象的共享需求 如果变量不具备上述特点 建议不要轻易使用静态变量 以免弄巧成拙 最后 再提一点内存的优化 就是有关对象的重用 比如 对象池和数据库连接池等 那样的话 是很节约内存空间的 不过 在用的时候要考虑各个方面 比如 运行环境的内存资源的限制等 为了防止对象池中的对象过多 要记得清除 内存管理有许多技巧和方式 其实内存管理有许多技巧和方式 在这 我给大家介绍一下 ( )要尽早的释放无用对象的引用 如果 该对象不用了 你可以把它设置为null 但要注意 如果该对象是某方法的返回值 千万不要这样处理 否则你从该方法中得到的返回值永远为空 而且这种错误不易被发现 因此这时很难及时抓住 排除NullPointerException异常 ( )尽量少用finalize函数 因为它会加大GC的工作量 因此尽量少用finalize方式回收资源 ( )如果需要使用经常用到的 可以使用soft应用类型(也就是转换为软引用类型) 它可以尽可能将保存在内存中 供程序调用 而不引起OutOfMemory ( )注意集合数据类型 包括数组 树 图 链表等数据结构 这些数据结构对于GC来说 回收更为复杂 另外 要注意那些全局变量 静态变量 这些对象往往容易引起悬挂对象 造成内存浪费 ( )尽量避免在类的默认构造器中创建 初始化大量的对象 防止在调用其子类的构造器时造成不必要的内存资源浪费 ( )尽量避免强制系统做垃圾内存回收(通过显式调用方法System gc()) 增长系统做垃圾回收的最终时间 降低系统性能 ( )尽量避免显式申请数组空间 当不得不显式申请数组空间时尽量准确的估计出其合理值 以免造成不必要的系统内存开销 ( )尽量在做远程方法调用(RMI)类应用开发时使用瞬间值变量 除非远程调用端需要获取该瞬间值变量的值 ( )尽量在合适的场景下使用对象池技术以提高系统的性能 缩减系统内存开销 但是要注意对象池的尺寸不易过大 及时清除无效对象释放内存资源 综合考虑应用运行环境的内存资源限制 避免过高估计运行环境所提供内存资源的数量 虽然 这些技巧提高不了多少性能 但是 在嵌入式开发 或者要求性能比较高的系统中却很有用 lishixinzhi/Article/program/Java/hx/201311/27137
主要内容:
进程是资源分配的最小单位,每个进程都有独立的代码和数据空间,一个进程包含 1 到 n 个线程。线程是 CPU 调度的最小单位,每个线程有独立的运行栈和程序计数器,线程切换开销小。
Java 程序总是从主类的 main 方法开始执行,main 方法就是 Java 程序默认的主线程,而在 main 方法中再创建的线程就是其他线程。在 Java 中,每次程序启动至少启动 2 个线程。一个是 main 线程,一个是垃圾收集线程。每次使用 Java 命令启动一个 Java 程序,就相当于启动一个 JVM 实例,而每个 JVM 实例就是在 *** 作系统中启动的一个进程。
多线程可以通过继承或实现接口的方式创建。
Thread 类是 JDK 中定义的用于控制线程对象的类,该类中封装了线程执行体 run() 方法。需要强调的一点是,线程执行先后与创建顺序无关。
通过 Runnable 方式创建线程相比通过继承 Thread 类创建线程的优势是避免了单继承的局限性。若一个 boy 类继承了 person 类,boy 类就无法通过继承 Thread 类的方式来实现多线程。
使用 Runnable 接口创建线程的过程:先是创建对象实例 MyRunnable,然后将对象 My Runnable 作为 Thread 构造方法的入参,来构造出线程。对于 new Thread(Runnable target) 创建的使用同一入参目标对象的线程,可以共享该入参目标对象 MyRunnable 的成员变量和方法,但 run() 方法中的局部变量相互独立,互不干扰。
上面代码是 new 了三个不同的 My Runnable 对象,如果只想使用同一个对象,可以只 new 一个 MyRunnable 对象给三个 new Thread 使用。
实现 Runnable 接口比继承 Thread 类所具有的优势:
线程有新建、可运行、阻塞、等待、定时等待、死亡 6 种状态。一个具有生命的线程,总是处于这 6 种状态之一。 每个线程可以独立于其他线程运行,也可和其他线程协同运行。线程被创建后,调用 start() 方法启动线程,该线程便从新建态进入就绪状态。
NEW 状态(新建状态) 实例化一个线程之后,并且这个线程没有开始执行,这个时候的状态就是 NEW 状态:
RUNNABLE 状态(就绪状态):
阻塞状态有 3 种:
如果一个线程调用了一个对象的 wait 方法, 那么这个线程就会处于等待状态(waiting 状态)直到另外一个线程调用这个对象的 notify 或者 notifyAll 方法后才会解除这个状态。
run() 里的代码执行完毕后,线程进入终结状态(TERMINATED 状态)。
线程状态有 6 种:新建、可运行、阻塞、等待、定时等待、死亡。
我们看下 join 方法的使用:
运行结果:
我们来看下 yield 方法的使用:
运行结果:
线程与线程之间是无法直接通信的,A 线程无法直接通知 B 线程,Java 中线程之间交换信息是通过共享的内存来实现的,控制共享资源的读写的访问,使得多个线程轮流执行对共享数据的 *** 作,线程之间通信是通过对共享资源上锁或释放锁来实现的。线程排队轮流执行共享资源,这称为线程的同步。
Java 提供了很多同步 *** 作(也就是线程间的通信方式),同步可使用 synchronized 关键字、Object 类的 wait/notifyAll 方法、ReentrantLock 锁、无锁同步 CAS 等方式来实现。
ReentrantLock 是 JDK 内置的一个锁对象,用于线程同步(线程通信),需要用户手动释放锁。
运行结果:
这表明同一时间段只能有 1 个线程执行 work 方法,因为 work 方法里的代码需要获取到锁才能执行,这就实现了多个线程间的通信,线程 0 获取锁,先执行,线程 1 等待,线程 0 释放锁,线程 1 继续执行。
synchronized 是一种语法级别的同步方式,称为内置锁。该锁会在代码执行完毕后由 JVM 释放。
输出结果跟 ReentrantLock 一样。
Java 中的 Object 类默认是所有类的父类,该类拥有 wait、 notify、notifyAll 方法,其他对象会自动继承 Object 类,可调用 Object 类的这些方法实现线程间的通信。
除了可以通过锁的方式来实现通信,还可通过无锁的方式来实现,无锁同 CAS(Compare-and-Swap,比较和交换)的实现,需要有 3 个 *** 作数:内存地址 V,旧的预期值 A,即将要更新的目标值 B,当且仅当内存地址 V 的值与预期值 A 相等时,将内存地址 V 的值修改为目标值 B,否则就什么都不做。
我们通过计算器的案例来演示无锁同步 CAS 的实现方式,非线程安全的计数方式如下:
线程安全的计数方式如下:
运行结果:
线程安全累加的结果才是正确的,非线程安全会出现少计算值的情况。JDK 15 开始,并发包里提供了原子 *** 作的类,AtomicBoolean 用原子方式更新的 boolean 值,AtomicInteger 用原子方式更新 int 值,AtomicLong 用原子方式更新 long 值。 AtomicInteger 和 AtomicLong 还提供了用原子方式将当前值自增 1 或自减 1 的方法,在多线程程序中,诸如 ++i 或 i++ 等运算不具有原子性,是不安全的线程 *** 作之一。 通常我们使用 synchronized 将该 *** 作变成一个原子 *** 作,但 JVM 为此种 *** 作提供了原子 *** 作的同步类 Atomic,使用 AtomicInteger 做自增运算的性能是 ReentantLock 的好几倍。
上面我们都是使用底层的方式实现线程间的通信的,但在实际的开发中,我们应该尽量远离底层结构,使用封装好的 API,例如 JUC 包(javautilconcurrent,又称并发包)下的工具类 CountDownLath、CyclicBarrier、Semaphore,来实现线程通信,协调线程执行。
CountDownLatch 能够实现线程之间的等待,CountDownLatch 用于某一个线程等待若干个其他线程执行完任务之后,它才开始执行。
CountDownLatch 类只提供了一个构造器:
CountDownLatch 类中常用的 3 个方法:
运行结果:
CyclicBarrier 字面意思循环栅栏,通过它可以让一组线程等待至某个状态之后再全部同时执行。当所有等待线程都被释放以后,CyclicBarrier 可以被重复使用,所以有循环之意。
相比 CountDownLatch,CyclicBarrier 可以被循环使用,而且如果遇到线程中断等情况时,可以利用 reset() 方法,重置计数器,CyclicBarrier 会比 CountDownLatch 更加灵活。
CyclicBarrier 提供 2 个构造器:
上面的方法中,参数 parties 指让多少个线程或者任务等待至 barrier 状态;参数 barrierAction 为当这些线程都达到 barrier 状态时会执行的内容。
CyclicBarrier 中最重要的方法 await 方法,它有 2 个重载版本。下面方法用来挂起当前线程,直至所有线程都到达 barrier 状态再同时执行后续任务。
而下面的方法则是让这些线程等待至一定的时间,如果还有线程没有到达 barrier 状态就直接让到达 barrier 的线程执行任务。
运行结果:
CyclicBarrier 用于一组线程互相等待至某个状态,然后这一组线程再同时执行,CountDownLatch 是不能重用的,而 CyclicBarrier 可以重用。
Semaphore 类是一个计数信号量,它可以设定一个阈值,多个线程竞争获取许可信号,执行完任务后归还,超过阈值后,线程申请许可信号时将会被阻塞。Semaphore 可以用来 构建对象池,资源池,比如数据库连接池。
假如在服务器上运行着若干个客户端请求的线程。这些线程需要连接到同一数据库,但任一时刻只能获得一定数目的数据库连接。要怎样才能够有效地将这些固定数目的数据库连接分配给大量的线程呢?
给方法加同步锁,保证同一时刻只能有一个线程去调用此方法,其他所有线程排队等待,但若有 10 个数据库连接,也只有一个能被使用,效率太低。另外一种方法,使用信号量,让信号量许可与数据库可用连接数为相同数量,10 个数据库连接都能被使用,大大提高性能。
上面三个工具类是 JUC 包的核心类,JUC 包的全景图就比较复杂了:
JUC 包(javautilconcurrent)中的高层类(Lock、同步器、阻塞队列、Executor、并发容器)依赖基础类(AQS、非阻塞数据结构、原子变量类),而基础类是通过 CAS 和 volatile 来实现的。我们尽量使用顶层的类,避免使用基础类 CAS 和 volatile 来协调线程的执行。JUC 包其他的内容,在其他的篇章会有相应的讲解。
Future 是一种异步执行的设计模式,类似 ajax 异步请求,不需要同步等待返回结果,可继续执行代码。使 Runnable(无返回值不支持上报异常)或 Callable(有返回值支持上报异常)均可开启线程执行任务。但是如果需要异步获取线程的返回结果,就需要通过 Future 来实现了。
Future 是位于 javautilconcurrent 包下的一个接口,Future 接口封装了取消任务,获取任务结果的方法。
在 Java 中,一般是通过继承 Thread 类或者实现 Runnable 接口来创建多线程, Runnable 接口不能返回结果,JDK 15 之后,Java 提供了 Callable 接口来封装子任务,Callable 接口可以获取返回结果。我们使用线程池提交 Callable 接口任务,将返回 Future 接口添加进 ArrayList 数组,最后遍历 FutureList,实现异步获取返回值。
运行结果:
上面就是异步线程执行的调用过程,实际开发中用得更多的是使用现成的异步框架来实现异步编程,如 RxJava,有兴趣的可以继续去了解,通常异步框架都是结合远程 >
你可以做一个下拉框,选项有教师和学生,不过这个的话,一般是通过权限来控制的,这样教师和学生登录之后,看到的菜单就不一样了。
JDBC连接数据库
•创建一个以JDBC连接数据库的程序,包含7个步骤:
1、加载JDBC驱动程序:
在连接数据库之前,首先要加载想要连接的数据库的驱动到JVM(Java虚拟机),
这通过javalangClass类的静态方法forName(String className)实现。
例如:
try{
//加载MySql的驱动类
ClassforName("commysqljdbcDriver") ;
}catch(ClassNotFoundException e){
Systemoutprintln("找不到驱动程序类 ,加载驱动失败!");
eprintStackTrace() ;
}
成功加载后,会将Driver类的实例注册到DriverManager类中。
2、提供JDBC连接的URL
•连接URL定义了连接数据库时的协议、子协议、数据源标识。
•书写形式:协议:子协议:数据源标识
协议:在JDBC中总是以jdbc开始
子协议:是桥连接的驱动程序或是数据库管理系统名称。
数据源标识:标记找到数据库来源的地址与连接端口。
例如:(MySql的连接URL)
jdbc:mysql:
//localhost:3306/testuseUnicode=true&characterEncoding=gbk ;
useUnicode=true:表示使用Unicode字符集。如果characterEncoding设置为
gb2312或GBK,本参数必须设置为true 。characterEncoding=gbk:字符编码方式。
3、创建数据库的连接
•要连接数据库,需要向javasqlDriverManager请求并获得Connection对象,
该对象就代表一个数据库的连接。
•使用DriverManager的getConnectin(String url , String username ,
String password )方法传入指定的欲连接的数据库的路径、数据库的用户名和
密码来获得。
例如:
//连接MySql数据库,用户名和密码都是root
String url = "jdbc:mysql://localhost:3306/test" ;
String username = "root" ;
String password = "root" ;
try{
Connection con =
DriverManagergetConnection(url , username , password ) ;
}catch(SQLException se){
Systemoutprintln("数据库连接失败!");
seprintStackTrace() ;
}
4、创建一个Statement
•要执行SQL语句,必须获得javasqlStatement实例,Statement实例分为以下3
种类型:
1、执行静态SQL语句。通常通过Statement实例实现。
2、执行动态SQL语句。通常通过PreparedStatement实例实现。
3、执行数据库存储过程。通常通过CallableStatement实例实现。
具体的实现方式:
Statement stmt = concreateStatement() ;
PreparedStatement pstmt = conprepareStatement(sql) ;
CallableStatement cstmt =
conprepareCall("{CALL demoSp( , )}") ;
5、执行SQL语句
Statement接口提供了三种执行SQL语句的方法:executeQuery 、executeUpdate
和execute
1、ResultSet executeQuery(String sqlString):执行查询数据库的SQL语句
,返回一个结果集(ResultSet)对象。
2、int executeUpdate(String sqlString):用于执行INSERT、UPDATE或
DELETE语句以及SQL DDL语句,如:CREATE TABLE和DROP TABLE等
3、execute(sqlString):用于执行返回多个结果集、多个更新计数或二者组合的
语句。
具体实现的代码:
ResultSet rs = stmtexecuteQuery("SELECT FROM ") ;
int rows = stmtexecuteUpdate("INSERT INTO ") ;
boolean flag = stmtexecute(String sql) ;
6、处理结果
两种情况:
1、执行更新返回的是本次 *** 作影响到的记录数。
2、执行查询返回的结果是一个ResultSet对象。
• ResultSet包含符合SQL语句中条件的所有行,并且它通过一套get方法提供了对这些
行中数据的访问。
• 使用结果集(ResultSet)对象的访问方法获取数据:
while(rsnext()){
String name = rsgetString("name") ;
String pass = rsgetString(1) ; // 此方法比较高效
}
(列是从左到右编号的,并且从列1开始)
7、关闭JDBC对象
*** 作完成以后要把所有使用的JDBC对象全都关闭,以释放JDBC资源,关闭顺序和声
明顺序相反:
1、关闭记录集
2、关闭声明
3、关闭连接对象
if(rs != null){ // 关闭记录集
try{
rsclose() ;
}catch(SQLException e){
eprintStackTrace() ;
}
}
if(stmt != null){ // 关闭声明
try{
stmtclose() ;
}catch(SQLException e){
eprintStackTrace() ;
}
}
if(conn != null){ // 关闭连接对象
try{
connclose() ;
}catch(SQLException e){
eprintStackTrace() ;
}
}
创建一个以JDBC连接数据库的程序,包含7个步骤: \x0d\ 1、加载JDBC驱动程序: \x0d\ 在连接数据库之前,首先要加载想要连接的数据库的驱动到JVM(Java虚拟机), \x0d\ 这通过javalangClass类的静态方法forName(String className)实现。 \x0d\ 例如: \x0d\ try{ \x0d\ //加载MySql的驱动类 \x0d\ ClassforName("commysqljdbcDriver") ; \x0d\ }catch(ClassNotFoundException e){ \x0d\ Systemoutprintln("找不到驱动程序类 ,加载驱动失败!"); \x0d\ eprintStackTrace() ; \x0d\ } \x0d\ 成功加载后,会将Driver类的实例注册到DriverManager类中。 \x0d\ 2、提供JDBC连接的URL \x0d\ •连接URL定义了连接数据库时的协议、子协议、数据源标识。 \x0d\ •书写形式:协议:子协议:数据源标识 \x0d\ 协议:在JDBC中总是以jdbc开始 \x0d\ 子协议:是桥连接的驱动程序或是数据库管理系统名称。 \x0d\ 数据源标识:标记找到数据库来源的地址与连接端口。 \x0d\ 例如:(MySql的连接URL) \x0d\ jdbc:mysql: \x0d\ //localhost:3306/testuseUnicode=true&characterEncoding=gbk ; \x0d\ useUnicode=true:表示使用Unicode字符集。如果characterEncoding设置为 \x0d\ gb2312或GBK,本参数必须设置为true 。characterEncoding=gbk:字符编码方式。 \x0d\ 3、创建数据库的连接 \x0d\ •要连接数据库,需要向javasqlDriverManager请求并获得Connection对象, \x0d\ 该对象就代表一个数据库的连接。 \x0d\ •使用DriverManager的getConnectin(String url , String username , \x0d\ String password )方法传入指定的欲连接的数据库的路径、数据库的用户名和 \x0d\ 密码来获得。 \x0d\ 例如: \x0d\ //连接MySql数据库,用户名和密码都是root \x0d\ String url = "jdbc:mysql://localhost:3306/test" ; \x0d\ String username = "root" ; \x0d\ String password = "root" ; \x0d\ try{ \x0d\ Connection con = \x0d\ DriverManagergetConnection(url , username , password ) ; \x0d\ }catch(SQLException se){ \x0d\ Systemoutprintln("数据库连接失败!"); \x0d\ seprintStackTrace() ; \x0d\ } \x0d\ 4、创建一个Statement \x0d\ •要执行SQL语句,必须获得javasqlStatement实例,Statement实例分为以下3 \x0d\ 种类型: \x0d\ 1、执行静态SQL语句。通常通过Statement实例实现。 \x0d\ 2、执行动态SQL语句。通常通过PreparedStatement实例实现。 \x0d\ 3、执行数据库存储过程。通常通过CallableStatement实例实现。 \x0d\ 具体的实现方式: \x0d\ Statement stmt = concreateStatement() ; \x0d\ PreparedStatement pstmt = conprepareStatement(sql) ; \x0d\ CallableStatement cstmt = \x0d\ conprepareCall("{CALL demoSp( , )}") ; \x0d\ 5、执行SQL语句 \x0d\ Statement接口提供了三种执行SQL语句的方法:executeQuery 、executeUpdate \x0d\ 和execute \x0d\ 1、ResultSet executeQuery(String sqlString):执行查询数据库的SQL语句 \x0d\ ,返回一个结果集(ResultSet)对象。 \x0d\ 2、int executeUpdate(String sqlString):用于执行INSERT、UPDATE或 \x0d\ DELETE语句以及SQL DDL语句,如:CREATE TABLE和DROP TABLE等 \x0d\ 3、execute(sqlString):用于执行返回多个结果集、多个更新计数或二者组合的 \x0d\ 语句。 \x0d\ 具体实现的代码: \x0d\ ResultSet rs = stmtexecuteQuery("SELECT FROM ") ; \x0d\ int rows = stmtexecuteUpdate("INSERT INTO ") ; \x0d\ boolean flag = stmtexecute(String sql) ; \x0d\ 6、处理结果 \x0d\ 两种情况: \x0d\ 1、执行更新返回的是本次 *** 作影响到的记录数。 \x0d\ 2、执行查询返回的结果是一个ResultSet对象。 \x0d\ • ResultSet包含符合SQL语句中条件的所有行,并且它通过一套get方法提供了对这些 \x0d\ 行中数据的访问。 \x0d\ • 使用结果集(ResultSet)对象的访问方法获取数据: \x0d\ while(rsnext()){ \x0d\ String name = rsgetString("name") ; \x0d\ String pass = rsgetString(1) ; // 此方法比较高效 \x0d\ } \x0d\ (列是从左到右编号的,并且从列1开始) \x0d\ 7、关闭JDBC对象 \x0d\ *** 作完成以后要把所有使用的JDBC对象全都关闭,以释放JDBC资源,关闭顺序和声 \x0d\ 明顺序相反: \x0d\ 1、关闭记录集 \x0d\ 2、关闭声明 \x0d\ 3、关闭连接对象 \x0d\ if(rs != null){ // 关闭记录集 \x0d\ try{ \x0d\ rsclose() ; \x0d\ }catch(SQLException e){ \x0d\ eprintStackTrace() ; \x0d\ } \x0d\ } \x0d\ if(stmt != null){ // 关闭声明 \x0d\ try{ \x0d\ stmtclose() ; \x0d\ }catch(SQLException e){ \x0d\ eprintStackTrace() ; \x0d\ } \x0d\ } \x0d\ if(conn != null){ // 关闭连接对象 \x0d\ try{ \x0d\ connclose() ; \x0d\ }catch(SQLException e){ \x0d\ eprintStackTrace() ; \x0d\ } \x0d\ }
以上就是关于Java中数据库连接池是如何做到,将Connection作为对象放入内存中不被回收机回收全部的内容,包括:Java中数据库连接池是如何做到,将Connection作为对象放入内存中不被回收机回收、java项目连不上mySQL、技术分享 把JVM移植到ARM环境中等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!
欢迎分享,转载请注明来源:内存溢出
微信扫一扫
支付宝扫一扫
评论列表(0条)