how2j.cn


工具版本兼容问题
使用 Socket(套接字)进行不同的程序之间的通信

本视频是解读性视频,所以希望您已经看过了本知识点的内容,并且编写了相应的代码之后,带着疑问来观看,这样收获才多。 不建议一开始就观看视频



10分39秒
本视频采用html5方式播放,如无法正常播放,请将浏览器升级至最新版本,推荐火狐,chrome,360浏览器 如果装有迅雷,播放视频呈现直接下载状态,请调整 迅雷系统设置-基本设置-启动-监视全部浏览器 (去掉这个选项)



步骤 1 : 建立连接   
步骤 2 : 收发数字   
步骤 3 : 收发字符串   
步骤 4 : 使用Scanner   
步骤 5 : 练习-服务端和客户端互聊   
步骤 6 : 答案-服务端和客户端互聊   
步骤 7 : 练习-聊天机器人数据库   
步骤 8 : 答案-聊天机器人数据库   

步骤 1 :

建立连接

1. 服务端开启8888端口,并监听着,时刻等待着客户端的连接请求
2. 客户端知道服务端的ip地址和监听端口号,发出请求到服务端
客户端的端口地址是系统分配的,通常都会大于1024
一旦建立了连接,服务端会得到一个新的Socket对象,该对象负责与客户端进行通信。
注意: 在开发调试的过程中,如果修改过了服务器Server代码,要关闭启动的Server,否则新的Server不能启动,因为8888端口被占用了
建立连接
package socket; import java.io.IOException; import java.net.ServerSocket; import java.net.Socket; 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) { // TODO Auto-generated catch block e.printStackTrace(); } } }
package socket; import java.io.IOException; import java.net.Socket; import java.net.UnknownHostException; public class Client { public static void main(String[] args) { try { //连接到本机的8888端口 Socket s = new Socket("127.0.0.1",8888); System.out.println(s); s.close(); } catch (UnknownHostException e) { // TODO Auto-generated catch block e.printStackTrace(); } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } } }
步骤 2 :

收发数字

一旦建立了连接,服务端和客户端就可以通过Socket进行通信了
1. 客户端打开输出流,并发送数字 110
2. 服务端打开输入流,接受数字 110,并打印
package socket; import java.io.IOException; import java.io.InputStream; import java.net.ServerSocket; import java.net.Socket; public class Server { 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) { // TODO Auto-generated catch block e.printStackTrace(); } } }
package socket; import java.io.IOException; import java.io.OutputStream; import java.net.Socket; import java.net.UnknownHostException; public class Client { public static void main(String[] args) { try { Socket s = new Socket("127.0.0.1", 8888); // 打开输出流 OutputStream os = s.getOutputStream(); // 发送数字110到服务端 os.write(110); os.close(); s.close(); } catch (UnknownHostException e) { // TODO Auto-generated catch block e.printStackTrace(); } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } } }
步骤 3 :

收发字符串

直接使用字节流收发字符串比较麻烦,使用数据流对字节流进行封装,这样收发字符串就容易了
1. 把输出流封装在DataOutputStream中
使用writeUTF发送字符串 "Legendary!"
2. 把输入流封装在DataInputStream
使用readUTF读取字符串,并打印
package socket; import java.io.DataInputStream; import java.io.IOException; import java.io.InputStream; import java.net.ServerSocket; import java.net.Socket; public class Server { 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 DataInputStream dis = new DataInputStream(is); //使用readUTF读取字符串 String msg = dis.readUTF(); System.out.println(msg); dis.close(); s.close(); ss.close(); } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } } }
package socket; import java.io.DataOutputStream; import java.io.IOException; import java.io.OutputStream; import java.net.Socket; import java.net.UnknownHostException; import java.util.Scanner; public class Client { 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("Legendary!"); dos.close(); s.close(); } catch (UnknownHostException e) { // TODO Auto-generated catch block e.printStackTrace(); } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } } }
步骤 4 :

使用Scanner

在上个步骤中,每次要发不同的数据都需要修改代码
可以使用Scanner读取控制台的输入,并发送到服务端,这样每次都可以发送不同的数据了。
package socket; import java.io.DataOutputStream; import java.io.IOException; import java.io.OutputStream; import java.net.Socket; import java.net.UnknownHostException; import java.util.Scanner; public class Client { 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); dos.close(); s.close(); } catch (UnknownHostException e) { // TODO Auto-generated catch block e.printStackTrace(); } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } } }
步骤 5 :

练习-服务端和客户端互聊

Or  姿势不对,事倍功半! 点击查看做练习的正确姿势
前面部分的学习效果是服务端接受数据,客户端发送数据。

做相应的改动,使得服务端也能发送数据,客户端也能接受数据,并且可以一直持续下去
练习-服务端和客户端互聊
步骤 6 :

答案-服务端和客户端互聊

在查看答案前,尽量先自己完成,碰到问题再来查看答案,收获会更多
本视频是解读性视频,所以希望您已经看过了本答案的内容,带着疑问来观看,这样收获才多。 不建议一开始就观看视频

3分53秒 本视频采用html5方式播放,如无法正常播放,请将浏览器升级至最新版本,推荐火狐,chrome,360浏览器 如果装有迅雷,播放视频呈现直接下载状态,请调整 迅雷系统设置-基本设置-启动-监视全部浏览器 (去掉这个选项)


package socket; import java.io.DataInputStream; import java.io.DataOutputStream; import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.net.ServerSocket; import java.net.Socket; import java.util.Scanner; public class Server { 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) { // TODO Auto-generated catch block e.printStackTrace(); } } }
package socket; import java.io.DataInputStream; import java.io.DataOutputStream; import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.net.Socket; import java.net.UnknownHostException; import java.util.Scanner; public class Client { 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) { // TODO Auto-generated catch block e.printStackTrace(); } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } } }
步骤 7 :

练习-聊天机器人数据库

Or  姿势不对,事倍功半! 点击查看做练习的正确姿势
首先创建一个数据库android(android现在是手机操作系统,其本意是机器人的意思)

然后创建一个表dictionary,字段:
id int
receive varchar(100)
response varchar(100)

receive 表示受到的信息
response 表示回应的信息

在这个表里准备一些数据:
你好 -> 好你妹!
你叫什么 -> 你想泡我啊?
打你哦 -> 来啊,来打我啊,不打有点挫
等等
等等

开发一个程序,当从scanner读取到消息,发给Server服务端,服务端用这个消息到表dictionary中找到对应的相应,返回出去。 看上去就像在自动回应一样。
如果一个recieve有多条response,则随机返回一条
练习-聊天机器人数据库
步骤 8 :

答案-聊天机器人数据库

在查看答案前,尽量先自己完成,碰到问题再来查看答案,收获会更多
本视频是解读性视频,所以希望您已经看过了本答案的内容,带着疑问来观看,这样收获才多。 不建议一开始就观看视频

6分25秒 本视频采用html5方式播放,如无法正常播放,请将浏览器升级至最新版本,推荐火狐,chrome,360浏览器 如果装有迅雷,播放视频呈现直接下载状态,请调整 迅雷系统设置-基本设置-启动-监视全部浏览器 (去掉这个选项)


create database android; use android; create table dictionary( id int AUTO_INCREMENT, receive varchar(100), response varchar(100), PRIMARY KEY (id) ) DEFAULT CHARSET=utf8; insert into dictionary values(null,'你好','好你妹!'); insert into dictionary values(null,'你叫什么','你想泡我啊?'); insert into dictionary values(null,'你叫什么','同志,不约'); insert into dictionary values(null,'打你哦','来啊,来打我啊,不打有点挫');
package socket; public class Dictionary { int id; String receive ; String response; }
package socket; import java.sql.Connection; import java.sql.DriverManager; import java.sql.PreparedStatement; import java.sql.ResultSet; import java.sql.SQLException; import java.util.ArrayList; import java.util.List; import charactor.Hero; public class DictionaryDAO { public DictionaryDAO() { try { Class.forName("com.mysql.jdbc.Driver"); } catch (ClassNotFoundException e) { e.printStackTrace(); } } public Connection getConnection() throws SQLException { return DriverManager.getConnection("jdbc:mysql://127.0.0.1:3306/android?characterEncoding=UTF-8", "root", "admin"); } public List<Dictionary> query(String recieve) { List<Dictionary> ds = new ArrayList<Dictionary>(); String sql = "select * from dictionary where receive = ? "; try (Connection c = getConnection(); PreparedStatement ps = c.prepareStatement(sql);) { ps.setString(1, recieve); ResultSet rs = ps.executeQuery(); while (rs.next()) { Dictionary d = new Dictionary(); int id = rs.getInt(1); String receive = rs.getString("receive"); String response = rs.getString("response"); d.id=id; d.receive=receive; d.response=response; ds.add(d); } } catch (SQLException e) { e.printStackTrace(); } return ds; } }
package socket; import java.io.DataInputStream; import java.io.DataOutputStream; import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.net.ServerSocket; import java.net.Socket; import java.util.ArrayList; import java.util.Collections; import java.util.List; public class Server { private static List<String> cannotUnderstand= new ArrayList<>(); static{ cannotUnderstand.add("听求不懂啊"); cannotUnderstand.add("说人话"); cannotUnderstand.add("再说一遍?"); cannotUnderstand.add("大声点"); cannotUnderstand.add("老子在忙,一边玩儿去"); } 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); List<Dictionary> ds= new DictionaryDAO().query(msg); String response = null; if(ds.isEmpty()){ Collections.shuffle(cannotUnderstand); response = cannotUnderstand.get(0); } else{ Collections.shuffle(ds); response = ds.get(0).response; } dos.writeUTF(response); } } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } } }
package socket; import java.io.DataInputStream; import java.io.DataOutputStream; import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.net.Socket; import java.net.UnknownHostException; import java.util.Scanner; public class Client { 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.nextLine(); dos.writeUTF(str); String msg = dis.readUTF(); System.out.println(msg); System.out.println(); } } catch (UnknownHostException e) { // TODO Auto-generated catch block e.printStackTrace(); } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } } }


HOW2J公众号,关注后实时获知布最新的教程和优惠活动,谢谢。


问答区域    
2018-05-16 没看答案 积分还一直少 为什么
jsy
没看答案 积分还一直少 为什么




2 个答案

jona 答案时间:2018-06-03
不用积分怎么看啊

nidehaowobd 答案时间:2018-05-16
其实站长留了一个福利给我们 不用积分y也可以看答案 就看你怎么搞了




答案 或者 代码至少填写一项, 如果是自己有问题,请重新提问,否则站长有可能看不到




2018-04-22 关于客户端和服务端同时读,造成的死锁问题?
Akuma



我想用Socket实现一个功能:客户端不停向服务端写信息,服务端处理完信息,把处理后的信息写回客户端,客户端再进行读。 但是我的代码会使得Server和Client端造成同时读的问题,从而两端都被阻塞,程序无法运行。 有没有办法,当客户端写完时,我可以提醒服务端该读;当服务端写完时,提醒客户端写。类似生产消费者模式。
//服务端代码
public class Server {
	public static void main(String args[]) {
		ServerSocket ss = null;
		byte[] buf = new byte[1024];
		InputStream is = null;
		try {
			ss = new ServerSocket(10000);
			System.out.println("服务器启动");
			String ip = InetAddress.getLocalHost().getHostName();
			Socket s = ss.accept();
			BufferedOutputStream br = new BufferedOutputStream(
					s.getOutputStream());
			is = s.getInputStream();
			while (true) {
				StringBuilder sb = new StringBuilder();
				// 此处被阻塞
				is.read(buf);
                                //服务端处理客户端信息
				sb.append(ip).append(":").append(new String(buf));
                                //服务端写
				br.write(sb.toString().getBytes());
			}
		} catch (Exception e) {
			e.printStackTrace();
		} finally {
		}
	}
}


//客户端代码
public class Client {
	public static void main(String args[]) throws UnknownHostException,
			IOException, InterruptedException {
		Socket so = new Socket(InetAddress.getLocalHost(), 10000);
		byte buf[] = new byte[1024];
		String s = "xxxx";
		OutputStream os = so.getOutputStream();
		InputStream is = so.getInputStream();
		while (true) {
			os.write(s.getBytes());
			// 此处被阻塞
			is.read(buf);
			System.out.println("返回:" + new String(buf));
		}
	}
}

							


3 个答案

Aresto_nicko 答案时间:2018-05-10
分开两个对象,读对象和写对象,针对两个对象分别实现不同功能,但是这样容易死锁,所以最好用trylock,弄两个lock一个读lock一个写lock,分别trylock,针对不同trylock不同操作,客户端服务器端都要对两个lock分开操作

Aresto_nicko 答案时间:2018-05-10
上面不是写了嘛,加锁,或者synchronized一个读写,你是不是没学过操作系统的死锁问题?大学没教过?

123456SHY 答案时间:2018-05-05
另开个线程,把读写分开在两个线程里




答案 或者 代码至少填写一项, 如果是自己有问题,请重新提问,否则站长有可能看不到





2018-04-18 聊天机器人数据库while, Collections.shuffle(ds);答案有问题
2018-04-15 请问通过Socket得到的输入输出流是否能够通过try with catch关闭呢
2018-02-12 封装成BufferedReader和BufferedWriter不行么?
2018-02-09 如何在关闭时将用的端口关闭并清空,使得下一次使用不会提示已被使用的错误
2017-12-28 客户端怎么重复输出
2017-12-06 答案-服务端和客户端互聊
2017-11-15 客户端和服务端的通信
2017-11-13 运行聊天机器人报错:java.net.SocketException: Connection reset
2017-10-23 麻烦老师帮助下
2017-10-07 eclipse中开启两个控制台分别查看Server和Client的输出
2017-09-04 实例中的服务端代码和客户端代码可以写在同一个myeclipse中吗
2017-09-04 8888端口是指什么端口?如何知道的?
2017-08-15 用PrintWriter 和 OutputStream 打开输出流 结果不同
2017-05-20 运行一次后出现错误
2017-03-28 写入的数字是1110的时候,服务端接收到的是86
2016-12-14 客户端发中文到服务器怎么输出控制台是乱码
2016-04-18 代码
2016-04-11 8888端口?




提问之前请登陆
关于 JAVA 中级-网络编程-Socket 的提问

尽量提供截图代码异常信息,有助于分析和解决问题。 也可进本站QQ群交流: 620943819
提问尽量提供完整的代码,环境描述,越是有利于问题的重现,您的问题越能更快得到解答。
对教程中代码有疑问,请提供是哪个步骤,哪一行有疑问,这样便于快速定位问题,提高问题得到解答的速度
在已经存在的几千个提问里,有相当大的比例,是因为使用了和站长不同版本的开发环境导致的,比如 jdk, eclpise, idea, mysql,tomcat 等等软件的版本不一致。
请使用和站长一样的版本,可以节约自己大量的学习时间。 站长把教学中用的软件版本整理了,都统一放在了这里, 方便大家下载: http://how2j.cn/k/helloworld/helloworld-version/1718.html

上传截图