网站首页
JSP空间
动态资讯
开源项目
技术文档
资源下载
J2EE资源
客户论坛
在线支付
 
  技术文档>>JAVA>>新手入门>>基础入门>查看文档  
  java5.0多线程编程实践     
  文章作者:未知  文章来源:水木森林  
  查看:67次  录入:管理员--2007-11-17  
       java5增加了新的类库并发集java.util.concurrent,该类库为并发程序提供了丰富的api多线程编程在java 5中更加容易,灵活。本文通过一个网络服务器模型,来实践java5的多线程编程,该模型中使用了java5中的线程池,阻塞队列,可重入锁等,还实践了callable, future等接口,并使用了java 5的另外一个新特性泛型。

  简介

  本文将实现一个网络服务器模型,一旦有客户端连接到该服务器,则启动一个新线程为该连接服务,服务内容为往客户端输送一些字符信息。一个典型的网络服务器模型如下:

  1. 建立监听端口。

  2. 发现有新连接,接受连接,启动线程,执行服务线程。 3. 服务完毕,关闭线程。

  这个模型在大部分情况下运行良好,但是需要频繁的处理用户请求而每次请求需要的服务又是简短的时候,系统会将大量的时间花费在线程的创建销毁。java 5的线程池克服了这些缺点。通过对重用线程来执行多个任务,避免了频繁线程的创建与销毁开销,使得服务器的性能方面得到很大提高。因此,本文的网络服务器模型将如下:

  1. 建立监听端口,创建线程池。

  2. 发现有新连接,使用线程池来执行服务任务。

  3. 服务完毕,释放线程到线程池。

  下面详细介绍如何使用java 5的concurrent包提供的api来实现该服务器。

  初始化

  初始化包括创建线程池以及初始化监听端口。创建线程池可以通过调用java.util.concurrent.executors类里的静态方法newchahedthreadpool或是newfixedthreadpool来创建,也可以通过新建一个java.util.concurrent.threadpoolexecutor实例来执行任务。这里我们采用newfixedthreadpool方法来建立线程池。

executorservice pool = executors.newfixedthreadpool(10);

  表示新建了一个线程池,线程池里面有10个线程为任务队列服务。

  使用serversocket对象来初始化监听端口。

private static final int port = 19527;
serverlistensocket = new serversocket(port);
serverlistensocket.setreuseaddress(true);
serverlistensocket.setreuseaddress(true);

  服务新连接

  当有新连接建立时,accept返回时,将服务任务提交给线程池执行。

while(true){
 socket socket = serverlistensocket.accept();
 pool.execute(new servicethread(socket));
}

  这里使用线程池对象来执行线程,减少了每次线程创建和销毁的开销。任务执行完毕,线程释放到线程池。

  服务任务

  服务线程servicethread维护一个count来记录服务线程被调用的次数。每当服务任务被调用一次时,count的值自增1,因此servicethread提供一个increasecount和getcount的方法,分别将count值自增1和取得该count值。由于可能多个线程存在竞争,同时访问count,因此需要加锁机制,在java 5之前,我们只能使用synchronized来锁定。java 5中引入了性能更加粒度更细的重入锁reentrantlock。我们使用reentrantlock保证代码线程安全。下面是具体代码:

private static reentrantlock lock = new reentrantlock ();
private static int count = 0;
private int getcount(){
 int ret = 0;
 try{
  lock.lock();
  ret = count;
 }finally{
  lock.unlock();
 }
 return ret;
}
private void increasecount(){
 try{
  lock.lock();
  ++count;
 }finally{
  lock.unlock();
 }
}

  服务线程在开始给客户端打印一个欢迎信息,

increasecount();
int curcount = getcount();
hellostring = "hello, id = " + curcount+"/r/n";
dos = new dataoutputstream(connectedsocket.getoutputstream());
dos.write(hellostring.getbytes());

  然后使用executorservice的submit方法提交一个callable的任务,返回一个future接口的引用。这种做法对费时的任务非常有效,submit任务之后可以继续执行下面的代码,然后在适当的位置可以使用future的get方法来获取结果,如果这时候该方法已经执行完毕,则无需等待即可获得结果,如果还在执行,则等待到运行完毕。

executorservice executor = executors.newsinglethreadexecutor();
future future = executor.submit(new timeconsumingtask());
dos.write("let's do soemthing other".getbytes());
string result = future.get();
dos.write(result.getbytes());

  其中timeconsumingtask实现了callable接口

class timeconsumingtask implements callable {
 public string call() throws exception {
  system.out.println("it's a time-consuming task, you'd better retrieve your result in the furture");
  return "ok, here's the result: it takes me lots of time to produce this result";
 }
}

  这里使用了java 5的另外一个新特性泛型,声明timeconsumingtask的时候使用了string做为类型参数。必须实现callable接口的call函数,其作用类似与runnable中的run函数,在call函数里写入要执行的代码,其返回值类型等同于在类声明中传入的类型值。在这段程序中,我们提交了一个callable的任务,然后程序不会堵塞,而是继续执行dos.write("let's do soemthing other".getbytes());当程序执行到string result = future.get()时如果call函数已经执行完毕,则取得返回值,如果还在执行,则等待其执行完毕。
 
 
上一篇: 从java应用程序动态生成pdf文件    下一篇: java中c/s通讯程序设计一例
  相关文档
java 各类本地接口——规范大全 11-17
初探.net中的delegate类型与.net事件 11-17
小tip:java里的时间比较 11-17
spring快速入门教程 11-17
j2se中的序列化详解(二) 11-16
java学习从入门到精通 4方法篇 11-16
提高ejb性能的十大技巧 11-17
nhibernate与代码生成器 11-17
j2ee综合:如何实现javabean的属性拷贝 03-14
java api之实现(上) 11-16
java学生成绩管理系统源代码 11-17
java高级日期概念 (献给那些要国际化时间及sql时间的兄弟) 11-17
在javascript程序中整合java函数 11-16
JAVA基础:浅谈Java与C#的事件处理机制 08-06
使用jdbc创建数据库对象(2) 11-17
java设计模式之observer 11-17
tomcat 服务器server.xml的关键参数配置 11-17
java异常处理机制的详细讲解和使用技巧 11-16
对比java语言中的覆盖和重载 11-17
round 方法 11-16
返回首页 | 关于我们 | J网章程 | JSP空间合租 | 客服中心 | 免责声明 | 常见问题 | 参观机房
本站主机空间代理至厦门市华众网络科技有限公司
《中华人民共和国增值电信业务经营许可证》
编号:闽B2-20050079
@2005-2008福建JSP技术网 版权所有 闽ICP备05000928号
厦门(总部):13616026886 福州:0591-87655121
邮箱:admin@fjjsp.com 站长QQ,点击这里给我发消息