java 网络编程——day23

这里写图片描述

找到本机主机名和IP地址

package day23;

import java.net.InetAddress;
import java.net.UnknownHostException;


public class IPDemo {
    public static void main(String[] args) throws UnknownHostException {

        InetAddress i = InetAddress.getLocalHost();

        System.out.println(i.toString());
        System.out.println("address:"+i.getHostAddress());
        System.out.println("name:"+i.getHostName());


        InetAddress ia = InetAddress.getByName("PC-20130712JEEW");
        System.out.println("address:"+ia.getHostAddress());
        System.out.println("name:"+ia.getHostName());
    }
}

这里写图片描述

这里写图片描述

在介绍Udp和Tcp之前,先介绍一个概念:Socket

Socket

Socket就是为网络服务提供的一种机制
通信两端都有Socket(就是两个端点)
网络通信其实就是Socket之间的通信
数据在两个Socket间通过IO传输

关于Udp传输

Udp发送:

需求:通过Udp传输方式,将一段文字数据发送出去

思路:
1,建立udpsocket服务。指定发送端口
2,提供数据,并将数据封装到数据包中。
3,通过socket服务的发送功能,将数据包发送出去。
4,关闭资源。

package day23;

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

class UdpSend {

    public static void main(String[] args) throws IOException {
//      1,创建udp服务,通过DatagramSocket对象建立发送端口
        DatagramSocket ds = new DatagramSocket(12121);

//      2,确定数据,并封装成数据包。
//      DatagramPacket(byte[] buf, int length, InetAddress address, int port) 
        byte[] buf = "shu ju lai le".getBytes();
        DatagramPacket dp = new DatagramPacket(buf, buf.length, InetAddress.getByName("PC-20130712JEEW"), 12222);

//      3,通过socket服务的发送功能,将数据包发送出去。通过send方法。
        ds.send(dp);

//      4,关闭资源。
        ds.close();
    }
}


Udp接收:
需求:定义一个应用程序,用于接受并处理数据

思路:
1,定义udpsocket服务。通常会监听一个端口,其实就是给这个接受网络应用程序定义数字标识。方便于明确哪些数据过来该应用程序可以处理。
2,定义一个数据包,存储接收到的字节数据。
3,通过socket服务中的receive方法将接收到的数据存入已定义好的 数据包中。
4,通过数据包对象的特有功能将这些不同的数据取出,打印在控制台上。
5,关闭资源。

package day23;

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



public class UdpRece {
    public static void main(String[] args) throws IOException {
        // 1,创建udp的socket服务,建立端点
        DatagramSocket ds = new DatagramSocket(12222);

        // 加入循环,可以一直接受数据
        while (true) {

            // 2,定义数据包,用于存储数据
            byte[] buf = new byte[1024];
            DatagramPacket dp = new DatagramPacket(buf, buf.length);

            // 3,通过服务的receive方法将收到的数据存入数据包中
            ds.receive(dp); //阻塞式方法

            // 4,通过数据包中的方法获取数据,并打印
            String ip = dp.getAddress().getHostAddress();
            String data = new String(dp.getData(), 0, dp.getLength());
            int port = dp.getPort();

            System.out.println("ip:" + ip);
            System.out.println("data:" + data);
            System.out.println("port:" + port);
        }
        // 5,关闭资源
        // ds.close();
    }
}

这里写图片描述

需求:在发送端写入数据并发出,在接收端打印传入数据的IP地址和该地址写入的数据

package day23;

import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;

class UdpSend2 {

    /**
     * @param args
     * @throws Exception 
     */
    public static void main(String[] args) throws Exception {

        DatagramSocket ds = new DatagramSocket();

        BufferedReader bufr = new BufferedReader(new InputStreamReader(System.in));

        String line  = null;
        while((line = bufr.readLine())!=null){
            if("886".equals(line))
                break;

            byte[] buf = line.getBytes();

            DatagramPacket dp = new DatagramPacket(buf, buf.length,InetAddress.getByName("PC-20130712JEEW"),20000);

            ds.send(dp);
        }

        ds.close();
    }


}
class UdpRece2 {

    /**
     * @param args
     * @throws Exception 
     */
    public static void main(String[] args) throws Exception {
        DatagramSocket ds = new DatagramSocket(20000);

        while(true){

            byte[] buf = new byte[1024*64];//每个包64K

            DatagramPacket dp = new DatagramPacket(buf, buf.length);

            ds.receive(dp);

            String ip = dp.getAddress().getHostAddress();
            String data = new String(dp.getData(),0,dp.getLength());

            System.out.println(ip+":::"+data);
        }
    }

}

发送端:

这里写图片描述
接收端:

这里写图片描述

编写一个聊天程序

有收数据的部分,和发数据的部分
两部分需要同时执行
那就需要用到多线程技术
一个线程控制收,一个线程控制发

因为收和发的动作是不一致的,所以要用到两个run方法
而且这两个方法要封装到不同的类中

package day23;

import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;


class Send implements Runnable {
    private DatagramSocket ds;

    public Send(DatagramSocket ds) {
        this.ds = ds;
    }

    public void run() {
        try {
            BufferedReader bufr = new BufferedReader(new InputStreamReader(
                    System.in));
            String line = null;

            while ((line = bufr.readLine()) != null) {
                if ("886".equals(line))
                    break;
                byte[] buf = line.getBytes();

                DatagramPacket dp = new DatagramPacket(buf, buf.length,
                        InetAddress.getByName("PC-20130712JEEW"), 10006);

                ds.send(dp);
            }
        } catch (Exception e) {
            throw new RuntimeException("发送端失败");
        }
    }
}

class Rece implements Runnable {
    private DatagramSocket ds;

    public Rece(DatagramSocket ds) {
        this.ds = ds;
    }

    public void run() {
        try {
            while (true) {
                byte[] buf = new byte[1024];
                DatagramPacket dp = new DatagramPacket(buf, buf.length);

                ds.receive(dp);

                String ip = dp.getAddress().getHostAddress();
                String data = new String(dp.getData(), 0, dp.getLength());

                System.out.println(ip + ":::" + data);
            }

        } catch (Exception e) {
            throw new RuntimeException("接收端失败");
        }
    }

}

class ChatDemo {
    public static void main(String[] args) throws Exception {
        DatagramSocket sendSocket = new DatagramSocket();
        DatagramSocket receSocket = new DatagramSocket(10006);

        new Thread(new Send(sendSocket)).start();
        new Thread(new Rece(receSocket)).start();
    }

}

这里写图片描述

Tcp传输

这里写图片描述

演示Tcp传输
1,Tcp分客户端和服务端
2,客户端对应的对象是Socket
服务端对应的对象是ServerSocket

客户端:
通过查阅Socket对象,发现在该对象建立时,就可以去连接指定主机
因为Tcp是面向连接的,所以在建立Socket服务时,就要有服务端存在,并连接成功。
也就是说,要先建立服务端
形成通路后在该通道进行数据的传输

步骤:
1,创建Socket服务,并指定要连接的主机和端口。

package day23;

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


class TcpClient {


    public static void main(String[] args) throws Exception {
//      创建客户端的Socket服务,指定目的主机和端口
        Socket s = new Socket("PC-20130712JEEW",10005);

//      为了发送数据,应该获取socket流中的输出流
        OutputStream out = s.getOutputStream();
        out.write("tcp数据来了".getBytes());

        s.close();
    }

}

服务端
定义端点接收数据并打印在控制台上
1,建立服务端的socket服务,通过ServerSocket建立
并监听一个窗口
2,获取连接过来的客户端对象
通过ServerSocket的accept方法完成。所以这个方法是阻塞式的
3,客户端如果发过来数据,那么服务端要使用对应的客户端对象,并获取到该客户端对象的读取流来读取发过来的数据。并打印在控制台上
4,关闭服务端

package day23;

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

class TcpServer{


    public static void main(String[] args) throws Exception {
//      建立服务端 的socket服务,并监听一个端口
        ServerSocket ss = new ServerSocket(10005);

//      通过accept方法获取连接过来的客户端对象
        Socket s = ss.accept();

        String ip = s.getInetAddress().getHostAddress();
        System.out.println(ip+"连接成功");

//      获取客户端发送过来的数据,那么使用客户端对象的读取流方法
        InputStream in = s.getInputStream();

        byte[] buf = new byte[1024];
        int len = in.read(buf);

        System.out.println(new String(buf,0,len));

        s.close();
//      只服务一次,接收完毕就关闭
        ss.close();

    }
}

这里写图片描述

演示tcp的传输的客户端和服务端的互访

需求:
客户端给服务端发送数据,服务端收到后给客户端反馈信息

客户端:
1,建立socket服务,指定要连接的主机和端口
2,获取socket流中的输出流,将数据写到该流中,通过网络发送给服务端
3,获取socket流中的输入流,将服务端反馈的数据获取到并打印
4,关闭客户端资源

package day23;

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

class TcpClient2 {
    public static void main(String[] args) throws Exception {

        Socket s = new Socket("PC-20130712JEEW",10006);

        OutputStream out = s.getOutputStream();
        out.write("服务端你好".getBytes());

        InputStream in = s.getInputStream();
        byte[] buf = new byte[1024];
        int len = in.read(buf);
        System.out.println(new String(buf,0,len));

        s.close();
    }
}

服务端:

package day23;

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

class TcpServer2 {
    public static void main(String[] args)throws Exception {

        ServerSocket ss = new ServerSocket(10006);

        Socket s  = ss.accept();

        String ip = s.getInetAddress().getHostAddress();
        System.out.println(ip+"连接成功");

        InputStream in = s.getInputStream();
        byte[] buf = new byte[1024];
        int len = in.read(buf);

        System.out.println(new String(buf,0,len));

        OutputStream out = s.getOutputStream();

        out.write("收到,你也好".getBytes());

        s.close();
        ss.close();


    }
}

这里写图片描述
这里写图片描述

需求:建立一个文本转换服务器

客户端给服务端发送文本,服务端会将文本转成大写再转会给客户端。
而且客户端可以不断的进行文本转换,当客户端输入over时转化结束

分析:
客户端:既然是操作设备上的数据,那么就可以使用IO技术,并按照IO的操作规律来思考
源:键盘录入
目的:网络设备,网络输出流
而且操作的是文本数据,可以选择字符流

package day23;

import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.InputStreamReader;
import java.io.OutputStreamWriter;
import java.net.ServerSocket;
import java.net.Socket;

/*
*
*发送端步骤:
*1,建立服务
*2,获取键盘录入
*3,将数据发给服务端
*4,获取服务端返回的大写数据
*5,关闭资源
*
*
*/

class TransClient{
    public static void main(String[] args)throws Exception {

        Socket s = new Socket("PC-20130712JEEW",10007);

//      定义读取键盘数据的流对象
        BufferedReader bufr = new BufferedReader(new InputStreamReader(System.in));

//      定义目的,将数据写入到socket输出流,发给服务端
        BufferedWriter bufOut = new BufferedWriter(new OutputStreamWriter(s.getOutputStream()));

//      定义一个socket读取流。读取服务端返回的大写信息
        BufferedReader bufIn = new BufferedReader(new InputStreamReader(s.getInputStream()));



        String line = null;

        while((line=bufr.readLine())!=null){

            if("over".equals(line))
                break;
            bufOut.write(line);
            bufOut.newLine();
            bufOut.flush();

            String str = bufIn.readLine();
            System.out.println("Server:"+str);
        }

        bufr.close();
        s.close();
    }
}
//和子由渑池怀旧
//          苏轼
//人生到处知何似,应似飞鸿踏雪泥。
//泥上偶然留指爪,鸿飞那复计东西。
//老僧已死成新塔,坏壁无由见旧题。
//往日崎岖还知否,路长人困蹇驴嘶。

/**
 * 
 * 服务端:
 * 源:socket读取流
 * 目的:socket输出流
 * 
 */
class TransServer {
    public static void main(String[] args) throws Exception{

        ServerSocket ss = new ServerSocket(10007);

        Socket s = ss.accept();

        String ip = s.getInetAddress().getHostAddress();
        System.out.println(ip+"连接成功");

//      读取socket读取流中的数据
        BufferedReader bufIn = new BufferedReader(new InputStreamReader(s.getInputStream()));


//      目的:socket输出流将大写数据写入到socket输出流并发送给客户端
        BufferedWriter bufOut = new BufferedWriter(new OutputStreamWriter(s.getOutputStream()));

        String line = null;
        while((line=bufIn.readLine())!=null){
            System.out.println(line);
            bufOut.write(line.toUpperCase());
            bufOut.newLine();
            bufOut.flush();
        }

        s.close();
        ss.close();
    }
}


/**
 * 该例子出现的问题
 * 现象:客户端和服务端都在莫名的等待
 * 为什么呢?
 * 因为客户端和服务端都有阻塞式方法,这些方法没有读到结束标记就会一直等
 * 导致两端都在等待
 */

发送端
这里写图片描述

服务端
这里写图片描述

数据上传

从客户端把数据传到服务器,服务器将数据保存,并返回“上传成功”字样

package day23;

import java.io.BufferedReader;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.InputStreamReader;
import java.io.PrintWriter;
import java.net.ServerSocket;
import java.net.Socket;

class TextClient {

    public static void main(String[] args) throws Exception{

        Socket s = new Socket("192.168.1.105",10010);

        BufferedReader bufr = new BufferedReader(new FileReader("IPDemo.java"));

        PrintWriter out = new PrintWriter(s.getOutputStream(),true); 

        String line = null;

        while((line = bufr.readLine())!=null){
            out.println(line);
        }

//      关闭客户端输出流
        s.shutdownOutput();

        BufferedReader bufIn = new BufferedReader(new InputStreamReader(s.getInputStream()));

        String str = bufIn.readLine();
        System.out.println(str);

        bufr.close();
        s.close();
    }
}


class TextServer {
    public static void main(String[] args) throws Exception{

        ServerSocket ss = new ServerSocket(10010);

        Socket s = ss.accept();

        String ip = s.getInetAddress().getHostAddress();
        System.out.println(ip+"连接成功");

        BufferedReader bufIn = new BufferedReader(new InputStreamReader(s.getInputStream()));

        PrintWriter out = new PrintWriter(new FileWriter("Server.txt",true));

        String line = null;

        while((line = bufIn.readLine())!=null){

            out.println(line);
        }

//      关闭服务端输入流
        s.shutdownInput();

        PrintWriter pw = new PrintWriter(s.getOutputStream(),true);
        pw.println("上传成功");

        out.close();
        s.close();
        ss.close();

    }
}

运行结果在文件夹中,已经复制成功。(因为在本机,所以相当于复制)

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值