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-10-05 关于对话的问题
ckdxc



我使用循环 来监听的 和 发出请求. 但是不知道 开始 两端都没有反应在 等待发送请求 . 但是发送完了 发送的一端 就不停的 循环报错 java.net.SocketException: Socket is closed at java.net.Socket.getOutputStream(Socket.java:943) at com.ckdxc.sockettest.ReqThreader.run(ReqThreader.java:36) at java.lang.Thread.run(Thread.java:748) 接收的一端 能正常接收
public class ListenThreader implements Runnable{
    ServerSocket ss = null;
    public ListenThreader(ServerSocket ss) {
        this.ss = ss;
    }
    @Override
    public void run() {
        Socket s = null;
        InputStream is = null;
        DataInputStream dis = null;
        while (true) {
            try {
                s = ss.accept();
                is = s.getInputStream();
                dis = new DataInputStream(is);
                System.out.println("接收到信息: " + dis.readUTF());
            } catch (IOException e) {
                e.printStackTrace();
            } finally {
                try {
                    if (null != dis) dis.close();
                    if (null != is) is.close();
                    if (null != s) s.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}


/**
 * 发出请求的线程类
 */
public class ReqThreader implements Runnable{
    Socket s = null;
    String ipstr = null;
    int port;
    public ReqThreader(String ipstr, int port) {
        this.ipstr = ipstr;
        this.port = port;
    }
    @Override
    public void run() {
        boolean flag = true;
        while (flag) {
            try {
                s = new Socket(ipstr, port);
                flag = !s.isConnected();
            } catch (IOException e) {
                //e.printStackTrace();   这里是一端开启 的  异常
            }
        }
        OutputStream os = null;
        DataOutputStream dos = null;
        Scanner scanner = new Scanner(System.in);
        while (true) {
            try {
                os = s.getOutputStream();
                dos = new DataOutputStream(os);
                String msg = scanner.next();
                dos.writeUTF(msg);
                System.out.println("发出信息: " + msg);
            } catch (IOException e) {
                e.printStackTrace();
            }finally {
                try {
                    if(null!=dos) dos.close();
                    if(null!=os) os.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}
public class Client01 {
    public static void main(String[] args){
        try {
            ServerSocket ss = new ServerSocket(8889);
            ListenThreader listenThreader = new ListenThreader(ss);
            ReqThreader reqThreader = new ReqThreader("127.0.0.1",8888);
            new Thread(listenThreader).start();
            new Thread(reqThreader).start();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}
public class Server01 {
    public static void main(String[] args) {
        try {
            ServerSocket ss = new ServerSocket(8888);
            ListenThreader listenThreader = new ListenThreader(ss);
            ReqThreader reqThreader = new ReqThreader("127.0.0.1", 8889);
            new Thread(listenThreader).start();
            new Thread(reqThreader).start();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}
异常一:
java.net.SocketException: Socket is closed
	at java.net.Socket.getOutputStream(Socket.java:943)
	at com.ckdxc.sockettest.ReqThreader.run(ReqThreader.java:36)
	at java.lang.Thread.run(Thread.java:748)
单单打开一端:
java.net.ConnectException: Connection refused: connect
	at java.net.DualStackPlainSocketImpl.connect0(Native Method)
	at java.net.DualStackPlainSocketImpl.socketConnect(DualStackPlainSocketImpl.java:79)
	at java.net.AbstractPlainSocketImpl.doConnect(AbstractPlainSocketImpl.java:350)
	at java.net.AbstractPlainSocketImpl.connectToAddress(AbstractPlainSocketImpl.java:206)
	at java.net.AbstractPlainSocketImpl.connect(AbstractPlainSocketImpl.java:188)
	at java.net.PlainSocketImpl.connect(PlainSocketImpl.java:172)
	at java.net.SocksSocketImpl.connect(SocksSocketImpl.java:392)
	at java.net.Socket.connect(Socket.java:589)
	at java.net.Socket.connect(Socket.java:538)
	at java.net.Socket.<init>(Socket.java:434)
	at java.net.Socket.<init>(Socket.java:211)
	at com.ckdxc.sockettest.ReqThreader.run(ReqThreader.java:25)
	at java.lang.Thread.run(Thread.java:748)


1 个答案

10517056 答案时间:2018-11-01
这个根本不用多线程就能实现的啊,你写多线程不说,两个主方法闹哪样,各个线程都要运行几次?,按题目逻辑,不应该是: 1-(客户端)启动线程a,在a线程中发送一条消息给服务端,并等待服务端的反馈,根据反馈打印该值 2-(服务端)启动线程b,在b线程中等待客户端发来的消息,根据该消息返回另一条值给客户端 这个用循环就可以完成了,而且端口只用一个就行,端口建立通信连接后,可以发送消息,也可以接受消息,不用两边都监听




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





2018-09-25 关于socket连接方式
叫我乌蝇哥
这个socket应该是基于TCP协议的socket连接吧,不是UDP连接吧?








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




2018-08-10 soc和ser是否相同呢?
2018-05-16 没看答案 积分还一直少 为什么
2018-04-22 关于客户端和服务端同时读,造成的死锁问题?
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群交流: 902680467
提问尽量提供完整的代码,环境描述,越是有利于问题的重现,您的问题越能更快得到解答。
对教程中代码有疑问,请提供是哪个步骤,哪一行有疑问,这样便于快速定位问题,提高问题得到解答的速度
在已经存在的几千个提问里,有相当大的比例,是因为使用了和站长不同版本的开发环境导致的,比如 jdk, eclpise, idea, mysql,tomcat 等等软件的版本不一致。
请使用和站长一样的版本,可以节约自己大量的学习时间。 站长把教学中用的软件版本整理了,都统一放在了这里, 方便大家下载: http://how2j.cn/k/helloworld/helloworld-version/1718.html

上传截图