本章的重点内容总结:网络协议分层思想,IP的概念,TCP/UDP的概念,TCP/UDP程序的写法。
TCP:比较慢 但可靠
UDP:比较快 但不可靠
一、通信协议的分层思想:
1.为什么要分层
由于结点之间联系很复杂,在制订协议时,把复杂成份分解成一些简单的成份,再将它们复合起来。最常用的复合方式是层次方式,即同层间可以通信,上一层可以条用下一层,而与再下一层不发生关系。各层互不影响,利于系统的开发和发展。
参考模型:
二、IP,TCP,UDP协议
IP协议:
IP(internet protocol)协议是网际层的主要协议,支持网间互联的数据报通信。主要提供的功能有:
1.无连接数据报传送
2.数据报路由选择和差错控制
TCP协议:
TCP(transmission control protocol)是专门设计用于在不可靠的因特网上提供 可靠的、端到端的字节流通信的协议。它是一种面向连接的协议。TCP连接是字节流而非报文流。
UDP协议:
UDP(user data protocol)向应用程序提供了一种发送封装的原始IP数据报的方法,并且发送时无需建立连接。是一种不可靠、效率高的连接。
三、Socket:
1.两个JAVA应用程序可通过一个双向的网络通信连接实现数据交换,这个双向链路的一端称为一个Socket。
2.Socket通常用来实现client-server连接。
3.java.net包中定义的两个类Socket和ServerSocket,分别用来实现双向连接的Client端和Server端。
4.建立连接时所需的寻址信息为远程计算机的IP地址和端口号(port number)
TCP Socket 通信模型
TCP代码举例:
TCPServer服务器端:new一个ServerSocket监听在某个端口,调用accept()方法接受一个连接,拿到输入流输出流getinputStream()/getOutputStream(),关闭close().
TCPClient客服端:new一个socket,指定本机地址和服务器的端口号练到服务器上,拿到输入流输出流,关闭colse().import java.net.*; import java.io.*; public class TCPServer { public static void main(String[] args) throws Exception { ServerSocket ss = new ServerSocket(6666); while(true) { Socket s = ss.accept(); System.out.println("a client connect!"); DataInputStream dis = new DataInputStream(s.getInputStream()); System.out.println(dis.readUTF()); dis.close(); s.close(); } } }
import java.net.*; import java.io.*; public class TCPClient { public static void main(String[] args) throws Exception { Socket s = new Socket("127.0.0.1", 6666); OutputStream os = s.getOutputStream(); DataOutputStream dos = new DataOutputStream(os); Thread.sleep(30000); dos.writeUTF("hello server!"); dos.flush(); dos.close(); s.close(); } }
UDP代码举例UDPServer端:new一个字节数组,new一个DatagramPacket,DatagramPacket接受任何东西都放到字节数组里。new一个DatagramSocket,通过调用receive()方法接收。
import java.net.*; import java.io.*; public class TestUDPServer { public static void main(String args[]) throws Exception { byte buf[] = new byte[1024]; DatagramPacket dp = new DatagramPacket(buf, buf.length); DatagramSocket ds = new DatagramSocket(5678); while(true) { ds.receive(dp); ByteArrayInputStream bais = new ByteArrayInputStream(buf); DataInputStream dis = new DataInputStream(bais); System.out.println(dis.readLong()); } } }
UDPClient端:new一个字节数组,new一个DatagramPacket,把数据转成字节数组之后装包,在包上写好地址。调用send()方法
import java.net.*; import java.io.*; public class TestUDPClient { public static void main(String args[]) throws Exception { long n = 10000L; ByteArrayOutputStream baos = new ByteArrayOutputStream(); DataOutputStream dos = new DataOutputStream(baos); dos.writeLong(n); byte[] buf = baos.toByteArray(); System.out.println(buf.length); DatagramPacket dp = new DatagramPacket(buf, buf.length,new InetSocketAddress("127.0.0.1", 5678));//每一个数据报都要指定地址 DatagramSocket ds = new DatagramSocket(9999);//客户端本身占据的端口 ds.send(dp); ds.close(); } }