出于多种目的要写这篇文章
简单的说是因为最近的工作和Java 的Socket相关
所以工作之余就有了写点东西的想法
同样我不希望把文章写的太复杂
因此浅出自己的一点心得
事实上网络编程简单的理解就是两台计算机相互通讯数据而已对于程序员而言去掌握一种编程接口并使用一种编程模型相对就会显得简单的多了Java SDK提供一些相对简单的Api来完成这些工作Socket就是其中之一对于Java而言这些Api存在于javanet 这个包里面因此只要导入这个包就可以准备网络编程了
网络编程的基本模型就是客户机到服务器模型简单的说就是两个进程之间相互通讯然后其中一个必须提供一个固定的位置而另一个则只需要知道这个固定的位置并去建立两者之间的联系然后完成数据的通讯就可以了这里提供固定位置的通常称为服务器而建立联系的通常叫做客户端基于这个简单的模型就可以进入网络编程啦
Java对这个模型的支持有很多种Api而这里我只想介绍有关Socket的编程接口对于Java而言已经简化了Socket的编程接口
首先我们来讨论有关提供固定位置的服务方是如何建立的Java提供了ServerSocket来对其进行支持事实上当你创建该类的一个实力对象并提供一个端口资源你就建立了一个固定位置可以让其他计算机来访问你ServerSocket server=new ServerSocket();这里稍微要注意的是端口的分配必须是唯一的因为端口是为了唯一标识每台计算机唯一服务的另外端口号是从-之间的前个端口已经被Tcp/Ip 作为保留端口因此你所分配的端口只能是个之后的
好了我们有了固定位置现在所需要的就是一根连接线了该连接线由客户方首先提出要求因此Java同样提供了一个Socket对象来对其进行支持只要客户方创建一个Socket的实例对象进行支持就可以了Socket client=new Socket(InetAddressgetLocalHost());客户机必须知道有关服务器的IP地址对于这一点Java也提供了一个相关的类InetAddress.该对象的实例必须通过它的静态方法来提供它的静态方法主要提供了得到本机IP和通过名字或IP直接得到InetAddress的方法
好了上面的方法基本可以建立一条连线让两台计算机相互交流了可是数据是如何传输的呢?事实上I/O操作总是和网络编程息息相关的因为底层的网络是传输数据的除非远程调用处理问题的核心在执行上否则数据的交互还是依赖于I/O操作的所以你必须导入javaio这个包.java的I/O操作也不复杂它提供了针对于字节流和Unicode的读者和写者然后也提供了一个缓沖用于数据的读写
BufferedReader in=new BufferedReader(new InputStreamReader(servergetInputStream()));
PrintWriter out=new PrintWriter(servergetOutputStream());
上面两句就是建立缓沖并把原始的字节流转变为Unicode可以操作而原始的字节流来源于Socket的两个方法getInputStream()和getOutputStream()方法分别用来得到输入和输出那么现在有了基本的模型和基本的操作工具我们可以做一个简单的Socket例程了
服务方:
import javaio*;
import javanet*;
public class MyServer {
public static void main(String[] args) throws IOException{
ServerSocket server=new ServerSocket();
Socket client=serveraccept();
BufferedReader in=new BufferedReader(new InputStreamReader(clientgetInputStream()));
PrintWriter out=new PrintWriter(clientgetOutputStream());
while(true){
String str=inreadLine();
Systemoutprintln(str);
outprintln(has receive);
outflush();
if(strequals(end))
break;
}
clientclose();
}
}
这个程序的主要目的在于服务器不断接收客户机所写入的信息直到客户机发送End字符串就退出程序并且服务器也会做出Receive为回应告知客户机已接收到消息
客户机代码:
import javanet*;
import javaio*;
public class Client{
static Socket server;
public static void main(String[] args)throws Exception{
server=new Socket(InetAddressgetLocalHost());
BufferedReader in=new BufferedReader(new InputStreamReader(servergetInputStream()));
PrintWriter out=new PrintWriter(servergetOutputStream());
BufferedReader wt=new BufferedReader(new InputStreamReader(Systemin));
while(true){
String str=wtreadLine();
outprintln(str);
outflush();
if(strequals(end)){
break;
}
Systemoutprintln(inreadLine());
}
serverclose();
}
}
客户机代码则是接受客户键盘输入并把该信息输出然后输出End用来做退出标识
这个程序只是简单的两台计算机之间的通讯如果是多个客户同时访问一个服务器呢?你可以试着再运行一个客户端结果是会抛出异常的那么多个客户端如何实现呢?
其实简单的分析一下就可以看出客户和服务通讯的主要通道就是Socket本身而服务器通过accept方法就是同意和客户建立通讯这样当客户建立Socket的同时服务器也会使用这一根连线来通讯那么既然如此只要我们存在多条连线就可以了那么我们的程序可以变为如下:
服务器:
import javaio*;
import javanet*;
public class MyServer {
public static void main(String[] args) throws IOException{
ServerSocket server=new ServerSocket();
while(true){
Socket client=serveraccept();
BufferedReader in=new BufferedReader(new InputStreamReader(clientgetInputStream()));
PrintWriter out=new PrintWriter(clientgetOutputStream());
while(true){
String str=inreadLine();
Systemoutprintln(str);
outprintln(has receive);
outflush();
if(strequals(end))
break;
}
clientclose();
}
}
}
这里仅仅只是加了一个外层的While循环这个循环的目的就是当一个客户进来就为它分配一个Socket直到这个客户完成一次和服务器的交互这里也就是接受到客户的End消息那么现在就实现了多客户之间的交互了但是问题又来了这样做虽然解决了多客户可是是排队执行的也就是说当一个客户和服务器完成一次通讯之后下一个客户才可以进来和服务器交互无法做到同时服务那么要如何才能同时达到既能相互之间交流又能同时交流呢?很显然这是一个并行执行的问题了所以线程是最好的解决方案
那么下面的问题是如何使用线程首先要做的事情是创建线程并使得其可以和网络连线取得联系然后由线程来执行刚才的操作要创建线程要么直接继承Thread要么实现Runnable接口.要建立和Socket的联系只要传递引用就可以了而要执行线程就必须重写run方法而run方法所做的事情就是刚才单线程版本main所做的事情因此我们的程序变成了这样:
import javanet*;
import javaio*;
public class MultiUser extends Thread{
private Socket client;
public MultiUser(Socket c){
thisclient=c;
}
public void run(){
try{
BufferedReader in=new BufferedReader(new InputStreamReader(clientgetInputStream()));
PrintWriter out=new PrintWriter(clientgetOutputStream());
//Mutil User but cant parallel
while(true){
String str=inreadLine();
Systemoutprintln(str);
outprintln(has receive);
outflush();
if(strequals(end))
break;
}
clientclose();
}catch(IOException ex){
}finally{
}
}
public static void main(String[] args)throws IOException{
ServerSocket server=new ServerSocket();
while(true){
//transfer location change Single User or Multi User
MultiUser mu=new MultiUser(serveraccept());
mustart();
}
}
}
我的类直接从Thread类继承了下来并且通过构造函数传递引用和客户Socket建立了联系这样每个线程就有了一个通讯管道同样我们可以填写run方法把之前的操作交给线程来完成这样多客户并行的Socket就建立起来了
我的文章写完了虽然我的文章到这里就结束了可是Java的Socket相关的操作还有很多各位大侠继续努力吧