介绍Java Socket编程的一篇不错的文章

爱被打了一巴掌 2022-08-24 00:52 225阅读 0赞

事实上网络编程简单的理解就是两台计算机相互通讯数据而已.对于程序员而言,去掌握一种编程接口并使用一种编程模型相对就会显得简单的多了.Java SDK提供一些相对简单的Api来完成这些工作.Socket就是其中之一.对于Java而言.这些Api存在与java.net 这个包里面.因此只要导入这个包就可以准备网络编程了.

  网络编程的基本模型就是客户机到服务器模型.简单的说就是两个进程之间相互通讯,然后其中一个必须提供一个固定的位置,而另一个则只需要知道这个固定的位置.并去建立两者之间的联系..然后完成数据的通讯就可以了.这里提供固定位置的通常称为服务器,而建立联系的通常叫做客户端.基于这个简单的模型,就可以进入网络编程啦.

  Java对这个模型的支持有很多种Api.而这里我只想介绍有关Socket的编程接口.对于Java而言已经简化了Socket的编程接口.首先我们来讨论有关提供固定位置的服务方是如何建立的.Java提供了ServerSocket来对其进行支持.事实上当你创建该类的一个实力对象并提供一个端口资源你就建立了一个固定位置可以让其他计算机来访问你.ServerSocket server=new ServerSocket(6789);这里稍微要注意的是端口的分配必须是唯一的.因为端口是为了唯一标识每台计算机唯一服务的.另外端口号是从0~65535之间的,前1024个端口已经被Tcp/Ip 作为保留端口,因此你所分配的端口只能是1024个之后的.好了.我们有了固定位置.现在所需要的就是一根连接线了.该连接线由客户方首先提出要求.因此Java同样提供了一个Socket对象来对其进行支持.只要客户方创建一个Socket的实例对象进行支持就可以了.Socket client

  =new Socket(InetAddress.getLocalHost(),5678);客户机必须知道有关服务器的IP地址.对于着一点Java也提供了一个相关的类InetAddress 该对象的实例必须通过它的静态方法来提供.它的静态方法主要提供了得到本机IP 和通过名字或IP直接得到InetAddress的方法.

  好了.上面的方法基本可以建立一条连线让两台计算机相互交流了.可是数据是如何传输的呢?事实上I/O操作总是和网络编程息息相关的.因为底层的网络是继续数据的.除非远程调用,处理问题的核心在执行上.否则数据的交互还是依赖于IO操作的.所以你也必须导入java.io这个包.java的IO操作也不复杂.它提供了针对于字节流和Unicode的读者和写者,然后也提供了一个缓冲用于数据的读写.

  1.   BufferedReader
  2. in
  3. =
  4. new
  5. BufferedReader(
  6. new
  7. InputStreamReader(server.getInputStream()));
  8.   PrintWriter
  9. out
  10. =
  11. new
  12. PrintWriter(server.getOutputStream());

  上面两句就是建立缓冲并把原始的字节流转变为Unicode可以操作.而原始的字节流来源于Socket的两个方法.getInputStream()和getOutputStream()方.分别用来得到输入和输出.那么现在有了基本的模型和基本的操作工具.我们可以做一个简单的Socket例程了.

  服务方:

  1.   import java.io.
  2. *
  3. ;
  4.   import java.net.
  5. *
  6. ;
  7.   
  8. public
  9. class
  10. MyServer {
  11.   
  12. public
  13. static
  14. void
  15. main(String[] args) throws IOException{
  16.   ServerSocket server
  17. =
  18. new
  19. ServerSocket(
  20. 5678
  21. );
  22.   Socket client
  23. =
  24. server.accept();
  25.   BufferedReader
  26. in
  27. =
  28. new
  29. BufferedReader(
  30. new
  31. InputStreamReader(client.getInputStream()));
  32.   PrintWriter
  33. out
  34. =
  35. new
  36. PrintWriter(client.getOutputStream());
  37.   
  38. while
  39. (
  40. true
  41. ){
  42.   String str
  43. =
  44. in
  45. .readLine();
  46.   System.
  47. out
  48. .println(str);
  49.   
  50. out
  51. .println(
  52. "
  53. has receive....
  54. "
  55. );
  56.   
  57. out
  58. .flush();
  59.   
  60. if
  61. (str.equals(
  62. "
  63. end
  64. "
  65. ))
  66.   
  67. break
  68. ;
  69.   }
  70.   client.close();
  71.   }
  72.   }

  这个程序的主要目的在于服务器不断接收客户机所写入的信息只到.客户机发送”End”字符串就退出程序.并且服务器也会做出”Receive”为回应.告知客户机已接收到消息.

  客户机代码:

  1.   import java.net.
  2. *
  3. ;
  4.   import java.io.
  5. *
  6. ;
  7.   
  8. public
  9. class
  10. Client{
  11.   
  12. static
  13. Socket server;
  14.   
  15. public
  16. static
  17. void
  18. main(String[] args)throws Exception{
  19.   server
  20. =
  21. new
  22. Socket(InetAddress.getLocalHost(),
  23. 5678
  24. );
  25.   BufferedReader
  26. in
  27. =
  28. new
  29. BufferedReader(
  30. new
  31. InputStreamReader(server.getInputStream()));
  32.   PrintWriter
  33. out
  34. =
  35. new
  36. PrintWriter(server.getOutputStream());
  37.   BufferedReader wt
  38. =
  39. new
  40. BufferedReader(
  41. new
  42. InputStreamReader(System.
  43. in
  44. ));
  45.   
  46. while
  47. (
  48. true
  49. ){
  50.   String str
  51. =
  52. wt.readLine();
  53.   
  54. out
  55. .println(str);
  56.   
  57. out
  58. .flush();
  59.   
  60. if
  61. (str.equals(
  62. "
  63. end
  64. "
  65. )){
  66.   
  67. break
  68. ;
  69.   }
  70.   System.
  71. out
  72. .println(
  73. in
  74. .readLine());
  75.   }
  76.   server.close();
  77.   }
  78.   }

  客户机代码则是接受客户键盘输入,并把该信息输出,然后输出”End”用来做退出标识.

  这个程序只是简单的两台计算机之间的通讯.如果是多个客户同时访问一个服务器呢?你可以试着再运行一个客户端,结果是会抛出异常的.那么多个客户端如何实现呢?

  其实,简单的分析一下,就可以看出客户和服务通讯的主要通道就是Socket本身.而服务器通过accept方法就是同意和客户建立通讯.这样当客户建立Socket的同时.服务器也会使用这一根连线来先后通讯.那么既然如此只要我们存在多条连线就可以了.那么我们的程序可以变为如下:

  服务器:

  1.   import java.io.
  2. *
  3. ;
  4.   import java.net.
  5. *
  6. ;
  7.   
  8. public
  9. class
  10. MyServer {
  11.   
  12. public
  13. static
  14. void
  15. main(String[] args) throws IOException{
  16.   ServerSocket server
  17. =
  18. new
  19. ServerSocket(
  20. 5678
  21. );
  22.   
  23. while
  24. (
  25. true
  26. ){
  27.   Socket client
  28. =
  29. server.accept();
  30.   BufferedReader
  31. in
  32. =
  33. new
  34. BufferedReader(
  35. new
  36. InputStreamReader(client.getInputStream()));
  37.   PrintWriter
  38. out
  39. =
  40. new
  41. PrintWriter(client.getOutputStream());
  42.   
  43. while
  44. (
  45. true
  46. ){
  47.   String str
  48. =
  49. in
  50. .readLine();
  51.   System.
  52. out
  53. .println(str);
  54.   
  55. out
  56. .println(
  57. "
  58. has receive....
  59. "
  60. );
  61.   
  62. out
  63. .flush();
  64.   
  65. if
  66. (str.equals(
  67. "
  68. end
  69. "
  70. ))
  71.   
  72. break
  73. ;
  74.   }
  75.   client.close();
  76.   }
  77.   }
  78.   }

  这里仅仅只是加了一个外层的While循环.这个循环的目的就是当一个客户进来就为它分配一个Socket直到这个客户完成一次和服务器的交互,这里也就是接受到客户的”End”消息.那么现在就实现了多客户之间的交互了.但是.问题又来了.这样做虽然解决了多客户,可是是排队执行的.也就是说当一个客户和服务器完成一次通讯之后下一个客户才可以进来和服务器交互.无法做到同时服务.那么要如何才能同时达到既能相互之间交流又能同时交流呢?很显然这是一个并行执行的问题了.所以线程是最好的解决方案.

  那么下面的问题是如何使用线程.首先要做的事情是创建线程并使得其可以和网络连线取得联系.然后由线程来执行刚才的操作.要创建线程要么直接继承Thread要么实现Runnable接口,要建立和Socket的联系只要传递引用就可以了.而要执行线程就必须重写run方法.而run方法所做的事情.就是刚才单线程版本main所做的事情.因此我们的程序变成了这样:

  1.   import java.net.
  2. *
  3. ;
  4.   import java.io.
  5. *
  6. ;
  7.   
  8. public
  9. class
  10. MultiUser extends Thread{
  11.   
  12. private
  13. Socket client;
  14.   
  15. public
  16. MultiUser(Socket c){
  17.   
  18. this
  19. .client
  20. =
  21. c;
  22.   }
  23.   
  24. public
  25. void
  26. run(){
  27.   
  28. try
  29. {
  30.   BufferedReader
  31. in
  32. =
  33. new
  34. BufferedReader(
  35. new
  36. InputStreamReader(client.getInputStream()));
  37.   PrintWriter
  38. out
  39. =
  40. new
  41. PrintWriter(client.getOutputStream());
  42.   
  43. //
  44. Mutil User but can't parallel
  45.   
  46. while
  47. (
  48. true
  49. ){
  50.   String str
  51. =
  52. in
  53. .readLine();
  54.   System.
  55. out
  56. .println(str);
  57.   
  58. out
  59. .println(
  60. "
  61. has receive....
  62. "
  63. );
  64.   
  65. out
  66. .flush();
  67.   
  68. if
  69. (str.equals(
  70. "
  71. end
  72. "
  73. ))
  74.   
  75. break
  76. ;
  77.   }
  78.   client.close();
  79.   }
  80. catch
  81. (IOException ex){
  82.   }
  83. finally
  84. {
  85.   }
  86.   }
  87.   
  88. public
  89. static
  90. void
  91. main(String[] args)throws IOException{
  92.   ServerSocket server
  93. =
  94. new
  95. ServerSocket(
  96. 5678
  97. );
  98.   
  99. while
  100. (
  101. true
  102. ){
  103.   
  104. //
  105. transfer location change Single User or Multi User
  106.   MultiUser mu
  107. =
  108. new
  109. MultiUser(server.accept());
  110.   mu.start();
  111.   }
  112.   }
  113.   }

  我的类直接从Thread类继承了下来.并且通过构造函数传递引用和客户Socket建立了联系.这样每个线程就有了.一个通讯管道.同样我们可以填写run方法.把之前的操作交给线程来完成.这样多客户并行的Socket就建立起来了.

发表评论

表情:
评论列表 (有 0 条评论,225人围观)

还没有评论,来说两句吧...

相关阅读