点我跳过黑哥的卑鄙广告行为,进入正文。

Java多线程系列更新中~

  正式篇:

  1. Java多线程(一) 什么是线程
  2. Java多线程(二)关于多线程的CPU密集型和IO密集型这件事
  3. Java多线程(三)如何创建线程
  4. Java多线程(四)java中的Sleep方法
  5. Java多线程(五)线程的生命周期

  番外篇(神TM番外篇):

  1. Java 过一下基础
  2. 转载:java中Thread.sleep()函数使用
  3. Java多线程 Socket使用

Socket是客户端和服务器端交流的端点。通过它可以实现客户端和服务器端的通信。顺着这篇文章操作下来,你应该会对Socket编程有比较系统的了解。注意,跟着敲代码,有什么不懂而我又没提到的地方,随时百度,不要积累误差。

客户端编程

Socket是有连接的,所以双方建立通信,需要知道彼此的网络位置(IP和端口)。

创建Socket对象:

Socket socket = new Socket("127.0.0.1",8852);
  • 第一个参数:服务器的IP地址(这里用的是本机地址)
  • 第二个参数:TCP端口号,其中0~1023是系统保留端口。

为什么需要端口号?例如你想要去找你的暗恋对象告白,你知道她的学校地址(IP地址),但是一个学校不会只有你暗恋对象一个人(一个主机不会只进行一个任务),你需要知道她的宿舍号(端口号),这样才能方便告白。

使用本机既做客户端,又做服务器是个什么情况?就是你和你暗恋对象在同一个学校,这没什么区别,你仍然需要她的宿舍号(端口号)才能找到她告白(客户端和服务前通信)。

初始版本代码

public class MyClient {
public static void main(String[] args) throws Exception{
Socket socket = new Socket("127.0.0.1",8852);
DataOutputStream dataOutputStream = new DataOutputStream(socket.getOutputStream());
dataOutputStream.writeUTF("Hello socket");
socket.close();
}
}

注意别忘了抛出异常,这里只是个演示作用,直接throws Exception了。这里你直接运行肯定是不能建立连接的,因为服务器端还没写。

这里,假如你知道服务器监听的是8852端口,你创建了一个socket来和它进行通信。socket.getOutputStream()得到这个socket的输出流,当你调用writeUTF()方法时,便会向其中写一个字符串。

典型三段式要牢记:打开、通信、关闭。

服务器端编程

注意了注意了,要开始造女票了。

初始版本代码

import java.io.DataInputStream;
import java.net.ServerSocket;
import java.net.Socket; public class MyServer {
public static void main(String[] args) throws Exception{
ServerSocket serverSocket = new ServerSocket(8852);
System.out.println("Server 等待接收数据~");
Socket socket = serverSocket.accept();
DataInputStream dataInputStream = new DataInputStream(socket.getInputStream());
String string = dataInputStream.readUTF();
System.out.println(string);
System.out.println("Server 接收数据完毕,拜拜了您~");
socket.close();
serverSocket.close();
}
}

这里服务器端用到了ServerSocket,负责接收客户端的连接请求。你的暗恋对象可能有好多人要向她告白,一个Socket自然无法处理。通过ServerSocket的accept方法从连接请求队列中取出一个进行连接。那这个ServerSocket就是你暗恋对象的室友,帮助她管理诸多“告白”。

这里只是把服务器端接收到的数据打印了一下便关闭了。至此,一个简单的服务器-客户端通信便建立了。但是这个服务器只能处理一个连接便关闭了,我们可以改进一下。

往下看之前请确保你大致理解了上述过程,最好自己敲一下代码

到这里你可能会想,只处理一个就关闭了,那我加个while循环,让它一直运行怎么样?

听着还可以,那我们来试验一下(下面只是分析一下,不用跟着敲)。

将服务器端代码更改为:

while(true)
{
  Socket socket = serverSocket.accept();
  DataInputStream dataInputStream = new DataInputStream(socket.getInputStream());
  String string = dataInputStream.readUTF();
  System.out.println(string);
  socket.close();
  serverSocket.close();
}

这个会出现什么问题?客户端正连着呢,啪,你服务器给人家断开了,而且accept()接受的是socket请求,所以这种情况,除非客户端再连一次,否则二者的通信就断了。

你可能会说,那我把Socket移到外面呢?

其实吧,这个仔细理一下就会明白,如果放到外面的话,只要你想接收多个socket,就要把

Socket socket = serverSocket.accept();放到while循环中,但是你只要把它放到while循环中,它就只能和一个socket进行通信。这里有兴趣可以再查阅一下资料,下面直接进入线程阶段。

服务器端线程代码

import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.IOException;
import java.net.ServerSocket;
import java.net.Socket; public class MyServer { public static void main(String[] args) throws Exception{
ServerSocket serverSocket = new ServerSocket(8852);
System.out.println("Server 等待接收数据~");
while(true) {
Socket socket = serverSocket.accept();
       //这里来一个连接请求就让deal去处理,自己在这里等下一个连接请求
deal(socket);
}
} public static void deal(Socket client){
     //这里创建了一个新的线程
new Thread(new Runnable() { @Override
public void run() {
DataInputStream dataInputStream=null;
DataOutputStream dataOutputStream = null;
String receiveFromClient="";
try {
dataInputStream = new DataInputStream(client.getInputStream());
dataOutputStream = new DataOutputStream(client.getOutputStream());
//只是这个线程中一直while循环,不影响其他的线程。
while(true) {
receiveFromClient = dataInputStream.readUTF();
System.out.println("receive: "+receiveFromClient);
if(receiveFromClient.equals("bye"))
{
dataOutputStream.writeUTF("终于厌倦我了吗?客户端都是大猪蹄子!!");
dataOutputStream.flush();
break;
}
else {
dataOutputStream.writeUTF("你TM给我发 "+receiveFromClient+" 干什么?");
dataOutputStream.flush();
}
}
}catch (Exception e) {
e.printStackTrace();
}finally {
try {
dataInputStream.close();
dataOutputStream.close();
client.close();
} catch (IOException e) {
e.printStackTrace();
}
} }
}).start();
}
}

客户端代码:

import java.io.BufferedReader;
import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.InputStreamReader;
import java.net.Socket; public class MyClient {
public static void main(String[] args) throws Exception{
Socket socket = new Socket("127.0.0.1",8852);
DataOutputStream dataOutputStream = new DataOutputStream(socket.getOutputStream());
DataInputStream dataInputStream = new DataInputStream(socket.getInputStream());
BufferedReader bReader = new BufferedReader(new InputStreamReader(System.in));
String string;
while(true)
{
string=bReader.readLine();
dataOutputStream.writeUTF(string);
System.out.println(dataInputStream.readUTF());
dataOutputStream.flush();
if(string.equals("bye"))
break;
}
socket.close();
}
}

结果展示:

服务器端:

客户端:

Java多线程 Socket使用的更多相关文章

  1. Java多线程Socket在控制台输出的多人聊天室编程

    服务器端代码 import java.io.IOException; import java.net.ServerSocket; import java.net.Socket; import java ...

  2. Java 多线程Socket编程通讯--实现聊天室代码

    1.创建服务器类 import java.io.IOException; import java.net.ServerSocket; import java.net.Socket; import ja ...

  3. Java 多线程 socket 取款例子 runnable callable

    socket部分参考 http://blog.csdn.net/kongxx/article/details/7259465 取款部分参考 http://blog.csdn.net/dayday198 ...

  4. Java多线程(五)线程的生命周期

    点我跳过黑哥的卑鄙广告行为,进入正文. Java多线程系列更新中~ 正式篇: Java多线程(一) 什么是线程 Java多线程(二)关于多线程的CPU密集型和IO密集型这件事 Java多线程(三)如何 ...

  5. Java多线程(四)java中的Sleep方法

    点我跳过黑哥的卑鄙广告行为,进入正文. Java多线程系列更新中~ 正式篇: Java多线程(一) 什么是线程 Java多线程(二)关于多线程的CPU密集型和IO密集型这件事 Java多线程(三)如何 ...

  6. Java多线程(三)如何创建线程

    点我跳过黑哥的卑鄙广告行为,进入正文. Java多线程系列更新中~ 正式篇: Java多线程(一) 什么是线程 Java多线程(二)关于多线程的CPU密集型和IO密集型这件事 Java多线程(三)如何 ...

  7. Java多线程(一) 什么是线程

    声明:本系列大多是翻译自https://www.javatpoint.com,加上自己的增删改,尽力写的系统而通俗易懂,后文不再重复声明. 点我跳过黑哥的卑鄙广告行为,进入正文. Java多线程系列更 ...

  8. Java多线程(二)关于多线程的CPU密集型和IO密集型这件事

    点我跳过黑哥的卑鄙广告行为,进入正文. Java多线程系列更新中~ 正式篇: Java多线程(一) 什么是线程 Java多线程(二)关于多线程的CPU密集型和IO密集型这件事 Java多线程(三)如何 ...

  9. 简单通过java的socket&serversocket以及多线程技术实现多客户端的数据的传输,并将数据写入hbase中

    业务需求说明,由于公司数据中心处于刚开始部署的阶段,这需要涉及其它部分将数据全部汇总到数据中心,这实现的方式是同上传json文件,通过采用socket&serversocket实现传输. 其中 ...

随机推荐

  1. Javascript 定时器调用传递参数的方法

    文章来源:  https://m.jb51.net/article/20880.htm 备注:先记下,以后整理: Javascript 定时器调用传递参数的方法,需要的朋友可以参考下. 无论是wind ...

  2. 29.C++- 异常处理

    C++内置了异常处理的语法元素 try catch try语句处理正常代码逻辑 当try语句发现异常时,则通过throw语句抛出异常,并退出try语句 catch语句处理异常情况 当throw语句抛出 ...

  3. 【转】AtomicReference与volatile的区别

    来源:AtomicReference与volatile的区别 AtomicReference与volatile的在可见性上的意义是一致的. volatile不能保证原子性,AutomicReferen ...

  4. python文件

    目录 1. 文件的概念 1.1 文件的概念和作用 1.2 文件的存储方式 2. 文件的基本操作 2.1 操作文件的套路 2.2 操作文件的函数/方法 2.3 read 方法 -- 读取文件 2.4 打 ...

  5. Android Studio 学习(四) 数据库

    文件存储 写数据 String data = "Data ti save"; FileOutputStream out =null; BufferedWriter writer = ...

  6. CSS选择器:子选择符号

    <html> <head> <style type="text/css"> .class>h2{color:red} </style ...

  7. oppor9手机怎么录制屏幕视频

    我们已经进入互联网时代,每个人都寸步不离手机.电脑等电子产品,看到美丽好看的视频总想记录下来,毕竟看到喜欢的视频还真不太容易,所以问题来了,oppor9手机怎么录制屏幕视频呢?安卓手机上怎么录制屏幕视 ...

  8. 腾讯.NET&PHP面试题

    在整个面试过程中,作为面试者的你,角色就是小怪兽,面试官的角色则是奥特曼,更不幸的是,作为小怪兽的你是孤身一人,而奥特曼却往往有好几个助攻,你总是被虐得不要不要的~ 作为复读一年才考上专科的我,遗憾的 ...

  9. python地理处理包——geopy使用之地理编码与反地理编码

    由于专业需要,经常接触一些地理处理的工具包,文档都是英文的,自己看的同时将其翻译一下,一方面自己学习的同时有个记录,要是能同时给一起的学习的童鞋们一些帮助,想想也是极好的.以下的文档内容主要翻译自官方 ...

  10. 教你分分钟搞定Docker私有仓库Registry

    一.什么是Docker私有仓库Registry 官方的Docker hub是一个用于管理公共镜像的好地方,我们可以在上面找到我们想要的镜像,也可以把我们自己的镜像推送上去.但是,有时候我们的服务器无法 ...