无法使用Atomikos启动UserTransaction

时间:2017-01-18 18:28:13

标签: java transactions atomikos

我正在测试atomikos事务和数据库连接的东西,作为其中的一部分,我试图执行下面的代码,

import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.util.Properties;

import javax.sql.XAConnection;
import javax.transaction.Transaction;
import javax.transaction.xa.XAResource;

import com.atomikos.datasource.xa.jdbc.JdbcTransactionalResource;
import com.atomikos.icatch.config.UserTransactionServiceImp;
import com.atomikos.icatch.jta.UserTransactionManager;
import com.atomikos.persistence.imp.StateRecoveryManagerImp;
import com.mysql.jdbc.jdbc2.optional.MysqlXADataSource;

/**
 * Working out how to use Atomikos, before building {@link Main}. It is not
 * intended that you write your applications like this - use JCA+EJB or Spring
 * instead! There is way too much boilerplate code here. Based on examples found
 * at the Atomikos website.
 */
public class TestAtomikos {

    public static void main(String[] args) throws Exception {

        MysqlXADataSource mysql = new MysqlXADataSource();
        mysql.setUser("root");
        mysql.setPassword("root");
        mysql.setUrl("jdbc:mysql://localhost:3306/world?useSSL=false");
        JdbcTransactionalResource mysqlResource = new JdbcTransactionalResource(
                "jdbc/mysql", mysql);

        UserTransactionServiceImp utsi = new UserTransactionServiceImp();
        utsi.registerResource(mysqlResource);
        Properties prop = new Properties();
        InputStream input = null;
        //StateRecoveryManagerImp srmi = new StateRecoveryManagerImp(null);

        try {

            input = new FileInputStream("C:\\Users\\abcd\\eclipse\\workspace_Tomcat\\JNDI\\src\\main\\resources\\jta.properties");

            // load a properties file
            prop.load(input);

        }
        catch (IOException ex) {
            ex.printStackTrace();
        } 

        utsi.init(prop);
        //utsi.init();

        UserTransactionManager utm = new UserTransactionManager();
        //utm.init();
        utm.begin();
        Transaction tx = utm.getTransaction();
        XAConnection xamysql = mysql.getXAConnection();
        XAResource db = xamysql.getXAResource();
        tx.enlistResource(db);

        Connection connection = xamysql.getConnection();
        PreparedStatement stmt=connection.prepareStatement("SELECT ID, Name FROM city");
        ResultSet rs = null;
        rs = stmt.executeQuery("SELECT ID, Name FROM city");
         while(rs.next())
         {
             System.out.println(rs.getInt("ID") + "  " +  rs.getString("Name"));
         }

    }
}

但是我收到了一个错误,

log4j:WARN No appenders could be found for logger (com.atomikos.logging.LoggerFactory).
log4j:WARN Please initialize the log4j system properly.
log4j:WARN See http://logging.apache.org/log4j/1.2/faq.html#noconfig for more info.
Exception in thread "main" java.lang.NoSuchMethodError: com.atomikos.persistence.imp.StateRecoveryManagerImp: method <init>()V not found
    at com.atomikos.icatch.standalone.UserTransactionServiceImp.createDefault(UserTransactionServiceImp.java:205)
    at com.atomikos.icatch.standalone.UserTransactionServiceImp.init(UserTransactionServiceImp.java:258)
    at com.atomikos.icatch.config.UserTransactionServiceImp.init(UserTransactionServiceImp.java:405)
    at com.atomikos.icatch.config.UserTransactionServiceImp.init(UserTransactionServiceImp.java:577)
    at com.test.abcd.TestAtomikos.main(TestAtomikos.java:57)

我尝试通过将atomikos与tomcat集成并调用一个调用db的servlet来执行相同的操作。我得到了同样的错误,我厌倦了在线检查StateRecoveryManagerImp的代码,我看到定义的init方法。我不确定是什么导致了这个问题。我只尝试了数据库的东西,它工作正常,我能够执行查询并获得结果。

我尝试使用各种版本的atomikos罐子,但没有运气。关于如何解决此问题的任何建议?

1 个答案:

答案 0 :(得分:0)

你得到的例外是:

  

的NoSuchMethodError:   com.atomikos.persistence.imp.StateRecoveryManagerImp:method()V   找不到

表示它在运行时尝试访问类StateRecoveryManagerImp的默认构造函数(不带参数的构造函数),但找不到它。

当您使用一个版本的库构建应用程序并且针对不兼容的其他版本运行它时,这通常是一个例外。

据我所知,从源代码中我可以看出,您使用transactions 3.9.0或更高版本编译了代码,但没有定义构造函数,这意味着我们拥有默认构造函数(如您可以看到here),并且您的应用程序在运行时使用的旧版本具有public StateRecoveryManagerImp ( ObjectLog objectlog )类型的构造函数,这意味着我们没有默认构造函数(如您所见here)所以当它试图在运行时调用它时它会失败。

要解决您的问题,只需检查运行时使用的类路径,并确保只有transactions的版本与编译时使用的版本相对应。