Atomikos中文说明文档_com.atomikos.icatch.service-程序员宅基地

技术标签: Atomikos-分布式事物管理  

Atomikos 翻译文档(英文文档来源:下载安装包中START_HERE.html)

                                 ----译者:周枫

请尊重劳动成果,转载请标明,英语水平有限,如有不准确地方请在评论中指出,谢谢

官网地址:http://www.atomikos.com/Main/WebHome

使用版本:AtomikosTransactionsEssentials-3.7.2

感谢您使用Atomikos,下面的说明文档可以让您正确使用,如果您有任何问题或者反馈,请访问我们的帮助网页http://www.atomikos.com/Main/SupportOverview,或者给我们发送邮件[email protected]

什么是Atomikos TransactionsEssentials

         Atomikos TransactionsEssentials 是一个为Java平台提供增值服务的并且开源类事务管理器,以下是包括在这个开源版本中的一些功能:

l  全面崩溃 / 重启恢复

l  兼容标准的SUN公司JTA API

l  嵌套事务

l  为XA和非XA提供内置的JDBC适配器

注释:XA:XA协议由Tuxedo首先提出的,并交给X/Open组织,作为资源管理器(数据库)与事务管理器的接口标准。目前,Oracle、Informix、DB2和Sybase等各大数据库厂家都提供对XA的支持。XA协议采用两阶段提交方式来管理分布式事务。XA接口提供资源管理器与事务管理器之间进行通信的标准接口。XA协议包括两套函数,以xa_开头的及以ax_开头的。

以下的函数使事务管理器可以对资源管理器进行的操作:

  1)xa_open,xa_close:建立和关闭与资源管理器的连接。

  2)xa_start,xa_end:开始和结束一个本地事务。

  3)xa_prepare,xa_commit,xa_rollback:预提交、提交和回滚一个本地事务。

  4)xa_recover:回滚一个已进行预提交的事务。

  5)ax_开头的函数使资源管理器可以动态地在事务管理器中进行注册,并可以对XID(TRANSACTION IDS)进行操作。

  6)ax_reg,ax_unreg;允许一个资源管理器在一个TMS(TRANSACTION MANAGER SERVER)中动态注册或撤消注册。

l  内置的JMS适配器XA-capable JMS队列连接器

注释:JMS:jms即Java消息服务(Java Message Service)应用程序接口是一个Java平台中关于面向消息中间件(MOM)的API,用于在两个应用程序之间,或分布式系统中发送消息,进行异步通信。Java消息服务是一个与具体平台无关的API,绝大多数MOM提供商都对JMS提供支持。

l  通过XA API兼容第三方适配器

l  更好的整合您的项目

l  集成Hibernate

如何使用Atomikos TransactionsEssentials

         Atomikos TransactionsEssentials 是一个可靠的库,可以加入到您的Java应用程序,也就是说为了使用这个产品,您必须添加一些jar文件(包括在dist和lib文件夹下)到您的应用程序或者应用程序服务器。

         请注意:Atomikos TransactionsEssentials是一个非常快速的嵌入式事务管理器,这就意味着,您不需要另外启动一个单独的事务管理器进程(不要查找任何的bin文件夹)。相反,您的应用服务器将有它自己的intra-VM事务管理器。

         配置需求:至少Java1.5 jdk,并且最少128M的内存

         性能优化:尽管这个软件有着很大的优势,但是想要更好的发挥其作用,可以按以下的方法优化:

l  更高的内存,意味着更高的吞吐量(每秒的事务数目)

l  使连接池尽可能的大

l  一旦你不需要的连接请马上关闭它们。不要把你的应用程序放在缓存里,让内部连接池为你做这些,这将促使更高效的连接使用

l  不要让活动的事务闲置:终止所有情况下的事务,尤其是在异常报错情况下的事务。这将减少数据库的锁定时间,并且最大效率的处理启用的使用。

如果想获取这些细节的更多信息,也要参阅文档说明部分。

值得注意的是,在我们所有的压力测试中,Atomikos TransactionsEssentials比J2EE的web容器更高效的吞吐量。这些测量值包括日志记录的高效的事务状态,同样,在我们所有的测量中,包括XA和non-XA,高效的效率是一样的。

         在J2SE中使用Atomikos Transactions Essentials,只需要按以下步骤

  1. 将idst和lib中的jar包全部放入的项目中
  2. 创建或者自定义你应用的transactions.properties(或者jta.properties)文件(事务管理器的配置),然后将它放入到classpath中,安装文件夹中包涵一个实例文件;在properties文件中注释(#)后面的是默认值,取消一行并且改变默认值。
    复制代码
    # SAMPLE PROPERTIES FILE FOR THE TRANSACTION SERVICE
    # THIS FILE ILLUSTRATES THE DIFFERENT SETTINGS FOR THE TRANSACTION MANAGER
    # UNCOMMENT THE ASSIGNMENTS TO OVERRIDE DEFAULT VALUES;
    
    # Required: factory implementation class of the transaction core.
    # NOTE: there is no default for this, so it MUST be specified! 
    # 
    com.atomikos.icatch.service=com.atomikos.icatch.standalone.UserTransactionServiceFactory
    
            
    # Set base name of file where messages are output 
    # (also known as the 'console file').
    #
    # com.atomikos.icatch.console_file_name = tm.out
    
    # Size limit (in bytes) for the console file;
    # negative means unlimited.
    #
    # com.atomikos.icatch.console_file_limit=-1
    
    # For size-limited console files, this option
    # specifies a number of rotating files to 
    # maintain.
    #
    # com.atomikos.icatch.console_file_count=1
    
    # Set the number of log writes between checkpoints
    #
    # com.atomikos.icatch.checkpoint_interval=500
    
    # Set output directory where console file and other files are to be put
    # make sure this directory exists!
    #
    # com.atomikos.icatch.output_dir = ./
    
    # Set directory of log files; make sure this directory exists!
    #
    # com.atomikos.icatch.log_base_dir = ./
    
    # Set base name of log file
    # this name will be  used as the first part of 
    # the system-generated log file name
    #
    # com.atomikos.icatch.log_base_name = tmlog
    
    # Set the max number of active local transactions 
    # or -1 for unlimited.
    #
    # com.atomikos.icatch.max_actives = 50
    
    # Set the default timeout (in milliseconds) for local transactions
    #
    # com.atomikos.icatch.default_jta_timeout = 10000
    
    # Set the max timeout (in milliseconds) for local transactions
    #
    # com.atomikos.icatch.max_timeout = 300000
    
    # The globally unique name of this transaction manager process
    # override this value with a globally unique name
    #
    # com.atomikos.icatch.tm_unique_name = tm
        
    # Do we want to use parallel subtransactions? JTA's default
    # is NO for J2EE compatibility
    #
    # com.atomikos.icatch.serial_jta_transactions=true
                        
    # If you want to do explicit resource registration then
    # you need to set this value to false.
    #
    # com.atomikos.icatch.automatic_resource_registration=true  
        
    # Set this to WARN, INFO or DEBUG to control the granularity
    # of output to the console file.
    #
    # com.atomikos.icatch.console_log_level=WARN
        
    # Do you want transaction logging to be enabled or not?
    # If set to false, then no logging overhead will be done
    # at the risk of losing data after restart or crash.
    #
    # com.atomikos.icatch.enable_logging=true
    
    # Should two-phase commit be done in (multi-)threaded mode or not?
    # Set this to false if you want commits to be ordered according
    # to the order in which resources are added to the transaction.
    #
    # NOTE: threads are reused on JDK 1.5 or higher. 
    # For JDK 1.4, thread reuse is enabled as soon as the 
    # concurrent backport is in the classpath - see 
    # http://mirrors.ibiblio.org/pub/mirrors/maven2/backport-util-concurrent/backport-util-concurrent/
    #
    # com.atomikos.icatch.threaded_2pc=false
    
    # Should shutdown of the VM trigger shutdown of the transaction core too?
    #
    # com.atomikos.icatch.force_shutdown_on_vm_exit=false
    复制代码

     

  3. 在你的应用程序中,创建一个实例com.atomikos.icatch.jta.UserTransactionImp或者com.atomikos.icatch.jta.UserTransactionManager(使用默认的无参数构造函数)
    复制代码
    /**
     * Copyright (C) 2000-2010 Atomikos <[email protected]>
     *
     * This code ("Atomikos TransactionsEssentials"), by itself,
     * is being distributed under the
     * Apache License, Version 2.0 ("License"), a copy of which may be found at
     * http://www.atomikos.com/licenses/apache-license-2.0.txt .
     * You may not use this file except in compliance with the License.
     *
     * While the License grants certain patent license rights,
     * those patent license rights only extend to the use of
     * Atomikos TransactionsEssentials by itself.
     *
     * This code (Atomikos TransactionsEssentials) contains certain interfaces
     * in package (namespace) com.atomikos.icatch
     * (including com.atomikos.icatch.Participant) which, if implemented, may
     * infringe one or more patents held by Atomikos.
     * It should be appreciated that you may NOT implement such interfaces;
     * licensing to implement these interfaces must be obtained separately from Atomikos.
     *
     * Unless required by applicable law or agreed to in writing, software
     * distributed under the License is distributed on an "AS IS" BASIS,
     * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     */
    
    package com.atomikos.icatch.jta;
    
    import java.io.Serializable;
    
    import javax.naming.NamingException;
    import javax.naming.Reference;
    import javax.naming.Referenceable;
    import javax.transaction.NotSupportedException;
    import javax.transaction.SystemException;
    import javax.transaction.TransactionManager;
    import javax.transaction.UserTransaction;
    
    import com.atomikos.icatch.admin.imp.SimpleLogAdministrator;
    import com.atomikos.icatch.config.TSInitInfo;
    import com.atomikos.icatch.config.UserTransactionService;
    import com.atomikos.icatch.config.UserTransactionServiceImp;
    import com.atomikos.util.SerializableObjectFactory;
    
    /**
     * 
     * 
     * Our UserTransaction implementation for J2SE transactions. This class is
     * special in that it automatically starts up and recover the transaction
     * service on first use. <b>Note: don't use this class in J2EE applications in
     * order to avoid starting different transaction engines in the same application
     * server! J2EE applications should use J2eeUserTransaction instead.</b>
     */
    
    public class UserTransactionImp implements UserTransaction, Serializable,
            Referenceable
    {
        private transient TransactionManager txmgr_;
    
        /**
         * No-argument constructor.
         */
    
        public UserTransactionImp ()
        {
        }
    
        /**
         * Referenceable mechanism requires later setup of txmgr_, otherwise binding
         * into JNDI already requires that TM is running.
         */
    
        private void checkSetup ()
        {
    
            // REMOVED FOLLOWING IF CHECK: DON'T CACHE THE TXMGR TO MAKE INSTANCES
            // RESILIENT TO RESTART IN TOMCAT. OTHERWISE, CLIENT APPS SEE THEIR
            // USERTX REFERENCES INVALIDATED AND THIS IS INTOLERABLE
            // if ( txmgr_ == null ) {
            // txmgr_ = TransactionManagerImp.getTransactionManager();
    
            synchronized ( TransactionManagerImp.class ) {
    
                txmgr_ = TransactionManagerImp.getTransactionManager ();
    
                // FOLLOWING COMMENTED OUT: NEW RECOVERY IN 2.0 ALLOWS US TO START
                // THE TM
                // IF NOT ALREADY RUNNING!!!
                // if ( txmgr_ == null )
                // throw new RuntimeException ( "No transaction monitor installed?"
                // );
    
                // NEW FROM 2.0: if TM is not running, just start it. Any resources
                // can be registered later.
                if ( txmgr_ == null ) {
                    UserTransactionService uts = new UserTransactionServiceImp ();
                    TSInitInfo info = uts.createTSInitInfo ();
                    uts.registerLogAdministrator ( SimpleLogAdministrator
                            .getInstance () );
                    uts.init ( info );
                    txmgr_ = TransactionManagerImp.getTransactionManager ();
                }
    
            }
    
            // }
        }
    
        /**
         * @see javax.transaction.UserTransaction
         */
    
        public void begin () throws NotSupportedException, SystemException
        {
            checkSetup ();
            txmgr_.begin ();
        }
    
        /**
         * @see javax.transaction.UserTransaction
         */
    
        public void commit () throws javax.transaction.RollbackException,
                javax.transaction.HeuristicMixedException,
                javax.transaction.HeuristicRollbackException,
                javax.transaction.SystemException, java.lang.IllegalStateException,
                java.lang.SecurityException
        {
            checkSetup ();
            txmgr_.commit ();
        }
    
        /**
         * @see javax.transaction.UserTransaction
         */
    
        public void rollback () throws IllegalStateException, SystemException,
                SecurityException
        {
            checkSetup ();
            txmgr_.rollback ();
        }
    
        /**
         * @see javax.transaction.UserTransaction
         */
    
        public void setRollbackOnly () throws IllegalStateException,
                SystemException
        {
            checkSetup ();
            txmgr_.setRollbackOnly ();
        }
    
        /**
         * @see javax.transaction.UserTransaction
         */
    
        public int getStatus () throws SystemException
        {
            checkSetup ();
            return txmgr_.getStatus ();
        }
    
        /**
         * @see javax.transaction.UserTransaction
         */
    
        public void setTransactionTimeout ( int seconds ) throws SystemException
        {
            checkSetup ();
            txmgr_.setTransactionTimeout ( seconds );
        }
    
        //
        //
        // IMPLEMENTATION OF REFERENCEABLE
        //
        //
    
        public Reference getReference () throws NamingException
        {
            return SerializableObjectFactory.createReference ( this );
        }
    }
    复制代码
    复制代码
    /**
     * Copyright (C) 2000-2010 Atomikos <[email protected]>
     *
     * This code ("Atomikos TransactionsEssentials"), by itself,
     * is being distributed under the
     * Apache License, Version 2.0 ("License"), a copy of which may be found at
     * http://www.atomikos.com/licenses/apache-license-2.0.txt .
     * You may not use this file except in compliance with the License.
     *
     * While the License grants certain patent license rights,
     * those patent license rights only extend to the use of
     * Atomikos TransactionsEssentials by itself.
     *
     * This code (Atomikos TransactionsEssentials) contains certain interfaces
     * in package (namespace) com.atomikos.icatch
     * (including com.atomikos.icatch.Participant) which, if implemented, may
     * infringe one or more patents held by Atomikos.
     * It should be appreciated that you may NOT implement such interfaces;
     * licensing to implement these interfaces must be obtained separately from Atomikos.
     *
     * Unless required by applicable law or agreed to in writing, software
     * distributed under the License is distributed on an "AS IS" BASIS,
     * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     */
    
    package com.atomikos.icatch.jta;
    
    import java.io.Serializable;
    
    import javax.naming.NamingException;
    import javax.naming.Reference;
    import javax.naming.Referenceable;
    import javax.transaction.HeuristicMixedException;
    import javax.transaction.HeuristicRollbackException;
    import javax.transaction.InvalidTransactionException;
    import javax.transaction.NotSupportedException;
    import javax.transaction.RollbackException;
    import javax.transaction.SystemException;
    import javax.transaction.Transaction;
    import javax.transaction.TransactionManager;
    import javax.transaction.UserTransaction;
    
    import com.atomikos.icatch.config.TSInitInfo;
    import com.atomikos.icatch.config.UserTransactionService;
    import com.atomikos.icatch.config.UserTransactionServiceImp;
    import com.atomikos.util.SerializableObjectFactory;
    
    /**
     * 
     * 
     * 
     * 
     * 
     * A straightforward, zero-setup implementation of a transaction manager. J2SE
     * applications can use an instance of this class to get a handle to the
     * transaction manager, and automatically startup or recover the transaction
     * service on first use. <b>J2EE applications should NOT use this class in order
     * to avoid the concurrent use of different transaction services. For J2EE
     * applications, we have the class J2eeTransactionManager instead.</b>
     */
    public class UserTransactionManager implements TransactionManager,
            Serializable, Referenceable, UserTransaction
    {
        private static final long serialVersionUID = -655789038710288096L;
    
        private transient TransactionManagerImp tm;
        
        private UserTransactionService uts;
    
        private boolean forceShutdown;
        
        private boolean startupTransactionService;
        
        private boolean closed;
    
        private void checkSetup () throws SystemException
        {
            if ( closed ) throw new SystemException ( "This UserTransactionManager instance was closed already. Call init() to reuse if desired." );
                
            synchronized ( TransactionManagerImp.class ) {
    
                tm = (TransactionManagerImp) TransactionManagerImp
                        .getTransactionManager ();
                if ( tm == null ) {
                    // not initialized -> startup TM
                    // System.out.println ( "STARTING UP TM!!!!!!");
                       if ( getStartupTransactionService() ) {
                        uts = new UserTransactionServiceImp ();
                        TSInitInfo info = uts.createTSInitInfo ();
                        uts.init ( info );
                        tm = (TransactionManagerImp) TransactionManagerImp
                                .getTransactionManager ();
                       }
                       else {
                           throw new SystemException ( "Transaction service not running" );
                       }
                }
            }
        }
        
        public UserTransactionManager()
        {
                //startup by default, to have backward compatibility
                this.startupTransactionService = true;
                this.closed = false;
        }
        
        /**
         * Sets whether the transaction service should be 
         * started if not already running. 
         * @param startup
         */
        public void setStartupTransactionService ( boolean startup )
        {
                this.startupTransactionService = startup;
        }
        
        /**
         * Returns true if the transaction service will 
         * be started if not already running.
         * @return
         */
        public boolean getStartupTransactionService()
        {
                return this.startupTransactionService;
        }
        
        /**
         * Performs initialization if necessary.
         * This will startup the TM (if not running)
         * and perform recovery, unless <b>getStartupTransactionService</b>
         * returns false.
         * 
         * @throws SystemException
         */
        
        public void init() throws SystemException
        {
                closed = false;
                checkSetup();
        }
    
        /**
         * @see javax.transaction.TransactionManager#begin()
         */
        public void begin () throws NotSupportedException, SystemException
        {
            checkSetup ();
            tm.begin ();
    
        }
        
        public boolean getForceShutdown()
        {
                return forceShutdown;
        }
        
        /**
         * Sets the force shutdown mode to use during close.
         * @param value 
         */
        public void setForceShutdown ( boolean value )
        {
                this.forceShutdown = value;
        }
    
        /**
         * @see javax.transaction.TransactionManager#commit()
         */
        public void commit () throws RollbackException, HeuristicMixedException,
                HeuristicRollbackException, SecurityException,
                IllegalStateException, SystemException
        {
            checkSetup ();
            tm.commit ();
    
        }
    
        /**
         * @see javax.transaction.TransactionManager#getStatus()
         */
        public int getStatus () throws SystemException
        {
            checkSetup ();
            return tm.getStatus ();
        }
    
        /**
         * @see javax.transaction.TransactionManager#getTransaction()
         */
        public Transaction getTransaction () throws SystemException
        {
            checkSetup ();
            return tm.getTransaction ();
        }
    
        /**
         * @see javax.transaction.TransactionManager#resume(javax.transaction.Transaction)
         */
        public void resume ( Transaction tx ) throws InvalidTransactionException,
                IllegalStateException, SystemException
        {
            checkSetup ();
            tm.resume ( tx );
    
        }
    
        /**
         * @see javax.transaction.TransactionManager#rollback()
         */
        public void rollback () throws IllegalStateException, SecurityException,
                SystemException
        {
            checkSetup ();
            tm.rollback ();
    
        }
    
        /**
         * @see javax.transaction.TransactionManager#setRollbackOnly()
         */
        public void setRollbackOnly () throws IllegalStateException,
                SystemException
        {
            checkSetup ();
            tm.setRollbackOnly ();
    
        }
    
        /**
         * @see javax.transaction.TransactionManager#setTransactionTimeout(int)
         */
        public void setTransactionTimeout ( int secs ) throws SystemException
        {
            checkSetup ();
            tm.setTransactionTimeout ( secs );
    
        }
    
        /**
         * @see javax.transaction.TransactionManager#suspend()
         */
        public Transaction suspend () throws SystemException
        {
            checkSetup ();
            return tm.suspend ();
        }
    
        /**
         * @see javax.naming.Referenceable#getReference()
         */
        public Reference getReference () throws NamingException
        {
            return SerializableObjectFactory.createReference ( this );
        }
        
        /**
         * Closes the transaction service, but only if it was 
         * implicitly started via this instance.
         * In other words, if the transaction service was started
         * in another way then this method will not do anything.
         *
         */
        public void close()
        {
                if ( uts != null ) {
                    uts.shutdown ( forceShutdown );
                    uts = null;
                }
                closed = true;
        }
    
    }
    复制代码

     

  4. 对于JDBC,使用我们的一个实例com.atomikos.jdbc.AtomikosDataSourceBean或者,对于non-XA驱动,可以使用com.atomikos.jdbc.nonxa.AtomikosNonXADataSourceBean
    复制代码
    /**
     * Copyright (C) 2000-2010 Atomikos <[email protected]>
     *
     * This code ("Atomikos TransactionsEssentials"), by itself,
     * is being distributed under the
     * Apache License, Version 2.0 ("License"), a copy of which may be found at
     * http://www.atomikos.com/licenses/apache-license-2.0.txt .
     * You may not use this file except in compliance with the License.
     *
     * While the License grants certain patent license rights,
     * those patent license rights only extend to the use of
     * Atomikos TransactionsEssentials by itself.
     *
     * This code (Atomikos TransactionsEssentials) contains certain interfaces
     * in package (namespace) com.atomikos.icatch
     * (including com.atomikos.icatch.Participant) which, if implemented, may
     * infringe one or more patents held by Atomikos.
     * It should be appreciated that you may NOT implement such interfaces;
     * licensing to implement these interfaces must be obtained separately from Atomikos.
     *
     * Unless required by applicable law or agreed to in writing, software
     * distributed under the License is distributed on an "AS IS" BASIS,
     * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     */
    
    package com.atomikos.jdbc;
    import java.util.Enumeration;
    import java.util.Properties;
    
    import javax.sql.XADataSource;
    
    import com.atomikos.beans.PropertyUtils;
    import com.atomikos.datasource.RecoverableResource;
    import com.atomikos.datasource.xa.jdbc.JdbcTransactionalResource;
    import com.atomikos.icatch.system.Configuration;
    import com.atomikos.jdbc.AbstractDataSourceBean;
    import com.atomikos.util.ClassLoadingHelper;
    
     /**
     * The preferred class for using Atomikos connection pooling. Use an instance of
     * this class if you want to use Atomikos JTA-enabled connection pooling. All
     * you need to do is construct an instance and set the required properties as
     * outlined below. The resulting bean will automatically register with the
     * transaction service (for recovery) and take part in active transactions.
     * All SQL done over connections (gotten from this class) will participate in JTA transactions.
     */
    
    public class AtomikosDataSourceBean 
    extends AbstractDataSourceBean
    {
        
        
        private static final long serialVersionUID = 1L;
        
        private Properties xaProperties = null;
        private String xaDataSourceClassName;
        private transient XADataSource xaDataSource;
        public AtomikosDataSourceBean() 
        {
            this.xaProperties = new Properties();
        }
        
        protected String printXaProperties()
        {
            StringBuffer ret = new StringBuffer();
            if ( xaProperties != null ) {
                Enumeration it = xaProperties.propertyNames();
                ret.append ( "[" );
                boolean first = true;
                while ( it.hasMoreElements() ) {
                    if ( ! first ) ret.append ( "," );
                    String name = ( String ) it.nextElement();
                    String value = xaProperties.getProperty( name);
                    ret.append ( name ); ret.append ( "=" ); ret.append ( value );
                    first = false;
                }
                ret.append ( "]" );
            }
            return ret.toString();
        }
        
        /**
         * Gets the properties used to
         * configure the XADataSource.  
         */
        
        public Properties getXaProperties()
        {
            return xaProperties;
        }
    
        /**
         * Sets the properties (name,value pairs) used to
         * configure the XADataSource. Required, unless you call setXaDataSource directly.
         * 
         * @param xaProperties 
         * 
         *
         */
        public void setXaProperties ( Properties xaProperties ) 
        {
            this.xaProperties = xaProperties;
        }
    
        /**
         * Get the XADataSource class name.
         */
        public String getXaDataSourceClassName() 
        {
            return xaDataSourceClassName;
        }
    
        /**
         * Sets the fully qualified underlying XADataSource class name. Required, unless you 
         * call setXaDataSource directly.
         * 
         * @param xaDataSourceClassName
         */
        public void setXaDataSourceClassName ( String xaDataSourceClassName ) 
        {
            this.xaDataSourceClassName = xaDataSourceClassName;
        }
        
        /**
         * Gets the configured XADataSource (if any).
         * @return The instance, or null if none.
         */
        
        public XADataSource getXaDataSource()
        {
            return xaDataSource;
        }
        
        /**
         * Sets the XADataSource directly - instead of providing the xaDataSourceClassName and xaProperties.
         * @param xaDataSource
         */
        public void setXaDataSource(XADataSource xaDataSource)
        {
            this.xaDataSource = xaDataSource;
        }
        
        
        protected com.atomikos.datasource.pool.ConnectionFactory doInit() throws Exception 
        {
            if (xaDataSource == null)
            {
                if (xaDataSourceClassName == null)
                    throwAtomikosSQLException("Property 'xaDataSourceClassName' cannot be null");
                if (xaProperties == null)
                    throwAtomikosSQLException("Property 'xaProperties' cannot be null");
            }
            
            
            if ( Configuration.isInfoLoggingEnabled() ) Configuration.logInfo(
                    this + ": initializing with [" +
                    " xaDataSourceClassName=" + xaDataSourceClassName + "," +
                    " uniqueResourceName=" + getUniqueResourceName() + "," +
                    " maxPoolSize=" + getMaxPoolSize() + "," +
                    " minPoolSize=" + getMinPoolSize() + "," +
                    " borrowConnectionTimeout=" + getBorrowConnectionTimeout() + "," +
                    " maxIdleTime=" + getMaxIdleTime() + "," +
                    " reapTimeout=" + getReapTimeout() + "," +
                    " maintenanceInterval=" + getMaintenanceInterval() + "," +
                    " testQuery=" + getTestQuery() + "," +
                    " xaProperties=" + printXaProperties() + 
                    " loginTimeout=" + getLoginTimeout() +
                    "]"
                    );
            
            
                if (xaDataSource == null)
                {
                    Class xadsClass = null;
                    try {
                        xadsClass = ClassLoadingHelper.loadClass ( getXaDataSourceClassName() );
                    } catch ( ClassNotFoundException nf ) {
                        AtomikosSQLException.throwAtomikosSQLException ( "The class '" + getXaDataSourceClassName() +
                                "' specified by property 'xaDataSourceClassName' could not be found in the classpath. Please make sure the spelling is correct, and that the required jar(s) are in the classpath." , nf );
                         
                    }
                    Object driver =  xadsClass.newInstance();
                    if ( ! ( driver instanceof XADataSource ) ) {
                        AtomikosSQLException.throwAtomikosSQLException (
                                 "The class '" + getXaDataSourceClassName() +
                                    "' specified by property 'xaDataSourceClassName' does not implement the required interface javax.jdbc.XADataSource. Please make sure the spelling is correct, and check your JDBC driver vendor's documentation." 
                        );
                    }
                    xaDataSource = (XADataSource) driver;
                    xaDataSource.setLoginTimeout ( getLoginTimeout() );
                    xaDataSource.setLogWriter ( getLogWriter() );
                    PropertyUtils.setProperties(xaDataSource, xaProperties );
                }
                
                JdbcTransactionalResource tr = new JdbcTransactionalResource(getUniqueResourceName() , xaDataSource);
                com.atomikos.datasource.pool.ConnectionFactory cf = new com.atomikos.jdbc.AtomikosXAConnectionFactory(xaDataSource, tr, this);
                Configuration.addResource ( tr );
                
                return cf;
        }
        
        protected void doClose() 
        {
            RecoverableResource res = Configuration.getResource ( getUniqueResourceName() );
            if ( res != null ) {
                Configuration.removeResource ( getUniqueResourceName() );
                //fix for case 26005
                res.close();
            }
        }    
    
        public String toString() 
        {
            String ret = "AtomikosDataSoureBean";
            String name = getUniqueResourceName();
            if ( name != null ) {
                ret = ret + " '" + name + "'";
            }
            return ret;
        }
    
    }
    复制代码
    复制代码
    /**
     * Copyright (C) 2000-2010 Atomikos <[email protected]>
     *
     * This code ("Atomikos TransactionsEssentials"), by itself,
     * is being distributed under the
     * Apache License, Version 2.0 ("License"), a copy of which may be found at
     * http://www.atomikos.com/licenses/apache-license-2.0.txt .
     * You may not use this file except in compliance with the License.
     *
     * While the License grants certain patent license rights,
     * those patent license rights only extend to the use of
     * Atomikos TransactionsEssentials by itself.
     *
     * This code (Atomikos TransactionsEssentials) contains certain interfaces
     * in package (namespace) com.atomikos.icatch
     * (including com.atomikos.icatch.Participant) which, if implemented, may
     * infringe one or more patents held by Atomikos.
     * It should be appreciated that you may NOT implement such interfaces;
     * licensing to implement these interfaces must be obtained separately from Atomikos.
     *
     * Unless required by applicable law or agreed to in writing, software
     * distributed under the License is distributed on an "AS IS" BASIS,
     * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     */
    
    package com.atomikos.jdbc.nonxa;
    
    
    import java.sql.Connection;
    import java.sql.SQLException;
    
    import com.atomikos.datasource.pool.ConnectionFactory;
    import com.atomikos.icatch.HeuristicMessage;
    import com.atomikos.icatch.system.Configuration;
    import com.atomikos.jdbc.AbstractDataSourceBean;
    import com.atomikos.util.DynamicProxy;
    
     /**
      * 
      * A Bean class for DataSource access to non-XA JDBC implementations. 
      * Instances are JTA transaction-aware and can rollback the work done 
      * over multiple connections (provided that all work was done in one and the same thread).
      * 
      *
      */
    public class AtomikosNonXADataSourceBean extends AbstractDataSourceBean 
    {
        private static final long serialVersionUID = 1L;
        
        private String url;
        
        private String user;
        
        private String password;
        
        private String driverClassName;
        
        private boolean readOnly;
        
        /**
         * Sets the URL to use for getting connections. Required.
         * 
         * @param url
         */
        
        public void setUrl ( String url ) 
        {
            this.url = url;
        }
        
        /**
         * Gets the URL to connect. 
         */
        
        public String getUrl()
        {
            return url;
        }
        
    
        /**
         * Marks this datasource as being used for read-only work. Optional.
         * 
         * Setting this to true will avoid warnings/errors upon recovery. ReadOnly mode
         * is intended to avoid XA configuration of databases where no updates are
         * being done.
         * 
         * @param readOnly Defaults to false.
         */
        
        public void setReadOnly ( boolean readOnly ) 
        {
            this.readOnly = readOnly;
        }
    
        /**
         * @return Whether or not this datasource is marked as readOnly.
         */
        
        public boolean getReadOnly()
        {
            return readOnly;
        }
    
        /**
         * @return The password.
         */
        
        public String getPassword ()
        {
            return password;
        }
    
        /**
         * Sets the password to use.
         * 
         * @param string
         */
        
        public void setPassword ( String string )
        {
            password = string;
        }
    
        /**
         * Set the user name to get connections with.
         * 
         * @param string
         */
        
        public void setUser ( String string )
        {
            user = string;
        }
    
        /**
         * @return The URL to connect with.
         */
        
        public String getUser ()
        {
            return user;
        }
        
        /**
         * 
         * @return The DriverManager class name.
         */
    
        public String getDriverClassName ()
        {
            return driverClassName;
        }
    
        /**
         * Sets the driver class name to be used by the DriverManager. Required.
         * 
         * @param string
         */
        public void setDriverClassName ( String string )
        {
            driverClassName = string;
        }
        
    
        protected void doClose() 
        {
            //nothing to do
        }
    
        protected ConnectionFactory doInit() throws Exception 
        {
            AtomikosNonXAConnectionFactory ret = null;
            if ( Configuration.isInfoLoggingEnabled() ) Configuration.logInfo(
                    this + ": initializing with [" +
                    " uniqueResourceName=" + getUniqueResourceName() + "," +
                    " maxPoolSize=" + getMaxPoolSize() + "," +
                    " minPoolSize=" + getMinPoolSize() + "," +
                    " borrowConnectionTimeout=" + getBorrowConnectionTimeout() + "," +
                    " maxIdleTime=" + getMaxIdleTime() + "," +
                    " reapTimeout=" + getReapTimeout() + "," +
                    " maintenanceInterval=" + getMaintenanceInterval() + "," +
                    " testQuery=" + getTestQuery() + "," +
                    " driverClassName=" + getDriverClassName() + "," +
                    " user=" + getUser() + "," +
                    " url=" + getUrl() + 
                    " loginTimeout=" + getLoginTimeout() +
                    "]"
                    );
            
            
            ret = new com.atomikos.jdbc.nonxa.AtomikosNonXAConnectionFactory ( this , url , driverClassName , user , password , getLoginTimeout() , readOnly ) ;
            ret.init();
            return ret;
        }
    
        public synchronized Connection getConnection ( HeuristicMessage hmsg ) throws SQLException
        {
            if ( Configuration.isInfoLoggingEnabled() ) Configuration.logInfo ( this + ": getConnection ( " + hmsg + " )..." );
            
            init();
            
            
            //let pool take care of reusing an existing handle
            Connection proxy = super.getConnection ( hmsg );          
    
            // here we are certain that proxy is not null -> increase the use count
            DynamicProxy dproxy = ( DynamicProxy ) proxy;
            com.atomikos.jdbc.nonxa.AtomikosThreadLocalConnection previous = (AtomikosThreadLocalConnection) dproxy.getInvocationHandler();
    
            previous.incUseCount();
            previous.addHeuristicMessage ( hmsg );
            if ( Configuration.isDebugLoggingEnabled() ) Configuration.logDebug ( this + ": returning " + proxy );
            return proxy;
        }
    
        
    
        public String toString() 
        {
            String ret = "AtomikosNonXADataSourceBean";
            String name = getUniqueResourceName();
            if ( name != null ) {
                ret = ret + " '" + name + "'";
            }
            return ret;
        }
        
    }
    复制代码

     

  5. 对于JMS,可以使用我们的实例com.atomikos.jms.AtomikosConnectionFactoryBeancom.atomikos.jms.extra.AbstractJmsSenderTemplate(发送信息时使用)com.atomikos.jms.extra.MessageDrivenContainer(接收时使用)
    复制代码
    /**
     * Copyright (C) 2000-2010 Atomikos <[email protected]>
     *
     * This code ("Atomikos TransactionsEssentials"), by itself,
     * is being distributed under the
     * Apache License, Version 2.0 ("License"), a copy of which may be found at
     * http://www.atomikos.com/licenses/apache-license-2.0.txt .
     * You may not use this file except in compliance with the License.
     *
     * While the License grants certain patent license rights,
     * those patent license rights only extend to the use of
     * Atomikos TransactionsEssentials by itself.
     *
     * This code (Atomikos TransactionsEssentials) contains certain interfaces
     * in package (namespace) com.atomikos.icatch
     * (including com.atomikos.icatch.Participant) which, if implemented, may
     * infringe one or more patents held by Atomikos.
     * It should be appreciated that you may NOT implement such interfaces;
     * licensing to implement these interfaces must be obtained separately from Atomikos.
     *
     * Unless required by applicable law or agreed to in writing, software
     * distributed under the License is distributed on an "AS IS" BASIS,
     * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     */
    
    package com.atomikos.jms;
    
    import java.io.Serializable;
    
    import javax.jms.ConnectionFactory;
    import javax.jms.JMSException;
    import javax.naming.NamingException;
    import javax.naming.Reference;
    import javax.naming.Referenceable;
    
    import com.atomikos.util.SerializableObjectFactory;
    
    /**
     * 
     * 
     * Common logic for the connection factory beans.
     *
     */
    
    public abstract class AbstractConnectionFactoryBean 
    implements Serializable, Referenceable, ConnectionFactory
    {
    
        protected String resourceName_;
        protected String xaFactoryJndiName_;
    
        protected AbstractConnectionFactoryBean ( )
        {
            this.xaFactoryJndiName_ = "";
            this.resourceName_ = "someUniqueName";
        }
        
        /**
         * Sets the JNDI name of the underlying XAConnectionFactory (optional). This is
         * optional and an alternative to directly supplying the required factory
         * through setXaConnectionFactory().
         * 
         * @param name
         *            The JNDI name where the XAConnectionFactory can be found.
         *            It is up to the client to make sure that the name exists and
     *            points to an existing XAConnectionFactory.
         */
        public void setXaFactoryJndiName ( String name ) 
        {
            xaFactoryJndiName_ = name;
        
        }
    
        /**
         * Retrieve the JNDI name where the XAConnectionFactory is expected.
         * 
         * @return String the name or an empty String if not set.
         */
        public String getXaFactoryJndiName() 
        {
            return xaFactoryJndiName_;
        }
    
        /**
         * Set the unique resource name for this factory (required). A unique
         * resource name is needed by the transaction service in order to register
         * and recover the underlying XA transactions. 
         * Note: the value you set here should not exceed 45 bytes in length.
         * 
         * <p><b>MQSeries NOTE:</b> For
         * IBM MQSeries, the name should include MQSeries_XA_RMI or the XA routines
         * will not work properly! 
         * 
         * @param name
         *            The unique resource name.
         */
        public void setResourceName ( String name ) 
        {
            resourceName_ = name;
        }
    
        /**
         * Get the resource name.
         * 
         * @return String the unique resource name as previously set.
         */
        public String getResourceName() 
        {
            return resourceName_;
        }
    
        public Reference getReference() throws NamingException 
        {
            return SerializableObjectFactory.createReference ( this );
        }
        
        /**
         * Initialization method to register the underlying resource for recovery
         * and other init code. 
         * 
         * @throws JMSException
         */
        
        public void init() throws JMSException
        {
                checkSetup();
        }
        
        protected abstract void checkSetup() throws JMSException;
    
    }
    复制代码
    复制代码
    /**
     * Copyright (C) 2000-2010 Atomikos <[email protected]>
     *
     * This code ("Atomikos TransactionsEssentials"), by itself,
     * is being distributed under the
     * Apache License, Version 2.0 ("License"), a copy of which may be found at
     * http://www.atomikos.com/licenses/apache-license-2.0.txt .
     * You may not use this file except in compliance with the License.
     *
     * While the License grants certain patent license rights,
     * those patent license rights only extend to the use of
     * Atomikos TransactionsEssentials by itself.
     *
     * This code (Atomikos TransactionsEssentials) contains certain interfaces
     * in package (namespace) com.atomikos.icatch
     * (including com.atomikos.icatch.Participant) which, if implemented, may
     * infringe one or more patents held by Atomikos.
     * It should be appreciated that you may NOT implement such interfaces;
     * licensing to implement these interfaces must be obtained separately from Atomikos.
     *
     * Unless required by applicable law or agreed to in writing, software
     * distributed under the License is distributed on an "AS IS" BASIS,
     * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     */
    
    package com.atomikos.jms.extra;
    
    import java.io.Serializable;
    import java.util.Map;
    
    import javax.jms.Connection;
    import javax.jms.Destination;
    import javax.jms.JMSException;
    import javax.jms.Queue;
    import javax.jms.Session;
    import javax.jms.Topic;
    import javax.transaction.Status;
    import javax.transaction.SystemException;
    
    import com.atomikos.icatch.jta.UserTransactionManager;
    import com.atomikos.icatch.system.Configuration;
    import com.atomikos.jms.AtomikosConnectionFactoryBean;
    import com.atomikos.jms.AtomikosJMSException;
    import com.atomikos.jms.AtomikosTransactionRequiredJMSException;
    
     /**
      * Common functionality for the sender templates.
      *
      */
    
    public abstract class AbstractJmsSenderTemplate 
    {
    
        protected AtomikosConnectionFactoryBean connectionFactoryBean;
        private String user;
        protected String password;
        protected Destination destination;
        private String destinationName;
        private Destination replyToDestination;
        private String replyToDestinationName;
        private int deliveryMode;
        private int priority;
        private long timeToLive;
        protected boolean inited;
    
        protected AbstractJmsSenderTemplate()
        {
            // set default values according to Sun's JMS javadocs
            setTimeToLive ( 0 );
            setDeliveryMode ( javax.jms.DeliveryMode.PERSISTENT );
            setPriority ( 4 );
        }
        
        protected abstract Session getOrRefreshSession ( Connection c ) throws JMSException;
        
        protected abstract Connection getOrReuseConnection() throws JMSException;
        
        protected abstract void afterUseWithoutErrors ( Connection c , Session s ) throws JMSException;
        
        protected void destroy ( Connection c , Session s)
        throws JMSException {
    
            try {
                if ( s != null ) s.close();
            } catch ( JMSException warn ) {
                Configuration.logWarning ( this + ": error closing session" , warn);
            }
    
            try {
                if ( c != null ) c.close();
            } catch ( JMSException warn ) {
                Configuration.logWarning ( this + ": error closing connection" , warn);
            }
    
        }    
        protected synchronized Connection refreshConnection() throws JMSException {
            Connection connection = null;
            if ( getDestinationName() == null )
                throw new JMSException ( "Please call setDestination or setDestinationName first!" );
        
            if ( user != null ) {
                connection = connectionFactoryBean.createConnection (
                        user, password );
        
            } else {
                connection = connectionFactoryBean.createConnection ();
            }
            connection.start ();
            return connection;
        }
    
    
        /**
         * Initializes the session for sending. 
         * Call this method first.
         */
        
        public void init() throws JMSException
        {
            if ( ! inited ) {
                if ( connectionFactoryBean == null ) throw new IllegalStateException ( "Property 'atomikosConnectionFactoryBean' must be set first!" );
                if ( getDestinationName() == null ) {
                    throw new IllegalStateException ( "Property 'destination' or 'destinationName' must be set first!" );
                }
                StringBuffer msg = new StringBuffer();
                msg.append ( this + ":configured with [" );
                msg.append ( "user=" ).append ( getUser() ).append ( ", " );
                msg.append ( "password=" ).append ( password ).append ( ", " );
                msg.append ( "deliveryMode=" ).append ( getDeliveryMode() ).append ( ", " );
                msg.append ( "timeToLive=" ).append ( getTimeToLive() ).append ( ", " );
                msg.append ( "priority=" ).append ( getPriority() ).append ( ", " );
                msg.append ( "destination=" ).append( getDestinationName() ).append ( ", " );
                msg.append ( "replyToDestination=" ).append ( getReplyToDestinationName() );
                msg.append ( "]" );
                if ( Configuration.isDebugLoggingEnabled() ) Configuration.logDebug ( msg.toString() );
                inited = true;        
            }
        }
        
        private void retrieveDestinationIfNecessary() throws JMSException 
        {
            if ( getDestination() == null ) {
                String dName = getDestinationName();
                RetrieveDestinationCallback cb = new RetrieveDestinationCallback ( dName );
                executeCallbackInternal ( cb );
                setDestination ( cb.getDestination() );
            }
            
        }
        
        private void retrieveReplyToDestinationIfNecessary() throws JMSException 
        {
            if ( getReplyToDestination() == null ) {
                String dName = getReplyToDestinationName();
                if ( dName != null ) {
                    RetrieveDestinationCallback cb = new RetrieveDestinationCallback ( dName );
                    executeCallbackInternal ( cb );
                    setReplyToDestination ( cb.getDestination() );
                }
                
            }
            
        }
    
        /**
         * Sets the connection factory to use. Required.
         * @param connectionFactory
         */
        public void setAtomikosConnectionFactoryBean(AtomikosConnectionFactoryBean connectionFactory) {
            this.connectionFactoryBean = connectionFactory;
        }
    
        public AtomikosConnectionFactoryBean getAtomikosConnectionFactoryBean() {
            return connectionFactoryBean;
        }
    
        public Destination getDestination() {
            return destination;
        }
        
    
        /**
         * Sets the (provider-specific) destination name in order
         * to lookup the destination (rather than providing one directly).
         * 
         * Required, unless you set the destination directly.
         * 
         * @param destinationName
         */
        
        public void setDestinationName ( String destinationName ) 
        {
            this.destinationName = destinationName;
        }
    
        /**
         * Sets the destination to send to. Required, unless
         * you set the destinationName instead.
         * 
         * @param destination
         */
        public void setDestination(Destination destination) {
            this.destination = destination;
        }
    
        private String getName(Destination d, String destinationName ) {
            String ret = destinationName;
            if ( ret == null ) {
                if ( d instanceof Queue ) {
                    Queue q = ( Queue ) d;
                    try {
                        ret = q.getQueueName();
                    } catch ( JMSException e ) {
                        if ( Configuration.isDebugLoggingEnabled() ) Configuration.logDebug ( this + ": error retrieving queue name" , e );
                    }
                } else if ( d instanceof Topic ) {
                    Topic t = ( Topic ) d;
                    try {
                        ret = t.getTopicName();
                    } catch ( JMSException e ) {
                        if ( Configuration.isDebugLoggingEnabled() ) Configuration.logDebug ( this + ": error retrieving topic name" , e );
                    }
                }
            }
            return ret;
        }
    
        protected String getDestinationName() {
            return getName ( getDestination() , destinationName );
        }
    
        protected String getReplyToDestinationName() {
            return getName ( getReplyToDestination() , replyToDestinationName );
        }
    
        /**
         * @return The user to connect with, or null if no explicit authentication
         *         is to be used.
         */
        public String getUser() {
            return user;
        }
    
        /**
         * If this session is used for sending request/reply messages, then this
         * property indicates the destination where the replies are to be sent (optional). The
         * session uses this to set the JMSReplyTo header accordingly. This property
         * can be omitted if no reply is needed.
         * 
         * <p>
         * The replyToDestination should be in the same JMS vendor domain as the send
         * queue. To cross domains, configure a bridge for both the request and the
         * reply channels.
         */
        public void setReplyToDestination(Destination destination) 
        {
            this.replyToDestination = destination;
        }
    
        /**
         * Sets the provider-specific replyToDestinationName. Optional.
         * 
         * @param replyToDestinationName
         */
    
        public void setReplyToDestinationName ( String replyToDestinationName ) 
        {
            this.replyToDestinationName = replyToDestinationName;
            
        }    
        
        /**
         * Gets the replyToDestination.
         * 
         * @return
         */
        public Destination getReplyToDestination() {
            return replyToDestination;
        }
    
        /**
         * Set the password for explicit authentication (optional). 
         * This is only required if
         * the user has also been set.
         * 
         * @param password
         *            The password.
         */
        public void setPassword(String password) {
            this.password = password;
        }
    
        /**
         * Set the user to use for explicit authentication (optional). If no explicit
         * authentication is required then this method should not be called.
         * 
         * @param user
         */
        public void setUser(String user) {
            this.user = user;
        }
        
        protected void executeCallbackInternal ( 
                JmsSenderTemplateCallback callback ) throws JMSException {
            
            init();
            Session  session = null;
            Connection conn = null;
            try {
                conn = getOrReuseConnection();
                session = getOrRefreshSession ( conn );    
                if ( Configuration.isDebugLoggingEnabled() ) Configuration.logDebug ( "Calling callback..." );
                callback.doInJmsSession ( session );
                if ( Configuration.isDebugLoggingEnabled() ) Configuration.logDebug ( "Callback done!" );
                afterUseWithoutErrors ( conn , session );
                
            } catch ( AtomikosTransactionRequiredJMSException notx ) {
                destroy ( conn , session );
                String msg = "The JMS session you are using requires a JTA transaction context for the calling thread and none was found." + "\n" +
                "Please correct your code to do one of the following: " + "\n" +            
                "1. start a JTA transaction before sending any message, or" + "\n" + 
                "2. increase the maxPoolSize of the AtomikosConnectionFactoryBean to avoid transaction timeout while waiting for a connection.";
                Configuration.logWarning ( msg );
                AtomikosTransactionRequiredJMSException.throwAtomikosTransactionRequiredJMSException ( msg );
        
            } catch ( JMSException e ) {
                e.printStackTrace();
                destroy ( conn , session );
                String msg = this + ": error in sending JMS message";
                AtomikosJMSException.throwAtomikosJMSException( msg , e );
            }
        }
    
        /**
         * Executes an application-level call-back within the managed session.
         * 
         * @param callback
         * @throws JMSException
         */
        public void executeCallback(JmsSenderTemplateCallback callback) throws JMSException {
            
            init();        
            
    
            retrieveDestinationIfNecessary();
            retrieveReplyToDestinationIfNecessary();
            
            UserTransactionManager tm = new UserTransactionManager ();
            try {
                if ( tm.getStatus () != Status.STATUS_ACTIVE )
                    throw new JMSException (
                            "This method requires an active transaction!" );
            } catch ( SystemException e ) {
                Configuration
                        .logWarning ( this +": error in getting transaction status", e );
                throw new RuntimeException ( e.getMessage () );
            }
            
            executeCallbackInternal ( callback );
            
        }
    
    
    
        /**
         * @return The deliverymode for messages sent in this session.
         */
        public int getDeliveryMode() {
            return deliveryMode;
        }
    
        /**
         * @return The priority for messages sent in this session.
         */
        public int getPriority() {
            return priority;
        }
    
        /**
         * @return The timeToLive for messages sent in this session.
         */
        public long getTimeToLive() {
            return timeToLive;
        }
    
        /**
         * 
         * Set the deliverymode for messages sent in this session (optional). Defaults to
         * persistent.
         * 
         * @param
         */
        public void setDeliveryMode(int i) {
            deliveryMode = i;
        }
    
        /**
         * Set the priority for messages sent in this session (optional). Defaults to 4.
         * 
         * @param
         */
        public void setPriority(int i) {
            priority = i;
        }
    
        /**
         * Set the time to live for messages sent in this session (optional). Defaults to 0.
         * 
         * @param
         */
        public void setTimeToLive(long l) {
            timeToLive = l;
        }
    
        /**
         * Sends a TextMessage.
         * 
         * @param content The text as a string.
         * @throws JMSException 
         */
        public void sendTextMessage(String content) throws JMSException {
            retrieveDestinationIfNecessary();
            retrieveReplyToDestinationIfNecessary();
            SendTextMessageCallback cb = new SendTextMessageCallback ( content , getDestination() , getReplyToDestination() , getDeliveryMode() , getPriority() , getTimeToLive() );
            executeCallback ( cb );
        }
    
        /**
         * Sends a MapMessage.
         * 
         * @param content The Map to get the content from.
         * 
         * @throws JMSException
         */
        public void sendMapMessage(Map content) throws JMSException {
            retrieveDestinationIfNecessary();
            retrieveReplyToDestinationIfNecessary();
            SendMapMessageCallback cb = new SendMapMessageCallback ( content , getDestination() , getReplyToDestination() , getDeliveryMode() , getPriority() , getTimeToLive() );
            executeCallback ( cb );        
        }
    
        /**
         * Sends an ObjectMessage.
         * 
         * @param content The serializable object content.
         * @throws JMSException
         */
        public void sendObjectMessage(Serializable content) throws JMSException {
            retrieveDestinationIfNecessary();
            retrieveReplyToDestinationIfNecessary();
            SendObjectMessageCallback cb = new SendObjectMessageCallback ( content , getDestination() , getReplyToDestination() , getDeliveryMode() , getPriority() , getTimeToLive() );
            executeCallback ( cb );        
        }
    
        /**
         * Sends a ByteMessage.
         * 
         * @param content The content as a byte array.
         * @throws JMSException
         */
        public void sendBytesMessage(byte[] content) throws JMSException {
            retrieveDestinationIfNecessary();
            retrieveReplyToDestinationIfNecessary();
            SendBytesMessageCallback cb = new SendBytesMessageCallback ( content , getDestination() , getReplyToDestination() , getDeliveryMode() , getPriority() , getTimeToLive() );
            executeCallback ( cb );
        }
    
        /**
         * Closes all resources.
         */
        public void close() {
            try {
                Connection c = getOrReuseConnection();
                Session s = getOrRefreshSession(c);
                destroy(c, s);
            } catch (JMSException e) {
                Configuration.logWarning ( this + ": error closing" , e );
            }
            connectionFactoryBean.close();
        }
    
    }
    复制代码
    复制代码
    /**
     * Copyright (C) 2000-2010 Atomikos <[email protected]>
     *
     * This code ("Atomikos TransactionsEssentials"), by itself,
     * is being distributed under the
     * Apache License, Version 2.0 ("License"), a copy of which may be found at
     * http://www.atomikos.com/licenses/apache-license-2.0.txt .
     * You may not use this file except in compliance with the License.
     *
     * While the License grants certain patent license rights,
     * those patent license rights only extend to the use of
     * Atomikos TransactionsEssentials by itself.
     *
     * This code (Atomikos TransactionsEssentials) contains certain interfaces
     * in package (namespace) com.atomikos.icatch
     * (including com.atomikos.icatch.Participant) which, if implemented, may
     * infringe one or more patents held by Atomikos.
     * It should be appreciated that you may NOT implement such interfaces;
     * licensing to implement these interfaces must be obtained separately from Atomikos.
     *
     * Unless required by applicable law or agreed to in writing, software
     * distributed under the License is distributed on an "AS IS" BASIS,
     * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     */
    
    package com.atomikos.jms.extra;
    
    import java.util.ArrayList;
    import java.util.Iterator;
    import java.util.List;
    
    import javax.jms.Destination;
    import javax.jms.ExceptionListener;
    import javax.jms.JMSException;
    import javax.jms.MessageListener;
    
    import com.atomikos.icatch.system.Configuration;
    import com.atomikos.jms.AtomikosConnectionFactoryBean;
    
     /**
      * 
      * A message-driven container for asynchronously receiving JMS messages
      * from a topic or queue, within a managed JTA transaction context.
      * 
      * Upon start, an instance of this class will create a number of
      * concurrent sessions that listen for incoming messages on the same destination.
      * MessageListener instances should be thread-safe if the pool size is larger
      * than one. Note: in general, after start() any changed properties are only
      * effective on the next start() event.
      * 
      * <p>
      * <b>IMPORTANT:</b> the transactional behaviour guarantees redelivery after failures.
      * As a side-effect, this can lead to so-called <em>poison messages</em>: messages
      * whose processing repeatedly fails due to some recurring error (for instance, a primary
      * key violation in the database, a NullPointerException, ...). Poison messages are problematic
      * because they can prevent other messages from being processed, and block the system.
      *  
      * To avoid poison messages, make sure that your MessageListener implementation 
      * only throws a <b>RuntimeException</b> when the problem is <em>transient</em>. In that
      * case, the system will perform rollback and the message will be redelivered
      * facing a clean system state. All non-transient errors (i.e., those that happen
      * each time a message is delivered) indicate problems at the application level
      * and should be dealt with by writing better application code.
      */
    
    public class MessageDrivenContainer 
    implements MessageConsumerSessionProperties
    {
        private static final int DEFAULT_TIMEOUT = 30;
    
        
        private AtomikosConnectionFactoryBean connectionFactoryBean;
        private MessageListener messageListener;
        private String user;
        private String password;
        private Destination destination;
        private String destinationName;
        private int transactionTimeout;
        private int poolSize;
        private List sessions;
        private boolean daemonThreads;
        private boolean notifyListenerOnClose;
        private String messageSelector;
        private ExceptionListener exceptionListener;
        private String subscriberName;
        private boolean noLocal;
        private boolean unsubscribeOnClose;
        private String clientID;
        
        private int receiveTimeout;
        
        public MessageDrivenContainer()
        {
            sessions = new ArrayList ();
            notifyListenerOnClose = false;
            setPoolSize ( 1 );
            setTransactionTimeout ( DEFAULT_TIMEOUT );
        }
        
        private MessageConsumerSession createSession() 
        {
            return new MessageConsumerSession ( this );
        }
        
        /**
         * Sets the clientID for durable subscriptions. Optional.
         * 
         * @param clientID
         */
        public void setClientID ( String clientID ) {
            this.clientID = clientID;
        }
        
        
        
        /**
         * Sets the connection factory to use. Required.
         * @param bean
         */
        public void setAtomikosConnectionFactoryBean ( AtomikosConnectionFactoryBean bean )
        {
            this.connectionFactoryBean = bean;
        }
        
        public AtomikosConnectionFactoryBean getAtomikosConnectionFactoryBean()
        {
            return connectionFactoryBean;
        }
        
        /**
         * Gets the destination. 
         * 
         * @return The destination, or null if not set.
         */
        public Destination getDestination()
        {
            return destination;
        }
        
        /**
         * Sets the JMS destination to listen on (required unless the destinationName is set instead).
         * 
         * @param dest
         */
        public void setDestination ( Destination dest )
        {
            this.destination = dest;
        }
        
        /**
         * Gets the destination name.
         * 
         * @return The name, or null if not set.
         */
        public String getDestinationName()
        {
            return destinationName;
        }
        
        /**
         * Sets the JMS provider-specific destination name
         * (required unless the destination is set directly). 
         * 
         * @param destinationName
         */
        public void setDestinationName ( String destinationName )
        {
            this.destinationName = destinationName;
        }
        
        
        /**
         * Sets whether threads should be daemon threads or not (optional).
         * Default is false.
         * @param value If true then threads will be daemon threads.
         */
        public void setDaemonThreads ( boolean value ) 
        {
                this.daemonThreads = value;
        }
    
        /**
         * Tests whether threads are daemon threads.
         * @return True if threads are deamons.
         */
        public boolean getDaemonThreads() 
        {
                return daemonThreads;
        }
    
        /**
         * 
         * Get the message listener if any.
         * 
         * @return
         */
        public MessageListener getMessageListener() 
        {
            return messageListener;
        }
    
        /**
         * Get the transaction timeout.
         * 
         * @return
         */
        public int getTransactionTimeout() 
        {
            return transactionTimeout;
        }
    
        /**
         * Get the user for connecting, or null if the default user should be used.
         * 
         * @return
         */
        public String getUser() 
        {
            return user;
        }
    
        /**
         * Set the message listener to use (required). 
         * The same instance will be used for each
         * session in the pool, meaning that instances need to be thread-safe. Only
         * one listener is allowed at a time. Call this method with a null argument
         * to unset the listener.
         * 
         * @param listener
         */
        public void setMessageListener ( MessageListener listener ) 
        {
        
            messageListener = listener;
            Iterator it = sessions.iterator ();
            while ( it.hasNext () ) {
                MessageConsumerSession s = (MessageConsumerSession) it.next ();
                s.setMessageListener ( listener );
            }
        }
    
        /**
         * Set the password if explicit authentication is needed (optional). 
         * You need to set this if the user is also set.
         * 
         * @param string
         */
        public void setPassword ( String string ) 
        {
            password = string;
        }
    
        /**
         * Set the transaction timeout in seconds (optional).
         * 
         * @param seconds
         */
        public void setTransactionTimeout ( int seconds ) 
        {
            transactionTimeout = seconds;
        }
    
        /**
         * Set the user to use for explicit authentication (optional). 
         * Don't set this property
         * if you want to use the default authentication.
         * 
         * @param string
         */
        public void setUser ( String string ) 
        {
            user = string;
        }
    
        /**
         * Get the message selector (if any)
         * 
         * @return The selector, or null if none.
         */
        public String getMessageSelector() 
        {
            return this.messageSelector;
        }
    
        /**
         * Set the message selector to use (optional).
         * 
         * @param selector
         */
        public void setMessageSelector ( String selector ) 
        {
            this.messageSelector = selector;
        }
    
        /**
         * Get the size of the pool.
         * 
         * @return
         */
        public int getPoolSize() 
        {
            return poolSize;
        }
    
        /**
         * Sets the size of the session pool (optional).
         * Default is 1.
         * 
         * @param size 
         */
        public void setPoolSize ( int size ) 
        {
            poolSize = size;
        }
    
        /**
         * Gets the exception listener (if any). 
         * @return Null if no ExceptionListener was set.
         */
        public ExceptionListener getExceptionListener() 
        {
            return exceptionListener;
        }
    
        /**
         * Sets the exception listener (optional). The listener will be
         * notified of connection-level JMS errors.
         * 
         * @param exceptionListener
         */
        public void setExceptionListener ( ExceptionListener exceptionListener ) 
        {
            this.exceptionListener = exceptionListener;
        }
        
        /**
         * Test if this instance will receive sends from the same connection.
         * 
         * @return
         */
        public boolean isNoLocal() {
            return noLocal;
        }
        
        /**
         * Sets whether or not this topic should receive sends from the 
         * same connection (optional). 
         * 
         * @param noLocal 
         */
    
        public void setNoLocal(boolean noLocal) {
            this.noLocal = noLocal;
        }
    
        /**
         * Gets the subscriber name (for durable subscribers).
         * @return The name, or null if not set (no durable subscriber).
         */
        
        public String getSubscriberName() {
            return subscriberName;
        }
    
        /**
         * Sets the name to use for durable subscriptions (optional).
         * <br>
         * <b>Note: this name will be appended with a suffix to ensure uniqueness
         * among instances in the pool. Otherwise, the JMS back-end would see 
         * multiple instances subscribing with the same name - an error.</b>
         * 
         * @param subscriberName
         */
        
        public void setSubscriberName(String subscriberName) {
            this.subscriberName = subscriberName;
        }
    
        protected boolean getNoLocal() {
            
            return isNoLocal();
        }
        
        /**
         * Start listening for messages.
         * 
         * @throws JMSException
         */
        public void start() throws JMSException 
        {
            if ( destination == null && destinationName == null )
                throw new JMSException (
                        "MessageDrivenContainer: destination not specified" );
            if ( connectionFactoryBean == null )
                throw new JMSException (
                        "MessageDrivenContainer: factory not set" );
            if ( messageListener == null )
                throw new JMSException (
                        "MessageDrivenContainer: messageListener not set" );
            for ( int i = 0; i < poolSize; i++ ) {
                MessageConsumerSession s = createSession();
                s.setMessageListener ( messageListener );
                s.setPassword ( password );
                s.setUser ( user );
                s.setDestination ( destination );
                s.setDestinationName ( destinationName );
                s.setAtomikosConnectionFactoryBean ( connectionFactoryBean );
                s.setDaemonThreads ( daemonThreads );
                s.setNotifyListenerOnClose ( notifyListenerOnClose );
                s.setMessageSelector ( getMessageSelector () );
                s.setExceptionListener ( exceptionListener );
                s.setNoLocal( noLocal );
                s.setSubscriberName( subscriberName );
                //set subscriber name with suffix to ensure unique names
                if ( getSubscriberName() != null ) s.setSubscriberName ( getSubscriberName() + "-" + i );
                s.setNoLocal ( getNoLocal() );
                s.setClientID(clientID);
                try {
                    s.startListening ();
                    // System.out.println ( "MessageDrivenContainer: started
                    // session");
                } catch ( Exception e ) {
                    Configuration.logWarning ( "Error starting pool", e );
                }
                sessions.add ( s );
            }
        
            // set listener again to trigger listening
            setMessageListener ( messageListener );
        }
    
        /**
         * Stop listening for messages. If <b>notifyListenerOnClose</b> is set then
         * calling this method will notify the listener by calling its onMessage
         * method with a null argument (and also without transaction context).
         * 
         * This method will wait for all active receive operations to unblock, which may take
         * up to <b>receiveTimeout</b> seconds per active thread.
         */
        public void stop() 
        {
            Iterator it = sessions.iterator ();
            while ( it.hasNext () ) {
                MessageConsumerSession s = (MessageConsumerSession) it.next ();
                s.stopListening ();
            }
        }
    
        /**
         * Getter to check whether the listener is notified on close.
         * 
         * @return
         */
        public boolean getNotifyListenerOnClose() 
        {
            return notifyListenerOnClose;
        }
    
        /**
         * Set whether the listener should be notified of close events on the pool
         * (optional). Default is false.
         * 
         * @param b
         *            If true, then the listener will receive a null message if the
         *            pool is closed.
         */
        public void setNotifyListenerOnClose ( boolean b ) 
        {
            notifyListenerOnClose = b;
            Iterator it = sessions.iterator ();
            while ( it.hasNext () ) {
                  MessageConsumerSession s = (MessageConsumerSession) it.next ();
                s.setNotifyListenerOnClose ( b );
            }
        }
        
        /**
         * Sets whether unsubscribe should be done at closing time (optional). Default is false.
         * 
         * @param b If true, then unsubscribe will be done at closing time. This only applies to 
         * durable subscribers (i.e., cases where subscriberName is set).
         */
        public void setUnsubscribeOnClose ( boolean b ) 
        {
            this.unsubscribeOnClose = b;
        }
        
        /**
         * Getter to test if unsubscribe should be called on close.
         */
        
        public boolean getUnsubscribeOnClose()
        {
            return unsubscribeOnClose;
        }
        
    
        /**
         * Gets the receive timeout in seconds.
         * 
         * @return
         */
        public int getReceiveTimeout() {
            int ret = receiveTimeout;
            if ( ret <=0 ) ret = getTransactionTimeout()/2;
            return ret;
        }
    
        
        /**
         * Sets the receive timeout in seconds, 
         * i.e. the number of seconds to wait for incoming messages in the message listener thread's event loop.
         *  
         * This property is optional and defaults to half the transactionTimeout, but typically this should be lower 
         * because the time required to shutdown (stop) this container will be bound by this value multiplied by 
         * the number of threads (as indicated by <b>poolSize</b>).
         * 
         * @param seconds
         */
        public void setReceiveTimeout(int seconds) {
            this.receiveTimeout = seconds;    
        }
    
    }
    复制代码

     

  6. 请检查我们完整的用户指南

注意:虽然这个版本包含了特定的第三方产品,如RDBMS软件和JMS代理软件,请注意,AtomikosTransactionsEssentials绝不是仅限制在这些特定的产品,除了RDBMS软件和JMS代理软件大多数的其他软件依然兼容。

J2SE实例:examples/jse这个文件夹包含带有源代码的各种各样的例子。可以执行这个脚本来运行这个程序(示例程序只有文本输出,而没有图形化界面输出)

         运行示例前最好安装Ant,然后打开命令窗口在 examples/jse 文件夹并且输入“ant”。

Linux/Unix/Mac OSX Note:在这些系统,你必须在终端输入'chmod u+x *.sh'命令,否则,示例将无法运行。

解决问题:偶尔,一些示例可能不能再继续启动使用(尽管在开始阶段可以正常运行),这个通常是由于进程原因引起的(比如JMS代理后台环境),关闭这期间的所有进程便会解决这个问题,或者,重新启动计算机,也可以解决这个问题。

在Spring中使用:Atomikos TransactionEssentials可以很方便的于Spring相结合,运行您程序使用企业级的J2EE应用程序,而不需要EJB甚至EJB容器。另外,我们额外提供了一个强大的功能,JMS消息驱动添加到Spring的内置特性。

         查看examples/spring文件夹,将展示Atomikos TransactionEssentials如何配置基于Spring的应用程序。

在J2EE的web容器中使用:在您的J2EE应用程序环境中使用Atomikos TransactionsEssentials最简单的方法是通过Spring作为pico-container(组件容器),如果想获取更多信息,请到Spring处查看。

Javadoc:

http://www.atomikos.com/downloads/transactions-essentials/com/atomikos/AtomikosTransactionsEssentials/javadoc/3.7/index.html

 

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/java_green_hand0909/article/details/80307047

智能推荐

什么是内部类?成员内部类、静态内部类、局部内部类和匿名内部类的区别及作用?_成员内部类和局部内部类的区别-程序员宅基地

文章浏览阅读3.4k次,点赞8次,收藏42次。一、什么是内部类?or 内部类的概念内部类是定义在另一个类中的类;下面类TestB是类TestA的内部类。即内部类对象引用了实例化该内部对象的外围类对象。public class TestA{ class TestB {}}二、 为什么需要内部类?or 内部类有什么作用?1、 内部类方法可以访问该类定义所在的作用域中的数据,包括私有数据。2、内部类可以对同一个包中的其他类隐藏起来。3、 当想要定义一个回调函数且不想编写大量代码时,使用匿名内部类比较便捷。三、 内部类的分类成员内部_成员内部类和局部内部类的区别

分布式系统_分布式系统运维工具-程序员宅基地

文章浏览阅读118次。分布式系统要求拆分分布式思想的实质搭配要求分布式系统要求按照某些特定的规则将项目进行拆分。如果将一个项目的所有模板功能都写到一起,当某个模块出现问题时将直接导致整个服务器出现问题。拆分按照业务拆分为不同的服务器,有效的降低系统架构的耦合性在业务拆分的基础上可按照代码层级进行拆分(view、controller、service、pojo)分布式思想的实质分布式思想的实质是为了系统的..._分布式系统运维工具

用Exce分析l数据极简入门_exce l趋势分析数据量-程序员宅基地

文章浏览阅读174次。1.数据源准备2.数据处理step1:数据表处理应用函数:①VLOOKUP函数; ② CONCATENATE函数终表:step2:数据透视表统计分析(1) 透视表汇总不同渠道用户数, 金额(2)透视表汇总不同日期购买用户数,金额(3)透视表汇总不同用户购买订单数,金额step3:讲第二步结果可视化, 比如, 柱形图(1)不同渠道用户数, 金额(2)不同日期..._exce l趋势分析数据量

宁盾堡垒机双因素认证方案_horizon宁盾双因素配置-程序员宅基地

文章浏览阅读3.3k次。堡垒机可以为企业实现服务器、网络设备、数据库、安全设备等的集中管控和安全可靠运行,帮助IT运维人员提高工作效率。通俗来说,就是用来控制哪些人可以登录哪些资产(事先防范和事中控制),以及录像记录登录资产后做了什么事情(事后溯源)。由于堡垒机内部保存着企业所有的设备资产和权限关系,是企业内部信息安全的重要一环。但目前出现的以下问题产生了很大安全隐患:密码设置过于简单,容易被暴力破解;为方便记忆,设置统一的密码,一旦单点被破,极易引发全面危机。在单一的静态密码验证机制下,登录密码是堡垒机安全的唯一_horizon宁盾双因素配置

谷歌浏览器安装(Win、Linux、离线安装)_chrome linux debian离线安装依赖-程序员宅基地

文章浏览阅读7.7k次,点赞4次,收藏16次。Chrome作为一款挺不错的浏览器,其有着诸多的优良特性,并且支持跨平台。其支持(Windows、Linux、Mac OS X、BSD、Android),在绝大多数情况下,其的安装都很简单,但有时会由于网络原因,无法安装,所以在这里总结下Chrome的安装。Windows下的安装:在线安装:离线安装:Linux下的安装:在线安装:离线安装:..._chrome linux debian离线安装依赖

烤仔TVの尚书房 | 逃离北上广?不如押宝越南“北上广”-程序员宅基地

文章浏览阅读153次。中国发达城市榜单每天都在刷新,但无非是北上广轮流坐庄。北京拥有最顶尖的文化资源,上海是“摩登”的国际化大都市,广州是活力四射的千年商都。GDP和发展潜力是衡量城市的数字指...

随便推点

java spark的使用和配置_使用java调用spark注册进去的程序-程序员宅基地

文章浏览阅读3.3k次。前言spark在java使用比较少,多是scala的用法,我这里介绍一下我在项目中使用的代码配置详细算法的使用请点击我主页列表查看版本jar版本说明spark3.0.1scala2.12这个版本注意和spark版本对应,只是为了引jar包springboot版本2.3.2.RELEASEmaven<!-- spark --> <dependency> <gro_使用java调用spark注册进去的程序

汽车零部件开发工具巨头V公司全套bootloader中UDS协议栈源代码,自己完成底层外设驱动开发后,集成即可使用_uds协议栈 源代码-程序员宅基地

文章浏览阅读4.8k次。汽车零部件开发工具巨头V公司全套bootloader中UDS协议栈源代码,自己完成底层外设驱动开发后,集成即可使用,代码精简高效,大厂出品有量产保证。:139800617636213023darcy169_uds协议栈 源代码

AUTOSAR基础篇之OS(下)_autosar 定义了 5 种多核支持类型-程序员宅基地

文章浏览阅读4.6k次,点赞20次,收藏148次。AUTOSAR基础篇之OS(下)前言首先,请问大家几个小小的问题,你清楚:你知道多核OS在什么场景下使用吗?多核系统OS又是如何协同启动或者关闭的呢?AUTOSAR OS存在哪些功能安全等方面的要求呢?多核OS之间的启动关闭与单核相比又存在哪些异同呢?。。。。。。今天,我们来一起探索并回答这些问题。为了便于大家理解,以下是本文的主题大纲:[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-JCXrdI0k-1636287756923)(https://gite_autosar 定义了 5 种多核支持类型

VS报错无法打开自己写的头文件_vs2013打不开自己定义的头文件-程序员宅基地

文章浏览阅读2.2k次,点赞6次,收藏14次。原因:自己写的头文件没有被加入到方案的包含目录中去,无法被检索到,也就无法打开。将自己写的头文件都放入header files。然后在VS界面上,右键方案名,点击属性。将自己头文件夹的目录添加进去。_vs2013打不开自己定义的头文件

【Redis】Redis基础命令集详解_redis命令-程序员宅基地

文章浏览阅读3.3w次,点赞80次,收藏342次。此时,可以将系统中所有用户的 Session 数据全部保存到 Redis 中,用户在提交新的请求后,系统先从Redis 中查找相应的Session 数据,如果存在,则再进行相关操作,否则跳转到登录页面。此时,可以将系统中所有用户的 Session 数据全部保存到 Redis 中,用户在提交新的请求后,系统先从Redis 中查找相应的Session 数据,如果存在,则再进行相关操作,否则跳转到登录页面。当数据量很大时,count 的数量的指定可能会不起作用,Redis 会自动调整每次的遍历数目。_redis命令

URP渲染管线简介-程序员宅基地

文章浏览阅读449次,点赞3次,收藏3次。URP的设计目标是在保持高性能的同时,提供更多的渲染功能和自定义选项。与普通项目相比,会多出Presets文件夹,里面包含着一些设置,包括本色,声音,法线,贴图等设置。全局只有主光源和附加光源,主光源只支持平行光,附加光源数量有限制,主光源和附加光源在一次Pass中可以一起着色。URP:全局只有主光源和附加光源,主光源只支持平行光,附加光源数量有限制,一次Pass可以计算多个光源。可编程渲染管线:渲染策略是可以供程序员定制的,可以定制的有:光照计算和光源,深度测试,摄像机光照烘焙,后期处理策略等等。_urp渲染管线

推荐文章

热门文章

相关标签