[置顶] think in java interview-高级开发人员面试宝典(八)

时间:2023-12-15 13:25:14

面经出了7套,收到许多读者的Email,有许多人说了,这些基础知识是不是为了后面进一步的”通向架构师的道路“做准备的?

对的,你们没有猜错,就是这样的,我一直在酝酿后面的”通向架构师的道路“如何开章。

说实话,我已经在肚子里准备好的后面的”通向架构师的道路“的内容自己觉得如果一下子全拿出来的话,很多人吃不消,因为架构越来越复杂,用到的知识越来越多,而且很多都是各知识点的混合应用。

所以,先以这几套面经来铺路,我们把基础打实了,才能把大楼造的更好。因为,一个架构师首先他是一个程序员,他的基础知识必须非常的扎实,API对于架构师来说已经不太需要eclipse的code insight(即在eclipse编辑器里打一个小点点就可以得到后面的函数),尤其是一些常用的JAVA API来说,是必须熟记于心的。

下面我们继续来几天面经,顺带便复习一下JAVA和数据库的一些基础。

Java IO流的复习

大家平时J2EE写多了,JAVA的IO操作可能都已经生疏了,面试时如果来上这么几道,是不是有点”其实这个问题很简单,可是我就是想不起来“的感觉啊?

呵呵!

JAVA的IO操作太多,我这边挑腾迅,盛大和百度的几道面试题,并整理出答案来供大家参考。

InputFromConsole

这个最简单不过了,从console接受用户输入的字符,如和用户有交互的命令行。

如果你不复习的话,嘿嘿,还真答不出,来看:

package org.sky.io;

public class InputFromConsole {

	/**
* @param args
*/
public static void main(String[] args) throws Exception {
int a = 0;
byte[] input = new byte[1024];
System.in.read(input);
System.out.println("your input is: " + new String(input)); } }

ListDir

列出给出路径下所有的目录,包括子目录

package org.sky.io;

import java.io.*;

public class ListMyDir {

	/**
* @param args
*/
public static void main(String[] args) {
String fileName = "D:" + File.separator + "tomcat2";
File f = new File(fileName);
File[] fs = f.listFiles();
for (int i = 0; i < fs.length; i++) {
System.out.println(fs[i].getName());
} } }

咦,上面这个程序只列出了一层目录,我们想连子目录一起List出来怎么办?

ListMyDirWithSubDir

package org.sky.io;

import java.io.*;

public class ListMyDirWithSubDir {

	/**
* @param args
*/
public void print(File f) {
if (f != null) {
if (f.isDirectory()) {
File[] fileArray = f.listFiles();
if (fileArray != null) {
for (int i = 0; i < fileArray.length; i++) {
print(fileArray[i]);
}
}
} else {
System.out.println(f);
}
}
} public static void main(String[] args) {
String fileName = "D:" + File.separator + "tomcat2";
File f = new File(fileName);
ListMyDirWithSubDir listDir = new ListMyDirWithSubDir();
listDir.print(f); }
}

InputStreamDemo

从外部读入一个文件

package org.sky.io;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.io.OutputStream; public class InputStreamDemo {
public void readFile(String fileName) {
File srcFile = new File(fileName);
InputStream in = null;
try {
in = new FileInputStream(srcFile);
byte[] b = new byte[(int) srcFile.length()];
for (int i = 0; i < b.length; i++) {
b[i] = (byte) in.read();
}
System.out.println(new String(b));
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
if (in != null) {
in.close();
in = null;
}
} catch (Exception e) {
}
}
} public static void main(String[] args) {
InputStreamDemo id = new InputStreamDemo();
String src = "D:" + File.separator + "hello.txt";
id.readFile(src);
} }

OutputStreamDemo

讲完了InputStream来讲OutputStream,输出内容至外部的一个文件

package org.sky.io;

import java.io.*;

public class OutputStreamDemo {

	public void writeWithByte() {
String fileName = "D:" + File.separator + "hello.txt";
OutputStream out = null;
File f = new File(fileName);
try {
out = new FileOutputStream(f, true);
String str = " [Publicity ministry of ShangHai Municipal committee of CPC]";
byte[] b = str.getBytes();
out.write(b);
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
if (out != null) {
out.close();
out = null;
}
} catch (Exception e) {
}
}
} public void writeWithByteArray() {
String fileName = "D:" + File.separator + "hello.txt";
OutputStream out = null;
File f = new File(fileName);
try {
out = new FileOutputStream(f, true);
String str = " [hello with byte yi ge ge xie]";
byte[] b = str.getBytes();
for (int i = 0; i < b.length; i++) {
out.write(b[i]);
}
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
if (out != null) {
out.close();
out = null;
}
} catch (Exception e) {
}
}
} public static void main(String[] args) {
OutputStreamDemo od = new OutputStreamDemo();
od.writeWithByte();
od.writeWithByteArray(); } }

这个Demo里分别用了”writeWithByte“和 ”writeWithByteArray“两种方法,注意查看

CopyFile

我们讲完了InputStream和OutputStream,我们就可以自己实现一个File Copy的功能了,来看

package org.sky.io;

import java.io.*;

public class CopyFile {

	public void copy(String src, String des) {
File srcFile = new File(src);
File desFile = new File(des);
InputStream in = null;
OutputStream out = null;
try {
in = new FileInputStream(srcFile);
out = new FileOutputStream(desFile);
byte[] b = new byte[(int) srcFile.length()];
for (int i = 0; i < b.length; i++) {
b[i] = (byte) in.read();
}
out.write(b);
System.out.println("copied [" + srcFile.getName() + "] with "
+ srcFile.length());
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
if (out != null) {
out.close();
out = null;
}
} catch (Exception e) {
}
try {
if (in != null) {
in.close();
in = null;
}
} catch (Exception e) {
}
}
} public static void main(String[] args) {
CopyFile cp = new CopyFile();
String src = "D:" + File.separator + "UltraEdit.zip";
String des = "D:" + File.separator + "UltraEdit_Copy.zip";
long sTime = System.currentTimeMillis();
cp.copy(src, des);
long eTime = System.currentTimeMillis();
System.out.println("Total spend: " + (eTime - sTime));
} }

运行后显示:

[置顶] think in java interview-高级开发人员面试宝典(八)

来看我们被Copy的这个文件的大小:

[置顶] think in java interview-高级开发人员面试宝典(八)

也不大,怎么用了7秒多?

原是我们没有使用Buffer这个东西,即缓冲,性能会相差多大呢?来看

BufferInputStreamDemo

package org.sky.io;

import java.io.*;

public class BufferInputStreamDemo {

	/**
* @param args
*/
public void copy(String src, String des) {
File srcFile = new File(src);
File desFile = new File(des);
BufferedInputStream bin = null;
BufferedOutputStream bout = null;
try {
bin = new BufferedInputStream(new FileInputStream(srcFile));
bout = new BufferedOutputStream(new FileOutputStream(desFile));
byte[] b = new byte[1024];
while (bin.read(b) != -1) {
bout.write(b);
}
bout.flush();
System.out.println("copied [" + srcFile.getName() + "] with "
+ srcFile.length());
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
if (bout != null) {
bout.close();
bout = null;
}
} catch (Exception e) {
}
try {
if (bin != null) {
bin.close();
bin = null;
}
} catch (Exception e) {
}
}
} public static void main(String[] args) {
BufferInputStreamDemo bd = new BufferInputStreamDemo();
String src = "D:" + File.separator + "UltraEdit.zip";
String des = "D:" + File.separator + "UltraEdit_Copy.zip";
long sTime = System.currentTimeMillis();
bd.copy(src, des);
long eTime = System.currentTimeMillis();
System.out.println("Total spend: " + (eTime - sTime)); } }

我们Copy同样一个文件,用了多少时间呢?来看!

[置顶] think in java interview-高级开发人员面试宝典(八)

丫只用了14毫秒,CALL!!!

ByteArrayDemo

来看看使用ByteArray输出文件吧

package org.sky.io;

import java.io.*;

public class ByteArrayDemo {

	/**
* @param args
*/
public void testByteArray() {
String str = "HOLLYJESUS";
ByteArrayInputStream input = null;
ByteArrayOutputStream output = null;
try {
input = new ByteArrayInputStream(str.getBytes());
output = new ByteArrayOutputStream();
int temp = 0;
while ((temp = input.read()) != -1) {
char ch = (char) temp;
output.write(Character.toLowerCase(ch));
}
String outStr = output.toString();
input.close();
output.close();
System.out.println(outStr);
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
if (output != null) {
output.close();
output = null;
}
} catch (Exception e) {
}
try {
if (input != null) {
input.close();
input = null;
}
} catch (Exception e) {
}
}
} public static void main(String[] args) {
ByteArrayDemo bd = new ByteArrayDemo();
bd.testByteArray(); } }

RandomAccess

有种输出流叫Random,你们还记得吗?学习时记得的,工作久了,HOHO,忘了,它到底有什么特殊的地方呢?来看:

package org.sky.io;

import java.io.*;

public class RandomAccess {
public void writeToFile() {
String fileName = "D:" + File.separator + "hello.txt";
RandomAccessFile randomIO = null;
try { File f = new File(fileName);
randomIO = new RandomAccessFile(f, "rw");
randomIO.writeBytes("asdsad");
randomIO.writeInt(12);
randomIO.writeBoolean(true);
randomIO.writeChar('A');
randomIO.writeFloat(1.21f);
randomIO.writeDouble(12.123);
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
if (randomIO != null) {
randomIO.close();
randomIO = null;
}
} catch (Exception e) {
}
}
} public static void main(String[] args) {
RandomAccess randomA = new RandomAccess();
randomA.writeToFile();
}
}

它输出后的文件是怎么样的呢?

[置顶] think in java interview-高级开发人员面试宝典(八)

PipeStream

这个流很特殊,我们在线程操作时,两个线程都在运行,这时通过发送一个指令让某个线程do something,我们在以前的jdk1.4中为了实现这样的功能,使用的就是这个PipeStream

先来看两个类,一个叫SendMessage,即发送一个指令。一个叫ReceiveMessage,用于接受指令。

SendMessage

package org.sky.io;

import java.io.*;

public class SendMessage implements Runnable {
private PipedOutputStream out = null; public PipedOutputStream getOut() {
return this.out;
} public SendMessage() {
this.out = new PipedOutputStream();
} public void send() { String msg = "start";
try {
out.write(msg.getBytes());
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
if (out != null) {
out.close();
out = null;
}
} catch (Exception e) {
}
}
} public void run() {
try {
System.out.println("waiting for signal...");
Thread.sleep(2000);
send();
} catch (Exception e) {
e.printStackTrace();
}
}
}

ReceiveMessage

package org.sky.io;

import java.io.*;

public class ReceiveMessage implements Runnable {
private PipedInputStream input = null; public PipedInputStream getInput() {
return this.input;
} public ReceiveMessage() {
this.input = new PipedInputStream();
} private void receive() { byte[] b = new byte[1000];
int len = 0;
String msg = "";
try {
len = input.read(b);
msg = new String(b, 0, len);
if (msg.equals("start")) {
System.out
.println("received the start message, receive now can do something......");
Thread.interrupted();
}
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
if (input != null) {
input.close();
input = null;
}
} catch (Exception e) {
}
}
} public void run() {
try {
receive();
} catch (Exception e) {
}
}
}

如何使用这两个类呢?

TestPipeStream

package org.sky.io;

public class TestPipeStream {

	/**
* @param args
*/
public static void main(String[] args) {
SendMessage send = new SendMessage();
ReceiveMessage receive = new ReceiveMessage();
try {
send.getOut().connect(receive.getInput());
Thread t1 = new Thread(send);
Thread t2 = new Thread(receive);
t1.start();
t2.start(); } catch (Exception e) {
e.printStackTrace();
} } }

注意这边有一个send.getOut().connect(receive.getInput());

这个方法就把两个线程”connect“起来了。

Serializable的IO操作

把一个类序列化到磁盘上,怎么做?

先来看我们要序列化的一个Java Bean

Person

package org.sky.io;

import java.io.Serializable;

public class Person implements Serializable {

	private String name = "";
private String age = "";
private String personId = ""; public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public String getAge() {
return age;
} public void setAge(String age) {
this.age = age;
} public String getPersonId() {
return personId;
} public void setPersonId(String personId) {
this.personId = personId;
} public String getCellPhoneNo() {
return cellPhoneNo;
} public void setCellPhoneNo(String cellPhoneNo) {
this.cellPhoneNo = cellPhoneNo;
} private String cellPhoneNo = "";
}

下面来看序列化的操作

SerializablePersonToFile

package org.sky.io;

import java.io.*;
import java.util.*; public class SerializablePersonToFile { /**
* @param args
*/
private List<Person> initList() {
List<Person> userList = new ArrayList<Person>();
Person loginUser = new Person();
loginUser.setName("sam");
loginUser.setAge("30");
loginUser.setCellPhoneNo("13333333333");
loginUser.setPersonId("111111111111111111");
userList.add(loginUser);
loginUser = new Person();
loginUser.setName("tonny");
loginUser.setAge("31");
loginUser.setCellPhoneNo("14333333333");
loginUser.setPersonId("111111111111111111");
userList.add(loginUser);
loginUser = new Person();
loginUser.setName("jim");
loginUser.setAge("28");
loginUser.setCellPhoneNo("15333333333");
loginUser.setPersonId("111111111111111111");
userList.add(loginUser);
loginUser = new Person();
loginUser.setName("Simon");
loginUser.setAge("30");
loginUser.setCellPhoneNo("17333333333");
loginUser.setPersonId("111111111111111111");
userList.add(loginUser);
return userList;
} private void serializeFromFile() {
FileInputStream fs = null;
ObjectInputStream ois = null;
try {
fs = new FileInputStream("person.txt");
ois = new ObjectInputStream(fs);
List<Person> userList = (ArrayList<Person>) ois.readObject();
for (Person p : userList) {
System.out.println(p.getName() + " " + p.getAge() + " "
+ p.getCellPhoneNo() + " " + p.getCellPhoneNo());
}
} catch (Exception ex) {
ex.printStackTrace();
} finally {
try {
if (ois != null) {
ois.close();
}
} catch (Exception e) {
}
try {
if (fs != null) {
fs.close();
}
} catch (Exception e) {
}
}
} private void serializeToFile() {
List<Person> userList = new ArrayList<Person>();
userList = initList();
FileOutputStream fs = null;
ObjectOutputStream os = null;
try {
fs = new FileOutputStream("person.txt");
os = new ObjectOutputStream(fs);
os.writeObject(userList);
} catch (Exception ex) {
ex.printStackTrace();
} finally {
try {
if (os != null) {
os.close();
}
} catch (Exception e) {
}
try {
if (fs != null) {
fs.close();
}
} catch (Exception e) {
}
}
} public static void main(String[] args) {
SerializablePersonToFile sf = new SerializablePersonToFile();
sf.serializeToFile();
sf.serializeFromFile();
} }

这边先把Person输出到Person.txt,再从Person.txt里反序列化出这个Person的Java Bean。

先讲这么些吧!

Java的流操作还有很多,这些是经常会被面试到的,很基础,因此经常被考到。

以前听一个读IT的同学说过,这些IO操作,就算没有Eclipse编辑器的话,用文本编辑器也应该能够写出来,你写不出只能代表你的基础太弱了。