http://www.cnblogs.com/mengdd/archive/2013/03/10/2952616.html

使用TCP/IP的套接字(Socket)进行通信

套接字Socket的引入

  为了能够方便地开发网络应用软件,由美国伯克利大学在Unix上推出了一种应用程序访问通信协议的操作系统用调用socket(套接字)

  socket的出现,使程序员可以很方便地访问TCP/IP,从而开发各种网络应用的程序。

  随着Unix的应用推广,套接字在编写网络软件中得到了极大的普及。后来,套接字又被引进了Windows等操作系统中。Java语言也引入了套接字编程模型。

什么是Socket?

  Socket是连接运行在网络上的两个程序间的双向通讯的端点。

使用Socket进行网络通信的过程

  服务器程序将一个套接字绑定到一个特定的端口,并通过此套接字等待和监听客户的连接请求。

  客户程序根据服务器程序所在的主机名和端口号发出连接请求。

  如果一切正常,服务器接受连接请求。并获得一个新的绑定到不同端口地址的套接字。(不可能有两个程序同时占用一个端口)。

  客户和服务器通过读写套接字进行通讯。

  使用ServerSocketSocket实现服务器端和客户端的Socket通信。

  

  其中:

  左边ServerSocket类的构造方法可以传入一个端口值来构建对象。

  accept()方法监听向这个socket的连接并接收连接。它将会阻塞直到连接被建立好。连接建立好后它会返回一个Socket对象。

  连接建立好后,服务器端和客户端的输入流和输出流就互为彼此,即一端的输出流是另一端的输入流。

总结:使用ServerSocket和Socket实现服务器端和客户端的Socket通信

  (1)建立Socket连接

  (2)获得输入/输出流

  (3)读/写数据

  (4)关闭输入/输出流

  (5)关闭Socket

通信程序测试

  建立服务器端和客户端如下: 

 

package com.example.network;

import java.net.ServerSocket;
import java.net.Socket; public class TcpServer
{
public static void main(String[] args) throws Exception
{
// 创建服务器端的socket对象
ServerSocket ss = new ServerSocket(5000); // 监听连接
Socket socket = ss.accept();
// 直到连接建立好之后代码才会往下执行 System.out.println("Connected Successfully!"); } }
package com.example.network;

import java.net.Socket;

public class TcpClient
{
public static void main(String[] args) throws Exception
{
Socket socket = new Socket("127.0.0.1", 5000);
} }

 

  然后先运行服务器端,再运行客户端,可以看到,运行客户端之后输出服务器端的后续代码。

  表明连接建立后才会往下执行。

  一个比较简陋的通信程序:

package com.example.network;

import java.io.InputStream;
import java.io.OutputStream;
import java.net.ServerSocket;
import java.net.Socket; public class TcpServer
{
public static void main(String[] args) throws Exception
{
// 创建服务器端的socket对象
ServerSocket ss = new ServerSocket(5000); // 监听连接
Socket socket = ss.accept();
// 直到连接建立好之后代码才会往下执行 System.out.println("Connected Successfully!"); // 获得服务器端的输入流,从客户端接收信息
InputStream is = socket.getInputStream();
// 服务器端的输出流,向客户端发送信息
OutputStream os = socket.getOutputStream(); byte[] buffer = new byte[200]; int length = 0;
length = is.read(buffer);
String str = new String(buffer, 0, length);
System.out.println(str); // 服务器端的输出
os.write("Welcome".getBytes()); // 关闭资源
is.close();
os.close();
socket.close(); } }
package com.example.network;

import java.io.InputStream;
import java.io.OutputStream;
import java.net.Socket; public class TcpClient
{
public static void main(String[] args) throws Exception
{
Socket socket = new Socket("127.0.0.1", 5000); // 客户端的输出流
OutputStream os = socket.getOutputStream(); // 将信息写入流,把这个信息传递给服务器
os.write("hello world".getBytes()); // 从服务器端接收信息 InputStream is = socket.getInputStream(); byte[] buffer = new byte[200]; int length = is.read(buffer);
String str = new String(buffer, 0, length);
System.out.println(str); // 关闭资源
is.close();
os.close();
socket.close();
} }

  先运行服务器,再运行客户端。之后可以在服务器和客户端的控制台上进行输入操作,另一端将会收到输入的信息并输出。

使用线程实现服务器端与客户端的双向通信

  用两个线程,一个线程专门用于处理服务器端的读,另一个线程专门用于处理服务器端的写。

  客户端同理。

  代码如下,程序共有六个类。

  服务器端和其输入输出线程:

MainServer
ServerInputThread
ServerOutputThread

 

  客户端和其输入输出线程(其输入输出线程和服务器端的完全一样):  

MainClient
ClientInputThread
ClientOutputThread

  经测试成功。即从服务器端控制台输入,可以从客户端接收到并输出;也可以反过来,从客户端控制台输入,那么服务器端会同时输出。

多个客户端的程序实验

  可以启动多个客户端,同时与服务器进行交互。这里还是采用上面的MainServer和MainClient及其输入输出线程代码。

  这部分做实验的时候需要使用命令行,因为Eclipse里面每次Run的时候都会重新启动程序,即想要Run第二个客户端的时候总是先关闭第一个客户端(因为它们运行的是同一个程序),这样,即只能有一个客户端存在。

  在命令行运行的方法如下:

  因为源文件带有包名,所以编译采用:

  javac –d . 源文件名.java

  注意d和.之间有一个空格。

  可以使用通配符编译所有的源文件,即使用:

  javac –d . *.java

  编译之后执行:

  java 完整包名+类名

  先启动服务器程序,之后新开命令行窗口启动客户端程序,结果如下:

  

(一个客户端时交互正常)

  

  

  (多个客户端交互异常)

  

  经实验,发现在一个服务器多个客户端的情况下,客户端可以流畅地向服务器发送信息,但是当服务器发送信息时,就会出现问题,并不是每一个客户端都能收到信息。

  如图中,当服务器发送语句时,第一个客户端收到了(并且是发送后多按下一个回车才收到),第二个客户端没有收到。

  后面试验了几个语句都是这样:

  

实现服务器支持多客户机通信

  服务器端的程序需要为每一个与客户机连接的socket建立一个线程,来解决同时通信的问题。

  服务器端应该管理一个socket的集合。

  即要完成一个功能完善的客户端和服务器通信程序,代码还是需要进一步完善的。

参考资料

  圣思园张龙老师Java SE系列视频教程。

使用TCP/IP的套接字(Socket)进行通信的更多相关文章

  1. Java 网络编程(五) 使用TCP/IP的套接字(Socket)进行通信

    链接地址:http://www.cnblogs.com/mengdd/archive/2013/03/10/2952616.html 使用TCP/IP的套接字(Socket)进行通信 套接字Socke ...

  2. 网络编程(二)--TCP协议、基于tcp协议的套接字socket

    一.TCP协议(Transmission Control Protocol 传输控制协议) 1.可靠传输,TCP数据包没有长度限制,理论上可以无限长,但是为了保证网络的效率,通常TCP数据包的长度不会 ...

  3. 网络编程(二)——TCP协议、基于tcp协议的套接字socket

    TCP协议与基于tcp协议的套接字socket 一.TCP协议(流式协议) 1.可靠传输,TCP数据包没有长度限制,理论上可以无限长,但是为了保证网络的效率,通常TCP数据包的长度不会超过IP数据包的 ...

  4. TCP/IP与套接字

    以前我们讲过进程间通信,通过进程间通信可以实现同一台计算机上不同的进程之间通信. 通过网络编程可以实现在网络中的各个计算机之间的通信. 进程能够使用套接字实现和其他进程或者其他计算机通信. 同样的套接 ...

  5. Python--day30--基于tcp协议的套接字socket

    socket 一开始被设计用在一台主机上多个应用程序之间通信. 是应用层与TCP/IP协议族通信的中间软件抽象层,它是一组接口. 是一个模块,是ip+port,门面模式是一种设计模式. socket通 ...

  6. 20181225 基于TCP/IP和基于UDP/IP的套接字编程

    一.TCP/IP的套接字编程 服务器端代码: import  socket​server = socket.socket() # 默认是基于TCP# 基于TCP的对象serve=socket.sock ...

  7. python 全栈开发,Day33(tcp协议和udp协议,互联网协议与osi模型,socket概念,套接字(socket)初使用)

    先来回顾一下昨天的内容 网络编程开发架构 B/S C/S架构网卡 mac地址网段 ip地址 : 表示了一台电脑在网络中的位置 子网掩码 : ip和子网掩码按位与得到网段 网关ip : 内置在路由器中的 ...

  8. Java套接字Socket编程--TCP参数

    在Java的Socket中,主要包含了以下可设置的TCP参数. 属性 说明 默认值 SO_TIMEOUT 对ServerSocket来说表示等待连接的最长空等待时间; 对Socket来说表示读数据最长 ...

  9. 网络编程----socket介绍、基于tcp协议的套接字实现、基于udp协议的套接字实现

    一.客户端/服务器架构(C/S架构)                                                即C/S架构,包括: 1.硬件C/S架构(打印机) 2.软件C/S架 ...

随机推荐

  1. python函数应用

    函数 定义: 函数是指将一组语句的集合通过一个名字(函数名)封装起来,要想执行这个函数,只需调用其函数名即可 面向过程:根据业务逻辑从上到下写垒代码 函数式:将某功能代码封装到函数中,日后便无需重复编 ...

  2. C# for和 foreach 的数组遍历 比较

    刚学习程序,感觉写代码 很有意思,所以把自己的感悟写下来啦,第一次写博客,可能是菜鸟中的菜鸟  时间久了,相信就会写的很好哦! for和 foreach 的数组遍历 比较 很简单的程序,不解释啦! u ...

  3. V-rep学习笔记:转动关节1

    V-REP(Virtual Robot Experimentation Platform),是全球领先的机器人及模拟自动化软件平台.V-REP让使用者可以模拟整个机器人系统或其子系统(如感测器或机械结 ...

  4. error: command 'gcc' failed with exit status 1 的解决办法

    yum install gcc python-devel 之前yum install gcc* 了 所以没成功. wget http://prdownloads.sourceforge.net/doc ...

  5. vim使用札记

    最近开始用vim编辑器了,从最开始的配置到现在慢慢使用,我在这儿会贴出一些我的使用上遇到过的问题和如何解决的方案,留给自己和一些会用到的人看看 1.vim怎么使汇编语法高亮 开始不知道,然后把文件的后 ...

  6. HDU 5723 Abandoned country(落后渣国)

    HDU 5723 Abandoned country(落后渣国) Time Limit: 8000/4000 MS (Java/Others)    Memory Limit: 65536/65536 ...

  7. 小型网站如何防范DDoS攻击

    ddos(Distributed Denial of Service,分布式拒绝服务攻击),俗称洪水攻击.是在传统的DoS攻击基础之上产生的新的破坏力更强的攻击方式.分布式拒绝服务攻击是指借助于客户/ ...

  8. [转]-Android Studio 快捷键整理分享-SadieYu

    文章编辑整理:Android Studio 中文组 - SadieYu Alt+回车 导入包,自动修正 Ctrl+N   查找类 Ctrl+Shift+N 查找文件 Ctrl+Alt+L  格式化代码 ...

  9. iOS - OC NSDate 时间

    前言 NSDate @interface NSDate : NSObject <NSCopying, NSSecureCoding> NSDate 用来表示公历的 GMT 时间(格林威治时 ...

  10. elastic

    学习链接 http://rfyiamcool.blog.51cto.com/1030776/1420811?utm_source=tuicool&utm_medium=referral