Java基础-22总结登录注册IO版,数据操作流,内存操作流,打印流,标准输入输出流,转换流,随机访问流,合并流,序列化流,Properties

本文涉及的产品
系统运维管理,不限时长
简介: 你需要的是什么,直接评论留言。 获取更多资源加微信公众号“Java帮帮” (是公众号,不是微信好友哦) 还有“Java帮帮”今日头条号,技术文章与新闻,每日更新,欢迎阅读 学习交流请加Java帮帮交流QQ群553841695 分享是一种美德,分享更快乐! 1:登录注册IO版本案例(掌握) 要求,对着写一遍。 cn.i

你需要的是什么,直接评论留言。

获取更多资源加微信公众号“Java帮帮” (是公众号,不是微信好友哦)

还有“Java帮帮”今日头条号,技术文章与新闻,每日更新,欢迎阅读

学习交流请加Java帮帮交流QQ群553841695

分享是一种美德,分享更快乐!


1:登录注册IO版本案例(掌握)

要求,对着写一遍。

cn.itcast.pojo User

cn.itcast.dao UserDao

cn.itcast.dao.impl UserDaoImpl(实现我不管)

cn.itcast.game GuessNumber

cn.itcast.test UserTest


package cn.itcast.pojo;(1)


/**

 * 这是用户基本描述类

 * 

 * @author 风清扬

 * @version V1.1

 * 

 */

public class User {

// 用户名

private String username;

// 密码

private String password;


public User() {

}


public String getUsername() {

return username;

}


public void setUsername(String username) {

this.username = username;

}


public String getPassword() {

return password;

}


public void setPassword(String password) {

this.password = password;

}

}


package cn.itcast.dao;(2)


import cn.itcast.pojo.User;


/**

 * 这是针对用户进行操作的接口

 * 

 * @author 风清扬

 * @version V1.1

 * 

 */

public interface UserDao {

/**

 * 这是用户登录功能

 * 

 * @param username

 *            用户名

 * @param password

 *            密码

 * @return 返回登录是否成功

 */

public abstract boolean isLogin(String username, String password);


/**

 * 这是用户注册功能

 * 

 * @param user

 *            要注册的用户信息

 */

public abstract void regist(User user);

}


package cn.itcast.dao.impl;(3)


import java.io.BufferedReader;

import java.io.BufferedWriter;

import java.io.File;

import java.io.FileNotFoundException;

import java.io.FileReader;

import java.io.FileWriter;

import java.io.IOException;


import cn.itcast.dao.UserDao;

import cn.itcast.pojo.User;


/**

 * 这是用户操作的具体实现类(IO版)

 * 

 * @author 风清扬

 * @version V1.1

 * 

 */

public class UserDaoImpl implements UserDao {

// 为了保证文件一加载就创建

private static File file = new File("user.txt");


static {

try {

file.createNewFile();

} catch (IOException e) {

System.out.println("创建文件失败");

// e.printStackTrace();

}

}


@Override

public boolean isLogin(String username, String password) {

boolean flag = false;


BufferedReader br = null;

try {

// br = new BufferedReader(new FileReader("user.txt"));

br = new BufferedReader(new FileReader(file));

String line = null;

while ((line = br.readLine()) != null) {

// 用户名=密码

String[] datas = line.split("=");

if (datas[0].equals(username) && datas[1].equals(password)) {

flag = true;

break;

}

}

} catch (FileNotFoundException e) {

System.out.println("用户登录找不到信息所在的文件");

// e.printStackTrace();

} catch (IOException e) {

System.out.println("用户登录失败");

// e.printStackTrace();

} finally {

if (br != null) {

try {

br.close();

} catch (IOException e) {

System.out.println("用户登录释放资源失败");

// e.printStackTrace();

}

}

}


return flag;

}


@Override

public void regist(User user) {

/*

 * 为了让注册的数据能够有一定的规则,我就自己定义了一个规则: 用户名=密码

 */

BufferedWriter bw = null;

try {

// bw = new BufferedWriter(new FileWriter("user.txt"));

// bw = new BufferedWriter(new FileWriter(file));

// 为了保证数据是追加写入,必须加true

bw = new BufferedWriter(new FileWriter(file, true));

bw.write(user.getUsername() + "=" + user.getPassword());

bw.newLine();

bw.flush();

} catch (IOException e) {

System.out.println("用户注册失败");

// e.printStackTrace();

} finally {

if (bw != null) {

try {

bw.close();

} catch (IOException e) {

System.out.println("用户注册释放资源失败");

// e.printStackTrace();

}

}

}

}

}


package cn.itcast.game;(4)


import java.util.Scanner;


/**

 * 这是猜数字小游戏

 * 

 * @author 风清扬

 * @version V1.1

 * 

 */

public class GuessNumber {

private GuessNumber() {

}


public static void start() {

// 产生一个随机数

int number = (int) (Math.random() * 100) + 1;


// 定义一个统计变量

int count = 0;


while (true) {

// 键盘录入一个数据

Scanner sc = new Scanner(System.in);

System.out.println("请输入数据(1-100):");

int guessNumber = sc.nextInt();


count++;


// 判断

if (guessNumber > number) {

System.out.println("你猜的数据" + guessNumber + "大了");

} else if (guessNumber < number) {

System.out.println("你猜的数据" + guessNumber + "小了");

} else {

System.out.println("恭喜你," + count + "次就猜中了");

break;

}

}

}

}



package cn.itcast.test;(5)


import java.util.Scanner;


import cn.itcast.dao.UserDao;

import cn.itcast.dao.impl.UserDaoImpl;

import cn.itcast.game.GuessNumber;

import cn.itcast.pojo.User;


/**

 * 用户测试类

 * 

 * @author 风清扬

 * @version V1.1

 * 

 */

public class UserTest {

public static void main(String[] args) {

// 为了能够回来

while (true) {

// 欢迎界面,给出选择项

System.out.println("--------------欢迎光临--------------");

System.out.println("1 登录");

System.out.println("2 注册");

System.out.println("3 退出");

System.out.println("请输入你的选择:");

// 键盘录入选择,根据选择做不同的操作

Scanner sc = new Scanner(System.in);

// 为了后面的录入信息的方便,我所有的数据录入全部用字符接收

String choiceString = sc.nextLine();


// switch语句的多个地方要使用,我就定义到外面

UserDao ud = new UserDaoImpl();


// 经过简单的思考,我选择了switch

switch (choiceString) {

case "1":

// 登录界面,请输入用户名和密码

System.out.println("--------------登录界面--------------");

System.out.println("请输入用户名:");

String username = sc.nextLine();

System.out.println("请输入密码:");

String password = sc.nextLine();


// 调用登录功能

// UserDao ud = new UserDaomImpl();


boolean flag = ud.isLogin(username, password);

if (flag) {

System.out.println("登录成功,可以开始玩游戏了");


System.out.println("你玩吗?y/n");

while (true) {

String resultString = sc.nextLine();

if (resultString.equalsIgnoreCase("y")) {

// 玩游戏

GuessNumber.start();

System.out.println("你还玩吗?y/n");

} else {

break;

}

}

System.out.println("谢谢使用,欢迎下次再来");

System.exit(0);

// break; //这里写break,结束的是switch

} else {

System.out.println("用户名或者密码有误,登录失败");

}

break;

case "2":

// 欢迎界面,请输入用户名和密码

System.out.println("--------------注册界面--------------");

System.out.println("请输入用户名:");

String newUsername = sc.nextLine();

System.out.println("请输入密码:");

String newPassword = sc.nextLine();


// 把用户名和密码封装到一个对象中

User user = new User();

user.setUsername(newUsername);

user.setPassword(newPassword);


// 调用注册功能

// 多态

// UserDao ud = new UserDaoImpl();

// 具体类使用

// UserDaoImpl udi = new UserDaoImpl();


ud.regist(user);

System.out.println("注册成功");

break;

case "3":

default:

System.out.println("谢谢使用,欢迎下次再来");

System.exit(0);

break;

}

}

}

}




2:数据操作流(操作基本类型数据的流)(理解)

(1)可以操作基本类型的数据

(2)流对象名称

DataInputStream

DataOutputStream


package cn.itcast_01;


import java.io.DataInputStream;

import java.io.DataOutputStream;

import java.io.FileInputStream;

import java.io.FileOutputStream;

import java.io.IOException;


/*

 * 可以读写基本数据类型的数据

 * 数据输入流:DataInputStream

 *  DataInputStream(InputStream in)

 * 数据输出流:DataOutputStream

 *  DataOutputStream(OutputStream out) 

 */

public class DataStreamDemo {

public static void main(String[] args) throws IOException {

// 写

// write();


// 读

read();

}


private static void read() throws IOException {

// DataInputStream(InputStream in)

// 创建数据输入流对象

DataInputStream dis = new DataInputStream(

new FileInputStream("dos.txt"));


// 读数据

byte b = dis.readByte();

short s = dis.readShort();

int i = dis.readInt();

long l = dis.readLong();

float f = dis.readFloat();

double d = dis.readDouble();

char c = dis.readChar();

boolean bb = dis.readBoolean();


// 释放资源

dis.close();


System.out.println(b);

System.out.println(s);

System.out.println(i);

System.out.println(l);

System.out.println(f);

System.out.println(d);

System.out.println(c);

System.out.println(bb);

}


private static void write() throws IOException {

// DataOutputStream(OutputStream out)

// 创建数据输出流对象

DataOutputStream dos = new DataOutputStream(new FileOutputStream(

"dos.txt"));


// 写数据了

dos.writeByte(10);

dos.writeShort(100);

dos.writeInt(1000);

dos.writeLong(10000);

dos.writeFloat(12.34F);

dos.writeDouble(12.56);

dos.writeChar('a');

dos.writeBoolean(true);


// 释放资源

dos.close();

}

}




3:内存操作流(理解)

(1)有些时候我们操作完毕后,未必需要产生一个文件,就可以使用内存操作流。

(2)三种

A:ByteArrayInputStream,ByteArrayOutputStream

B:CharArrayReader,CharArrayWriter

C:StringReader,StringWriter


package cn.itcast_02;


import java.io.ByteArrayInputStream;

import java.io.ByteArrayOutputStream;

import java.io.IOException;


/*

 * 内存操作流:用于处理临时存储信息的,程序结束,数据就从内存中消失。

 * 字节数组:

 *  ByteArrayInputStream

 *  ByteArrayOutputStream

 * 字符数组:

 *  CharArrayReader

 *  CharArrayWriter

 * 字符串:

 *  StringReader

 *  StringWriter

 */

public class ByteArrayStreamDemo {

public static void main(String[] args) throws IOException {

// 写数据

// ByteArrayOutputStream()

ByteArrayOutputStream baos = new ByteArrayOutputStream();


// 写数据

for (int x = 0; x < 10; x++) {

baos.write(("hello" + x).getBytes());

}


// 释放资源

// 通过查看源码我们知道这里什么都没做,所以根本需要close()

// baos.close();


// public byte[] toByteArray()

byte[] bys = baos.toByteArray();


// 读数据

// ByteArrayInputStream(byte[] buf)

ByteArrayInputStream bais = new ByteArrayInputStream(bys);


int by = 0;

while ((by = bais.read()) != -1) {

System.out.print((char) by);

}


// bais.close();

}

}


4:打印流(掌握)

(1)字节打印流,字符打印流

(2)特点:

A:只操作目的地,不操作数据源

B:可以操作任意类型的数据

C:如果启用了自动刷新,在调用println()方法的时候,能够换行并刷新

D:可以直接操作文件

问题:哪些流可以直接操作文件呢?

看API,如果其构造方法能够同时接收File和String类型的参数,一般都是可以直接操作文件的

package cn.itcast_03;


import java.io.IOException;

import java.io.PrintWriter;


/*

 * 打印流

 * 字节流打印流 PrintStream

 * 字符打印流 PrintWriter

 * 

 * 打印流的特点:

 *  A:只有写数据的,没有读取数据。只能操作目的地,不能操作数据源。

 *  B:可以操作任意类型的数据。

 *  C:如果启动了自动刷新,能够自动刷新。

 *  D:该流是可以直接操作文本文件的。

 *  哪些流对象是可以直接操作文本文件的呢?

 *  FileInputStream

 *  FileOutputStream

 *  FileReader

 *  FileWriter

 *  PrintStream

 *  PrintWriter

 *  看API,查流对象的构造方法,如果同时有File类型和String类型的参数,一般来说就是可以直接操作文件的。

 * 

 *  流:

 *  基本流:就是能够直接读写文件的

 *  高级流:在基本流基础上提供了一些其他的功能

 */

public class PrintWriterDemo {

public static void main(String[] args) throws IOException {

// 作为Writer的子类使用

PrintWriter pw = new PrintWriter("pw.txt");


pw.write("hello");

pw.write("world");

pw.write("java");

pw.close();

}

}


package cn.itcast_03;


import java.io.FileWriter;

import java.io.IOException;

import java.io.PrintWriter;


/*

 * 1:可以操作任意类型的数据。

 *  print()

 *  println()

 * 2:启动自动刷新

 *  PrintWriter pw = new PrintWriter(new FileWriter("pw2.txt"), true);

 *  还是应该调用println()的方法才可以

 *  这个时候不仅仅自动刷新了,还实现了数据的换行。

 * 

 *  println()

 * 其实等价于于:

 * bw.write();

 * bw.newLine();

 * bw.flush();

 */

public class PrintWriterDemo2 {

public static void main(String[] args) throws IOException {

// 创建打印流对象

// PrintWriter pw = new PrintWriter("pw2.txt");

PrintWriter pw = new PrintWriter(new FileWriter("pw2.txt"), true);


// write()是搞不定的,怎么办呢?

// 我们就应该看看它的新方法

// pw.print(true);

// pw.print(100);

// pw.print("hello");


pw.println("hello");

pw.println(true);

pw.println(100);


pw.close();

}

}

(3)打印流复制文本文件

BufferedReader br = new BufferedReader(new FileReader("a.txt"));

PrintWriter pw = new PrintWriter(new FileWriter("b.txt"),true);

String line = null;

while((line=br.readLine())!=null) {

pw.println(line);

}

pw.close();

br.close();


package cn.itcast_03;


import java.io.BufferedReader;

import java.io.BufferedWriter;

import java.io.FileReader;

import java.io.FileWriter;

import java.io.IOException;

import java.io.PrintWriter;


/*

 * 需求:DataStreamDemo.java复制到Copy.java中

 * 数据源:

 *  DataStreamDemo.java -- 读取数据 -- FileReader -- BufferedReader

 * 目的地:

 *  Copy.java -- 写出数据 -- FileWriter -- BufferedWriter -- PrintWriter

 */

public class CopyFileDemo {

public static void main(String[] args) throws IOException {

// 以前的版本

// 封装数据源

// BufferedReader br = new BufferedReader(new FileReader(

// "DataStreamDemo.java"));

// // 封装目的地

// BufferedWriter bw = new BufferedWriter(new FileWriter("Copy.java"));

//

// String line = null;

// while ((line = br.readLine()) != null) {

// bw.write(line);

// bw.newLine();

// bw.flush();

// }

//

// bw.close();

// br.close();


// 打印流的改进版

// 封装数据源

BufferedReader br = new BufferedReader(new FileReader(

"DataStreamDemo.java"));

// 封装目的地

PrintWriter pw = new PrintWriter(new FileWriter("Copy.java"), true);

String line = null;

while((line=br.readLine())!=null){

pw.println(line);

}

pw.close();

br.close();

}

}



5:标准输入输出流(理解)

(1)System类下面有这样的两个字段

in 标准输入流

out 标准输出流


package cn.itcast_04;


import java.io.PrintStream;


/*

 * 标准输入输出流

 * System类中的两个成员变量:

 * public static final InputStream in “标准”输入流。

 *  public static final PrintStream out “标准”输出流。

 * 

 *  InputStream is = System.in;

 *  PrintStream ps = System.out;

 */

public class SystemOutDemo {

public static void main(String[] args) {

// 有这里的讲解我们就知道了,这个输出语句其本质是IO流操作,把数据输出到控制台。

System.out.println("helloworld");


// 获取标准输出流对象

PrintStream ps = System.out;

ps.println("helloworld");

ps.println();

// ps.print();//这个方法不存在

// System.out.println();

// System.out.print();

}

}



(2)三种键盘录入方式

A:main方法的args接收参数

B:System.in通过BufferedReader进行包装

BufferedReader br = new BufferedReader(new InputStreamReader(System.in));

C:Scanner

Scanner sc = new Scanner(System.in);


package cn.itcast_04;


import java.io.BufferedReader;

import java.io.IOException;

import java.io.InputStreamReader;


/*

 * System.in 标准输入流。是从键盘获取数据的

 * 

 * 键盘录入数据:

 *  A:main方法的args接收参数。

 *  java HelloWorld hello world java

 *  B:Scanner(JDK5以后的)

 *  Scanner sc = new Scanner(System.in);

 *  String s = sc.nextLine();

 *  int x = sc.nextInt()

 *  C:通过字符缓冲流包装标准输入流实现

 *  BufferedReader br = new BufferedReader(new InputStreamReader(System.in));

 */

public class SystemInDemo {

public static void main(String[] args) throws IOException {

// //获取标准输入流

// InputStream is = System.in;

// //我要一次获取一行行不行呢?

// //行。

// //怎么实现呢?

// //要想实现,首先你得知道一次读取一行数据的方法是哪个呢?

// //readLine()

// //而这个方法在哪个类中呢?

// //BufferedReader

// //所以,你这次应该创建BufferedReader的对象,但是底层还是的使用标准输入流

// // BufferedReader br = new BufferedReader(is);

// //按照我们的推想,现在应该可以了,但是却报错了

// //原因是:字符缓冲流只能针对字符流操作,而你现在是字节流,所以不能是用?

// //那么,我还就想使用了,请大家给我一个解决方案?

// //把字节流转换为字符流,然后在通过字符缓冲流操作

// InputStreamReader isr = new InputStreamReader(is);

// BufferedReader br= new BufferedReader(isr);

BufferedReader br = new BufferedReader(new InputStreamReader(System.in));


System.out.println("请输入一个字符串:");

String line = br.readLine();

System.out.println("你输入的字符串是:" + line);


System.out.println("请输入一个整数:");

// int i = Integer.parseInt(br.readLine());

line = br.readLine();

int i = Integer.parseInt(line);

System.out.println("你输入的整数是:" + i);

}

}


(3)转换流:


package cn.itcast_04;


import java.io.BufferedWriter;

import java.io.IOException;

import java.io.OutputStreamWriter;

/*

 * 转换流的应用。

 */

public class SystemOutDemo2 {

public static void main(String[] args) throws IOException {

// 获取标准输入流

// // PrintStream ps = System.out;

// // OutputStream os = ps;

// OutputStream os = System.out; // 多态

// // 我能不能按照刚才使用标准输入流的方式一样把数据输出到控制台呢?

// OutputStreamWriter osw = new OutputStreamWriter(os);

// BufferedWriter bw = new BufferedWriter(osw);

BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(

System.out));


bw.write("hello");

bw.newLine();

// bw.flush();

bw.write("world");

bw.newLine();

// bw.flush();

bw.write("java");

bw.newLine();

bw.flush();

bw.close();

}

}

(4)输出语句的原理和如何使用字符流输出数据

A:原理

System.out.println("helloworld");

PrintStream ps = System.out;

ps.println("helloworld");

B:把System.out用字符缓冲流包装一下使用

BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(System.out));


6:随机访问流(理解)

(1)可以按照文件指针的位置写数据和读数据。

(2)案例:

A:写数据

B:读数据

C:获取和改变文件指针的位置


package cn.itcast_05;


import java.io.IOException;

import java.io.RandomAccessFile;


/*

 * 随机访问流:

 *  RandomAccessFile类不属于流,是Object类的子类。

 *  但它融合了InputStream和OutputStream的功能。

 *  支持对文件的随机访问读取和写入。

 * 

 * public RandomAccessFile(String name,String mode):第一个参数是文件路径,第二个参数是操作文件的模式。

 *  模式有四种,我们最常用的一种叫"rw",这种方式表示我既可以写数据,也可以读取数据 

 */

public class RandomAccessFileDemo {

public static void main(String[] args) throws IOException {

// write();

read();

}


private static void read() throws IOException {

// 创建随机访问流对象

RandomAccessFile raf = new RandomAccessFile("raf.txt", "rw");


int i = raf.readInt();

System.out.println(i);

// 该文件指针可以通过 getFilePointer方法读取,并通过 seek 方法设置。

System.out.println("当前文件的指针位置是:" + raf.getFilePointer());


char ch = raf.readChar();

System.out.println(ch);

System.out.println("当前文件的指针位置是:" + raf.getFilePointer());


String s = raf.readUTF();

System.out.println(s);

System.out.println("当前文件的指针位置是:" + raf.getFilePointer());


// 我不想重头开始了,我就要读取a,怎么办呢?

raf.seek(4);

ch = raf.readChar();

System.out.println(ch);

}


private static void write() throws IOException {

// 创建随机访问流对象

RandomAccessFile raf = new RandomAccessFile("raf.txt", "rw");


// 怎么玩呢?

raf.writeInt(100);

raf.writeChar('a');

raf.writeUTF("中国");


raf.close();

}

}



7:合并流(理解)

(1)把多个输入流的数据写到一个输出流中。

(2)构造方法:

A:SequenceInputStream(InputStream s1, InputStream s2) 

B:SequenceInputStream(Enumeration<? extends InputStream> e) 


案例:A:    a.txt+b.txt -- c.txt


package cn.itcast_06;


import java.io.BufferedOutputStream;

import java.io.FileInputStream;

import java.io.FileOutputStream;

import java.io.IOException;

import java.io.InputStream;

import java.io.SequenceInputStream;


/*

 * 以前的操作:

 * a.txt -- b.txt

 * c.txt -- d.txt

 * 

 * 现在想要:

 * a.txt+b.txt -- c.txt

 */

public class SequenceInputStreamDemo {

public static void main(String[] args) throws IOException {

// SequenceInputStream(InputStream s1, InputStream s2)

// 需求:把ByteArrayStreamDemo.java和DataStreamDemo.java的内容复制到Copy.java中

InputStream s1 = new FileInputStream("ByteArrayStreamDemo.java");

InputStream s2 = new FileInputStream("DataStreamDemo.java");

SequenceInputStream sis = new SequenceInputStream(s1, s2);

BufferedOutputStream bos = new BufferedOutputStream(

new FileOutputStream("Copy.java"));


// 如何写读写呢,其实很简单,你就按照以前怎么读写,现在还是怎么读写

byte[] bys = new byte[1024];

int len = 0;

while ((len = sis.read(bys)) != -1) {

bos.write(bys, 0, len);

}


bos.close();

sis.close();

}

}



B:    a.txt+b.txt+c.txt -- d.txt



package cn.itcast_06;


import java.io.BufferedOutputStream;

import java.io.FileInputStream;

import java.io.FileOutputStream;

import java.io.IOException;

import java.io.InputStream;

import java.io.SequenceInputStream;

import java.util.Enumeration;

import java.util.Vector;


/*

 * 以前的操作:

 * a.txt -- b.txt

 * c.txt -- d.txt

 * e.txt -- f.txt

 * 

 * 现在想要:

 * a.txt+b.txt+c.txt -- d.txt

 */

public class SequenceInputStreamDemo2 {

public static void main(String[] args) throws IOException {

// 需求:把下面的三个文件的内容复制到Copy.java中

// ByteArrayStreamDemo.java,CopyFileDemo.java,DataStreamDemo.java


// SequenceInputStream(Enumeration e)

// 通过简单的回顾我们知道了Enumeration是Vector中的一个方法的返回值类型。

// Enumeration<E> elements()

Vector<InputStream> v = new Vector<InputStream>();

InputStream s1 = new FileInputStream("ByteArrayStreamDemo.java");

InputStream s2 = new FileInputStream("CopyFileDemo.java");

InputStream s3 = new FileInputStream("DataStreamDemo.java");

v.add(s1);

v.add(s2);

v.add(s3);

Enumeration<InputStream> en = v.elements();

SequenceInputStream sis = new SequenceInputStream(en);

BufferedOutputStream bos = new BufferedOutputStream(

new FileOutputStream("Copy.java"));


// 如何写读写呢,其实很简单,你就按照以前怎么读写,现在还是怎么读写

byte[] bys = new byte[1024];

int len = 0;

while ((len = sis.read(bys)) != -1) {

bos.write(bys, 0, len);

}


bos.close();

sis.close();

}

}




8:序列化流(理解)

(1)可以把对象写入文本文件或者在网络中传输

(2)如何实现序列化呢?

让被序列化的对象所属类实现序列化接口。

该接口是一个标记接口。没有功能需要实现。

(3)注意问题:

把数据写到文件后,在去修改类会产生一个问题。

如何解决该问题呢?

在类文件中,给出一个固定的序列化id值。

而且,这样也可以解决黄色警告线问题


package cn.itcast_07;


import java.io.FileInputStream;

import java.io.FileOutputStream;

import java.io.IOException;

import java.io.ObjectInputStream;

import java.io.ObjectOutputStream;


/*

 * 序列化流:把对象按照流一样的方式存入文本文件或者在网络中传输。对象 -- 流数据(ObjectOutputStream)

 * 反序列化流:把文本文件中的流对象数据或者网络中的流对象数据还原成对象。流数据 -- 对象(ObjectInputStream)

 */

public class ObjectStreamDemo {

public static void main(String[] args) throws IOException,

ClassNotFoundException {

// 由于我们要对对象进行序列化,所以我们先自定义一个类

// 序列化数据其实就是把对象写到文本文件

// write();


read();

}


private static void read() throws IOException, ClassNotFoundException {

// 创建反序列化对象

ObjectInputStream ois = new ObjectInputStream(new FileInputStream(

"oos.txt"));


// 还原对象

Object obj = ois.readObject();


// 释放资源

ois.close();


// 输出对象

System.out.println(obj);

}


private static void write() throws IOException {

// 创建序列化流对象

ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream(

"oos.txt"));


// 创建对象

Person p = new Person("林青霞", 27);


// public final void writeObject(Object obj)

oos.writeObject(p);


// 释放资源

oos.close();

}

}


package cn.itcast_07;


import java.io.Serializable;


/*

 * NotSerializableException:未序列化异常

 * 

 * 类通过实现 java.io.Serializable 接口以启用其序列化功能。未实现此接口的类将无法使其任何状态序列化或反序列化。

 * 该接口居然没有任何方法,类似于这种没有方法的接口被称为标记接口。

 * 

 * java.io.InvalidClassException: 

 * cn.itcast_07.Person; local class incompatible: 

 * stream classdesc serialVersionUID = -2071565876962058344, 

 * local class serialVersionUID = -8345153069362641443

 * 

 * 为什么会有问题呢?

 *  Person类实现了序列化接口,那么它本身也应该有一个标记值。

 *  这个标记值假设是100。

 *  开始的时候:

 *  Person.class -- id=100

 *  wirte数据: oos.txt -- id=100

 *  read数据: oos.txt -- id=100

 * 

 *  现在:

 *  Person.class -- id=200

 *  wirte数据: oos.txt -- id=100

 *  read数据: oos.txt -- id=100

 * 我们在实际开发中,可能还需要使用以前写过的数据,不能重新写入。怎么办呢?

 * 回想一下原因是因为它们的id值不匹配。

 * 每次修改java文件的内容的时候,class文件的id值都会发生改变。

 * 而读取文件的时候,会和class文件中的id值进行匹配。所以,就会出问题。

 * 但是呢,如果我有办法,让这个id值在java文件中是一个固定的值,这样,你修改文件的时候,这个id值还会发生改变吗?

 * 不会。现在的关键是我如何能够知道这个id值如何表示的呢?

 * 不用担心,你不用记住,也没关系,点击鼠标即可。

 * 你难道没有看到黄色警告线吗?

 * 

 * 我们要知道的是:

 *  看到类实现了序列化接口的时候,要想解决黄色警告线问题,就可以自动产生一个序列化id值。

 *  而且产生这个值以后,我们对类进行任何改动,它读取以前的数据是没有问题的。

 * 

 * 注意:

 *  我一个类中可能有很多的成员变量,有些我不想进行序列化。请问该怎么办呢?

 *  使用transient关键字声明不需要序列化的成员变量

 */

public class Person implements Serializable {

private static final long serialVersionUID = -2071565876962058344L;


private String name;


// private int age;


private transient int age;


// int age;


public Person() {

super();

}


public Person(String name, int age) {

super();

this.name = name;

this.age = age;

}


public String getName() {

return name;

}


public void setName(String name) {

this.name = name;

}


public int getAge() {

return age;

}


public void setAge(int age) {

this.age = age;

}


@Override

public String toString() {

return "Person [name=" + name + ", age=" + age + "]";

}

}



(4)面试题:

什么时候序列化?

如何实现序列化?

什么是反序列化?


9:Properties(理解)

(1)是一个集合类,Hashtable的子类


package cn.itcast_08;


import java.util.Properties;

import java.util.Set;


/*

 * Properties:属性集合类。是一个可以和IO流相结合使用的集合类。

 * Properties 可保存在流中或从流中加载。属性列表中每个键及其对应值都是一个字符串。 

 * 

 * 是Hashtable的子类,说明是一个Map集合。

 */

public class PropertiesDemo {

public static void main(String[] args) {

// 作为Map集合的使用

// 下面这种用法是错误的,一定要看API,如果没有<>,就说明该类不是一个泛型类,在使用的时候就不能加泛型

// Properties<String, String> prop = new Properties<String, String>();


Properties prop = new Properties();


// 添加元素

prop.put("it002", "hello");

prop.put("it001", "world");

prop.put("it003", "java");


// System.out.println("prop:" + prop);


// 遍历集合

Set<Object> set = prop.keySet();

for (Object key : set) {

Object value = prop.get(key);

System.out.println(key + "---" + value);

}

}

}


(2)特有功能

A:public Object setProperty(String key,String value)

B:public String getProperty(String key)

C:public Set<String> stringPropertyNames()


package cn.itcast_08;


import java.util.Properties;

import java.util.Set;


/*

 * 特殊功能:

 * public Object setProperty(String key,String value):添加元素

 * public String getProperty(String key):获取元素

 * public Set<String> stringPropertyNames():获取所有的键的集合

 */

public class PropertiesDemo2 {

public static void main(String[] args) {

// 创建集合对象

Properties prop = new Properties();


// 添加元素

prop.setProperty("张三", "30");

prop.setProperty("李四", "40");

prop.setProperty("王五", "50");


// public Set<String> stringPropertyNames():获取所有的键的集合

Set<String> set = prop.stringPropertyNames();

for (String key : set) {

String value = prop.getProperty(key);

System.out.println(key + "---" + value);

}

}

}


/*

 * class Hashtalbe<K,V> { public V put(K key,V value) { ... } }

 * 

 * class Properties extends Hashtable { public V setProperty(String key,String

 * value) { return put(key,value); } }

 */



(3)和IO流结合的方法

把键值对形式的文本文件内容加载到集合中

public void load(Reader reader)

public void load(InputStream inStream)


把集合中的数据存储到文本文件中

public void store(Writer writer,String comments)

public void store(OutputStream out,String comments)


package cn.itcast_08;


import java.io.FileReader;

import java.io.FileWriter;

import java.io.IOException;

import java.io.Reader;

import java.io.Writer;

import java.util.Properties;


/*

 * 这里的集合必须是Properties集合:

 * public void load(Reader reader):把文件中的数据读取到集合中

 * public void store(Writer writer,String comments):把集合中的数据存储到文件

 * 

 * 单机版游戏:

 *  进度保存和加载。

 *  三国群英传,三国志,仙剑奇侠传...

 * 

 *  吕布=1

 *  方天画戟=1

 */

public class PropertiesDemo3 {

public static void main(String[] args) throws IOException {

// myLoad();


myStore();

}


private static void myStore() throws IOException {

// 创建集合对象

Properties prop = new Properties();


prop.setProperty("林青霞", "27");

prop.setProperty("武鑫", "30");

prop.setProperty("刘晓曲", "18");

//public void store(Writer writer,String comments):把集合中的数据存储到文件

Writer w = new FileWriter("name.txt");

prop.store(w, "helloworld");

w.close();

}


private static void myLoad() throws IOException {

Properties prop = new Properties();


// public void load(Reader reader):把文件中的数据读取到集合中

// 注意:这个文件的数据必须是键值对形式

Reader r = new FileReader("prop.txt");

prop.load(r);

r.close();


System.out.println("prop:" + prop);

}

}



(4)案例:

A:根据给定的文件判断是否有键为"lisi"的,如果有就修改其值为100


package cn.itcast_08;


import java.io.FileReader;

import java.io.FileWriter;

import java.io.IOException;

import java.io.Reader;

import java.io.Writer;

import java.util.Properties;

import java.util.Set;


/*

 * 我有一个文本文件(user.txt),我知道数据是键值对形式的,但是不知道内容是什么。

 * 请写一个程序判断是否有“lisi”这样的键存在,如果有就改变其实为”100”

 * 

 * 分析:

 *  A:把文件中的数据加载到集合中

 *  B:遍历集合,获取得到每一个键

 *  C:判断键是否有为"lisi"的,如果有就修改其值为"100"

 *  D:把集合中的数据重新存储到文件中

 */

public class PropertiesTest {

public static void main(String[] args) throws IOException {

// 把文件中的数据加载到集合中

Properties prop = new Properties();

Reader r = new FileReader("user.txt");

prop.load(r);

r.close();


// 遍历集合,获取得到每一个键

Set<String> set = prop.stringPropertyNames();

for (String key : set) {

// 判断键是否有为"lisi"的,如果有就修改其值为"100"

if ("lisi".equals(key)) {

prop.setProperty(key, "100");

break;

}

}


// 把集合中的数据重新存储到文件中

Writer w = new FileWriter("user.txt");

prop.store(w, null);

w.close();

}

}



B:写一个程序实现控制猜数字小游戏程序不能玩超过5次


package cn.itcast_08;


import java.io.FileReader;

import java.io.FileWriter;

import java.io.IOException;

import java.io.Reader;

import java.io.Writer;

import java.util.Properties;


/*

 * 我有一个猜数字小游戏的程序,请写一个程序实现在测试类中只能用5次,超过5次提示:游戏试玩已结束,请付费。

 */

public class PropertiesTest2 {

public static void main(String[] args) throws IOException {

// 读取某个地方的数据,如果次数不大于5,可以继续玩。否则就提示"游戏试玩已结束,请付费。"

// 创建一个文件

// File file = new File("count.txt");

// if (!file.exists()) {

// file.createNewFile();

// }


// 把数据加载到集合中

Properties prop = new Properties();

Reader r = new FileReader("count.txt");

prop.load(r);

r.close();


// 我自己的程序,我当然知道里面的键是谁

String value = prop.getProperty("count");

int number = Integer.parseInt(value);


if (number > 5) {

System.out.println("游戏试玩已结束,请付费。");

System.exit(0);

} else {

number++;

prop.setProperty("count", String.valueOf(number));

Writer w = new FileWriter("count.txt");

prop.store(w, null);

w.close();


GuessNumber.start();

}

}

}




10:NIO(了解)

(1)JDK4出现的NIO,对以前的IO操作进行了优化,提供了效率。但是大部分我们看到的还是以前的IO

(2)JDK7的NIO的使用

Path:路径

Paths:通过静态方法返回一个路径

Files:提供了常见的功能

复制文本文件

把集合中的数据写到文本文件


package cn.itcast_09;


import java.io.IOException;

import java.nio.charset.Charset;

import java.nio.file.Files;

import java.nio.file.Paths;

import java.util.ArrayList;


/*

 * nio包在JDK4出现,提供了IO流的操作效率。但是目前还不是大范围的使用。

 * 有空的话了解下,有问题再问我。

 * 

 * JDK7的之后的nio:

 * Path:路径

 * Paths:有一个静态方法返回一个路径

 *  public static Path get(URI uri)

 * Files:提供了静态方法供我们使用

 *  public static long copy(Path source,OutputStream out):复制文件

 *  public static Path write(Path path,Iterable<? extends CharSequence> lines,Charset cs,OpenOption... options)

 */

public class NIODemo {

public static void main(String[] args) throws IOException {

// public static long copy(Path source,OutputStream out)

// Files.copy(Paths.get("ByteArrayStreamDemo.java"), new

// FileOutputStream(

// "Copy.java"));


ArrayList<String> array = new ArrayList<String>();

array.add("hello");

array.add("world");

array.add("java");

Files.write(Paths.get("array.txt"), array, Charset.forName("GBK"));

}

}




Java帮帮今日头条号



Java帮帮微信公众号



Java帮帮交流群

目录
相关文章
|
4月前
|
存储 Java
【IO面试题 四】、介绍一下Java的序列化与反序列化
Java的序列化与反序列化允许对象通过实现Serializable接口转换成字节序列并存储或传输,之后可以通过ObjectInputStream和ObjectOutputStream的方法将这些字节序列恢复成对象。
|
7月前
|
存储 算法 Java
从零开始学习 Java:简单易懂的入门指南之IO序列化、打印流、压缩流(三十三)
从零开始学习 Java:简单易懂的入门指南之IO序列化、打印流、压缩流(三十三)
|
4月前
|
XML 存储 JSON
【IO面试题 六】、 除了Java自带的序列化之外,你还了解哪些序列化工具?
除了Java自带的序列化,常见的序列化工具还包括JSON(如jackson、gson、fastjson)、Protobuf、Thrift和Avro,各具特点,适用于不同的应用场景和性能需求。
|
4月前
|
JSON Java 数据格式
【IO面试题 七】、 如果不用JSON工具,该如何实现对实体类的序列化?
除了JSON工具,实现实体类序列化可以采用Java原生序列化机制或第三方库如Protobuf、Thrift、Avro等。
|
5月前
|
存储 Java
JaveSE—IO流详解:对象输入输出流(序列化及反序列化)
JaveSE—IO流详解:对象输入输出流(序列化及反序列化)
|
7月前
|
分布式计算 Java 大数据
IO流【Java对象的序列化和反序列化、File类在IO中的作用、装饰器模式构建IO流体系、Apache commons-io工具包的使用】(四)-全面详解(学习总结---从入门到深化)
IO流【Java对象的序列化和反序列化、File类在IO中的作用、装饰器模式构建IO流体系、Apache commons-io工具包的使用】(四)-全面详解(学习总结---从入门到深化)
103 0
|
7月前
|
Java API 数据库
IO流【转换流、字符输出流、数据流、对象流】(三)-全面详解(学习总结---从入门到深化)
IO流【转换流、字符输出流、数据流、对象流】(三)-全面详解(学习总结---从入门到深化)
79 0
|
1月前
|
JSON 数据格式 索引
Python中序列化/反序列化JSON格式的数据
【11月更文挑战第4天】本文介绍了 Python 中使用 `json` 模块进行序列化和反序列化的操作。序列化是指将 Python 对象(如字典、列表)转换为 JSON 字符串,主要使用 `json.dumps` 方法。示例包括基本的字典和列表序列化,以及自定义类的序列化。反序列化则是将 JSON 字符串转换回 Python 对象,使用 `json.loads` 方法。文中还提供了具体的代码示例,展示了如何处理不同类型的 Python 对象。
|
1月前
|
存储 安全 Java
Java编程中的对象序列化与反序列化
【10月更文挑战第22天】在Java的世界里,对象序列化和反序列化是数据持久化和网络传输的关键技术。本文将带你了解如何在Java中实现对象的序列化与反序列化,并探讨其背后的原理。通过实际代码示例,我们将一步步展示如何将复杂数据结构转换为字节流,以及如何将这些字节流还原为Java对象。文章还将讨论在使用序列化时应注意的安全性问题,以确保你的应用程序既高效又安全。
|
2月前
|
存储 Java
Java编程中的对象序列化与反序列化
【10月更文挑战第9天】在Java的世界里,对象序列化是连接数据持久化与网络通信的桥梁。本文将深入探讨Java对象序列化的机制、实践方法及反序列化过程,通过代码示例揭示其背后的原理。从基础概念到高级应用,我们将一步步揭开序列化技术的神秘面纱,让读者能够掌握这一强大工具,以应对数据存储和传输的挑战。