FIX: Thread Blocking Issues When Using JDBC-ODBC Bridge (191932)



The information in this article applies to:

  • Microsoft SDK for Java 1.0
  • Microsoft SDK for Java 1.5
  • Microsoft SDK for Java 1.51
  • Microsoft SDK for Java 2.0
  • Microsoft SDK for Java 3.0
  • Microsoft SDK for Java 2.01
  • Microsoft SDK for Java 2.02

This article was previously published under Q191932

SYMPTOMS

When using the Microsoft supplied JDBC-ODBC Bridge component for JDBC (Msjdbc10.dll and associated Java class files) with multiple threads in a Java application or applet, the threads using JDBC appear to hang. Running a JDBC statement on more than one thread results in thread blocking, even when using a thread safe non-blocking ODBC driver such as the Microsoft SQL Server ODBC driver.

CAUSE

The Microsoft JDBC-ODBC Bridge globally synchronizes all method calls. This means that a call to any JDBC method that does not immediately return will block all other JDBC calls from other threads. If your JDBC code calls executeQuery for example, and the query takes five seconds to run, then all other JDBC using threads in your application will hang for five seconds.

This can easily cause a deadlock when one JDBC using thread has locked a table and another JDBC using thread tries to lock the same table. Because the second thread is blocked by the first threads table lock, the second thread waits inside the JDBC call to lock the table -- therefore preventing the first thread from making the necessary JDBC calls to unlock the table.

RESOLUTION

Update your virtual machine to the latest version.

STATUS

Microsoft has confirmed this to be a bug in the Microsoft products listed at the beginning of this article. This bug has been corrected in the Virtual Machine that is included with SDK for Java 3.1.

MORE INFORMATION

Steps to Reproduce the Behavior

The following code sample demonstrates the global thread blocking behavior with the Microsoft SQL Server ODBC driver. Run the sample using jview so you can watch the output as the code runs. If you run the sample as-is (modifying the connection string to point to your SQL Server if you don't have a local server), you will notice that all 10 threads block on the Java side but SQL Server only reports that one spid is blocked (using the sp_who2 command from an ISQL window).

If you uncomment the setQueryTimeout line:
   // stmt.setQueryTimeout(5);
				
you will notice that the threads can continue after the timeout expires, but there is still quite a bit of blocking.

If you uncomment the synchronization code block:
   /*synchronized static*/ 
				
you will notice that the threads no longer appear to hang; you have effectively synchronized all thread's access to the locking code.
   // START CODE SAMPLE
   import java.sql.*;
   import java.util.*;

   public class TestJDBC
   {
     static int THREAD_COUNT = 10;
     static int LOOP_COUNT   = 50;
     public static void main(String args[])
     {
       JDBCThread jdbcThreads[];
       int i;

       // Load the JDBC-ODBC bridge driver.
       try
       {
         Class.forName( "com.ms.jdbc.odbc.JdbcOdbcDriver" );
       }
       catch (ClassNotFoundException cnfEX)
       {
         System.out.println( "ClassNotFoundException: " +
           cnfEX.toString() );
         return;
       }

       // Create THREAD_COUNT worker threads.
       jdbcThreads = new JDBCThread[THREAD_COUNT];
       for (i=0; i<THREAD_COUNT; i++ )
       {
         jdbcThreads[i] = new JDBCThread( i, LOOP_COUNT );
       }

       // Start all worker threads.
       System.out.println( "Starting worker threads." );
       for (i=0; i<THREAD_COUNT; i++ )
       {
         jdbcThreads[i].start();
       }
       System.out.println( "All worker threads started." );
     }
   }

   // 
   // JDBC worker thread.
   // 
   class JDBCThread extends Thread
   {
     private int tid = 0;
     private int loopcount = 0;

     // Modify following connect string to point to your SQL Server.
     private static String connectInfo =
       "JDBC:ODBC:DRIVER={SQL Server};" +
       "SERVER=(local);DATABASE=pubs;UID=sa;PWD=;";
     private static String sql =
       "select * from authors (tablockx)";
     private java.sql.Connection conn = null;
     private java.sql.Statement stmt  = null;

     JDBCThread( int threadid, int iterations )
     {
       tid = threadid;
       loopcount = iterations;
       try
       {
         // Open connection to database.
         conn = DriverManager.getConnection ( connectInfo, "", "" );

         // Set connection to manual transaction mode so tablockx works.
         conn.setAutoCommit( false );
       }
       catch( Exception e )
       {
         System.out.println( e.toString() );
         conn = null;
       }
     }

     public /*synchronized static*/ void
       doExecute( int tid, Connection conn,
     Statement stmt, String sql ) throws SQLException
     {
       java.sql.ResultSet rs = null;

       System.out.println( "JDBCThread[" + tid +
         "] entering doExecute()" );

       // Open resultset.  Setting query timeout is recommended.
       // stmt.setQueryTimeout(5);
       rs = stmt.executeQuery( sql );

       // Close resultset and statement.
       rs.close();
       stmt.close();
       stmt = null;

       // Clear transaction and flag success.
       conn.rollback();
       System.out.println( "JDBCThread[" + tid + "] exiting doExecute()" );
     }
     public void run()
     {
       int i;

       if ( null == conn )
       {
         System.out.println( "Connection not open, exiting." );
         return;
       }

       System.out.println( "JDBCThread[" + tid + "] starting." );
       for ( i=1; i<=loopcount; i++ )
       {
         try
         {
           // Create statement and do some statement work.
           stmt = conn.createStatement();
           doExecute( tid, conn, stmt, sql );
         }
         catch( SQLException sqlEX )
         {
           while ( null != sqlEX )
           {
             System.out.println( "Thread[" + tid + "] SQLException: " +
                                                      sqlEX.toString() );
             sqlEX = sqlEX.getNextException();
           }
         }
         catch( Exception e)
         {
           if ( null != stmt )
           {
             try { stmt.close(); } catch( Exception e1) {};
           }
           stmt = null;
         };
       }
       System.out.println( "JDBCThread[" + tid + "] exiting." );
     }
   }
   // END CODE SAMPLE
				

REFERENCES

For the latest Knowledge Base articles and other support information about Visual J++ and the SDK for Java, see the following pages on the Microsoft Technical Support site: For more information about Virtual Machine for Java and SDK for Java, visit the following Microsoft Web site:

Modification Type:MinorLast Reviewed:11/17/2005
Keywords:kbbug kbfix kbJava kbprb KB191932