使用JAVA编写网络通信程序
佚名 2005-12-23
摘要 本文阐述了使用JAVA编程语言对基于客户/服务器模式的应用编写网络通信程序,讨论了SOCKET机制、输入输出流以及程序实现代码。
关键词 JAVA,网络,SOCKET,APPLET
网络上的系统结构多为客户/服务器模式,服务器端负责数据和图像等的存储、维护、管理以及传递,客户端则负责人机界面的操作、送出需求及显示收回的数据。 下面介绍一下如何使用JAVA来进行网络编程: 1) 由于客户端通过IE同服务器建立联系,所以客户端使用Applet,服务器端使用Application; 2) 服务器应设置成多线程,应答多个客户的请求; 3) 两端通信使用SOCKET机制。 1 Java中输入/输出流概念: 过滤流DataInputStream 和DataOutputStream 除了分别作为FilterInputStream 和FilterOutputStream的子类外,还分别实现了接口DataInput 和DataOutput。接口DataInput 中定义的方法主要包括从流中读取基本类型的数据、读取一行数据、或者读取指定长度的字节数,如readBoolean() readInt()、readLine()、readFully()等。接口DataOutput中定义的方法主要是向流中写入基本类型的数据或者写入一定长度的字节数组,如writeChar()、writeDouble() DataInputStream可以从所连接的输入流中读取与机器无关的基本类型数据,用以实现一种独立于具体平台的输入方式;DataInputStream 可以向所连接的输出流写入基本类型的数据。 2 Socket 机制 Socket是面向客户/服务器模型设计的,网络上的两个程序通过一个双向的通讯连接实现数据的交换,这个双向链路的一端称为一个Socket。 Socket通常用来实现客户方和服务方的连接。客户程序可以向Socket写请求,服务器将处理此请求,然后通过Socket将结果返回给用户。 Socket通信机制提供了两种通讯方式:有联接和无联接方式,分别面向不同的应用需求。使用有联接方式时,通信链路提供了可靠的,全双工的字节流服务。在该方式下,通信双方必须创建一个联接过程并建立一条通讯链路,以后的网络通信操作完全在这一对进程之间进行,通信完毕关闭此联接过程。使用无联接方式时其系统开销比无联接方式小,但通信链路提供了不可靠的数据报服务,不能保证信源所传输的数据一定能够到达信宿。在该方式下,通信双方不必创建一个联接过程和建立一条通讯链路,网络通信操作在不同的主机和进程之间转发进行。 3 Java语言 Java语言的优点主要表现在:简单、面向对象、多线程、分布性、体系结构中立、安全性等方面。 (1) 简单性 Java与C++语言非常相近,但Java比C++简单,它抛弃了C++中的一些不是绝对必要的功能,如头文件、预处理文件、指针、结构、运算符重载、多重继承以及自动强迫同型。 Java实现了自动的垃圾收集,简化了内存管理的工作。这使程序设计更加简便,同时减少了出错的可能。 (2) 面向对象 Java提供了简单的类机制和动态的构架模型。对象中封装了它的状态变量和方法,很好地实现了模块化和信息隐藏;而类则提供了一类对象的原型,通过继承和重载机制,子类可以使用或重新定义父类或超类所提供的方法,从而既实现了代码的复用,又提供了一种动态的解决方案。 Java是一种完全面向对象的程序设计语言,它除了数组、布尔和字符三个基本数据类型外的其它类都是对象,它不再支持全局变量。在Java中,如果不创建新类就无法创建程序,Java程序在运行时必须先创建一个类的实例,然后才能提交运行。 Java同样支持继承特性,Java的类可以从其它类中继承行为,但Java只支持类的单重继承,即每个类只能从一个类中继承。 Java支持界面,界面允许程序员定义方法但又不立即实现,一个类可以实现多个界面,利用界面可以得到多重继承的许多优点而又没有多重继承的问题。 (3) 多线程 多线程使应用程序可以同时进行不同的操作,处理不同的事件。在多线程机制中,不同的线程处理不同的任务,他们之间互不干涉,不会由于一处等待影响其他部分,这样容易实现网络上的实时交互操作。 Java程序可以有多个执行线程,如可以让一个线程进行复杂的计算,而让另一个线程与用户进行交互,这样用户可以在不中断计算线程的前提下与系统进行交互。多线程保证了较高的执行效率。
(4) 分布性 Java是面向网络的语言。通过它提供的类库可以处理TCP/IP协议,用户可以通过URL地址在网络上很方便的访问其他对象。 (5) 体系结构中立 Java是一种网络语言,为使Java程序能在网络的任何地方运行,Java解释器生成与体系结构无关的字节码结构的文件格式。Java为了做到结构中立,除生成机器无关的字节码外,还制定了完全统一的语言文本,如Java的基本数据类型不会随目标机的变化而变化,一个整型总是32位,一个长整型总是64位。 为了使Java的应用程序能不依赖于具体的系统,Java语言环境还提供了用于访问底层操作系统功能的类组成的包,当程序使用这些包时,可以确保它能运行在各种支持Java的平台上。 java.lang: 一般的语言包。其中包括用于字符串处理、多线程、异常处理和数字函数等的类,该包是实现Java程序运行平台的基本包 java.util: 实用工具包。其中包括哈希表、堆栈、时间和日期等 java.io: 基于流模型的输入/输出包。该包用统一的流模型实现了各种格式的输入/输出,包括文件系统、网络和设备的输入/输出等 java.net: 网络包。该包支持TCP/IP协议,其中提供了socket、URL和WWW的编程接口 java.awt: 抽象窗口工具集。其中实现了可以跨平台的图形用户界面组件,包括窗口、菜单、滚动条和对话框等 java.applet: 支持applet程序设计的基本包 (6) 安全性 用于网络、分布环境下的Java必须要防止病毒的入侵,Java不支持指针,一切对内存的访问都必须通过对象的实例变量来实现,这样就防止了程序员使用欺骗手段访问对象的私有成员,同时也避免了指针操作中容易产生的错误。 4 JAVA工具 (1) JDK 1) Java编译器 Java编译器将Java源代码文件编译成可执行的Java字节码。Java源代码文件的扩展名为 .java,Java编译器把这种扩展名的文件编译成扩展名为.class的文件。源文件中的每个类在编译后都将产生一个class文件,这意味一个Java源代码文件可能编译生成多个class文件。 2) Java解释器 Java解释器对编译生成的字节码格式的可执行程序的运行提供支持,它是运行非图形Java程序的命令行工具。 3) Appletviewer 它是Java Applet的简单测试工具,可使用它来测试Java Applet程序,而不需要WWW浏览器的支持。 (2) Visual J++ Visual J++ 集成了可视化界面设计、交互式调试、代码编辑、联机帮助信息和介绍如何快速掌握该开发环境的实用向导等多项功能,同时具有能充分利用Active X和COM新技术的优势。利用Visual J++可创建交互性很强的Internet应用程序,是难得的Java 开发系统。
5 客户机/服务器通信的实现: (1) Application 同 Applet 的通信 两端通过Socket机制进行连接: 1) 客户端的编程流程: 打开Socket,新建一个套接字; 为套接字建立一个输入和输出流; 根据服务器协议从套接字读入或向套接字写入; 清除套接字和输入/输出流; 2)服务器端的编程流程: 打开Server Socket,创建一个服务器型套接字和一个普通套接字,服务器型套接字在指定端口为客户端请求的Socket 服务; 使用ServerSocket类的accept()方法使服务器型套接字处于监听状态并把监听结果返回给普通套接字; 为该普通套接字创建输入和输出流; 从输入和输出流中读入或写入字节流,进行相应的处理,并将结果返回给客户端; 在客户端和服务器工作结束后关闭所有的对象,如服务器型的套接字,普通套接字,输入和输出流。 正是由于Java系统具有基于Socket的灵活通信机制,因而其应用程序能自由地打开和访问网络上的对象,就象在本地文件系统中一样。 (2) Applet之间的通信: Applet之间的通信使用Applet Context类的getApplet()方法。
6 程序 //服务器端程序S.java 负责与客户端通信 import java.io.*; import java.net.*; import java.lang.*; import T2;
class ThreadEchoHandler extends Thread //创建线程 { T2 theT2=new T2(); Socket incoming; int counter; ThreadEchoHandler(Socket i,int c) { incoming=i; counter=c; }
public void run() { try { DataInputStream in=new DataInputStream(incoming.getInputStream()); DataOutputStream out=new DataOutputStream(incoming.getOutputStream()); System.out.println ("hello"); boolean done=false;
while(!done) { String aa=""; String str=in.readUTF(); //从客户端得到字符串
//在此加入各自的服务程序 System.out.println (str); theT2.pass(str); //解码 theT2.tongji(); //修改监控库中的信息
aa=theT2.guan(); //操纵数据库 System.out.println ("string z is:"+aa);
if(aa.compareTo("null")!=0 ) //若是查询数据库,返回查询后的结果 { //若不是查询数据库,不向客户端输出信息 out.writeUTF(aa); out.flush(); } }//while
incoming.close();//线程关闭 }//try
catch(IOException e) {System.out.println(e);} }//end run }
//---------------------------------------- class S { public static void main(String[] args) { int i=1; try { ServerSocket s=new ServerSocket(1111); for(;;) { Socket incoming=s.accept(); System.out.println("connect: "+i); new ThreadEchoHandler(incoming,i).start(); i++; } } catch(Exception e) { System.out.println(e); } } }
//客户端通信小应用程序 Echo.java
import java.io.*; import java.net.*; import java.awt.*; import java.applet.*;
public class Echo extends Applet { TextArea ta; Socket echoSocket; DataOutputStream os; DataInputStream is; StringLine;
public void init() { setBackground(Color.white); ta=new TextArea(5,80); ta.setEditable(false); add(ta); try {echoSocket=new Socket("10.102.4.41",1111);} //与服务器建立连接 catch(IOException e) {System.out.println("error");} }
public void st(String stri) //发送字符串的方法 { try { DataOutputStream os=new DataOutputStream(echoSocket.getOutputStream()); DataInputStream is=new DataInputStream(echoSocket.getInputStream()); os.writeUTF(""+ stri ); //向服务器输送string os.flush(); } catch(IOException e) {System.out.println(" error:"+e); } } public String st1() //接收字符串的方法 { String Line=""; try { DataOutputStream os=new DataOutputStream(echoSocket.getOutputStream()); DataInputStream is=new DataInputStream(echoSocket.getInputStream()); Line=is.readUTF();//从服务器读来的信息 ta.appendText(""+Line); //在文本域中输出信息 } catch(IOException e) {System.out.println(" error:"+e); } return Line; } }
7 程序调试心得: 1) 在建立Socket连接时,两端的端口号必须设为一致,否则建立不了连接。服务器端必须有主机IP地址或主机名参数。 2) 连接建立好之后应确定输入和输出流。起初程序中用的是DataInputStream和PrintStream,结果只能传输英文,传输中文时产生乱码,将PrintStream改为DataOutputStream,使用readUTF()和writeUTF()方法后,中文传输问题得到解决。 3) 如果一个使用某端口的程序没有关闭,另一个程序就不能使用这个端口。 4) 开始进行通信的程序均为 Application,因不符合客户机/服务器机制,应将客户端的Application改为Applet。其转化的主要步骤如下: 创建一个包含APPLET标签的HTML文件; 去掉应用程序中的main()方法; 类名应继承Applet类,而不是Frame类,并在程序开头加入 import java.applet.*;语句; 用init()方法代替Application程序中的构造方法,当浏览器创建Applet类对象的时候,它自动执行init()方法; 如Application中缺省使用了BorderLayout布局管理器,应在Applet的init()方法中重新设定; 如果Application中有setTitle()方法,必须将其去掉,如Application中使用了菜单,在Applet 中用按钮来替换。 5) 懂得了在一程序中如何引用自定义的类中的方法和变量,在程序开头加入import 类名;在程序中加入类名 实例=new 类名(); 然后使用 实例.方法(),实例.变量即可。