IBM®
跳转到主要内容
    中国 [选择]    使用条款
 
 
Select a scope: Search for:    
    首页    产品    服务与解决方案     支持与下载    个性化服务    
跳转到主要内容

developerWorks 中国  >  WebSphere  >

最佳实践: 避免或最小化 Servlet 中的同步

developerWorks
文档选项

未显示需要 JavaScript 的文档选项


级别: 初级

Harvey W. Gunther (hgunther@us.ibm.com), WebSphere 产品开发小组中的资深性能分析师, IBM

2002 年 3 月 01 日

请将 servlet 中同步的使用最小化。因为 servlet 是多线程的,主要代码路径的同步会严重地且极为有害地影响性能。

读者: 开发人员

产品: WebSphere Application Server

版本: 版本 3.0.2.x,3.5.x 和 4.0

平台: 所有平台

关键字: Servlet,JSP

摘要

最小化 servlet 中同步的使用。因为 servlet 是多线程的,主要代码路径的同步会严重地且极为有害地影响性能。

建议

servlet 是多线程的。基于 servlet 的应用程序必须认识并适当地处理这一点。如果应用程序有很多大段的代码是同步的,那么这个应用程序实际上就变成单线程的,而且吞吐量会显著下降。

在 servlet 中不出现同步是最佳选择,然而,如果应用程序设计无法避免同步,那么请使用“锁对象(lock Object)”并且锁定可用性最小的代码路径。请不要同步 servlet 的 service 方法或 doGet 以及 doPost 方法。这些方法是主要代码路径。同步这些方法或任何这些 servlet 方法之一将锁定整个 servlet 实例。下列代码显示了一个使用“锁对象”来保护 servlet 实例变量 numberOfRows 的示例。

最小同步代码路径

        public class BpAllBadThingsServletsV1b extends HttpServlet
{
    private int numberOfRows = 0;
    private javax.sql.DataSource ds = null;
    private Object lockObject = new Object();
    public void doGet(HttpServletRequest request, HttpServletResponse response)
                        throws ServletException, IOException
     {
               Connection conn = null;
               ResultSet rs = null;
               PreparedStatement pStmt = null;
               int startingRows = 0;
               synchronize(lockObject)               {
                                 startingRows = numberOfRows;
               }
               try
               {
                         String employeeInformation = null;
                         conn = ds.getConnection("db2admin", "db2admin");
                         pStmt = conn.prepareStatement
                              ("select * from db2admin.employee");
                         rs = pStmt.executeQuery();
               }
               catch (Exception es)
               {
                        // Error handling code here
               }
      }
}
    
      

应被取代的方法

以下代码显示如何同步主要代码路径来保护称为 numberOfRows 的 servlet 实例变量。

使用 javax.servlet.SingleThreadModel 仍是另一种保护 servlet 实例变量的方法,但最好还是避免使用这种方法。

下面的图 1 显示了同步的性能影响

锁定主要代码路径:过度的同步

        public class BpAllBadThingsServletsV1a extends HttpServlet
{
     private int numberOfRows = 0;
     private javax.sql.DataSource ds = null;
     public void doGet(HttpServletRequest request, HttpServletResponse response)
                     throws ServletException, IOException
     {
             Connection conn = null;
             ResultSet rs = null;
             PreparedStatement pStmt = null;
             int startingRows;
             try
             {
                 synchronized(this) // Locks out Most of the Servlet Processing
                 {
                         startingRows = numberOfRows;
                         String employeeInformation = null;
                         conn = ds.getConnection("db2admin", "db2admin");
                         pStmt = conn.prepareStatement
                              ("select * from db2admin.employee");
                         rs = pStmt.executeQuery();
                 }
              }
             catch (Exception es)
             {
                 // Error handling code here
             }
     }
}               
    
      



图 1 -- 性能影响 -- 同步
性能影响


参考资料



关于作者

Harvey W. Gunther 是 IBM 在北卡罗莱纳州 Raleigh 的 WebSphere 产品开发小组中的资深性能分析师。可以通过 hgunther@us.ibm.com与他联系。




对本文的评价

太差! (1)
需提高 (2)
一般;尚可 (3)
好文章 (4)
真棒!(5)

建议?







回页首


IBM 公司保留在 developerWorks 网站上发表的内容的著作权。未经IBM公司或原始作者的书面明确许可,请勿转载。如果您希望转载,请通过 提交转载请求表单 联系我们的编辑团队。
    关于 IBM 隐私条约 联系 IBM 使用条款