IO知识点整理(序列化,管道流,数据流,字节数组流,与编码)
一:序列化的问题
1.序列号的使用问题
关于在序列化中的序列号的使用问题,一般要是使用。
因为,每次要序列化的类产生都会产生一个一个新的序列号,如果将这个类的程序修改后,就会产生新的序列号,以前序列化后的文件将不会被读取。
所以,为了程序修改后,以前序列化后的程序仍然可以被读取,使用静态的序列号十分有必要。
2.将数据进行序列化
中在ObjectOutputStream类。
import java.io.FileOutputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable; public class Test114 { public static void main(String[] args) throws Exception{
writeObj();
}
public static void writeObj() throws Exception{
ObjectOutputStream objo=new ObjectOutputStream(
new FileOutputStream("op.txt"));
objo.writeObject(new People0("lisi",10));
objo.close();
}
} class People0 implements Serializable{
//序列号
public static final long serialVersionUID = 42L;
//属性
private String name;
private int age;
public People0(String name,int age){
this.name=name;
this.age=age;
}
public String toString() {
return name+":"+age;
}
}
3.运行结果
4.读序列化后的文件
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable; public class Test114 { public static void main(String[] args) throws Exception{
//writeObj();
readObj();
}
public static void readObj() throws Exception{
ObjectInputStream readObj=new ObjectInputStream(
new FileInputStream("op.txt"));
People0 p=(People0)readObj.readObject();
//因为有toString方法,所以可以直接打印
System.out.println(p);
//
readObj.close();
}
public static void writeObj() throws Exception{
ObjectOutputStream objo=new ObjectOutputStream(
new FileOutputStream("op.txt"));
objo.writeObject(new People0("lisi",10));
objo.close();
}
} class People0 implements Serializable{
//序列号
public static final long serialVersionUID = 42L;
//属性
private String name;
private int age;
public People0(String name,int age){
this.name=name;
this.age=age;
}
public String toString() {
return name+":"+age;
}
}
5.运行结果
6.不能序列化的情况
类中的static不能够序列化,因为static的属性在方法区,而序列化主要是序列化的是栈里的文件数据。
同时transient修饰的属性不能够序列化。
7.不能序列化的程序演示
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable; public class Test114 { public static void main(String[] args) throws Exception{
//writeObj();
readObj();
}
public static void readObj() throws Exception{
ObjectInputStream readObj=new ObjectInputStream(
new FileInputStream("op.txt"));
People0 p=(People0)readObj.readObject();
//因为有toString方法,所以可以直接打印
System.out.println(p);
//
readObj.close();
}
public static void writeObj() throws Exception{
ObjectOutputStream objo=new ObjectOutputStream(
new FileOutputStream("op.txt"));
objo.writeObject(new People0("lisi",10,"usa"));
objo.close();
}
} class People0 implements Serializable{
//序列号
public static final long serialVersionUID = 42L;
//属性
private String name;
//添加transient的修饰
private transient int age;
//新加static
private static String contry="cn";
public People0(String name,int age,String contry){
this.name=name;
this.age=age;
this.contry=contry;
}
public String toString() {
return name+":"+age+":"+contry;
}
}
8.运行结果
运行的顺序是,先序列化,序列化后生成新的文件后,再进行反序列化。
这时,才会发现static与transient都没有被序列化。
二:管道流与RandomAccessFile
1.管道流
可以将读写进行相连。
管道输入流应该连接到管道输出流;管道输入流提供要写入管道输出流的所有数据字节。
通常,数据由某个线程从 PipedInputStream
对象读取,并由其他线程将其写入到相应的 PipedOutputStream
。
不建议对这两个对象尝试使用单个线程,因为这样可能死锁线程。
2.管道流程序(多线程)
import java.io.IOException;
import java.io.PipedInputStream;
import java.io.PipedOutputStream; public class Test115 { public static void main(String[] args) throws Exception {
PipedInputStream in=new PipedInputStream();
PipedOutputStream out=new PipedOutputStream();
in.connect(out); //链接connect Writer w=new Writer(out);
Reader r=new Reader(in);
new Thread(w).start();
new Thread(r).start();
} }
class Writer implements Runnable{
private PipedOutputStream out;
public Writer(PipedOutputStream out){
this.out=out;
}
public void run(){
try{
Thread.sleep(5000);
out.write("sjhdbs".getBytes());
out.close();
}catch(Exception e){
e.toString();
}
}
}
class Reader implements Runnable{
private PipedInputStream in;
public Reader(PipedInputStream in){
this.in=in;
}
public void run(){
try{
byte[] buf=new byte[1024];
int len=in.read(buf);
String str=new String(buf,0,len);
System.out.println("str="+str);
in.close();
}catch(Exception e){
e.toString();
}
}
}
3.运行结果
4.RandomAccessFile
该类不是算是IO体系中子类。
而是直接继承自Object。
但是它是IO包中成员。因为它具备读和写功能。
内部封装了一个数组,而且通过指针对数组的元素进行操作。
可以通过getFilePointer获取指针位置,
同时可以通过seek改变指针的位置。
通过构造函数可以看出,该类只能操作文件。
而且操作文件还有模式:只读r,,读写rw等。
5.写入程序
在程序中使用writeInt,这个方法是每次写入4个字节,如果使用write则是每次写入1个字节。
考虑到数字的越界,这里使用writeInt方法。
这个方法的写方式在下面的程序中没有什么特殊的地方。
import java.io.FileNotFoundException;
import java.io.RandomAccessFile; public class Test116 {
public static void main(String[] args) throws Exception{
randomWrite();
}
/**
* 简单的写入
* @throws Exception
*/
public static void randomWrite() throws Exception{
RandomAccessFile raf=new RandomAccessFile("pu.txt","rw");
raf.write("张三".getBytes());
raf.writeInt(97);
raf.write("李四".getBytes());
raf.writeInt(20);
raf.close();
}
}
6.结果
7.分别使用seek与skipBytes的读方式(待探求为啥读取有问题)
import java.io.FileNotFoundException;
import java.io.RandomAccessFile; public class Test117 { public static void main(String[] args) throws Exception{
System.out.println("seek function");
randomReadSeek();
System.out.println("skipBytes function");
randomReadSkipbytes();
}
/**
* seek function
* @throws Exception
*/
public static void randomReadSeek() throws Exception{
RandomAccessFile raf=new RandomAccessFile("pu.txt", "r");
raf.seek(0);
byte[] buf = new byte[8]; raf.read(buf); String name = new String(buf); int age = raf.readInt(); System.out.println("name="+name);
System.out.println("age="+age); raf.close();
}
public static void randomReadSkipbytes()throws Exception{
RandomAccessFile raf=new RandomAccessFile("pu.txt", "r");
raf.skipBytes(2);
byte[] buf = new byte[8]; raf.read(buf); String name = new String(buf); int age = raf.readInt(); System.out.println("name="+name);
System.out.println("age="+age); raf.close();
} }
三:数据流对象的操作
1.关于DataInputStream与DataOutputStream的介绍
其主要是将数据与流进行结合。
2.写数据程序
import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream; public class Test118 { public static void main(String[] args) throws Exception {
dataWrite(); }
public static void dataWrite()throws Exception{
DataOutputStream dos=new DataOutputStream(new FileOutputStream("data.txt"));
dos.writeInt(78);
dos.writeBoolean(true);
dos.writeDouble(89.90907);
dos.close();
}
}
3.读的程序
注意点是,读取数据的时候,必须与写的数据类型相对应,保持读写的一致性。
import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream; public class Test118 { public static void main(String[] args) throws Exception {
//dataWrite();
dataRead();
}
public static void dataRead() throws Exception{
DataInputStream dis=new DataInputStream(new FileInputStream("data.txt"));
int num=dis.readInt();
boolean state=dis.readBoolean();
double twonum=dis.readDouble();
System.out.println("num="+num+",state="+state+",twonum="+twonum);
}
public static void dataWrite()throws Exception{
DataOutputStream dos=new DataOutputStream(new FileOutputStream("data.txt"));
dos.writeInt(78);
dos.writeBoolean(true);
dos.writeDouble(89.90907);
dos.close();
}
}
4.运行结果
5.关于DataInputStream中的writeUIF的特殊性(包含readUIF)
所以在下面,通过程序进行验证。
6.带编码的可以读写字串的程序
import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.FileInputStream;
import java.io.FileOutputStream; public class Test119 { public static void main(String[] args)throws Exception {
writeUtf();
readUtf();
}
public static void writeUtf()throws Exception{
DataOutputStream dos=new DataOutputStream(new FileOutputStream("string.txt"));
dos.writeUTF("shdfcsdcnklbsnkl");
dos.close();
}
public static void readUtf()throws Exception{
DataInputStream dis=new DataInputStream(new FileInputStream("String.txt"));
String str=dis.readUTF();
System.out.println("str="+str);
} }
7.运行结果
8.使用FileOutputStream的编码方式对字符串的读写
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.io.UnsupportedEncodingException; public class Test120 {
public static void main(String[] args) throws Exception{
OutputStreamWriter fos=new OutputStreamWriter(new FileOutputStream("gnk.txt"),"gbk");
fos.write("你好");
fos.close();
}
}
9.结果
四:字节数组流的操作
1.程序
/*
用于操作字节数组的流对象。 ByteArrayInputStream :在构造的时候,需要接收数据源,。而且数据源是一个字节数组。 ByteArrayOutputStream: 在构造的时候,不用定义数据目的,因为该对象中已经内部封装了可变长度的字节数组。
这就是数据目的地。
因为这两个流对象都操作的数组,并没有使用系统资源。
所以,不用进行close关闭。
在流操作规律讲解时:
源设备,
键盘 System.in,硬盘 FileStream,内存 ArrayStream。
目的设备:
控制台 System.out,硬盘FileStream,内存 ArrayStream。
用流的读写思想来操作数据。
*/
import java.io.*;
class ByteArrayStream
{
public static void main(String[] args)
{
//数据源。
ByteArrayInputStream bis = new ByteArrayInputStream("ABCDEFD".getBytes());
//数据目的
ByteArrayOutputStream bos = new ByteArrayOutputStream(); int by = 0; while((by=bis.read())!=-1)
{
bos.write(by);
}
System.out.println(bos.size());
System.out.println(bos.toString());
}
}
五:编码
1.介绍
编码:将字符串变成字节数组
解码:将字节数组变成字符串
String-->byte[]; str.getBytes(charsetName);
byte[] -->String: new String(byte[],charsetName);
2.程序
import java.util.Arrays; public class Test121 {
public static void main(String[] args)throws Exception {
String str="嘿嘿";
byte[] buf1=str.getBytes("utf-8");
//打印字节码
System.out.println(Arrays.toString(buf1));
//utf-8解码
String s1=new String(buf1,"gbk");
System.out.println("s1="+s1);
//再使用utf-8编码
byte[] buf2=s1.getBytes("gbk");
System.out.println(Arrays.toString(buf2));
//打印
String s2=new String(buf2,"utf-8");
System.out.println("s2="+s2);
}
}
3.结果
IO知识点整理(序列化,管道流,数据流,字节数组流,与编码)的更多相关文章
- 6.5(java学习笔记)其他流(字节数组流,数据流,对象流,打印流)
一.字节数组流 之前使用输入输出流的操作的对象是文件,而这里字节数组流操作的对象是内存,内存可以看做是一个字节数组. 使用字节数组流读写就可以看做是从内存A到内存B的读写,对象时内存即字节数组. 1. ...
- JAVA IO分析二:字节数组流、基本数据&对象类型的数据流、打印流
上一节,我们分析了常见的节点流(FileInputStream/FileOutputStream FileReader/FileWrite)和常见的处理流(BufferedInputStream/B ...
- Java:IO流其他类(字节数组流、字符数组流、数据流、打印流、Properities、对象流、管道流、随机访问、序列流、字符串读写流)
一.字节数组流: 类 ByteArrayInputStream:在构造函数的时候,需要接受数据源,而且数据源是一个字节数组. 包含一个内部缓冲区,该缓冲区包含从流中读取的字节.内部计数器跟踪 read ...
- Java IO学习笔记(三)转换流、数据流、字节数组流
转换流 1.转换流:将字节流转换成字符流,转换之后就可以一个字符一个字符的往程序写内容了,并且可以调用字符节点流的write(String s)方法,还可以在外面套用BufferedReader()和 ...
- Java文件与io——字节数组流数据流字符串流
字节数组流 ByteArrayInputStream:包含一个内部缓冲区,该缓冲区包含从流中读取的字节.内部计数器跟踪read方法要提供的下一个字节.关闭ByteArrayInputStream无效. ...
- IO知识点整理(文件File类的使用)
一: 1.API 2.构造函数的程序 注意这集中构造函数的特点. 同时,字段separator的使用. import java.io.File; public class Test101 { publ ...
- Java基础IO类之字节数组流
package IODemo; //字节数组流 :内部维护这着一个字节数组,我们可以利用流的读取机制来处理字符串 无需关闭,不会报IO异常 // ByteArrayInputstream ByteAr ...
- Java IO 流-- 字节数组流ByteArrayInPutStream ByteArrayOutPutStream
字节数组流输于缓冲流,放在jvm内存中,java可以直接操作.我们使用时可以不用关闭,交给GC垃圾回收机制处理. 当然我们为了保持良好习惯和代码一致性也可以加上关闭语句. 当其实我么打开ByteArr ...
- Java字节数组流学习
字节数组流 基于内存操作,内部维护着一个字节数组,我们可以利用流的读取机制来处理字符串.无需关闭,不会报IOException. ByteArrayInputStream 包含一个内部缓冲区,该缓冲区 ...
随机推荐
- unity3d 几种镜头畸变
1.Fisheye distortion 鱼眼镜头 解释来自百度百科:鱼眼镜头是一种焦距为16mm或更短的并且视角接近或等于180°. 它是一种极端的广角镜头,“鱼眼镜头”是它的俗称.为使镜头达到最 ...
- bzoj3612 平衡 (dp)
设f[i][j]为把i拆成j个不重复的.大于0小于等于N的数的方案数 我们考虑一个方案是怎么来的:(初始状态是f[0][0]=1) 如果这个方案里有1,那它是先把原来的状态的每个数加1.然后再增加一个 ...
- 【bzoj3196】 Tyvj1730—二逼平衡树
http://www.lydsy.com/JudgeOnline/problem.php?id=3196 (题目链接) 题意 1.查询k在区间内的排名:2.查询区间内排名为k的值:3.修改某一位值上的 ...
- box-sizing border-box 的理解
http://blog.csdn.net/isaisai/article/details/20449827 -webkit-box-sizing: border-box; 则div 设置的宽高将包含 ...
- python独立环境virtualenv(并在pycharm中指定)
上篇文章中讲到 Django 如何启动以及配置 sessions 功能.sessions 功能用是跟踪用户的状态,经常结合 Cookie 功能实现自动登录功能. 所谓的“自动登录”指的是:我们登录一些 ...
- nginx 重写URL尾部斜杠
1. 在URL结尾添加斜杠 在虚拟主机中这么添加一条改写规则: rewrite ^(.*[^/])$ $1/ permanent;或者rewrite ^([/\w-_]*[^/])$ $1/ perm ...
- C#获取文件超大图标256*256(转)
从Bing搜索得到,保存于此 using System; using System.Collections.Generic; using System.Linq; using System.Text; ...
- 前端 ajax 改写登录界面
SSM 整合项目开发到一个阶段,想慢慢地把前台框架等技术引入进来 突然碰到一个困惑好久的问题: ajax 替换原本 form 表单 post 提交登录: 一直 404 错误,心塞.... 最后发现原来 ...
- BZOJ4103 异或运算
4103: [Thu Summer Camp 2015]异或运算 Time Limit: 20 Sec Memory Limit: 512 MB Description 给定长度为n的数列X={x1 ...
- 如何得到Slave应用relay-log的时间
官方社区版MySQL 5.7.19 基于Row+Position搭建的一主一从异步复制结构:Master->{Slave} ROLE HOSTNAME BASEDIR DATADIR IP PO ...