什么是网络编程?

目录

一.UDP

        DatagramSocket

        DatagramPacket

       服务器

       客户端

二.TCP

       ServerSocket

       Socke

       服务器

       客户端


网络编程指的就是网络上的主机通过不同的进程,以编程的方式实现网络信息传输,而提到网络编程的话,就需要知道一个网络编程套接字socket(操作系统提供的网络编程的API),这是传输层和应用层之间进行调用的接口,也就是操作系统给应用程序提供的API,一调用这个就可以和内核之间进行信息传递,那么在传输层内两个主要的协议TCP/UDP,就得了解了解了,而这两个协议的差别也是很大的,所以分别给这两组协议各自都提供了合适的API,因此这两套API都得加以了解!

一.UDP

基本特点:无连接(不需要与服务器建立连接就可以直接发送数据),不可靠(并不知道对方是否接受到数据),面向数据流,全双工(一条通道,双向通信)

两个核心类:

DatagramSocket

创建了一个UDP版本的socket对象(代表着操作系统中的socket文件(代表这着网卡硬件设备的抽象体现))

核心方法:receive:接收数据

send:发送数据

close:释放数据

DatagramPacket

表示了一个UDP数据报,意味着每次发送/接收的都是一个DatagramPacket对象

使用这两个类,来简单实现一个客户端服务器程序,回显服务(请求啥就响应啥)

服务器

import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.SocketException;


public class UDPEchoServer {
    //首先需要有一个DataframSocket实例,这是进行网络编程的前提
    private DatagramSocket socket = null;
    //其次在构造函数里面完成对象的构造,构造socket对象的时候需要显式的绑定一个端口号,这个端口号就是用来区分一个应用程序的,主机收到网卡上的数据时,应该给哪个程序就得明确

    public UDPEchoServer(int port) throws SocketException {
        //这里有这个异常表示socket对象有可能创建失败:1.端口号已经被占用了,同一主机的两个程序不能有相同的端口号,但是一个进程可以绑定多个端口(人和手机号),一个进程创建多个socket对象,每个对象都有自己的端口号这是可以的
        // 2.每个进程能打开的文件个数是有上限的,如果在这之前已经打开了很多文件,那么这个socket文件很有可能打不开
        socket = new DatagramSocket(port);
    }

    public void start() throws IOException {
        System.out.println("启动服务器!");
        //由于UDP不需要建立连接,因此可以直接接收从客户端传来的数据
        while(true){
            //先接收客户端的请求,因为服务器是被动接收数据的,因此开始都是先接收,而不是发送
            DatagramPacket requestPacket = new DatagramPacket(new byte[1024],1024);//需要指定大小
            //这里的receive如果客户端没有发送请求的话,会一直在阻塞状态,直到客户端请求发送才会接收
            socket.receive(requestPacket);//想要接收数据,就需要有一个空的DatagramPacket对象,来让这个方法填充数据
            //此时已经将接收到的数据放在requestPacket里面了,然后将这里面的数据解析出来,解析成字符串
            String request = new String(requestPacket.getData(),0,requestPacket.getLength(),"UTF-8");//也可以指定字符编码
            //然后根据请求计算其所能得到的响应,存放在另一个字符串内
            String response = prcoess(request);
            //然后就要把服务器响应的数据发送回到客户端去
            DatagramPacket responsePacket = new DatagramPacket(response.getBytes(),response.getBytes().length,requestPacket.getSocketAddress());//这里不仅要放回响应内容,而且要指定好要发送到的IP地址和端口号,这样才可以正确发送,要不然是没有办法准确发送的
            socket.send(responsePacket);//发送过来的是DatagramPacket数据报,返回的也就是这样的数据报,但是发送回去的就不再是一个空的了,而是应该把从上面所得到的响应字符串放到里面然后再发送
            //可以再手动输出一下客户端的IP地址和端口号
            System.out.printf("[%s:%d] req: %s, resp: %s\n",requestPacket.getAddress().toString(),requestPacket.getPort(),request,response);
        }
    }
    //由于这里是回显服务,因此不需要做出改变,其实对于一个服务器来说最难的地方也就是计算响应了
    public String prcoess(String request) {
        return request;
    }

    public static void main(String[] args) throws IOException {
        UDPEchoServer udpEchoServer = new UDPEchoServer(9090);
        udpEchoServer.start();
    }
}

客户端

import java.io.IOException;
import java.net.*;
import java.util.Scanner;

//只有服务器还不行,还得来个客户端,才能真正实现
public class UDPEchoClient {
    private DatagramSocket socket = null;
    private String severIP;//IP地址
    private int severPort;//端口号
    //构造函数
    public UDPEchoClient(String ip,int port) throws SocketException {
        socket = new DatagramSocket();
        //这里就是用无参版本的构造socket对象,就是让系统随机分配一个端口号
        severIP = ip;//这个指定的是服务器的IP地址和端口号和客户端没有关系
        severPort = port;
    }
    Scanner input = new Scanner(System.in);
    public void start() throws IOException {
        while(true){
            //首先先从控制台读取到用户输入的信息,然后才能发送
            System.out.print("->");
            String request = input.next();
            //然后把用户写的信息构造成一个UDP请求,才能发送
            DatagramPacket requestPacket = new DatagramPacket(request.getBytes(),request.getBytes().length,
                    InetAddress.getByName(severIP),severPort);//这里需要指定发送给谁也就是指定服务器的IP地址和端口号
            //然后发送UDP数据报
            socket.send(requestPacket);

            //发送完之后需要再接收从服务器得到的响应数据
            DatagramPacket responsePacket = new DatagramPacket(new byte[1024],1024);
            socket.receive(responsePacket);

            //然后再把响应的数据写成字符串形式,然后打印到控制台
            String response = new String(responsePacket.getData(),0,responsePacket.getLength(),"UTF-8");
            System.out.printf("req:%s,resp:%s\n",request,response);
        }


    }

    public static void main(String[] args) throws IOException {
        UDPEchoClient udpEchoClient = new UDPEchoClient("127.0.0.1",9090);
        udpEchoClient.start();
    }
}

什么是网络编程?_第1张图片什么是网络编程?_第2张图片

此时我们就可以看到当客户端先发出请求之后,服务器这边就可以产生响应了,而一个服务器是可以响应多个客户端的,但是在idea里面一个程序只能启动一次,那该怎么办呢? 

当然是可以解决的 

 什么是网络编程?_第3张图片

然后就可以启动多个客户端了,并且也有响应 

什么是网络编程?_第4张图片

什么是网络编程?_第5张图片

二.TCP

基本特点:有连接,可靠(知道对方是否接受到数据),面向字节流,全双工(一条通道,双向通信)

两个核心类:

ServerSocket

专门给TCP服务器用的

Socke

既需要给服务器使用,有需要给客户端使用

这里通过这样的类,来描述一个socket文件即可,而不需要专门的类,表示"传输的包",这是面向字节流的.是以字节为单位进行传输的

使用这两个类,来简单实现一个客户端服务器程序,回显服务(请求啥就响应啥)

服务器


import java.io.InputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.io.PrintWriter;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.Scanner;

public class TCPEchoServer {
    private ServerSocket serverSocket = null;



    public TCPEchoServer(int port) throws IOException {
        serverSocket = new ServerSocket(port);
    }
    public void start() throws IOException {
        System.out.println("服务器启动!");
        while(true){
            //由于TCP是有连接的,不能一上来就直接读取请求,得先建立连接(也是相当于接电话)
            //accept就是在接电话,而接电话的前提是有人给你打电话 了,如果没有客户端尝试建立连接,那么服务器就会陷入阻塞状态
            //accept是返回了一个Socket对象,称为clientSocket,后续如果和客户端之间沟通都是通过clientSocket来完成的,而serverSocket就只做了第一步也就是只接了个电话
            Socket clientSocket = serverSocket.accept();
            //然后再处理响应
            processConnection(clientSocket);
        }
    }

    private void processConnection(Socket clientSocket) {
        System.out.printf("[%s,%d] 客户端建立连接\n",clientSocket.getInetAddress().toString(),clientSocket.getPort());//IP地址和端口号
        //接下来就是处理请求和响应

        //这里针对TCP socket的读写就和读写文件是一摸一样的
        try(InputStream inputStream = clientSocket.getInputStream()){
            try (OutputStream outputStream = clientSocket.getOutputStream()){
                Scanner scanner = new Scanner(inputStream);//从控制台写入到inputStream中去

                while (true){
                    //循环处理每个请求分别返回请求
                    if(!scanner.hasNext()){
                        System.out.printf("[%s,%d] 客户端断开连接!\n",clientSocket.getInetAddress().toString(),clientSocket.getPort());
                        break;
                    }
                    //此处用scanner更方便,如果使用inputStream的read也是可以的
                    String request = scanner.next();
                    //根据请求计算响应
                    String response = process(request);
                    //把这个响应返回给客户端
                    
                    //为了方便起见,可以使用PrintWriter把OutputStream包裹一下
                    PrintWriter printWriter = new PrintWriter(outputStream);//填充性参数
                    printWriter.println(response);
                    //刷新缓冲区,如果没有这个刷新,可能客户端就不能第一时间看到响应结果
                    printWriter.flush();
                    System.out.printf("[%s,%d] req :%s,resp:%s\n",clientSocket.getInetAddress().toString(),clientSocket.getPort(),request,response);
                }
            }
        } catch (IOException e) {
            e.printStackTrace();
        }finally {
            try {
                clientSocket.close();//但是这个是每个连接有一个的,这个的数目很多,并且连接断开也就不用了,因此这个是会持续积累的,所以每次都得保证处理完的连接都进行释放
                //记得关闭文件(释放资源)   
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }

    private String process(String request) {
        return request;
    }

    public static void main(String[] args) throws IOException {
        TCPEchoServer tcpEchoServer = new TCPEchoServer(9090);
        tcpEchoServer.start();
    }
}

客户端


import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.PrintWriter;
import java.net.Socket;
import java.util.Scanner;

public class TCPEchoClient {
    //此时使用普通的socket就可以了,不用ServerSocket了
    private Socket socket = null;

    //构造函数
    public TCPEchoClient(String serverIP,int serverPort) throws IOException {
        //对于UDP的socket和TCP的serverSocket来说,构造函数里面给定的端口号,就是指定自己要绑定哪个端口号
        //而这里对于TCP的Socket来说,这里指定的IP和端口号是表示自己要与哪一个服务器进行连接,这里的IP和端口号都是服务器的
        socket = new Socket(serverIP,serverPort);//调用这个构造方法就会和服务器进行连接(类似于打电话拨号码)

    }
    //启动
    public void start(){
        System.out.println("和服务器连接成功!");
        Scanner input = new Scanner(System.in);
        try(InputStream inputStream = socket.getInputStream()) {
            try(OutputStream outputStream = socket.getOutputStream()) {
                while(true){
                    //先从控制台读取字符串
                    System.out.print("->");
                    String request = input.next();
                    //然后再把读到的字符串,构造成请求,把请求发给服务器
                    PrintWriter printWriter = new PrintWriter(outputStream);
                    printWriter.println(request);//填充性参数,把里面的值写回到outputStream里面
                    printWriter.flush();//每添加一个就刷新一下,保证服务器能及时看到数据

                    //然后从服务器中读取响应,并解析
                    Scanner respScanner = new Scanner(inputStream);
                    String response = respScanner.next();
                    //打印结果
                    System.out.printf("req:%s,resp:%s\n",request,response);
                }
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    public static void main(String[] args) throws IOException {
        TCPEchoClient tcpEchoClient = new TCPEchoClient("127.0.0.1",9090);
        tcpEchoClient.start();
    }
}

什么是网络编程?_第6张图片

什么是网络编程?_第7张图片

此时我们发现服务器启动之后,只要启动客户端就会连接服务器,发出的请求也可以得到响应,貌似是正常了,但是如果当我们启动多个客户端的时候就会发现问题所在了:

什么是网络编程?_第8张图片

 什么是网络编程?_第9张图片什么是网络编程?_第10张图片

 多个客户端启动之后就会发现服务器那边并没有与这个客户端建立连接,而且客户端2发出任何请求都没有得到响应,只有在第一个客户端断开连接后,第二的客户端才能建立连接,这肯定是不现实的,一个服务器不可能只连接一个客户端,只为一个客户端提供服务,那么这个问题出在哪里了呢?

仔细研究一下就会发现:

什么是网络编程?_第11张图片

什么是网络编程?_第12张图片

 accept接通连接之后,就会进入processConnection的内层循环里面去,然后一直发出请求,只要这个客户端的请求不中断,那么这个循环就不会结束,然后就会一直处于这个连接当中,其他客户端发出连接请求之后,只能等待第一个客户端的连接断开之后,才能与下一个客户端建立连接,也就是执行下一条accept,一直都只能与一个客户端进行连接,这就不满足我们想与多个客户端进行连接的想法了,这也就是问题所在了,那么怎么解决这个问题呢?  很简单,只需要使用以前学习过的多线程或者线程池来组织一下这些连接,把这些客户端的连接变成并行操作的就可以了:

 多线程版本:

import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.PrintWriter;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.Scanner;

public class TCPThreadEchoServer {
    private ServerSocket serverSocket = null;



    public TCPThreadEchoServer(int port) throws IOException {
        serverSocket = new ServerSocket(port);
    }
    public void start() throws IOException {
        System.out.println("服务器启动!");
        while(true){
            
            Socket clientSocket = serverSocket.accept();
            //使用多线程:
           
            Thread t = new Thread(()->{
                processConnection(clientSocket);//这样之后每一个客户端的连接就从串行变成并行的了,然后每一个客户端的连接就可以成功了
            });
            t.start();

        }
    }

    private void processConnection(Socket clientSocket) {
        System.out.printf("[%s,%d] 客户端建立连接\n",clientSocket.getInetAddress().toString(),clientSocket.getPort());//IP地址和端口号
      
        try(InputStream inputStream = clientSocket.getInputStream()){
            try (OutputStream outputStream = clientSocket.getOutputStream()){
                Scanner scanner = new Scanner(inputStream);

                while (true){
                    
                    if(!scanner.hasNext()){
                        System.out.printf("[%s,%d] 客户端断开连接!\n",clientSocket.getInetAddress().toString(),clientSocket.getPort());
                        break;
                    }
                    
                    String request = scanner.next();
                    //根据请求计算响应
                    String response = process(request);
                    
                    PrintWriter printWriter = new PrintWriter(outputStream);
                    printWriter.println(response);

                    printWriter.flush();
                    System.out.printf("[%s,%d] req :%s,resp:%s\n",clientSocket.getInetAddress().toString(),clientSocket.getPort(),request,response);
                }
            }
        } catch (IOException e) {
            e.printStackTrace();
        }finally {
            try {
                clientSocket.close();/
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }

    public String process(String request) {
        return request;
    }

    public static void main(String[] args) throws IOException {
        TCPThreadEchoServer tcpThreadServer = new TCPThreadEchoServer(9090);
        tcpThreadServer.start();
    }
}

 线程池版本:



import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.PrintWriter;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.Scanner;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

public class TCPThreadPoolEchoServer {
    private ServerSocket serverSocket = null;



    public TCPThreadPoolEchoServer(int port) throws IOException {
        serverSocket = new ServerSocket(port);
    }
    public void start() throws IOException {
        System.out.println("服务器启动!");
        ExecutorService pool = Executors.newCachedThreadPool();
        while(true){
           
            Socket clientSocket = serverSocket.accept();
            //线程池来解决:
            //然后再处理响应
            pool.submit(new Runnable() {
                @Override
                public void run() {
                    processConnection(clientSocket);

                }
            });

        }
    }

    private void processConnection(Socket clientSocket) {
        System.out.printf("[%s,%d] 客户端建立连接\n",clientSocket.getInetAddress().toString(),clientSocket.getPort());
        
        try(InputStream inputStream = clientSocket.getInputStream()){
            try (OutputStream outputStream = clientSocket.getOutputStream()){
                Scanner scanner = new Scanner(inputStream);/

                while (true){
                   
                    if(!scanner.hasNext()){
                        System.out.printf("[%s,%d] 客户端断开连接!\n",clientSocket.getInetAddress().toString(),clientSocket.getPort());
                        break;
                    }
                    
                    String request = scanner.next();
                    //根据请求计算响应
                    String response = process(request);
                    //把这个响应返回给客户端
                    //为了方便起见,可以使用PrintWriter把OutputStream包裹一下
                    PrintWriter printWriter = new PrintWriter(outputStream);//填充性参数
                    printWriter.println(response);
                   
                    printWriter.flush();
                    System.out.printf("[%s,%d] req :%s,resp:%s\n",clientSocket.getInetAddress().toString(),clientSocket.getPort(),request,response);
                }
            }
        } catch (IOException e) {
            e.printStackTrace();
        }finally {
            try {
                clientSocket.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }

    public String process(String request) {
        return request;
    }

    public static void main(String[] args) throws IOException {
        TCPThreadPoolEchoServer server = new TCPThreadPoolEchoServer(9090);
        server.start();
    }
}

然后随便使用一个线程版本的服务器就可以与多个客户端进行连接了:

什么是网络编程?_第13张图片

什么是网络编程?_第14张图片

什么是网络编程?_第15张图片

 这样就可以与多个客户端进行连接了这样问题也就解决了!!!

你可能感兴趣的:(操作系统,计算机网络,java)