Java的Socket通讯基础编程完全指南

 更新时间:2015年08月06日 10:14:33   作者:zinss26914  
这篇文章主要介绍了Java的Socket通讯基础编程,包括对Socket服务器的并发访问方法,是Java网络编程中的重要知识,相当推荐!需要的朋友可以参考下

什么是Socket
网络上的两个程序通过一个双向的通讯连接实现数据的交换,这个双向链路的一端称为一个Socket。Socket通常用来实现客户方和服务方的连接。Socket是TCP/IP协议的一个十分流行的编程界面,一个Socket由一个IP地址和一个端口号唯一确定。
但是,Socket所支持的协议种类也不光TCP/IP一种,因此两者之间是没有必然联系的。在Java环境下,Socket编程主要是指基于TCP/IP协议的网络编程。

Socket通讯的过程
Server端Listen(监听)某个端口是否有连接请求,Client端向Server 端发出Connect(连接)请求,Server端向Client端发回Accept(接受)消息。一个连接就建立起来了。Server端和Client 端都可以通过Send,Write等方法与对方通信。
对于一个功能齐全的Socket,都要包含以下基本结构,其工作过程包含以下四个基本的步骤:
  (1) 创建Socket;
  (2) 打开连接到Socket的输入/出流;
  (3) 按照一定的协议对Socket进行读/写操作;
  (4) 关闭Socket.(在实际应用中,并未使用到显示的close,虽然很多文章都推荐如此,不过在我的程序中,可能因为程序本身比较简单,要求不高,所以并未造成什么影响。)


创建Socket
java在包java.net中提供了两个类Socket和ServerSocket,分别用来表示双向连接的客户端和服务端。这是两个封装得非常好的类,使用很方便。其构造方法如下:

  Socket(InetAddress address, int port);
  Socket(InetAddress address, int port, boolean stream);
  Socket(String host, int prot);
  Socket(String host, int prot, boolean stream);
  Socket(SocketImpl impl)
  Socket(String host, int port, InetAddress localAddr, int localPort)
  Socket(InetAddress address, int port, InetAddress localAddr, int localPort)
  ServerSocket(int port);
  ServerSocket(int port, int backlog);
  ServerSocket(int port, int backlog, InetAddress bindAddr)

  其中address、host和port分别是双向连接中另一方的IP地址、主机名和端 口号,stream指明socket是流socket还是数据报socket,localPort表示本地主机的端口号,localAddr和 bindAddr是本地机器的地址(ServerSocket的主机地址),impl是socket的父类,既可以用来创建serverSocket又可 以用来创建Socket。count则表示服务端所能支持的最大连接数。例如:学习视频网 http://www.xxspw.com

  Socket client = new Socket("127.0.01.", 80);
  ServerSocket server = new ServerSocket(80);

  注意,在选择端口时,必须小心。每一个端口提供一种特定的服务,只有给出正确的端口,才 能获得相应的服务。0~1023的端口号为系统所保留,例如http服务的端口号为80,telnet服务的端口号为21,ftp服务的端口号为23, 所以我们在选择端口号时,最好选择一个大于1023的数以防止发生冲突。
  在创建socket时如果发生错误,将产生IOException,在程序中必须对之作出处理。所以在创建Socket或ServerSocket是必须捕获或抛出例外。

代码

server

   

 package socket; 
   
  import java.io.*; 
  import java.net.*; 
   
  public class TcpServer { 
    public static void main(String[] args) throws Exception { 
      ServerSocket server = new ServerSocket(9091); 
      try { 
        Socket client = server.accept(); 
        try { 
          BufferedReader input = 
              new BufferedReader(new InputStreamReader(client.getInputStream())); 
          boolean flag = true; 
          int count = 1; 
   
          while (flag) { 
            System.out.println("客户端要开始发骚了,这是第" + count + "次!"); 
            count++; 
             
            String line = input.readLine(); 
            System.out.println("客户端说:" + line); 
             
            if (line.equals("exit")) { 
              flag = false; 
              System.out.println("客户端不想玩了!"); 
            } else { 
              System.out.println("客户端说: " + line); 
            } 
   
          } 
        } finally { 
          client.close(); 
        } 
         
      } finally { 
        server.close(); 
      } 
    } 
  } 


client

  package socket; 
   
  import java.io.*; 
  import java.net.*; 
  import java.util.Scanner; 
   
  public class TcpClient { 
    public static void main(String[] args) throws Exception { 
      Socket client = new Socket("127.0.0.1", 9091); 
      try { 
        PrintWriter output = 
            new PrintWriter(client.getOutputStream(), true); 
        Scanner cin = new Scanner(System.in); 
        String words; 
   
        while (cin.hasNext()) { 
          words = cin.nextLine(); 
   
          output.println(words); 
   
          System.out.println("写出了数据: " + words); 
        } 
   
        cin.close(); 
      } finally { 
        client.close(); 
      } 
    } 
  } 

Server绑定ip

用c写socket的时候,struct sockaddr_in 结构体是可以指定sin_addr.s_addr的,也就是可以指定ip地址,为什么会有这种需求呢,例如我的网络链接是这样的:

201586101125110.png (804×634)

我可能只想绑定eth0这个网卡的ip地址,因为我的lo和wlan0都可能在用一端口做了nginx的虚拟主机,因此在服务器端开启ServerSocket的时候,有指定ip的需求

方案
ServerSocket的一个构造函数如下:

public ServerSocket(int port, int backlog, InetAddress bindAddr) throws IOException

参数:

    port - 本地 TCP 端口
    backlog - 侦听 backlog
    bindAddr - 要将服务器绑定到的 InetAddress


因为InetAddress无构造函数,我在这里纠结了好一段时间,查看stackoverflow上,可以使用InetAddress的getByName方法

示例代码

  InetAddress bindip = InetAddress.getByName("192.168.1.168"); 
   
  ServerSocket server = new ServerSocket(9091, 0, bindip); 

并发访问
服务器端通过增加多线程来同时处理多个客户端的请求,其实实现还是很水的,毕竟java对多线程封装也足够好了,我是在Server服务器端用一个内部类实现了Runnable接口,在run方法里处理客户端的请求,将数据打印出来

server代码

   

package capitalsocket; 
   
  import java.io.BufferedReader; 
  import java.io.IOException; 
  import java.io.InputStreamReader; 
  import java.net.InetAddress; 
  import java.net.ServerSocket; 
  import java.net.Socket; 
   
  public class CapitalizeServer { 
    private static int clientNum = 0; 
   
    public static void main(String args[]) throws Exception { 
      ServerSocket listener = new ServerSocket(9898, 0, InetAddress.getByName("192.168.1.168")); 
      try { 
        while (true) { 
          Capitalizer multip = new Capitalizer(listener.accept(), CapitalizeServer.clientNum ++); 
          Thread t = new Thread(multip); 
          t.start(); 
        } 
      } finally { 
        listener.close(); 
      } 
    } 
   
    private static class Capitalizer implements Runnable { 
      private Socket client; 
      private int id; 
   
      public Capitalizer(Socket s, int id) { 
        this.client = s; 
        this.id = id; 
      } 
   
      public void run() { 
        try { 
          BufferedReader input = 
              new BufferedReader(new InputStreamReader(this.client.getInputStream())); 
           
          while (true) { 
            String data = input.readLine(); 
             
            if (data.equals("bye")) { 
              System.out.println("当前第" + this.id + "个客户端度不想玩了!"); 
              break; 
            } else { 
              System.out.println("当前第" + this.id + "个客户端说:" + data); 
            } 
          } 
   
        } catch (IOException e) { 
          e.printStackTrace(); 
        } finally { 
          try { 
            this.client.close(); 
          } catch (IOException e) { 
            e.printStackTrace(); 
          } 
        } 
      } 
    } 
   
  } 


client代码
客户端代码基本没变,增加了一个退出操作

   

package capitalsocket; 
   
  import java.io.PrintWriter; 
  import java.net.Socket; 
  import java.util.Scanner; 
   
  public class CapitalizeClient { 
    public static void main(String[] args) throws Exception { 
      Socket client = new Socket("192.168.1.168", 9898); 
      try { 
        PrintWriter output = new PrintWriter(client.getOutputStream(), true); 
        Scanner cin = new Scanner(System.in); 
        String words; 
   
        while (cin.hasNext()) { 
          words = cin.nextLine(); 
          output.println(words); 
           
          if (words.equals("bye")) { 
            break; 
          } 
           
          // 每写一次数据需要sleep一会 
          Thread.sleep(3000); 
        } 
   
        cin.close(); 
      } finally { 
        client.close(); 
      } 
    } 
  } 

相关文章

  • Java使用正则表达式判断独立字符的存在(代码示例)

    Java使用正则表达式判断独立字符的存在(代码示例)

    通过使用正则表达式,我们可以更加灵活地判断字符串中是否包含特定的字符,并且可以控制匹配的条件,如独立的字符,这为我们处理字符串提供了更多的选择和功能,这篇文章主要介绍了Java使用正则表达式判断独立字符的存在,需要的朋友可以参考下
    2023-10-10
  • 实例解析观察者模式及其在Java设计模式开发中的运用

    实例解析观察者模式及其在Java设计模式开发中的运用

    观察者模式定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象,这个主题对象在状态上发生变化时,会通知所有观察者对象,使它们能够自动更新自己.下面就以实例解析观察者模式及其在Java设计模式开发中的运用
    2016-05-05
  • springboot中nacos-client获取配置的实现方法

    springboot中nacos-client获取配置的实现方法

    本文主要介绍了springboot中nacos-client获取配置的实现方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-04-04
  • SpringBoot jackson 精度处理问题解决

    SpringBoot jackson 精度处理问题解决

    由于JavaScript处理的最大数值限制,较大的雪花ID在JS中容易溢出,为解决此问题,可在SpringMVC或SpringBoot中使用@RequestBody注解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2024-10-10
  • SpringBoot+kaptcha实现验证码花式玩法详解

    SpringBoot+kaptcha实现验证码花式玩法详解

    这篇文章主要想和大家聊聊kaptcha的用法,毕竟这个已经有16年历史的玩意还在有人用,说明它的功能还是相当强大的,感兴趣的小伙伴可以了解一下
    2022-05-05
  • 详解Java匿名内部类

    详解Java匿名内部类

    这篇文章介绍了Java匿名内部类的实现,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-12-12
  • Java中用POI实现将数据导出到Excel

    Java中用POI实现将数据导出到Excel

    这篇文章主要介绍了Java中用POI实现将数据导出到Excel,文中有非常详细的代码示例,对正在学习java的小伙伴们有非常好的帮助,需要的朋友可以参考下
    2021-04-04
  • ThreadLocal简介_动力节点Java学院整理

    ThreadLocal简介_动力节点Java学院整理

    这篇文章主要为大家详细介绍了ThreadLocal简介的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-08-08
  • idea如何关闭右侧类显示方法

    idea如何关闭右侧类显示方法

    这篇文章主要介绍了idea如何关闭右侧类显示方法,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-07-07
  • Java如何将任意类型的Object对象转换为相应的实体对象

    Java如何将任意类型的Object对象转换为相应的实体对象

    这篇文章主要介绍了Java如何将任意类型的Object对象转换为相应的实体对象问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-01-01

最新评论