使用 Socket(套接字)可以进行不同的程序之间的通信。
1.建立连接
- 服务端开启8888端口,并监听着,时刻等待着客户端的连接请求
- 客户端知道服务端的ip地址和监听端口号,发出请求到服务端
客户端的端口地址是系统分配的,通常都会大于1024
一旦建立了连接,服务端会得到一个新的Socket对象,该对象负责与客户端进行通信。
注意: 在开发调试的过程中,如果修改过了服务器Server代码,要关闭启动的Server,否则新的Server不能启动,因为8888端口被占用了(如下图所示)
代码实现:
服务器:
package socket2;
import java.io.*;
import java.net.*;
public class Server {
public static void main(String[] args) {
try {
//服务端打开端口8888
ServerSocket ss = new ServerSocket(8888);
//在8888端口上监听,看是否有连接请求过来
System.out.println("监听在端口号:8888");
Socket s = ss.accept();
System.out.println("有连接过来" + s);
s.close();
ss.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
客户端:
package socket2;
import java.io.*;
import java.net.*;
public class Client {
public static void main(String[] args) {
try {
Socket s = new Socket("127.0.0.1",8888);
System.out.println(s);
s.close();
}catch (UnknownHostException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
}
先运行客户端,再运行服务端才有效,因为服务端需要接受到来自客户端的邀请才能执行后面步骤。
服务端输出:
监听在端口号:8888
有连接过来Socket[addr=/127.0.0.1,port=57555,localport=8888]
客户端输出:
Socket[addr=/127.0.0.1,port=8888,localport=57555]
localport = 8888是服务端提供的端口,port = 57555是客户端系统匹配的端口地址。
当然上面的这些只是实现了连接,连接了之后如何通信才是最重要的,下面来介绍各种通信方式。
2.收发数字
一旦建立了连接,服务端和客户端就可以通过Socket进行通信了
- 客户端打开输出流,并发送数字 111
- 服务端打开输入流,接受数字 111,并打印
代码如下:
服务端:
package socket2;
import java.io.*;
import java.net.*;
public class Server1 {
public static void main(String[] args){
try {
ServerSocket ss = new ServerSocket(8888);
System.out.println("监听在端口号:8888");
Socket s = ss.accept();
//打开输入流
InputStream is = s.getInputStream();
//读取客户端发来的数据
int msg = is.read();
System.out.println(msg);
is.close();
s.close();
ss.close();
}catch (IOException e) {
e.printStackTrace();
}
}
}
客户端:
package socket2;
import java.io.*;
import java.net.*;
public class Client1 {
public static void main(String[] args) {
try {
Socket s = new Socket("127.0.0.1",8888);
//打开输出流
OutputStream os = s.getOutputStream();
os.write(111);
os.close();
s.close();
}catch(UnknownHostException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
}
结果:
服务端输出:
监听在端口号:8888
111
通信的过程中不可能只有数字,内容也可以包括字符串,下面来介绍如何来收发字符串。
3.收发字符串
直接使用字节流收发字符串比较麻烦,使用数据流对字节流进行封装,这样收发字符串就容易了
- 把输出流封装在DataOutputStream中
使用writeUTF发送字符串 “abc” - 把输入流封装在DataInputStream
使用readUTF读取字符串,并打印
代码如下:
服务端:
package socket2;
import java.io.*;
import java.net.*;
public class Server2 {
public static void main(String[] args) {
try {
ServerSocket ss = new ServerSocket(8888);
Socket s = ss.accept();
InputStream is = s.getInputStream();
//把输入流封装在DataInputStream
DataInputStream dis = new DataInputStream(is);
使用readUTF读取字符串
String msg = dis.readUTF();
System.out.println(msg);
dis.close();
s.close();
ss.close();
}catch (IOException e) {
e.printStackTrace();
}
}
}
客户端:
package socket2;
import java.net.*;
import java.io.*;
public class Client2 {
public static void main(String[] args) {
try {
Socket s = new Socket("127.0.0.1",8888);
OutputStream os = s.getOutputStream();
//把输出流封装在DataOutputStream中
DataOutputStream dos = new DataOutputStream(os);
//使用writeUTF发送字符串
dos.writeUTF("abc");
dos.close();
s.close();
}catch (UnknownHostException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
}
服务端输出:
abc
4.使用Scanner
为了使得交流更灵活可以使用Scanner直接输入进行通信。
代码如下:
服务端:
package socket2;
import java.net.*;
import java.io.*;
public class Server3 {
public static void main(String[] args) {
try {
ServerSocket ss = new ServerSocket(8888);
Socket s = ss.accept();
InputStream is = s.getInputStream();
//把输入流封装在DataInputStream
DataInputStream dis = new DataInputStream(is);
使用readUTF读取字符串
String msg = dis.readUTF();
System.out.println(msg);
dis.close();
s.close();
ss.close();
}catch (IOException e) {
e.printStackTrace();
}
}
}
客户端:
package socket2;
import java.net.*;
import java.io.*;
import java.util.*;
public class Client3 {
public static void main(String[] args) {
try {
Socket s = new Socket("127.0.0.1", 8888);
OutputStream os = s.getOutputStream();
DataOutputStream dos = new DataOutputStream(os);
//使用Scanner读取控制台的输入,并发送到服务端
Scanner sc = new Scanner(System.in);
String str = sc.next();
dos.writeUTF(str);
sc.close();
dos.close();
s.close();
} catch (UnknownHostException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
客户端输入:
123ab
服务端输出:
123ab
5.练习
前面部分的学习效果是服务端接受数据,客户端发送数据。
做相应的改动,使得服务端也能发送数据,客户端也能接受数据,并且可以一直持续下去
代码如下:
服务端:
package socket2;
import java.io.*;
import java.net.*;
import java.util.*;
public class Server4{
public static void main(String[] args) {
try {
ServerSocket ss = new ServerSocket(8888);
System.out.println("监听在端口号:8888");
Socket s = ss.accept();
InputStream is = s.getInputStream();
DataInputStream dis = new DataInputStream(is);
OutputStream os = s.getOutputStream();
DataOutputStream dos = new DataOutputStream(os);
while (true) {
String msg = dis.readUTF();
System.out.println("收到客户端信息"+msg);
Scanner sc = new Scanner(System.in);
String str = sc.next();
dos.writeUTF(str);
}
} catch (IOException e) {
e.printStackTrace();
}
}
}
客户端:
package socket2;
import java.io.*;
import java.net.*;
import java.util.*;
public class Client4 {
public static void main(String[] args) {
try {
Socket s = new Socket("127.0.0.1", 8888);
OutputStream os = s.getOutputStream();
DataOutputStream dos = new DataOutputStream(os);
InputStream is = s.getInputStream();
DataInputStream dis = new DataInputStream(is);
while(true){
Scanner sc = new Scanner(System.in);
String str = sc.next();
dos.writeUTF(str);
String msg = dis.readUTF();
System.out.println("收到服务端信息"+msg);
}
} catch (UnknownHostException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
}