`
liudaoru
  • 浏览: 1573914 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

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

    博客分类:
  • java
阅读更多

  From:java.ccidnet.com/art/297/20051223/399191_3.html

事实上网络编程简单的理解就是两台计算机相互通讯数据而已。对于程序员而言,去掌握一种编程接口并使用一种编程模型相对就会显得简单的多了。Java SDK提供一些相对简单的Api来完成这些工作,Socket就是其中之一。

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

Java对这个模型的支持有很多种Api。而这里我只想介绍有关Socket的编程接口。对于Java而言已经简化了Socket的编程接口。首先我们来讨论有关提供固定位置的服务方是如何建立的。Java提供了ServerSocket来对其进行支持。事实上当你创建该类的一个实力对象并提供一个端口资源你就建立了一个固定位置可以让其他计算机来访问你。

 

<ccid_nobr></ccid_nobr>
<ccid_code></ccid_code>ServerSocket server=new  
ServerSocket(6789);

这里稍微要注意的是端口的分配必须是唯一的。因为端口是为了唯一标识每台计算机唯一服务的。另外端口号是从0~65535之间的,前1024个端口已经被Tcp/Ip 作为保留端口,因此你所分配的端口只能是1024个之后的。

好了。我们有了固定位置。现在所需要的就是一根连接线了。该连接线由客户方首先提出要求。因此Java同样提供了一个Socket对象来对其进行支持。只要客户方创建一个Socket的实例对象进行支持就可以了。

 

<ccid_nobr></ccid_nobr>
<ccid_code></ccid_code>Socket client=new
Socket(InetAddress.getLocalHost(),5678);

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

 

 

好了,上面的方法基本可以建立一条连线让两台计算机相互交流了。可是数据是如何传输的呢?事实上I/O操作总是和网络编程息息相关的。因为底层的网络是继续数据的。除非远程调用,处理问题的核心在执行上。

否则数据的交互还是依赖于IO操作的。所以你也必须导入Java。io这个包。Java的IO操作也不复杂。它提供了针对于字节流和Unicode的读者和写者,然后也提供了一个缓冲用于数据的读写。



 

<ccid_nobr></ccid_nobr>
<ccid_code></ccid_code>BufferedReader in=new 
BufferedReader(new InputStreamReader
(server.getInputStream()));
  PrintWriter out=new 
  PrintWriter(server.getOutputStream());

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

 

<ccid_nobr></ccid_nobr>
<ccid_code></ccid_code>import Java。io。*;
import Java。net。*;

public class MyServer 
{
 public static void main
 (String[] args) throws IOException
 {
  ServerSocket server=new 
  ServerSocket(5678);
  Socket client=server.accept();
BufferedReader 
in=new BufferedReader(new
InputStreamReader
(client.getInputStream()));
PrintWriter out=new 
PrintWriter(client.getOutputStream());
while(true)
{
 String str=in.readLine();
System。out。println(str);
 out。println("has receive...");
 out。flush();
 if(str。equals("end"))
 break;
}
client。close();
 } 
}

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

 

<ccid_nobr></ccid_nobr>
<ccid_code></ccid_code>import Java.net.*;
import Java.io.*;

public class Client{
 static Socket server;
 
 public static void 
 main(String[] args)
 throws Exception
 {
  server=new Socket
  (InetAddress.getLocalHost(),5678);
  BufferedReader in=new BufferedReader(new 
  InputStreamReader(server。getInputStream()));
  PrintWriter out=new
  PrintWriter(server.getOutputStream());
  BufferedReader wt=new BufferedReader(new 
  InputStreamReader(System.in));
  
  while(true)
  {
   String str=wt.readLine();
   out.println(str);
   out.flush();
   if(str.equals("end"))
   {
    break;
   }
   System.out.println(in.readLine());
  }
  server.close();
 }
}
 

 

客户机代码则是接受客户键盘输入,并把该信息输出,然后输出"End"用来做退出标识。这个程序只是简单的两台计算机之间的通讯。如果是多个客户同时访问一个服务器呢?你可以试着再运行一个客户端,结果是会抛出异常的。那么多个客户端如何实现呢?

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

服务器:

 

<ccid_nobr></ccid_nobr>
<ccid_code></ccid_code>import Java.io.*;
import Java.net.*;

public class MyServer 
{
 public static void 
 main(String[] args) 
 throws IOException
 {
  ServerSocket server=new 
  ServerSocket(5678);
while(true)
{
  Socket client=server.accept();
BufferedReader in=new BufferedReader(new 
InputStreamReader(client.getInputStream()));
PrintWriter out=new
PrintWriter(client.getOutputStream());
while(true)
{
 String str=in.readLine();
System.out.println(str);
 out.println("has receive...");
 out.flush();
 if(str.equals("end"))
 break;
}
client。close();
}
 } 
}

这里仅仅只是加了一个外层的While循环。这个循环的目的就是当一个客户进来就为它分配一个Socket直到这个客户完成一次和服务器的交互,这里也就是接受到客户的"End"消息。那么现在就实现了多客户之间的交互了。

但是问题又来了,这样做虽然解决了多客户,可是是排队执行的。也就是说当一个客户和服务器完成一次通讯之后下一个客户才可以进来和服务器交互。无法做到同时服务。那么要如何才能同时达到既能相互之间交流又能同时交流呢?很显然这是一个并行执行的问题了。所以线程是最好的解决方案。

那么下面的问题是如何使用线程。首先要做的事情是创建线程并使得其可以和网络连线取得联系。然后由线程来执行刚才的操作。要创建线程要么直接继承Thread要么实现Runnable接口,要建立和Socket的联系只要传递引用就可以了。

而要执行线程就必须重写run方法。而run方法所做的事情。就是刚才单线程版本main所做的事情。因此我们的程序变成了这样:

 

<ccid_nobr></ccid_nobr>
<ccid_code></ccid_code>import Java.net.*;
import Java.io.*;

public class MultiUser extends Thread
{
 private Socket client;
 
 public MultiUser(Socket c)
 {
  this。client=c;
 }
 
 public void run()
 {
  try
  {   
   BufferedReader in=new BufferedReader(new 
   InputStreamReader(client.getInputStream()));
   PrintWriter out=new
   PrintWriter(client.getOutputStream());
    //Mutil User but can't parallel
    while(true)
	{
     String str=in.readLine();
     System。out.println(str);
     out.println("has receive...");
     out.flush();
     if(str.equals("end"))
      break;
    }
   client.close();  
   }catch(IOException ex)
   {
   }finally
   {
    
   }
 }
 
 public static void 
 main(String[] args)throws IOException
 {
  ServerSocket 
  server=new ServerSocket(5678);
  while(true)
  {
   //transfer location 
   change Single User or Multi User
   MultiUser mu=new MultiUser(server.accept());
   mu.start();
  }
 }
}

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

分享到:
评论

相关推荐

    基于JAVA的ICQ系统.7z

    本篇文章将深入探讨一个基于Java语言实现的ICQ系统,帮助读者理解如何使用Java技术栈构建类似的实时通信应用。 首先,我们要明白Java在开发此类系统中的优势。Java作为一种跨平台的编程语言,其“一次编写,到处...

    Tcp服务端与客户端的JAVA实例源代码.7z

    TCP服务端和客户端之间的通信是基于Socket编程的。本篇将深入讲解如何使用Java实现TCP服务端与客户端的交互,并通过提供的源代码进行实践。 一、TCP服务端 1. 建立ServerSocket:服务端首先需要创建一个`...

    Java游戏使命的召唤源码.7z

    在本篇文章中,我们将深入探讨“Java游戏使命的召唤源码”,了解如何通过Java实现一款类似《使命的召唤》这样的游戏。 首先,我们要明白《使命的召唤》是一款射击类游戏,它涉及到多个关键的技术领域,如图形渲染、...

    Java学习导图.7z

    - **网络编程**:Socket编程,TCP/IP协议的理解。 - **并发编程**:并发工具类,如ExecutorService、Semaphore、CountDownLatch等。 4. **框架篇**: - **Spring框架**:依赖注入、AOP、MVC架构。 - **MyBatis...

    Java面试.doc

    本篇文章将深入探讨Java的基础、容器、多线程、反射、对象拷贝等关键概念,并提供相关的面试题及答案。 **一、Java基础** 1. **JDK与JRE的区别** JDK是Java Development Kit的缩写,包含了Java开发所需的一切,...

    软件开发工程师(JAVA)笔试题.doc

    这篇文档是一个软件开发工程师(JAVA)的笔试题目集,主要涉及Java语言的基础知识和编程概念。以下是根据题目内容提取的相关知识点: 1. **标识符**:标识符是用来命名变量、类、接口等的名称。在Java中,合法的...

    面试专题--基础篇资料

    12. **网络编程**:Java提供了丰富的Socket类和ServerSocket类,用于实现TCP/IP网络通信。此外,还有URL、URLConnection等类用于HTTP协议的网络请求。 13. **反射机制**:反射允许程序在运行时动态获取类的信息并...

    J2me手机游戏飞机射击完整源码.7z

    在本篇中,我们将深入探讨一款基于J2ME技术的手机游戏——“飞机射击”的完整源码,旨在帮助开发者理解J2ME游戏开发的核心概念和技术。 一、J2ME基础架构 J2ME由配置(Configurations)和 profiles(Profiles)组成...

    Android源码——高仿QQ的即时通讯(附带服务端).7z

    Android中的Socket编程和AsyncTask或IntentService是实现客户端与服务端通信的基础。 2. **Android架构设计** 高仿QQ的Android应用应该遵循MVP(Model-View-Presenter)或MVVM(Model-View-ViewModel)架构,以保证...

    vc代码合集

    2012-06-11 10:37 151,588 多线程Java Socket编程示例.html 2012-06-11 09:46 56,832 存储管理动态分区分配算法的模拟.doc 2012-06-11 09:45 32,389 完整的串口通信程序VC++源码.7z 2012-06-11 10:41 2,183,866 对...

Global site tag (gtag.js) - Google Analytics