Java高级特性(三)网络编程

网络编程

两台及以上的电脑相互连接形成的网络,连接的本质就是网络通讯的问题,通讯的实现就产生了一系列的协议,IP,tcp,udp等协议,所谓网络编程就是数据的通讯操作,这个通讯操作,分为客户端和服务器端,形成了以下两种模型,
c/s(client/server,客户端与服务器端),要开发出两套程序,一套程序为服务端,服务端改变后,客户端也发生相应改变,
b/s(Browse//server ,浏览器与服务端,) 只开发一套服务端的程序,服务端利用浏览器访问,开发维护成本低,由于使用http协议,并且使用80端口,故安全性较差,现在的开发大都以B/S为主。
C/S,分为两种,tcp(可靠的数据连接),udp(不可靠的数据连接)

echo程序模型

tcp程序的基本实现,

tcp开发,其核心时使用两个类实现数据的交互处理:serversocket(服务器端,),socket(客户端)
在这里插入图片描述
serversocket的主要目的时设置服务器的监听端口,
而socket要指明主机和端口,

实现一个简单的数据处理操作,使用echo模型,是一个客户端和服务器端的服务模型,

在这里插入图片描述

package demo;
import java.io.PrintWriter;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.Scanner;
public class mythread {
    public static void main(String[] args) throws Exception {
        ServerSocket server=new ServerSocket(999); //  设置服务端口
        System.out.println("d等待客户端连接*****");
        Socket client =server.accept();     //有客户连接
        /***
         * 首先需要先接收客户端发来的信息,而后才可以将信息处理之后发送给客户端
         */
        Scanner scan=new Scanner(client.getInputStream());
        scan.useDelimiter("\n"); //设置分隔符
        PrintWriter out=new PrintWriter(client.getOutputStream());
        boolean flag=true; //作为一个循环标记
        while(flag) {
            if (scan.hasNext()){
                String val=scan.next().trim();
                if ("byebye".equalsIgnoreCase(val)){
                    out.println("byebye***");
                    flag=false;    //结束循环
                }else {
                    out.println("[echo]"+val);
                    out.flush(); //强制刷新缓冲区
                }
            }
        }
        scan.close();
        out.close();
        client.close();
    }}
package demo;
import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.io.PrintWriter;
import java.net.Socket;
import java.util.Scanner;
public class EchoClient {
    private static final BufferedReader KEYBOARD_INPUT=new BufferedReader(new InputStreamReader(System.in));
    public static void main(String[] args) throws Exception {
        Socket client=new Socket("localhost",9999); //定义服务端连接信息,
        //现在客户端需要输入和输出的操作支持,所以依然要准备出scanner和printwriter
        Scanner scan=new Scanner(client.getInputStream()); //接受服务端内容
        scan.useDelimiter("\n");
        PrintWriter out=new PrintWriter(client.getOutputStream());  //向服务端发送内容
        boolean flag=true;
        while (flag){
            String input=getString("请输入要发送的内容").trim();
            out.println(input);    //加换行
            if (scan.hasNext()){
                System.out.println(scan.next());
            }
            if ("byebye".equalsIgnoreCase(input)){
                flag=false;
            }
    }
        scan.close();
        out.close();
        client.close();
    }
    public static String getString(String prompt)throws Exception{
        System.out.print(prompt);
        String str = KEYBOARD_INPUT.readLine();
        return str;
    }
}

这里需要注意,自己电脑的telnet cilent服务端要打开,然后运行EchoClient.java,打开电脑的命令行,输入telnet,然后输入open localhost 9999,就可以输入操作了,
(window10版本 没有telnet服务端,要自行下载,网上很难找到破解版,鹰酱要收费,哎)

BIO处理模型

上一个程序存在严重的性能缺陷,仅能为一个线程提供服务,

单线程的服务器开发本身就是不合理的,
可以通过,服务器上启动多个线程,为每一个客户端提供服务,

在这里插入图片描述

修改服务器端程序实现多线程接受请求

package demo;

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

public class mythread {
    private static class ClientThread implements Runnable{
        private Socket Client=null;  //描述每一个不同的客户端
        private Scanner scan=null;
        private PrintStream out=null;
        private boolean flag=true;


        public ClientThread(Socket client) throws Exception {
            this.Client=client;
            this.scan=new Scanner(client.getInputStream());
            this.scan.useDelimiter("\n"); //设置分隔符
            this.out=new PrintStream(client.getOutputStream());
        }

        @Override
        public void run() {
            while(this.flag) {
                if (this.scan.hasNext()){
                    String val=scan.next().trim();
                    if ("byebye".equalsIgnoreCase(val)){
                        out.println("byebye***");
                        this.flag=false;    //结束循环
                    }else {
                        out.println("[echo]"+val);
                        out.flush(); //强制刷新缓冲区
                    }
                }
            }
            try {
                this.scan.close();
                this.out.close();
                this.Client.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }
    public static void main(String[] args) throws Exception {
        ServerSocket server=new ServerSocket(999); //  设置服务端口
        System.out.println("d等待客户端连接*****");


        boolean flag=true; //作为一个循环标记
        while (flag){
            Socket client =server.accept();     //有客户连接
            /***
             * 首先需要先接收客户端发来的信息,而后才可以将信息处理之后发送给客户端
             */
            new Thread(new ClientThread(client)).start();
        }
        server.close();
    }}

多线程意味着能够处理更多的服务处理,

udp程序

基于数据包的网路编程实现,

实现UDP程序需要两个类:datagramPacket(数据内容),datagramsoket(网路传输)数据包是否接收与发送者无关,

package demo;

import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;

public class UDPserver {
    public static void main(String[] args) throws Exception{
        DatagramSocket server=new DatagramSocket(9999); //连接到9999端口
        String str="www.mldn.cn";
        DatagramPacket packet=new DatagramPacket(str.getBytes(),0,str.length(), InetAddress.getByName("localhost"),9999); //接收数据
        server.send(packet);  //发送消息
        System.out.println("发送完毕");
        server.close();
    }
}

package demo;

import java.net.DatagramPacket;
import java.net.DatagramSocket;

public class UDPClient {
    public static void main(String[] args) throws Exception{
        DatagramSocket Client=new DatagramSocket(9999); //连接到9999端口
        byte data[]=new byte[1024]; //接受消息
        DatagramPacket packet=new DatagramPacket(data,data.length); //接收数据
        System.out.println("客户端等待接收消息*****");
        Client.receive(packet); // 接收消息,所有的消息在data字节数据中
        System.out.println("接收到的消息内容为:"+new String(data,0,packet.getLength()));
        Client.close();
    }
}

tcp连接可靠但是需要的资源也越多,这也就是差别

/***下面补以下stream数据流

stream数据流

jdk1.8之后进入大数据时代,在类集里面也支持有数据的流式分析处理操作,为此专门提供了一个stream的接口,同时在collections接口中提供了实例化的方法,
获取stream接口对象:public default streamstream();

主要是对数据到分析处理,同时主要时是针对集合中的数据进行分析处理,
基础操作:

package demo;

import java.sql.SQLOutput;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.stream.Collectors;
import java.util.stream.Stream;

public class stream {
    public static void main(String[] args) throws Exception{
        List<String> all=new ArrayList<>();
        Collections.addAll(all,"java","php","python","javascript");
        Stream<String> stream=all.stream();
        System.out.println(stream.count());  // 输出元素个数,
        //要求将每一个元素的字母变为小写字母,然后在判断字母j是否存在
        System.out.println(stream.filter(ele->ele.toLowerCase().contains("j")).count());
        //将包含j的数据转化为list集合
        List<String> result=stream.filter((ele)->ele.toLowerCase().contains("j")).collect(Collectors.toList());
        System.out.println(result);

    }
}

处理大量数据时,选取一些数据
在这里插入图片描述

Java大数据处理了解一下就行。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值