【Java基础教程】(四十六)IO篇 · 下:System类对IO的支持:错误输出、信息输出、系统输入,字符缓冲流、扫描流和对象序列化流~

news2024/12/27 5:07:25

Java基础教程之IO操作 · 下

  • 🔹本节学习目标
  • 1️⃣ System类对 IO 的支持
    • 1.1 错误输出:System.err
    • 1.2 信息输出:System.out
    • 1.3 系统输入:System. in
  • 2️⃣ 字符缓冲流:BufferedReader
  • 3️⃣ 扫描流:Scanner
  • 4️⃣ 对象序列化
    • 4.1 序列化接口:Serializable
    • 4.2 实现序列化与反序列化
    • 4.3 transient 关键字
  • 🌾 总结

在这里插入图片描述

🔹本节学习目标

  • 了解System类对 IO 的支持:错误输出(System.err)、信息输出(System.out)和系统输入(System.in);
  • 掌握缓冲字符流、打印流及扫描流的使用;
  • 掌握对象序列化流的应用场景及使用,掌握接口(Serializable)及transient关键字的作用;

1️⃣ System类对 IO 的支持

System 类是现在为止使用最多的一个类,所有的信息输出都会使用到 “System.out.println()”或 “System.out.print()”两个方法完成。实际上 System 类中也专门提供了与 I0 有关的3个对象常量,如下所示。

常量类型描述
public static final PrintStream err常量显示器上错误输出
public static final PrintStream out常量显示器上信息输出
public static final InputStream in常量键盘数据输入

可以发现, errout 两个对象的类型都属于 PrintStream 类型, in 对象类型属于 InputStream, 而最早使用的"System.out.println()" 代码从本质上来讲就是调用了 System 类中的out 常量,由于此常量类型为 PrintStream, 所以可以继续调用 PrintStream 类中的 print()println()方法,也就证明 Java 的任何输出操作实际上都是 I0操作。

1.1 错误输出:System.err

System.errPrintStream 类对象,此对象专门负责进行错误信息的输出操作。

//	范例 1: 错误输出
package com.xiaoshan.demo;

public class TestDemo  {
	public static void main(String[] args) throws Exception {
		try{
			Integer.parseInt("abc");	//此处一定会发生异常
		} catch (Exception e){
			System.err.println(e);	//错误输出
		}
	}
}

程序执行结果:

java.lang.NumberFormatException: For input string: "abc"

本程序试图将字符串 “abc” 转换为 int 型数据,而由于字符串不是由数字所组成,所以此处一定会产生异常,而本程序在异常处理中利用System.err 实现了错误信息的打印。

从实际的效果来看,如果本程序中使用了System.out, 其信息内容是完全一样的, 那么为什么非要用System.err而不使用System.out呢?

System.errSystem.out都属于PrintStream类型的对象,所以两者的功能肯定完全相同,但是在Java最初设计时,希望某些错误信息不被使用者看见,所以就定义了 System.err,而System.errSystem.out默认情况下都是在屏幕上输出(可以使用 System类中的"setErr(PrintStream err)"或"setOut(PrintStream err)"改变输出位置,但一般不会改变out的输出位置),所以区别就不这么大了。

另外如果是在IDE工具中显示,会发现System.err输出错误信息使用的是红色字体, 而System.out输出错误信息使用的是黑色字体。

1.2 信息输出:System.out

Sysetm.out 是进行屏幕输出的操作对象,是一个 PrintStream 类的实例化对象,并且由JVM 负责该对象的实例化,由于 PrintStream 也是 OutputStream 的子 类,所以可以利用 System.outOutputStream 类进行实例化。

从实际上讲,下边范例的代码意义并不大,因为 OutputStreamPrintStream 相比,明显输出时使用 PrintStream 更加方便 (因为它提供了各种 print()println()方法), 而本程序的意义只是在于验证不同的子类为 OutputStream 的位置也不同这一对象多态性的概念应用。

//	范例 2: 利用 OutputStream 实现屏幕输出
package com.xiaoshan.demo;

import java.io.OutputStream;

public class TestDemo{
	public static void main(String[] args) throws Exception{
		// OutputStream 就为屏幕输出
		OutputStream out = System.out;
		out.write("https://lst66.blog.csdn.net".getBytes());
	}
}

程序执行结果:

https://lst66.blog.csdn.net

由于System.out 属于PrintStream 类的实例,所以可以直接利用其为 OutputStream 对象实例化, 此时调用 write() 方法输出时就变为了屏幕显示。

在JDK 1.8提供的函数式接口里面,有一个消费型的函数式接口,此接口中的方法不返回结果,但是会接收参数。此时就可以利用方法引用的概念,利用System.out.println() 实现 Consumer 接口的实例化。

//	范例 3: 消费型函数式接口与方法引用
package com.xiaoshan.demo;

import java.util.function.Consumer;

public class TestDemo {
	public static void main(String[] args) throws Exception{
		Consumer<String> con = System.out::println;       //方法引用
		con.accept("更多文章请访问:https://lst66.blog.csdn.net"); 	//   输   出
	}
}

程序执行结果:

更多文章请访问:https://lst66.blog.csdn.net

程序采用方法引用的概念,将 System.out 对象中的 println() 方法引用到 Consumer 接口中,而后利用accept()方法就可以实现屏幕信息输出。

1.3 系统输入:System. in

在许多编程语言中为了方便用户的交互操作,都会直接提供一种键盘输入数据的操作功能,但是在 Java 中并没有提供这样可以直接使用的键盘输入操作,而要想实现此类操作必须采用 IO 处理的形式完成,操作的核心就是利用 System.in ( 此为 InputStream 类实例化对象)完成。

//	范例 4: 实现键盘的数据输入
package com.xiaoshan.demo;

import java.io.InputStream;

public class TestDemo  {
	public static void main(String[] args) throws Exception{//此处直接抛出
		//为了方便读者理解,本处将System.in使用InputStream接收,但实际上不需要此操作
		InputStream input = System.in;	 //System.in为InputStream类实例
		//开辟空间接收数据
		//信息提示,此处没有换行
		//读取数据并返回长度
		byte data[] = new byte[1024];
		System.out.print("请输入数据:");
		int len = input.read(data);
		System.out.println("输入数据为:"+ new String(data, O, len));
	}
}

程序执行结果:

请输入数据:更多文章请访问:https://lst66.blog.csdn.net
输入数据为:更多文章请访问:https://lst66.blog.csdn.net

本程序利用 System.in 实现了键盘数据的读取操作,与文件读取不同的地方在于,本程序的文件内容已经是固定好的,所以读取时会直接进行加载 (如下图(a) 所示)。通过键盘读取时,由于内容还未输入,所以会出现一个等待用户输入的操作界面,用户输入完成按回车后才会开始读取数据(如下图(b) 所示)。

图1 文件与键盘输入

虽然上边范例的代码实现了键盘输入的操作功能,但是此时就会出现一个问题:本处读取的数据为1024个字节,如果读取的内容超过了1024个字节,就会出现漏读的问题 ,也即超过数组长度的部分不会被接收。

在上边范例程序中,只是读取了一次,所以当出现内容超过预定义数组长度后就会出现漏读的问题。但是也可以利用内存流将其进行改进,例如:可以将所有读取的数据暂时保存在内存输出流中,并且ByteArraryOutputStream类中提供了 “toByteArray()” 方法可以取得全部的字节数据,这样结合循环读取就可以将内容全部读取。此处提供一种思路,不再编写复杂代码演示, 有兴趣的朋友可以自行实现。

而除了上述这种方式,要想解决此时的漏读问题,也可以不使用数组接收,而是采用循环的方式进行读取,并且将每次读取进来的数据利用 StringBuffer 类对象保存。

//	范例 5: 改进输入操作设计
package com.xiaoshan.demo;

import java.io.InputStream;

public class TestDemo{
	public static void main(String[] args) throws Exception{
		InputStream input = System.in;
		StringBuffer buf = new StringBuffer(); 	//接收输入数据
		System.out.print("请输入数据:");			//提示信息
		int temp = 0;							//接收每次读取数据长度
		while((temp = input.read()) != -1){		//判断是否有输入数据
			if (temp == '\n'){ 	//判断是否为回车符
				break;			//停止接收
			}
			buf.append((char) temp);	//保存读取数据
		}
		System.out.println("输入数据为:" + buf);	//输出内容
	}
}

程序执行结果(输入英文时):

请输入数据:www.baidu.com
输入数据为:www.baidu.com

程序执行结果(输入中文):

请输入数据:我是小山
输入数据为:我是小山

此程序由于使用了 StringBuffer 类对象保存数据,所以没有数据长度的限制,但是此时的代码在输入英文数据时没有任何问题,而输入中文数据时却会出现乱码。造成乱码的原因也很简单,这是一个中文的编码拆分成了两半(每次读取一个字节) 而造成的编码出错。要想解决此类问题,就可以利用字符缓冲输入流完成。

2️⃣ 字符缓冲流:BufferedReader

为了可以进行完整数据的输入操作,最好的做法是采用缓冲区的方式对输入的数据进行暂存,而后程序可以利用输入流一次性读取内容,如下图所示,这样就可以避免输入中文时的读取错乱问题。

图2 缓冲区操作

如果要使用缓冲区进行数据操作,java.io 包提供了以下两种操作流。

  • 字符缓冲区流: BufferedReaderBufferedWriter;
  • 字节缓冲区流: BufferedInputStreamBufferedOutputStream

以上给出的4个操作类中,最为重要的就是BufferedReader 类,此类是Reader 的子类,属于字符缓冲输入流,而如果要处理中文数据,字符流是最方便的。 BufferedReader 类的常用方法如下所示。

方法类型描述
public BufferedReader(Reader in)构造方法设置字符输入流
public String readLine() throws IOException普通方法读取一行数据,默认以“\n”为分隔符

如果要使用 BufferedReader 类来处理 System.in 的操作就会出现一个问题, BufferedReaderReader 的子类,并且构造方法中也要接收 Reader 类型,而 System.inInputStream 类型,所以此处必须将 InputStream 类型转换为 Reader 类型,那么就可以利用 InputStreamReader 类来实现这一转换操作。字节输入流转字符缓冲输入流结构如下图所示。

图3 字节输入流转字符缓冲输入流结构

//	范例 6: 键盘数据输入的标准格式
package com.xiaoshan.demo;

import java.io.BufferedReader;
import java.io.InputStreamReader;

public class TestDemo  {
	public static void main(String[] args) throws Exception{
		//System.in 是InputStream 类对象,BufferedReader 的构造方法里面需要接收 Reader 类对象
		//利用InputStreamReader 将字节流对象变为字符流对象
		BufferedReader buf = new BufferedReader(new InputStreamReader(System.in));
		System.out.print("请输入数据:");
		String str = buf.readLine();         //以回车作为换行
		System.out.println("输入的内容:"+str);
	}
}

程序执行结果:

请输入数据:小山code
输入的内容:小山code

此程序将 System.in 字节输入流对象转换为了 BufferedReader 类对象,可以利用 readLine() 方法直接读取键盘输入的数据,并且可以很好地进行中文处理。

那么既然System.in属于 InputStream类型对象,并且对于字节流也提供了一个 BufferedInputStream 的字节缓冲输入流对象,为什么不使用这个类,而非要使用 BufferedReader类呢?

原因是字符流方便处理中文,并且支持String返回。打开 BufferedInputStream 类的方法可以发现,在该类中并没有类似 readLine() 功能的方法,也就是说只有 BufferedReader 类可以直接将一行输入数据以字符串的形式返回,这样用户可以进行更多操作,例如:数据类型转换、正则验证、字符串 操作等。

//	范例 7: 判断输入内容
package com.xiaoshan.demo;

import java.io.BufferedReader;
import java.io.InputStreamReader;

public class TestDemo {
	public static void main(String[] args)throws Exception{
		BufferedReader buf = new BufferedReader(new InputStreamReader(System.in));
		boolean flag = true;						//编写一个循环的逻辑
		while (flag){								//判断标志位
			System.out.print("请输入年龄:");			//提示信息
			String str = buf.readLine();				//以回车作为换行
			if (str.matches("\\d{1,3}")){			//输入数据由数字组成
				System.out.println("年龄是:"+ Integer.parseInt(str));
				flag = false;	//退出循环
			}else {
				System.out.println("年龄输入错误,应该由数字所组成。");
			}
		}
	}
}

程序执行结果:

请输入年龄:https://lst66.blog.csdn.net/
年龄输入错误,应该由数字所组成。
请输入年龄:27
年龄是:27

此程序实现了一个简单的逻辑,要求用户在输入年龄时必须输入数字,由于BufferedReader读取数据时使用的readLine()方法返回的是String类型,所以可以利用正则进行判断。当判断通过时会利用包装类将字符串转换为 int型数据,同时会退出while循环;当判断失败时会提示用户错误信息,并且等待用户重新输入。本程序流程如下图所示。

图4 程序执行流程

这个程序实际上是所有实际项目开发的一个缩影,在程序中由于输入数据的返回类型是 String , 就方便利用正则进行判断 ,或者进行数据类型的转换。所以只要是输入数据一般都会采用 String类型,像在 JSP 技术中接收表单参数使用的 request.getParameter() 方法实际上返回的也是 String 类型的对象 。

字符缓冲流除了可以接收输入信息外,也可以利用缓冲区进行文件的读取,此时只需要在实例化
BufferedReader 类对象时传递 FileReader 类(实际上也可以使用 FileInputStream, 但是依然需要 InputStreamReader 转换)。

//	范例 8: 读取文件
package com.xiaoshan.demo;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileReader;

public class TestDemo {
	public static void main(String[] args) throws Exception{
		File file = new File("D:" + File.separator + "xiaoshan.txt");
		//使用文件输入流实例化 BufferedReader类对象
		BufferedReader buf = new BufferedReader(new FileReader(file));
		String str = null;		//接收输入数据
		//读取数据并判断是否存在 //输出读取内容
		while ((str = buf.readLine()) != null){
			System.out.println(str);
		}
		buf.close();
	}
}

运行结果:

姓名:小山, 年龄:19, 成绩:59.96

此程序使用 BufferedReader 读取输入的数据信息,以“\n”作为读取数据的分隔符,并在读取时采用循环的模式,将每一行数据读取后直接进行输出,利用这样的处理方式要比直接使用 InputStream 读取更加简单。

3️⃣ 扫描流:Scanner

从 JDK 1.5 开始增加了一个 java.util.Scanner 的程序类,利用这个类可以方便地实现数据的输入操作。 Scanner 类并没有定义在 java.io 包中,而是定义在了 java.util 包中,所以此类是一个工具类,此类的继承结构如下图所示。

图5 Scanner 类的继承结构

在JDK 1.5之前如果要进行数据的输入操作,使用 java.io.BufferedReader 类是最方便的,但是 BufferedReader 类会存在以下两个问题。
(1)它读取数据时只能按照字符串返回:public String readLine() throws IOException
(2)所有的分隔符都是固定的。

为此,从JDK 1.5开始增加了新的Scanner类,而Scanner类增加后对于键盘数据输入的实现也会变得更加简单,这一点大家可以观察到。

Scanner类的定义中可以发现其 实现了Iterator接口,它在整个Java开发中是一个重要的接口,在 Iterator接口里面主要定义了两个抽象方法:hasNext() (判断是否有数据)、next()(取得数据),同时这两个方法也会出现在Scanner类中,并且Scanner类提供了具体数据类型的判断与取得操作。

通过Scanner 类的继承关系可以发现, Scanner 不仅实现了 Iterator (迭代)接口与 Closeable 接口,而且
Scanner 类的构造方法还可以接收 InputStreamFile 等类型以实现输入流操作。Scanner 类中定义的常用方法如下所示。

方法类型描述
public Scanner(InputStream source)构造接收InputStream输入流对象,此为输入源
public boolean hasNext()普通判断是否有数据输入
public String next()普通取出输入数据,以String形式返回
public boolean hasNextXxx()普通判断是否有指定类型数据存在
public 数据类型 nextXxx()普通取出指定数据类型的数据
public Scanner useDelimiter(String pattern)普通设置读取的分隔符
//	范例 9: 利用 Scanner 实现键盘数据输入
package com.xiaoshan.demo;

import java.util.Scanner;

public class TestDemo {
	public static void main(String[] args) throws Exception{
		Scanner scan = new Scanner(System.in);		//准备接收键盘输入数据
		System.out.print("请输入内容:");	//提示信息
		if (scan.hasNext()){	//是否有输入数据
			System.out.println("输入内容:"+ scan.next());	//存在内容则输出
		}
		scan.close();
	}
}

程序执行结果:

请输入内容:小山的博客,https://lst66.blog.csdn.net
输入内容:小山的博客,https://lst66.blog.csdn.net

程序实现了键盘输入数据的操作,通过对比可以发现,利用 Scanner 实现的键盘输入数据操作代码要比 BufferedReader 更加简单。

实际上上边范例的代码属于Scanner使用的标准形式,即先使用 hasNextXxx() 进行判断,有数据之后再进行输入。对于字符串的操作中是否有 hasNextXxx() 方法判断意义不大 (可以直接使用 next(), 但是这样做不标准), 因为即使此时代码不输入任何字符串也表示接收(因为不为NULL, 是一个空字符串), 但是如果是具体的数据类型输入就有意义了。

//	范例 10: 输入一个数字—— double
package com.xiaoshan.demo;

import java.util.Scanner;

public class TestDemo  {
	public static void main(String[] args) throws Exception{
		Scanner scan = new Scanner(System.in);	//准备接收键盘输入数据
		System.out.print("请输入成绩:");
		if(scan.hasNextDouble()){	//表示输入的是一个小数
			double score = scan.nextDouble();	//省略了转型
			System.out.println("输入内容:"+ score);
		}else{						//表示输入的不是一个小数
			System.out.println("输入的不是数字,错误!");
		}
		scan.close();
	}
}

程序执行结果(输入错误):

请输入成绩:xiaoshan
输入的不是数字,错误!

程序执行结果(输入正确):

请输入成绩:100.0
输入内容:100.0

本程序在输入数据时使用了 Scanner 类的 hasNextDouble() 方法来判断输入的数据是否是 double 数值,如果是则利用 nextDouble() 直接将输入的字符串转化为 double 型数据后返回。

Scanner 类中除了支持各种常用的数据类型外,也可以在输入数据时使用正则表达式来进行格式验证。

//	范例 11: 正则验证
package com.xiaoshan.demo;

import java.util.Scanner;

public class TestDemo{
	public static void main(String[] args) throws Exception  {
		Scanner scan = new Scanner(System.in); 	//准备接收键盘输入数据
		System.out.print("请输入生日:");		//提示文字
		if (scan.hasNext("\\d{4}-\\d{2}-\\d{2}")){		//正则验证
			String bir = scan.next("\\d{4}-\\d{2}-\d{2}");		//接收数据
			System.out.println("输入内容:"+ bir);
		}else{								//数据格式错误
			System.out println("输入的生日格式错误!");
		}
		scan.close();
	}
}

程序执行结果:

请输入生日:1997-12-15
输入内容:1997-12-15

此程序利用正则验证了输入的字符串数据,如果数据符合正则规则,则进行接收,如果不符合则提示信息错误。
Scanner 类的构造里面由于接收的类型是 InputStream, 所以依然可以设置一个文件的数据流。

考虑到文件本身会存在多行内容,所以需要考虑读取的分隔符问题(默认是空字符为分隔符,例如:空格或换行), 这样在读取前就必须使用 “useDelimiter()” 方法来设置分隔符。

//	范例 12: 读取文件
package com.xiaoshan.demo;

import java.io.File;
import java.io.FilelnputStream;
import java.util.Scanner;

public class TestDemo  {
	public static void main(String[] args) throws Exception{
		Scanner scan = new Scanner(new FilelnputStream(new File("D:" + File.separator + "xiaoshan.txt")));                         //设置读取的文件输入流
		scan.useDelimiter("\n");         	//设置读取的分隔符
		while (scan.hasNext()){     //循环读取       
			System.out.println(scan.next());   	//直接输出读取数据
		}
		scan.close();
	}
}

此程序利用 Scanner 实现了文件数据的读取操作,在读取前可以使用 useDelimiter() 方法设置指定的读取分隔符,随后就可以利用 while 循环来读取数据。

通过一系列分析,大家应该已经清楚了 InputStreamOutputStream的不足,同时也应该发现利用 PrintStream(或PrintWriter) 可以加强程序输出数据的操作支持, Scanner(或 BufferedReader) 可以加强程序输入数据的操作支持。所以在日后的开发 中,只要操作的是文本数据 (不是二进制数据), 输出时都使用打印流,输入时都使用扫描流(或者使用字符缓冲区输入流)。

4️⃣ 对象序列化

Java 允许用户在程序运行中进行对象的创建,但是这些创建的对象都只保存在内存中,所以这些对象的生命周期都不会超过 JVM 进程。但是在很多时候可能需要在 JVM 进程结束后对象依然可以被保存下来,或者在不同的 JVM 进程中要进行对象传输,那么在这样的情况下就可以采用对象序列化的方式进行处理。

4.1 序列化接口:Serializable

对象序列化的本质实际上就是将内存中所保存的对象数据转换为二进制数据流进行传输的操作。但 是并不是所有类的对象都可以直接进行序列化操作,要被序列化的对象所在的类一定要实现 java.io.Serializable 接口。而通过文档观察可以发现,序列化接口里面并没有任何操作方法存在,因为它是一个标识接口,表示一种能力。

//	范例 13: 定义一个可以被序列化对象的类
package com.xiaoshan.demo;

import java.io.Serializable;

@SuppressWarnings("serial")		//压制序列化版本号警告信息 
class Book implements Serializable{		//此类对象可以被序列化
	private String title;
	private double price;
	
	public Book(String title, double price){
		this.title = title;
		this.price = price;
	}

	@Override
	public String toString(){
		return "书名:"+ this.title + ",价格:" + this.price;
	}
}

4.2 实现序列化与反序列化

实现了Serializable 接口后并不意味着对象可以实现序列化操作。实际上在对象序列化与反序列化的操作中,还需要以下两个类的支持。

  • 序列化操作类: java.io.ObjectOutputStream,将对象序列化为指定格式的二进制数据;
  • 反序列化操作类: java.io.ObjectInputStream,将序列化的二进制对象信息转换回对象内容。

ObjectOutputStream类的常用方法如下所示:

方法类型描述
public ObjectOutputStream(OutputStream out) throws IOException构造指定对象序列化的输出流
public final void writeObject(Object obj) throws IOException普通序列化对象

ObjectInputStream类的常用方法如下所示:

方法类型描述
public ObjectInputStream(InputStream in) throws IOException构造指定对象反序列化的输入流
public final Object readObject() throws IOException,ClassNotFoundException普通从输入流中读取对象

通过 ObjectOutputStreamObjectInputStream 类的方法定义可以发现,序列化对象时
(writeObject()) 接收的参数统一为 Object, 而反序列化对象时 (readObject()) 返回的类型也为 Object, 所以这两个类可以序列化或反序列化Java 中的所有数据类型(Object 可以接收所有引用类型, 将基本类型自动装箱为包装类后接收)。

//	范例 14: 实现序列化对象操作——ObjectOutputStream
package com.xiaoshan.demo;

import java.io.File;
import java.io.FileOutputStream;
import java.io.ObjectOutputStream;

public class TestDemo {
	public static void main(String[] args) throws Exception {
		ser();
	}
	public static void ser() throws Exception {
		ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream( new File("D:" + File.separator + "book.ser")));
		oos.writeObject(new Book("Java开发实战经典",79.8));//序列化对象
		oos.close();
	}
}

此程序实现了 Book 类对象序列化的操作,在实例化 ObjectOutputStream 类对象时需要设置一个 OutputStream 类对象,此时设置的为 FileOutputStream 子类,表示对象将被序列化到文件中。book.ser 文件保存内容如图所示。

在这里插入图片描述

下面演示反序列化操作。

//	范例 15: 实现反序列化操作——ObjectInputStream
package com.xiaoshan.demo;

import java.io.File;
import java.io.FileInputStream;
import java.io.ObjectInputStream;

public class TestDemo {
	public static void main(String[] args) throws Exception {
		dser();
	}
	public static void dser() throws Exception {
		ObjectInputStream ois = new ObjectInputStream(new FilelnputStream(new File("D:"+  File.separator + "book.ser")));
		Object obj= ois.readObject();    	//反序列化对象
		Book book =(Book) obj;            	//转型
		System.out.println(book);
		ois.close();
	}
}	

程序执行结果:

书名:Java开发实战经典,价格:79.8

此程序首先利用 ObjectInputStream 类通过指定 InputStream 子类对象确定对象读取的输入流为文件,然后利用 readObject() 方法可以将被序列化的对象反序列化回来。由于返回的对象类型都是 Object, 所以如果有需要可以利用向下转型的操作,将返回对象转化为具体的子类类型。

4.3 transient 关键字

Java 中对象最有意义的内容就是对象的属性信息,所以在默认情况下,如果要进行对象的序列化操作,所序列化的一定是对象的属性信息,并且该对象中的所有属性信息都将被序列化。如果某些属性的内容不需要被保存,就可以通过 transient 关键字来定义。

//	范例 16: 定义不需要序列化的属性
package com.xiaoshan.demo;

import java.io.Serializable;

@SuppressWarnings("'serial")
class Book implements Serializable { 	//此类对象可以被序列化 
	private transient String title;		//此属性无法被序列化
	private double price;
	
	public Book(String title, double price){
		this.title = title;
		this.price = price;
	}
	
	@Override
	public String toString(){
		return "书名:"+this.title+",价格:"+ this.price;
	}
}

此程序中 Book 类的 title 属性上使用了 transient 关键字进行定义,这样当进行对象序列化操作时, 此属性的内容将不能被保存。

既然 java.io.Serializable 接口中没有定义任何抽象方法,那么是不是意味着,开发中所有的类都实现 Serializable接口会比较好呢?

在实际的开发中,并不是所有的类都要求去实现序列化接口,只有需要传输对象所在的类时才需要实现Serializable接口,而这样的类最主要的就是简单Java类。由于简单Java 类的实际定义与数据表结构相似,所以在很多情况下,很少会使用 transient关键字。

🌾 总结

在本文中,我们详细探讨了Java中System类对IO操作的支持,以及字符缓冲流、扫描流和对象序列化流的应用。

首先,我们了解到System类提供了三个重要的I/O流:错误输出(System.err)、信息输出(System.out)和系统输入(System.in)。通过这些流,我们可以方便地处理错误信息、输出结果和用户输入。

接下来,我们研究了字符缓冲流(BufferedReader),它提供了高效读取文本数据的能力。通过使用缓冲机制,字符缓冲流可以减少磁盘或网络读取的次数,从而提高性能。

我们还讨论了扫描流(Scanner),它提供了一种简便的方式来解析基本类型和字符串。通过Scanner类,我们可以方便地从输入源(如键盘或文件)中读取数据,并以各种类型的形式进行处理。

最后,我们探索了对象序列化流,它提供了一种将对象转换为字节流并写入到文件中的机制。我们介绍了序列化接口(Serializable)的作用以及如何实现对象的序列化与反序列化。同时,我们了解到transient关键字的用途,它可以让对象的某些字段在序列化过程中被忽略。

通过掌握System类对IO的支持以及字符缓冲流、扫描流和对象序列化流的应用,我们可以更有效地处理输入输出操作,并实现数据的读取、写入和持久化。这些技术在日常开发中非常实用,特别是在涉及文本处理、数据交互和对象存储等方面。


温习回顾上一篇(点击跳转)
《【Java基础教程】(四十五)IO篇 · 中:转换流、内存流和打印流(探索装饰设计模式与PrintStream类的进阶),文件操作案例实践、字符编码问题~》

继续阅读下一篇(点击跳转)
《【Java基础教程】(四十七)网络编程篇:》

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/787902.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Latex】官方文档教你长公式对齐方法

具体请参见下面链接 Aligning equations with amsmath - Overleaf, Online LaTeX Editor 详细介绍了各种公式的对齐方式&#xff0c;特别是 长公式的换行方法 以及多公式排列情况&#xff1a;

ctfshow-web2

0x00 前言 CTF 加解密合集CTF Web合集 0x01 题目 最简单的SQL注入0x02 Write Up 这道题很多wp&#xff0c;但是为了完整性还是写一下 知道这里是sql注入&#xff0c;确认一下是什么类型的闭合 使用a or 11 #进行测试 可以知道闭合是单引号闭合&#xff0c;然后查一下显示的…

openGauss学习笔记-20 openGauss 简单数据管理-DISTINCT

文章目录 openGauss学习笔记-20 openGauss 简单数据管理-DISTINCT20.1 语法格式20.2 参数说明20.3 示例 openGauss学习笔记-20 openGauss 简单数据管理-DISTINCT DISTINCT关键字与SELECT语句一起使用&#xff0c;用于去除重复记录&#xff0c;只获取唯一的记录。 当一个表中有…

【python中级】 base64编码将图片数据转化为成字符串

【python中级】 base64编码将图片数据转化为成字符串 1、背景2、图片base64编码1、背景 在Qt编程的时候,有的地方比如logo需要加载图片,在程序打包之后,需要将图片拷贝进项目。 能不能将图片转化成一串字符串,使用该字符串代替图像地址。 即这样就可以字符串变量的形式编…

Terraform学习日记-AWS-EC2

terraform install https://developer.hashicorp.com/terraform/tutorials/aws-get-started/install-cli 这里我们使用 aws-linux-2022 作为执行环境 # sudo yum install -y yum-utils# sudo yum-config-manager --add-repo https://rpm.releases.hashicorp.com/AmazonLinux/…

选择合适的图表,高效展现数据魅力

随着大数据时代的来临&#xff0c;数据的重要性愈发凸显&#xff0c;数据分析和可视化成为了决策和传递信息的重要手段。在数据可视化中&#xff0c;选择合适的图表是至关重要的一环&#xff0c;它能让数据更加生动、直观地呈现&#xff0c;为观众提供更有说服力的信息。本文将…

从娱乐产品到效率工具,ARknovv首款AR眼镜回归“AR本质”

如果说2022年是AR的元年&#xff0c;2023年则有望成为消费级AR眼镜的新拐点。 今年AR眼镜行业发展明显加快&#xff0c;且不断有大厂入局&#xff1a;今年2月小米发布无线AR眼镜探索版&#xff1b;3月荣耀也推出了一款全新的观影眼镜&#xff1b;而苹果在6月发布的MR头显Visio…

【java】2208. 将数组和减半的最少操作次数

给你一个正整数数组 nums 。每一次操作中&#xff0c;你可以从 nums 中选择 任意 一个数并将它减小到 恰好 一半。&#xff08;注意&#xff0c;在后续操作中你可以对减半过的数继续执行操作&#xff09; 请你返回将 nums 数组和 至少 减少一半的 最少 操作数。 示例 1&#…

Android性能优化之游戏的Theme背景图

近期&#xff0c;对游戏的内存优化&#xff0c;通过内存快照发现&#xff0c;某个Activity的theme背景图 占用3M 多。考虑着手对齐进行优化。 问题 查看游戏中的内存快照&#xff0c;发现有一个图片bitmap 占用3M 多&#xff0c;设置在Activity的背景中&#xff1a; 查看Phon…

用户档案与用户画像:很容易混淆

用户档案与用户画像&#xff1a;很容易混淆 需求分析或设计需要梳理 趣讲大白话&#xff1a;搞清楚怎么来&#xff0c;才能用好 【趣讲信息科技236期】 **************************** User Profile&#xff0c;不知道谁提出来的。被译为用户档案或用户简要&#xff0c;它是基于…

一篇文章搞定《APP的启动流程》

一篇文章搞定《APP的启动流程》 前言冷启动、温启动、热启动启动中的重要成员简介zygote进程InstrumentationSystemServer进程ActivityManagerServiceBinderActivityThread 启动的步骤详解一、点击桌面图标二、创建进程三、初始化APP进程四、APP进程与System_server的绑定五、初…

《深度解析Docker与微服务架构:构建灵活可扩展的现代应用》

&#x1f337;&#x1f341; 博主 libin9iOak带您 Go to New World.✨&#x1f341; &#x1f984; 个人主页——libin9iOak的博客&#x1f390; &#x1f433; 《面试题大全》 文章图文并茂&#x1f995;生动形象&#x1f996;简单易学&#xff01;欢迎大家来踩踩~&#x1f33…

【SQL语句复习】第1-2章

SQL复习 学习目标&#xff1a;复习SQL语句 学习地址&#xff1a;https://linklearner.com/learn/detail/70 第一章 初始数据库 数据库是将大量数据保存起来&#xff0c;通过计算机加工而成的可以进行高效访问的数据集合。该数据集合称为数据库&#xff08;Database&#xf…

大学生用一周时间给麦当劳做了个App(Vue版)

背景 有个大学生粉丝最近私信联系我&#xff0c;说基于我之前开源的多语言项目做了个仿麦当劳的项目&#xff0c;虽然只是个样子货&#xff0c;但是收获颇多&#xff0c;希望把自己写的代码开源出来供大家一起学习进度。这个小伙伴确实是非常积极上进&#xff0c;很多大学生&a…

C盘空间不足:解决办法完整教程

当C盘空间不足时&#xff0c;你可以尝试以下几种解决方案&#xff1a; 1. 清理临时文件&#xff1a;使用Windows自带的磁盘清理工具&#xff0c;可以删除临时文件、回收站中的文件和其他不必要的系统文件&#xff0c;释放一些空间&#xff0c;推荐使用工具分区助手。 2. 卸载不…

数据库版本管理工具Flyway入门实战

From version control to continuous delivery, Flyway helps individuals, teams, and enterprises build on application delivery processes to automate database development. 1.引言 在项目开发中&#xff0c;一直在探索如何进行数据库的版本管理。关注的公众号推送了…

踩坑 视觉SLAM 十四讲第二版 ch8 编译及运行问题

1.fmt相关 CMakeLists.txt中&#xff1a;在后面加上 fmt target_link_libraries(optical_flow ${OpenCV_LIBS} fmt ) target_link_libraries(direct_method ${OpenCV_LIBS} ${Pangolin_LIBRARIES} fmt )2.不存在用户定义的从 "std::_Bind<void (OpticalFlowTracker::…

动态内存管理函数的使用与优化技巧(内存函数、柔性数组)

目录 前言 一、动态内存函数 为什么存在动态内存分配 动态内存函数介绍 malloc和free calloc realloc 常见的错误 经典笔试题目 二、C/C程序的内存开辟 三、柔性数组 柔性数组的特点&#xff1a; 柔性数组的使用 柔性数组的优势 前言 动态内存管理函数是C语言中非常重要的一部…

Verilog 学习之路(三)——牛客刷题篇

1.输入序列连续的序列检测 题面 [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-kJH9kHFH-1690301233803)(https://s2.loli.net/2023/07/26/HJPXR2mhbaVCG6d.png)]思路 对于序列检测题目&#xff0c;常规的解法有两种&#xff1a;状态机法和序列缓存…

需求管理全过程流程图及各阶段核心关注点详解

分析报告指出&#xff0c;多达76%的项目失败是因为差劲的需求管理&#xff0c;这个是项目失败的最主要原因&#xff0c;比落后的技术、进度失控或者混乱的变更管理还要关键。很多项目往往在开始的时候已经决定了失败&#xff0c;谜底就在谜面上&#xff0c;开始就注定的失败&am…