Java中,static代码块创建的IO流需要手动关闭吗?
Posted
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Java中,static代码块创建的IO流需要手动关闭吗?相关的知识,希望对你有一定的参考价值。
class A
******
static
FileInputStream fis = new FileInputStream("a.txt");
******
但系统进行垃圾回收的时机是不确定的,static 代码块执行完后,不会马上进行回收。在这段时间内,会造成无用的内存及IO等其他资源占用。所以最好还是用完即关。
推荐写在 try 里,try 块执行完自动关闭。追问
网上好多人写的JDBC工具类里面,在static代码块加载配置文件,这里创建的io流都没手动关闭。
你知道是为什么吗?应该不是因为他们不知道io流要手动关闭吧?
表面上是 "网上好多人写的JDBC工具类",实际上是几个人写了,然后一大堆人复制粘贴。
如果不关闭 FileInputStream,在学习或者练习项目用用也没什么问题。真实项目肯定不允许这样写的。
在一个 FileInputStream 不关闭的情况下,另一个 FileInputStream 也是可以读这个文件的,再说配置文件也基本只有一个流会去读。并且 fis 作为 static 块里的临时变量,执行完后,是可以被 GC 的。所以即使你不关闭流,项目也确实可以正常运行。
但是,如果你打开 1000、10000 个 FileInputStream,就会造成系统资源的严重浪费。
除非是写 try(FileInputStream xxx ....) 这样的,可以在 try 执行完后自动关闭。还有一些工具类,使用完流后会自动关闭流。不然就需要手动调用 close()。
你也可以试下,执行完 static 后,流并未被关闭。
现在很多的框架集成了自动创建关闭资源(如:spring框架),你只是没找到这些代码,但关闭资源的相关代码都存在于包内。 参考技术B 不管你是类变量还是实例变量,都需要关闭啊追问
我看网上好多人写的JDBC工具类这个地方都没有手动关闭,好多教程也是这样
追答不关闭有可能导致内存、数据库资源占用越来越多,
这不是马上就凸显的问题,
是操作一段时间后才会发现
你的意思是这样写:xxxxxx.close()写到图片中这个finally里面?
我觉得你这样写是合理的
java静态方法非静态代码块{}静态代码块static{}
(一)java 静态代码块 静态方法区别
一般情况下,如果有些代码必须在项目启动的时候就执行的时候,需要使用静态代码块,这种代码是主动执行的;需要在项目启动的时候就初始化,在不创建对象的情况下,其他程序来调用的时候,需要使用静态方法,这种代码是被动执行的. 静态方法在类加载的时候 就已经加载 可以用类名直接调用
比如main方法就必须是静态的 这是程序入口
两者的区别就是:静态代码块是自动执行的;
静态方法是被调用的时候才执行的.
静态方法
(1)在Java里,可以定义一个不需要创建对象的方法,这种方法就是静态方法。要实现这样的效果,只需要在类中定义的方法前加上static关键字。例如:
public static int maximum(int n1,int n2)
使用类的静态方法时,注意:
a在静态方法里只能直接调用同类中其他的静态成员(包括变量和方法),而不能直接访问类中的非静态成员。这是因为,对于非静态的方法和变量,需要先创建类的实例对象后才可使用,而静态方法在使用前不用创建任何对象。
b 静态方法不能以任何方式引用this和super关键字,因为静态方法在使用前不用创建任何实例对象,当静态方法调用时,this所引用的对象根本没有产生。
(2)静态变量是属于整个类的变量而不是属于某个对象的。注意不能把任何方法体内的变量声明为静态,例如:
fun()
static int i=0;//非法。
(3)一个类可以使用不包含在任何方法体中的静态代码块,当类被载入时,静态代码块被执行,且只被执行一次,静态块常用来执行类属性的初始化。例如:
static
类装载步骤
在Java中,类装载器把一个类装入Java虚拟机中,要经过三个步骤来完成:装载、链接和初始化,其中链接又可以分成校验、准备和解析三步,除了解析外,其它步骤是严格按照顺序完成的,各个步骤的主要工作如下:
装载:查找和导入类或接口的二进制数据;
链接:执行下面的校验、准备和解析步骤,其中解析步骤是可以选择的;
校验:检查导入类或接口的二进制数据的正确性;
准备:给类的静态变量分配并初始化存储空间;
解析:将符号引用转成直接引用;
初始化:激活类的静态变量的初始化Java代码和静态Java代码块。
初始化类中属性是静态代码块的常用用途,但只能使用一次。
(二)
package statictest;
class Code
System.out.println("Code的构造块");
static
System.out.println("Code的静态代码块");
public Code()
System.out.println("Code的构造方法");
public class statictest
System.out.println("statictest的构造块");
static
System.out.println("statictest的静态代码块");
public statictest()
System.out.println("statictest的构造方法");
public static void main(String[] args)
System.out.println("statictest的主方法");
new Code();
new Code();
new statictest();
new statictest();
/*
statictest的静态代码块
statictest的主方法
Code的静态代码块
Code的构造块
Code的构造方法
Code的构造块
Code的构造方法
statictest的构造块
statictest的构造方法
statictest的构造块
statictest的构造方法
*/
(三)
package statictest1;
public class TestStaticCon
public static int a = 0;
static
a = 10;
System.out.println("父类的静态代码块在执行a=" + a);
a = 8;
System.out.println("父类的非静态代码块在执行a=" + a);
public TestStaticCon()
this("a在父类带参构造方法中的值:" + TestStaticCon.a); // 调用另外一个构造方法
System.out.println("父类无参构造方法在执行a=" + a);
public TestStaticCon(String n)
System.out.println("n: " + n);
System.out.println("a: " + a);
public static void main(String[] args)
TestStaticCon tsc = null;
System.out.println("!!!!!!!!!!!!!!!!!!!!!");
tsc = new TestStaticCon();
/*
父类的静态代码块在执行a=10
!!!!!!!!!!!!!!!!!!!!!
父类的非静态代码块在执行a=8
n: a在父类带参构造方法中的值:10
a: 8
父类无参构造方法在执行a=8
*/
总结:
对象的初始化顺序:首先执行父类静态的内容,父类静态的内容执行完毕后,接着去执行子类的静态的内容,当子类的静态内容执行完毕之后,再去看父类有没有非静态代码块,如果有就执行父类的非静态代码块,父类的非静态代码块执行完毕,接着执行父类的构造方法;父类的构造方法执行完毕之后,它接着去看子类有没有非静态代码块,如果有就执行子类的非静态代码块。子类的非静态代码块执行完毕再去执行子类的构造方法。总之一句话,静态代码块内容先执行,接着执行父类非静态代码块和构造方法,然后执行子类非静态代码块和构造方法。
注意:子类的构造方法,不管这个构造方法带不带参数,默认的它都会先去寻找父类的不带参数的构造方法。如果父类没有不带参数的构造方法,那么子类必须用supper关键子来调用父类带参数的构造方法,否则编译不能通过。
以上是关于Java中,static代码块创建的IO流需要手动关闭吗?的主要内容,如果未能解决你的问题,请参考以下文章