TCP通信的文件上传案例练习优化
- 服务器端
文件命名&循环结束&多线程提高效率
public static void main(String[] args) throws IOException {
ServerSocket server = new ServerSocket(8888);
// 让服务器一直处于监听状态(死循环accept方法):有一个客户端上传文件,就保存一个文件。
while(true){
Socket socket = server.accept();
// 使用多线程技术,提高程序的效率:有一个客户端上传文件,就开启一个线程,完成文件的上传。
new Thread(new Runnable() {
@Override
public void run() {
try{
InputStream is = socket.getInputStream();
// 4. 判断d:\\upload文件夹是否存在,不存在则创建
File file = new File("d:\\upload");
if(!file.exists()){
file.mkdirs();
}
// 自定义一个文件的命名规则:防止同名的文件被覆盖。规则:域名+毫秒值+随机数
String fileName = "itcast" + System.currentTimeMillis() + new Random().nextInt(999999) + ".jpg";
FileOutputStream fos = new FileOutputStream(file + "\\" + fileName);
int len = 0;
byte[] bytes = new byte[1024];
while((len = is.read(bytes)) != -1){
fos.write(bytes,0,len);
}
socket.getOutputStream().write("上传成功".getBytes());
fos.close();
socket.close();
}catch(IOException e){
System.out.println(e);
}
}
}).start();
}
// 服务器就不用关闭
// server.close();
}
模拟BS版本TCP服务器
- 分析
客户端是ie浏览器 Socket
http://127.0.0.1:8080/day11-code/web/index.html
访问服务器
服务器端 ServerSocket
读取客户端的请求信息
GET /day11-code/web/index.html HTTP/1.1
...
服务器要给客户端回写一个信息,回写一个html页面(文件)
我们需要读取index.html文件,就必须得知道这个文件的地址?
而这个地址就是请求信息的第一行中间的部分 GET /day11-code/web/index.html HTTP/1.1
可以使用BufferedReader中的方法readLine读取一行 InputStream is = socket.getInputStream();
new BufferedReader(new InputStreamReader(is)); 把网络字节输入流,转为字符缓冲输入流
GET /day11-code/web/index.html HTTP/1.1
可以使用String类得方法split(" ")切割字符串,获取中间的部分
arr[1] /day11-code/web/index.html
使用String类得方法substring(1),获取html文件的路径
day11-code/web/index.html
服务器创建一个本地的字节输入流,根据获取到的文件路径,读取html文件
【//写入HTTP协议响应头,固定写法
os.write("HTTP/1.1 200 OK\r\n".getBytes());
os.write("Content-Type:text/html\r\n".getBytes());
//必须要写入空行,否则浏览器不解析
os.write("\r\n".getBytes());】
服务器端使用网络字节输出流把读取到的文件,写到客户端(浏览器)显示
- 代码实现
public static void main(String[] args) throws IOException {
ServerSocket server = new ServerSocket(8080);
Socket socket = server.accept();
InputStream is = socket.getInputStream();
// 把is网络字节输入流对象,转换为字符缓冲输入流
BufferedReader br = new BufferedReader(new InputStreamReader(is));
// 把客户端请求信息的第一行读取处理 GET /day11-code/web/index.html HTTP/1.1
String line = br.readLine();
// 把读取的信息进行切割,只要中间部分 /day11-code/web/index.html
String[] arr = line.split(" ");
// 把路径前边的/去掉,进行截取 day11-code/web/index.html
String htmlpath = arr[1].substring(1);
FileInputStream fis = new FileInputStream(htmlpath);
OutputStream os = socket.getOutputStream();
// 写入HTTP协议响应头,固定写法
os.write("HTTP/1.1 200 OK\r\n".getBytes());
os.write("Content-Type:text/html\r\n".getBytes());
// 必须要写入空行,否则浏览器不解析
os.write("\r\n".getBytes());
int len = 0;
byte[] bytes = new byte[1024];
while((len = fis.read(bytes)) != -1){
os.write(bytes,0,len);
}
fis.close();
socket.close();
server.close();
}
模拟BS版本TCP服务器优化
循环结束&多线程提高效率
public static void main(String[] args) throws IOException {
ServerSocket server = new ServerSocket(8080);
/*浏览器解析服务器回写的html页面,页面中如果有图片,那么浏览器就会单独的开启一个线程,读取服务器的图片
我们就得让服务器一直处于监听状态,客户端请求一次,服务器就回写一次*/
while(true){
Socket socket = server.accept();
new Thread(new Runnable() {
@Override
public void run() {
try{
InputStream is = socket.getInputStream();
// 把is网络字节输入流对象,转换为字符缓冲输入流
BufferedReader br = new BufferedReader(new InputStreamReader(is));
// 把客户端请求信息的第一行读取处理 GET /day11-code/web/index.html HTTP/1.1
String line = br.readLine();
// 把读取的信息进行切割,只要中间部分 /day11-code/web/index.html
String[] arr = line.split(" ");
// 把路径前边的/去掉,进行截取 day11-code/web/index.html
String htmlpath = arr[1].substring(1);
FileInputStream fis = new FileInputStream(htmlpath);
OutputStream os = socket.getOutputStream();
// 写入HTTP协议响应头,固定写法
os.write("HTTP/1.1 200 OK\r\n".getBytes());
os.write("Content-Type:text/html\r\n".getBytes());
// 必须要写入空行,否则浏览器不解析
os.write("\r\n".getBytes());
int len = 0;
byte[] bytes = new byte[1024];
while((len = fis.read(bytes)) != -1){
os.write(bytes,0,len);
}
fis.close();
socket.close();
}catch(IOException e){
e.printStackTrace();
}
}
}).start();
}
// server.close();
}