java-网络通信相关

2020-12-24 06:28

阅读:895

标签:XML   lang   服务端   rand   loading   返回   navig   flat   响应   

 

技术图片

 

 

java中 

技术图片

 

 

 

Socket类

技术图片

 

 

 

import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.Socket;

public class Client {
    public static void main(String[] args) throws IOException {
        //创建socket,指定服务器的ip 和端口
        Socket socket = new Socket("127.0.0.1",8888);
        //从socket里面获取字节输出流,准备给服务器发送数据
        OutputStream ops = socket.getOutputStream();
        ops.write("你好,全植强的服务器".getBytes());
//没有服务器会抛出异常:thread "main" java.net.ConnectException: Connection refused: connect
        //获取字节输入流,接受来自服务器的回馈数据
        InputStream ips =socket.getInputStream();

        byte[] bytes = new byte[1024];//创建接受字节数组,数组长度初定为
        int len = ips.read(bytes);//将数据读入数组,返回数组的长度
        System.out.println(new String(bytes,0,len));//输出数据

        socket.close();
    }
}

 

 

ServerSocket类

 

技术图片

 

 

 

import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.ServerSocket;
import java.net.Socket;

public class Server {
    public static void main(String[] args) throws IOException {
        //创建ServerSocket 指定端口号,不指定就有系统随机安排!!
        ServerSocket server = new ServerSocket(8888);
        //监听获取客户端的socket
        Socket socket= server.accept();
        //从获取到客户端的socket创建字节输入流,准备读取客户端发来的数据
        InputStream ips =socket.getInputStream();

        byte[] bytes = new byte[1024];
        int len = ips.read(bytes);//将数据读入数组,返回数据的长度
        System.out.println(new String(bytes,0,len));

        OutputStream ops = socket.getOutputStream();
        ops.write("收到小弟".getBytes());
        socket.close();
        server.close();
    }
}

 

 

TCP文件上传是咧

技术图片

 

 

 

FileClient

技术图片

 

 

 

/*
文件上传的客服端,读取本地文件,上传到服务器,读取服务器回写的数据
    数据源:Snipaste_2020-06-25_23-37-15.png
    目的地:服务器
 */
public class TcpSocketFileClient {
    public static void main(String[] args) throws IOException {
        FileInputStream fis = new FileInputStream("Snipaste_2020-06-25_23-37-15.png");
        Socket socket = new Socket("127.0.0.1",8888);
        OutputStream os = socket.getOutputStream();
        int len = 0;
        byte[] bytes = new byte[1024];
        /*
        @return     the total number of bytes read into the buffer, or
     *             -1 if there is no more data because the end of
     *             the file has been reached.
         */
        while ((len = fis.read(bytes)) != -1){
            System.out.println(len);
            os.write(bytes,0,len);
        }

        InputStream ips =socket.getInputStream();
        while ((len = ips.read(bytes)) != -1){
            System.out.println(new String(bytes,0,len));
        }

        fis.close();
        socket.close();
    }
}

 

 

FileServer::

技术图片

 

 

 

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.ServerSocket;
import java.net.Socket;

public class TcpSocketFileServer {
    public static void main(String[] args) throws IOException {
        //创建服务器socket指定端口号
        ServerSocket server = new ServerSocket(8888);
        //监听客户端,获取socket
        Socket socket = server.accept();
        //获取socket中的字节输入流,读取客户端传进来的数据
        InputStream ips = socket.getInputStream();
        //判断文件对象是否存在
        File file = new File("d:\\upload");
        if(!file.exists()){
            file.mkdir();
        }
        //获取本地文件输出字节流,准备将从客户端获取到的文件写出到本地磁盘中
        FileOutputStream fops = new FileOutputStream(file+"\\up.png");
        //读取客户端发来的输入字节流
        int len = 0;
        byte[] bytes = new byte[1024];
        while ((len = ips.read(bytes))!= -1){
            //网本地输出字节流写入客户端的数据
            fops.write(bytes,0,len);
        }
        //使用socket中的字节输出字节流往客户端输出数据
        socket.getOutputStream().write("done".getBytes());
        fops.close();
        socket.close();
        server.close();

    }
}

运行服务器再运行客户端最后发现:

技术图片

 

 

 

 

当时两个程序都没有停止下来,为什么:

技术图片

 

 

 

解决方法;

技术图片

 

 

 

客户端加入:

技术图片

 

 

 结果,两个程序都结束,client也收到结果:

技术图片

 

 

 

 

优化服务端

  /*
        优化,自定义一个文件的命名规则,防止同盟的文件被覆盖掉
            规则:域名+毫秒数+随机数

         */
        String filename = "\\better.quan"+System.currentTimeMillis()+new Random().nextInt(787878)+".png";
        FileOutputStream fops = new FileOutputStream(file+filename);
        //读取客户端发来的输入字节流
        int len = 0;
        byte[] bytes = new byte[1024];
        while ((len = ips.read(bytes))!= -1){
            //网本地输出字节流写入客户端的数据
            fops.write(bytes,0,len);
        }

结果:

技术图片

 

 

 

优化2:让服务器不停下,持续监听,客户端就能不断上传文件:

   /*
        让服务器一直处于监听状态(死循环accept方法)
        有一个客户端上传文件,就保存一个文件
         */

        while (true){
            Socket socket = server.accept();
            //获取socket中的字节输入流,读取客户端传进来的数据
            InputStream ips = socket.getInputStream();
            //判断文件对象是否存在
            File file = new File("d:\\upload");
            if(!file.exists()){
                file.mkdir();
            }
            //获取本地文件输出字节流,准备将从客户端获取到的文件写出到本地磁盘中

        /*
        优化,自定义一个文件的命名规则,防止同盟的文件被覆盖掉
            规则:域名+毫秒数+随机数

         */
            String filename = "\\better.quan"+System.currentTimeMillis()+new Random().nextInt(787878)+".png";
            FileOutputStream fops = new FileOutputStream(file+filename);
            //读取客户端发来的输入字节流
            int len = 0;
            byte[] bytes = new byte[1024];
            while ((len = ips.read(bytes))!= -1){
                //网本地输出字节流写入客户端的数据
                fops.write(bytes,0,len);
            }
            //使用socket中的字节输出字节流往客户端输出数据
            socket.getOutputStream().write("done".getBytes());
            fops.close();
            socket.close();
        }
       //不用关闭服务器
//        server.close();

 

多次启动客户端

技术图片

 

 

 

 

优化三:提高效率,利用多线程

来一个客户端上传文件请求,开启一个线程去处理

public class TcpSocketFileServer {
    public static void main(String[] args) throws IOException {
        //创建服务器socket指定端口号
        ServerSocket server = new ServerSocket(8888);
        //监听客户端,获取socket
        /*
        让服务器一直处于监听状态(死循环accept方法)
        有一个客户端上传文件,就保存一个文件
         */
        while (true) {
            Socket socket = server.accept();
            //获取socket中的字节输入流,读取客户端传进来的数据
            /*
            使用多线程,提高程序的效率
                有一个客户端上传文件,就开启一个线程,完成文件的上传
             */
            new Thread(new Runnable() {
                @Override
                public void run() {
                    try {//由于接口类当中没有throw异常,使用try catch
                        InputStream ips = socket.getInputStream();
                        //判断文件对象是否存在
                        File file = new File("d:\\upload");
                        if (!file.exists()) {
                            file.mkdir();
                        }
                        //获取本地文件输出字节流,准备将从客户端获取到的文件写出到本地磁盘中
                        /*
                        优化,自定义一个文件的命名规则,防止同盟的文件被覆盖掉
                            规则:域名+毫秒数+随机数
                         */
                        String filename = "\\better.quan" + System.currentTimeMillis() + new Random().nextInt(787878) + ".png";
                        FileOutputStream fops = new FileOutputStream(file + filename);
                        //读取客户端发来的输入字节流
                        int len = 0;
                        byte[] bytes = new byte[1024];
                        while ((len = ips.read(bytes)) != -1) {
                            //网本地输出字节流写入客户端的数据
                            fops.write(bytes, 0, len);
                        }
                        //使用socket中的字节输出字节流往客户端输出数据
                        socket.getOutputStream().write("done".getBytes());
                        fops.close();
                        socket.close();
                    } catch (FileNotFoundException e) {
                        e.printStackTrace();
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                }
            }).start();
            //不用关闭服务器
//        server.close();
        }
    }
}

 

 

模拟B\S服务器

//BS服务器版本TCP服务器
public class BsTcpServer {
    public static void main(String[] args) throws IOException {
        ServerSocket server = new ServerSocket(8080);
        Socket socket = server.accept();

        InputStream ips = socket.getInputStream();
        byte[] bytes = new byte[1024];
        int len = 0;
        while ((len = ips.read(bytes))!= -1){
            System.out.println(new String(bytes,0,len));
        }
    }
}

运行并访问

技术图片

 

 

 服务器输出为:

GET /testCOM.html HTTP/1.1   //地址请求的资源位置
Host: localhost:8080
Connection: keep-alive
Cache-Control: max-age=0
Upgrade-Insecure-Requests: 1
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/83.0.4103.106 Safari/537.36 Edg/83.0.478.54
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.9
Sec-Fetch-Site: cross-site
Sec-Fetch-Mode: navigate
Sec-Fetch-User: ?1
Sec-Fetch-Dest: document
Accept-Encoding: gzip, deflate, br
Accept-Language: zh-CN,zh;q=0.9,en;q=0.8,en-GB;q=0.7,en-US;q=0.6
Cookie: goSessionid=xf82SLPFkJGHj5qZcl_lsO6XFT-FTHwjgVlxkkQdzWs%3D

 

GET /testCOM.html HTTP/1.1 可以使用BufferReader的方法readline读取一行
new BufferedReader(new InputStreamReader(is)) 将网络字节输入流转换为字符缓冲输入流

再使用String 类方法split(" ") 切割字符串,获取中间的部分
再使用String的方法substring(1)获取html文件的路径
testCOM.html

  然后服务器建立一个本地字节输入流,根据获取到的文件路径读取html文件

  服务器使用网络字节输出流把读取到的文件写道客户端即浏览器显示

固定的浏览器写入:

//写入HTTP协议响应投,固定写法
out.write("HTTP/1.1 200 OK\r\n".getBytes());
out.write("Content-Type:text/html\r\n".getBytes());

//必须写入空行
out.write("\r\n".getBytes());

 

Server辰戌:

import java.io.*;
import java.net.ServerSocket;
import java.net.Socket;

//BS服务器版本TCP服务器
public class BsTcpServer {
    public static void main(String[] args) throws IOException {
        ServerSocket server = new ServerSocket(8080);
        Socket socket = server.accept();

        InputStream ips = socket.getInputStream();
//        byte[] bytes = new byte[1024];
//        int len = 0;
//        while ((len = ips.read(bytes))!= -1){
//            System.out.println(new String(bytes,0,len));
//        }
        BufferedReader br =new BufferedReader(new InputStreamReader(ips));
        String line = br.readLine();//GET /resources/testCOM.html HTTP/1.1
        String[] strings = line.split(" ");// /resources/testCOM.html
        String htmlpath = strings[1].substring(1);//  resources/testCOM.html

        FileInputStream fips = new FileInputStream(htmlpath); //  resources/testCOM.html

        OutputStream ops =socket.getOutputStream();

        //写入HTTP协议响应投,固定写法
        ops.write("HTTP/1.1 200 OK\r\n".getBytes());
        ops.write("Content-Type:text/html\r\n".getBytes());
        //必须写入空行
        ops.write("\r\n".getBytes());

        int len = 0;
        byte[] bytes = new  byte[1024];
        while ((len = fips.read(bytes)) != -1){
            ops.write(bytes,0,len);
        }
        fips.close();
        socket.close();
        server.close();

    }
}

 

 

 

结果:

技术图片

 

 

 

技术图片

 

java-网络通信相关

标签:XML   lang   服务端   rand   loading   返回   navig   flat   响应   

原文地址:https://www.cnblogs.com/java-quan/p/13193657.html


评论


亲,登录后才可以留言!