`

读懂Java中的Socket编程

    博客分类:
  • Java
阅读更多

Socket,又称为套接字,Socket是计算机网络通信的基本的技术之一。如今大多数基于网络的软件,如浏览器,即时通讯工具甚至是P2P下载都是基于Socket实现的。本文会介绍一下基于TCP/IP的Socket编程,并且如何写一个客户端/服务器程序。

餐前甜点

Unix的输入输出(IO)系统遵循Open-Read-Write-Close这样的操作范本。当一个用户进程进行IO操作之前,它需要调用Open来指定并获取待操作文件或设备读取或写入的权限。一旦IO操作对象被打开,那么这个用户进程可以对这个对象进行一次或多次的读取或写入操作。Read操作用来从IO操作对象读取数据,并将数据传递给用户进程。Write操作用来将用户进程中的数据传递(写入)到IO操作对象。 当所有的Read和Write操作结束之后,用户进程需要调用Close来通知系统其完成对IO对象的使用。

在Unix开始支持进程间通信(InterProcess Communication,简称IPC)时,IPC的接口就设计得类似文件IO操作接口。在Unix中,一个进程会有一套可以进行读取写入的IO描述符。IO描述符可以是文件,设备或者是通信通道(socket套接字)。一个文件描述符由三部分组成:创建(打开socket),读取写入数据(接受和发送到socket)还有销毁(关闭socket)。

在Unix系统中,类BSD版本的IPC接口是作为TCP和UDP协议之上的一层进行实现的。消息的目的地使用socket地址来表示。一个socket地址是由网络地址和端口号组成的通信标识符。

进程间通信操作需要一对儿socket。进程间通信通过在一个进程中的一个socket与另一个进程中得另一个socket进行数据传输来完成。当一个消息执行发出后,这个消息在发送端的socket中处于排队状态,直到下层的网络协议将这些消息发送出去。当消息到达接收端的socket后,其也会处于排队状态,直到接收端的进程对这条消息进行了接收处理。

TCP和UDP通信

关于socket编程我们有两种通信协议可以进行选择。一种是数据报通信,另一种就是流通信。

数据报通信

数据报通信协议,就是我们常说的UDP(User Data Protocol 用户数据报协议)。UDP是一种无连接的协议,这就意味着我们每次发送数据报时,需要同时发送本机的socket描述符和接收端的socket描述符。因此,我们在每次通信时都需要发送额外的数据。

流通信

流通信协议,也叫做TCP(Transfer Control Protocol,传输控制协议)。和UDP不同,TCP是一种基于连接的协议。在使用流通信之前,我们必须在通信的一对儿socket之间建立连接。其中一个socket作为服务器进行监听连接请求。另一个则作为客户端进行连接请求。一旦两个socket建立好了连接,他们可以单向或双向进行数据传输。

读到这里,我们多少有这样的疑问,我们进行socket编程使用UDP还是TCP呢。选择基于何种协议的socket编程取决于你的具体的客户端-服务器端程序的应用场景。下面我们简单分析一下TCP和UDP协议的区别,或许可以帮助你更好地选择使用哪种。

在UDP中,每次发送数据报时,需要附带上本机的socket描述符和接收端的socket描述符。而由于TCP是基于连接的协议,在通信的socket对之间需要在通信之前建立连接,因此会有建立连接这一耗时存在于TCP协议的socket编程。

在UDP中,数据报数据在大小上有64KB的限制。而TCP中也不存在这样的限制。一旦TCP通信的socket对建立了连接,他们之间的通信就类似IO流,所有的数据会按照接受时的顺序读取。

UDP是一种不可靠的协议,发送的数据报不一定会按照其发送顺序被接收端的socket接受。然后TCP是一种可靠的协议。接收端收到的包的顺序和包在发送端的顺序是一致的。

简而言之,TCP适合于诸如远程登录(rlogin,telnet)和文件传输(FTP)这类的网络服务。因为这些需要传输的数据的大小不确定。而UDP相比TCP更加简单轻量一些。UDP用来实现实时性较高或者丢包不重要的一些服务。在局域网中UDP的丢包率都相对比较低。

Java中的socket编程

下面的部分我将通过一些示例讲解一下如何使用socket编写客户端和服务器端的程序。

注意:在接下来的示例中,我将使用基于TCP/IP协议的socket编程,因为这个协议远远比UDP/IP使用的要广泛。并且所有的socket相关的类都位于java.net包下,所以在我们进行socket编程时需要引入这个包。

客户端编写

开启Socket

如果在客户端,你需要写下如下的代码就可以打开一个socket。

1
2
3
String host = "127.0.0.1";
int port = 8919;
Socket client = new Socket(host, port);

上面代码中,host即客户端需要连接的机器,port就是服务器端用来监听请求的端口。在选择端口时,需要注意一点,就是0~1023这些端口都已经被系统预留了。这些端口为一些常用的服务所使用,比如邮件,FTP和HTTP。当你在编写服务器端的代码,选择端口时,请选择一个大于1023的端口。

写入数据

接下来就是写入请求数据,我们从客户端的socket对象中得到OutputStream对象,然后写入数据后。很类似文件IO的处理代码。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
public class ClientSocket {
  public static void main(String args[]) {
        String host = "127.0.0.1";
        int port = 8919;
        try {
          Socket client = new Socket(host, port);
          Writer writer = new OutputStreamWriter(client.getOutputStream());
          writer.write("Hello From Client");
          writer.flush();
          writer.close();
          client.close();
        } catch (IOException e) {
          e.printStackTrace();
        }
    }
}

关闭IO对象

类似文件IO,在读写数据完成后,我们需要对IO对象进行关闭,以确保资源的正确释放。

服务器端编写

打开服务器端的socket

1
2
3
int port = 8919;
ServerSocket server = new ServerSocket(port);
Socket socket = server.accept();

上面的代码创建了一个服务器端的socket,然后调用accept方法监听并获取客户端的请求socket。accept方法是一个阻塞方法,在服务器端与客户端之间建立联系之前会一直等待阻塞。

读取数据

通过上面得到的socket对象获取InputStream对象,然后安装文件IO一样读取数据即可。这里我们将内容打印出来。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
public class ServerClient {
  public static void main(String[] args) {
        int port = 8919;
        try {
            ServerSocket server = new ServerSocket(port);
                Socket socket = server.accept();
            Reader reader = new InputStreamReader(socket.getInputStream());
            char chars[] = new char[1024];
            int len;
            StringBuilder builder = new StringBuilder();
            while ((len=reader.read(chars)) != -1) {
               builder.append(new String(chars, 0, len));
            }
            System.out.println("Receive from client message=: " + builder);
            reader.close();
            socket.close();
            server.close();
        } catch (Exception e) {
            e.printStackTrace();
        }
  }
}

关闭IO对象

还是不能忘记的,最后需要正确地关闭IO对象,以确保资源的正确释放。

附注一个例子

这里我们增加一个例子,使用socket实现一个回声服务器,就是服务器会将客户端发送过来的数据传回给客户端。代码很简单。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
import java.io.*;
import java.net.*;
public class EchoServer {
    public static void main(String args[]) {
        // declaration section:
        // declare a server socket and a client socket for the server
        // declare an input and an output stream
        ServerSocket echoServer = null;
        String line;
        DataInputStream is;
        PrintStream os;
        Socket clientSocket = null;
        // Try to open a server socket on port 9999
        // Note that we can't choose a port less than 1023 if we are not
        // privileged users (root)
        try {
           echoServer = new ServerSocket(9999);
        }
        catch (IOException e) {
           System.out.println(e);
        }
        // Create a socket object from the ServerSocket to listen and accept 
        // connections.
        // Open input and output streams
        try {
               clientSocket = echoServer.accept();
               is = new DataInputStream(clientSocket.getInputStream());
               os = new PrintStream(clientSocket.getOutputStream());
               // As long as we receive data, echo that data back to the client.
               while (true) {
                 line = is.readLine();
                 os.println(line);
               }
        } catch (IOException e) {
               System.out.println(e);
            }
        }
}

编译运行上面的代码,进行如下请求,就可以看到客户端请求携带的数据的内容。

1
2
3
4
5
15:00 $ curl http://127.0.0.1:9999/?111
GET /?111 HTTP/1.1
User-Agent: curl/7.37.1
Host: 127.0.0.1:9999
Accept: */*

总结

进行客户端-服务器端编程还是比较有趣的,同时在Java中进行socket编程要比其他语言(如C)要简单快速编写。

java.net这个包里面包含了很多强大灵活的类供开发者进行网络编程,在进行网络编程中,建议使用这个包下面的API。同时Sun.*这个包也包含了很多的网络编程相关的类,但是不建议使用这个包下面的API,因为这个包可能会改变,另外这个包不能保证在所有的平台都有包含。

原文信息

分享到:
评论

相关推荐

    Java课程设计案例精编-源代码

    7. **网络编程**:Java提供Socket编程接口,可以实现客户端-服务器通信,案例可能包含简单的网络应用。 8. **图形用户界面(GUI)编程**:使用Swing或JavaFX库创建交互式的桌面应用程序,如按钮、文本框、菜单等...

    java实战源代码

    Java是一种广泛使用的面向对象的编程语言,以其跨平台、高性能和丰富的类库而闻名。"Java实战源代码"是学习和提升Java...在学习过程中,记得不仅要读懂代码,还要尝试自己修改和扩展,这样才能更好地吸收和巩固知识。

    自考Java语言程序设计例题大全

    Java语言程序设计是计算机科学领域中的重要组成部分...在学习过程中,不仅要读懂代码,还应尝试修改和扩展,以提升问题解决能力。同时,理论知识与实践相结合,不断巩固,才能在自考Java语言程序设计的道路上取得成功。

    飞信 java 1.0.18版源代码

    开发者需要对Java语法有深入理解,才能读懂并修改代码。 2. **多线程编程**:飞信作为实时通讯工具,可能涉及到多个并发任务,如消息接收、发送、心跳检测等,因此源码中可能会包含大量多线程编程的应用,如`Thread...

    30天学通Java项目案例例开发+源代码

    《30天学通Java项目案例开发》是一个旨在帮助初学者快速掌握Java编程语言和实践应用的教程。...记住,理论知识与实践相结合是提升技能的关键,所以不仅要读懂代码,还要动手编写,才能真正掌握Java的魅力。

    一文读懂2014年全球互联网广告新生态

    有些概念是你必须知道的,而且知道的程度就如同你知道一年...10个C#Socket编程代码示例 http://download.csdn.net/album/detail/631 6份GDI+程序设计资源整合[全零分] http://download.csdn.net/album/detail/625

    JAVA自学之路

    在这儿之后,你可以选择向J2ME、或者Java GUI、Socket编程等方向努力,但是通过第一篇的办法,你可以很容易就将这方向过滤掉(永远不要忘了我们是为了就业),好吧,暂时和它们诀别,还是向着J2EE的方向前进吧。...

    [14本经典Android开发教程]-7-Android编程入门教程

    [14本经典Android开发教程]-7-Android编程入门教程 -------------------------------------------------- ...一文读懂2014年全球互联网广告新生态 http://download.csdn.net/detail/cleopard/8340303

    Android游戏源码—3D迷宫

    Android游戏源码-3D迷宫 ...10个C#Socket编程代码示例 http://download.csdn.net/album/detail/631 6份GDI+程序设计资源整合[全零分] http://download.csdn.net/album/detail/625 2014年移动游戏行业数据分析 ...

    源程序OOXX

    理解这些概念对于读懂源码至关重要。 2. **类与对象**:在Java中,类是创建对象的蓝图,而对象是类的实例。类定义了对象的状态(数据成员)和行为(方法)。 3. **包(Package)**:Java中的包是命名空间,用来...

    TCP/IP协议与网络编程技术

    (3)考虑到TCP/IP协议比较抽象,学习起来有一定的难度,所以全书尽量避免使用晦涩难懂专业术语,而用浅显易懂的语言说明问题,努力将书打造成一本人人都读懂书籍。 (4)初学网络程序设计的人员,往往感到网络程序...

    [14本经典Android开发教程]-2-Android开发手册—API函数详解

    一文读懂2014年全球互联网广告新生态 http: download csdn net detail cleopard 8340303">[14本经典Android开发教程] 2 Android开发手册 API函数详解 一 TextView的API 1 1 结构 java lang Object android view ...

    cocos2d-x多平台高仿微信打飞机

    cocos2d x多平台高仿微信打...10个C#Socket编程代码示例 http: download csdn net album detail 631 6份GDI+程序设计资源整合[全零分] http: download csdn net album detail 625 2014年移动游戏行业数据分析 ...

    COCOS2DX 3.2微信打飞机可执行源码

    COCOS2DX 3 2微信打飞机可...10个C#Socket编程代码示例 http: download csdn net album detail 631 6份GDI+程序设计资源整合[全零分] http: download csdn net album detail 625 2014年移动游戏行业数据分析 ...

    最透彻的的Netty高性能原理和框架架构解析,使用文件和图形详细描述了netty的原理

    * 《高性能网络编程(五):一文读懂高性能网络编程中的 I/O 模型》 * 《高性能网络编程(六):一文读懂高性能网络编程中的线程模型》 * 《有关“为何选择 Netty”的 11 个疑问及解答》 * 《开源 NIO 框架八卦——到底...

    C++贪吃蛇游戏源码

    C++贪吃蛇游戏源码 @或直接从这里寻找@ ... @更多@ ... 福利 ...17份软件测试文档 ...10个[精品资源]Java学习资料合辑[一] ...一文读懂2014年全球互联网广告新生态 http://download.csdn.net/detail/cleopard/8340303

    C++拼图游戏源码

    C++拼图游戏源码 @或直接从这里寻找@ ... @更多@ ... ...10个[精品资源]Java学习资料合辑[一] ...10个C#Socket编程代码示例 ...一文读懂2014年全球互联网广告新生态 http://download.csdn.net/detail/cleopard/8340303

    C++扫雷游戏源码

    经典游戏-扫雷游戏C++源码 @或直接从这里寻找@ ... @更多@ ... 福利 ...17份软件测试文档 ...10个[精品资源]Java学习资料合辑[一] ...一文读懂2014年全球互联网广告新生态 http://download.csdn.net/detail/cleopard/8340303

    2014年移动游戏行业数据分析

    一文读懂2014年全球互联网广告新生态 http://download.csdn.net/detail/cleopard/8340303 @更多@ http://cleopard.download.csdn.net/ 13份WPF经典开发教程 http://download.csdn.net/album/detail/1115 C#资料...

Global site tag (gtag.js) - Google Analytics