很简单的教程哦!

1.socket 简介

Socket 又称"套接字",应用程序通常通过"套接字"向网络发出请求或者应答网络请求。ServerSocket 用于服务器端,Socket 是建立网络连接时使用的。在连接成功时,应用程序两端都会产生一个 Socket 实例,操作这个实例,完成所需的会话。

2.ServerSocket 的建立与使用

最简单的建立服务器ServerSocket

public class MyServerSocket {

    public static void main(String[] args) {
try {
//1-65535 监听12345端口
ServerSocket serverSocket = new ServerSocket(12345);
//监听客户端链接,调用accept()方法 accept方法是一个阻塞的方法,会阻塞当前线程
Socket socket = serverSocket.accept();
//客户端有请求时,弹出提示框
JOptionPane.showMessageDialog(null, "有客户链接到了本机的12345端口");
} catch (IOException e) {
e.printStackTrace();
}
}
}

运行该代码,会发现程序阻塞在serverSocket.accept()处,打开浏览器请求127.0.0.1:12345  会有弹窗,程序结束

3.使用 ServerSocket 建立聊天服务器-1

上述方法是不合理的,因为accept()方法会造成程序阻塞,这样,主线程就会被阻塞,对于阻塞的代码,需要放置到独立线程中,修改如下

public class MyServerSocket {
public static void main(String[] args) {
new ServerListener().start();
}
}

监听链接的线程

/**
* 监听连接的线程
*/
public class ServerListener extends Thread{
@Override
public void run() {
try {
//1-65535 监听12345端口
ServerSocket serverSocket = new ServerSocket(12345);
//监听客户端链接,调用accept()方法 accept方法是一个阻塞的方法,会阻塞当前线程 //每当有一个客户端连接到当前的serversocket就会返回一个新的socket对象,所以当有多个的时候
//就要创建一个while循环来监听来自客户端的链接
while (true) {//true,让他一直处于循环,不会结束
Socket socket = serverSocket.accept();
//客户端有请求时,弹出提示框
JOptionPane.showMessageDialog(null, "有客户链接到了本机的12345端口");
//由于每个socket又要与独立的客户端进行通讯,所以将socket传递给新的线程
new ChatSocket(socket).start();
}
} catch (IOException e) {
e.printStackTrace();
}
} }

用于通讯的线程

/**
* 创建用于socket通信的线程
*/
public class ChatSocket extends Thread { Socket socket;//本地需要有socket来接受传入的s值 public ChatSocket(Socket s){
this.socket=s;
} public void out(String out){
try {
// 执行数据的输出和相关功能的包装
socket.getOutputStream().write(out.getBytes("UTF-8"));
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
} @Override
public void run() {
int count = 0;
while (true) {
count++;
out("loop:"+count);
try {
sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}

此时一个线程版的socket通信就算结束了,打开cmd 输入telnet localhost 12345会有惊喜哦

这种惊喜('telnet' 不是内部或外部命令,也不是可运行的程序)的自行百度

此时可以同时打开多个cmd 效果相同

4.使用 ServerSocket 建立聊天服务器-2

以上是一个简单的socket聊天服务器,但是当前的服务器只有向客户端发送数据的功能,并没有从客户端读取数据的功能!

每个ChatSocket线程都是独立的,不能相互共同数据,建立ChatManager沟通所有数据,代码修改如下:

public class MyServerSocket {
public static void main(String[] args) {
new ServerListener().start();
}
}

监听链接的线程

/**
* 监听连接的线程
*/
public class ServerListener extends Thread{
@Override
public void run() {
try {
//1-65535 监听12345端口
ServerSocket serverSocket = new ServerSocket(12345);
//监听客户端链接,调用accept()方法 accept方法是一个阻塞的方法,会阻塞当前线程 //每当有一个客户端连接到当前的serversocket就会返回一个新的socket对象,所以当有多个的时候
//就要创建一个while循环来监听来自客户端的链接
while (true) {//true,让他一直处于循环,不会结束
Socket socket = serverSocket.accept();
//客户端有请求时,弹出提示框
JOptionPane.showMessageDialog(null, "有客户链接到了本机的12345端口");
//由于每个socket又要与独立的客户端进行通讯,所以将socket传递给新的线程
ChatSocket cs = new ChatSocket(socket);
cs.start();
ChatManager.getChatManager().add(cs);
}
} catch (IOException e) {
e.printStackTrace();
}
} }

创建用于socket通信的线程

/**
* 创建用于socket通信的线程
*/
public class ChatSocket extends Thread { Socket socket;//本地需要有socket来接受传入的s值 public ChatSocket(Socket s){
this.socket=s;
} public void out(String out){
try {
// 执行数据的输出和相关功能的包装
socket.getOutputStream().write(out.getBytes("UTF-8"));
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
} @Override
public void run() {//run方法中加入接收数据的功能
try {
BufferedReader br = new BufferedReader(
new InputStreamReader(
socket.getInputStream(),"UTF-8"));
String line = null;
while ((line = br.readLine())!=null) {//客户端的数据
//发给聊天室的所有人
ChatManager.getChatManager().publish(this, line);
}
br.close();
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
}

聊天管理类

/**
* 聊天管理类
*/
public class ChatManager { //单例化该ChatManager
private ChatManager(){}
private static final ChatManager cm = new ChatManager();
public static ChatManager getChatManager(){
return cm;
}
Vector<ChatSocket> vector = new Vector<ChatSocket>(); public void add(ChatSocket cs){
vector.add(cs);
} public void publish(ChatSocket cs,String out){
for(int i=0;i<vector.size();i++){
ChatSocket cschatSocket = vector.get(i);
if(!cs.equals(cschatSocket)){//发送消息的对象不接受消息本身
cschatSocket.out(out);
}
}
}
}

打开多个cmd,实现聊天功能,无论在哪个终端发,都能接收到消息

Socket 基础解析使用ServerSocket建立聊天服务器的更多相关文章

  1. 使用 ServerSocket 建立聊天服务器-1

    1.代码目录 2.ChatSocket.java --------------------------------------------------------------------------- ...

  2. 使用ServerSocket建立聊天服务器(一)

    -------------siwuxie095                             工程名:TestMyServerSocket 包名:com.siwuxie095.socket ...

  3. 使用ServerSocket建立聊天服务器(二)

    -------------siwuxie095                         工程名:TestMyServerSocket 包名:com.siwuxie095.socket 类名:M ...

  4. 使用 ServerSocket 建立聊天服务器-2

    1. 从serverListener中可以看出,每一个客户端创建新的请求之后,都会把它分配给一个独立的chatsocket ,但是每一个ChatSocket都是相互独立的,他们之间并不能沟通,所以要新 ...

  5. 使用Java建立聊天客户端

    ---------------siwuxie095                             关于 聊天服务器,详见本人博客的分类:来一杯Java, 里面的 使用ServerSocket ...

  6. Socket Server-基于线程池的TCP服务器

    了解线程池 在http://blog.csdn.net/ns_code/article/details/14105457(读书笔记一:TCP Socket)这篇博文中,服务器端采用的实现方式是:一个客 ...

  7. 从零开始的Python学习Episode 21——socket基础

    socket基础 网络通信要素: A:IP地址   (1) 用来标识网络上一台独立的主机 (2) IP地址 = 网络地址 + 主机地址(网络号:用于识别主机所在的网络/网段.主机号:用于识别该网络中的 ...

  8. Python学习 :socket基础

    socket基础 什么是socket? - socket为接口通道,内部封装了IP地址.端口.协议等信息:我们可以看作是以前的通过电话机拨号上网的年代,socket即为电话线 socket通信流程 我 ...

  9. 【Java TCP/IP Socket】基于线程池的TCP服务器(含代码)

    了解线程池 在http://blog.csdn.net/ns_code/article/details/14105457(读书笔记一:TCP Socket)这篇博文中,服务器端采用的实现方式是:一个客 ...

随机推荐

  1. PHP中的一个很好用的文件上传类

    <?php    class FileUpload{      private $filepath; //设置上传文件的路径   private $allowtype=array('jpg',' ...

  2. HDU1506 ( Largest Rectangle in a Histogram ) [dp]

    近期情绪太不稳定了.可能是由于在找实习这个过程碰壁了吧.第一次面试就跪了,可能是我面的是一个新公司,制度不完好,我感觉整个面试过程全然不沾编程,我面试的还是软件开发-后来我同学面试的时候.说是有一道数 ...

  3. linux系统的安装

    安装linux系统须要选择一个linux操作系统,有redhat,ubuntu,centos,这里选择centos进行linux系统的安装 首先在centos的官方站点下载镜像文件CentOS-6.5 ...

  4. block 解析 - 内存

    block结构体相应的也有一个成员引用,这样会增加对局部变量的 _para1引用,在Block销毁的时候引用就释放掉了 我们了解到了用__block修饰的变量,可以在block内部修改,__block ...

  5. openGl学习之加入颜色

    OpenGL 支持两种颜色模式:一种是 RGBA模式.一种是 颜色索引模式. 不管哪种颜色模式.计算机都必须为每个像素保存一些数据,即通过每个像素的颜色,来改变总体图形的颜色.不同的是. RGBA 模 ...

  6. Lucene站内搜索的设计思路

    正好近期部门有一个小需求需要做商品的搜索,虽然最终由于工作量等原因先做数据库搜索,我依然用刚接触的Lucene弄了一套自嗨. 首先看需求:搜索:根据商品标题和内容搜索 没错,就这么简单! 我想了想,数 ...

  7. mvc模式jsp+servel+jdbc oracle基本增删改查demo

    mvc模式jsp+servel+jdbc oracle基本增删改查demo 下载地址

  8. GDI+ 对象释放崩溃的问题

    确保在Gdiplus::GdiplusShutdown(m_gdiplusToken); 之前delete 掉GDI+的对象,例如:delete *pBitmap; 如果先Gdiplus::Gdipl ...

  9. BZOJ 1016: [JSOI2008]最小生成树计数( kruskal + dfs )

    不同最小生成树中权值相同的边数量是一定的, 而且他们对连通性的贡献是一样的.对权值相同的边放在一起(至多10), 暴搜他们有多少种方案, 然后乘法原理. ----------------------- ...

  10. BZOJ 2424: [HAOI2010]订货(最小费用最大流)

    最小费用最大流..乱搞即可 ------------------------------------------------------------------------------ #includ ...