SAP接口编程之 NCo3.0系列(06) : 会话管理 这篇文章中,对会话管理的相关知识点已经说得很详细了,请参考。现在用JCo3.0来实现。

如果SAP中多个函数需要在一个session中运行,需要JCoContext来提供保证。如果在同一个线程中,大体模式这样:

  1. JCoContext.begin(sapDestination);
  2. fm1.execute(sapDestination);
  3. fm2.execute(sapDestination);
  4. JCoContext.end(destination);

begin()和end()之间的函数execute之后,SAP不会释放连接,确保同一个session之中。

第二种情况:如果不同的函数不在同一个线程中,需要由开发人员实现SessionReferenceProvider接口,在类中提供session id。逻辑跟nco3.0也是一样的。JCo3.0提供了一个示例代码,但是搞的太复杂,我弄了一个简单的,方便理解。

我们要使用的函数是从标准系统函数INCREMENT_COUNTER
GET_COUNTER拷贝而来的。在SAP系统中INCREMENT_COUNTER
GET_COUNTER在同一个function group中,共享一个变量count(计数器),每次运行INCREMENT_COUNTER
, count就会加一,GET_COUNTER函数
可以获得这个count。因为这两个函数不能被远程调用,所以我们将这两个函数拷贝出另外两个函数ZINCREMENT_COUNTER和ZGET_COUNTER。

首先我们把两个函数定义在一个类RfcFunctions中:

  1. package jco3.demo6;
  2. import com.sap.conn.jco.JCoDestination;
  3. import com.sap.conn.jco.JCoException;
  4. import com.sap.conn.jco.JCoFunction;
  5. public class RfcFunctions
  6. {
  7. public static int runGetCounter(JCoDestination dest) throws JCoException
  8. {
  9. JCoFunction counterFM = dest.getRepository().getFunction("ZGET_COUNTER");
  10. counterFM.execute(dest);
  11. int counter = (int) counterFM.getExportParameterList().getValue("GET_VALUE");
  12. return counter;
  13. }
  14. public static void runIncrement(JCoDestination dest) throws JCoException
  15. {
  16. JCoFunction increment = dest.getRepository().getFunction("ZINCREMENT_COUNTER");
  17. increment.execute(dest);
  18. }
  19. }

然后编写测试类进行测试:

  1. package jco3.demo6;
  2. import com.sap.conn.jco.JCoContext;
  3. import com.sap.conn.jco.JCoDestination;
  4. import com.sap.conn.jco.JCoDestinationManager;
  5. import com.sap.conn.jco.JCoException;
  6. public class TestSessionSameThread
  7. {
  8. public static void main(String[] args) throws JCoException, InterruptedException
  9. {
  10. // get JCoDestination object instance
  11. JCoDestination destination = JCoDestinationManager.getDestination("ECC");
  12. // make sure the two functions will be executed in the same session
  13. JCoContext.begin(destination);
  14. // Before increment
  15. System.out.println("Before execution of ZINCREMENT_COUNTER:");
  16. System.out.println("Counter:" + RfcFunctions.runGetCounter(destination));
  17. // Run incrementCounter five times
  18. for(int i = 0; i < 5; i++){
  19. RfcFunctions.runIncrement(destination);
  20. System.out.println("Add:" + (i + 1));
  21. }
  22. // After increment
  23. System.out.println("After execution of ZINCREMENT_COUNTER:");
  24. System.out.println("Counter:" + RfcFunctions.runGetCounter(destination));
  25. // release the connection
  26. JCoContext.end(destination);
  27. }
  28. }

代码很直观,就不多说了。函数执行前,counter的值为0,运行函数5次之后,counter的值为5。如果我们注释掉JCoContext.begin(destination);JCoContext.end(destination);,可以对比出不同的效果。

如果在不同的线程中执行不同的函数,需要开发者提供session id。我准备将两个函数放在不同的线程中:

  • 在JVM的主线程中调用ZGET_COUNTER,查看counter的结果。
  • 在另外一个线程中运行ZINCREMENT_COUNTER,两个线程通过JCoContext,保持在同一个session ID下。

JCoSessionRefence实现类的主要作用是提供session ID:

  1. package jco3.session;
  2. import java.util.concurrent.atomic.AtomicInteger;
  3. import com.sap.conn.jco.ext.JCoSessionReference;
  4. public class JCoSessionRefenceImpl implements JCoSessionReference
  5. {
  6. private AtomicInteger atomInt = new AtomicInteger(0);
  7. private String id = "session"+String.valueOf(atomInt.addAndGet(1));
  8. public void contextFinished()
  9. {
  10. }
  11. public void contextStarted()
  12. {
  13. }
  14. @Override
  15. public String getID()
  16. {
  17. /**
  18. * We need to override getID() method
  19. */
  20. return id;
  21. }
  22. }

SessionReferenceProvider接口的实现类中,改写getCurrentSessionReference()方法,获取上面定义的JCoSessionRefence,从而获得session ID。其他方法保持不动。

  1. package jco3.session;
  2. import com.sap.conn.jco.ext.JCoSessionReference;
  3. import com.sap.conn.jco.ext.SessionException;
  4. import com.sap.conn.jco.ext.SessionReferenceProvider;
  5. public class SessionReferencProviderImpl implements SessionReferenceProvider
  6. {
  7. @Override
  8. public JCoSessionReference getCurrentSessionReference(String scopeType)
  9. {
  10. /**
  11. * We need to override getCurrentSessionReference() method
  12. */
  13. JCoSessionRefenceImpl sessionRef = new JCoSessionRefenceImpl();
  14. return sessionRef;
  15. }
  16. @Override
  17. public boolean isSessionAlive(String sessionID)
  18. {
  19. return false;
  20. }
  21. public void jcoServerSessionContinued(String sessionID) throws SessionException
  22. {
  23. }
  24. public void jcoServerSessionFinished(String sessionID)
  25. {
  26. }
  27. public void jcoServerSessionPassivated(String sessionID) throws SessionException
  28. {
  29. }
  30. public JCoSessionReference jcoServerSessionStarted() throws SessionException
  31. {
  32. return null;
  33. }
  34. }

注册SessionReferenceProvider接口的实现类,这样JCoDestination就有状态管理功能了。

  1. package jco3.session;
  2. import com.sap.conn.jco.JCoDestination;
  3. import com.sap.conn.jco.JCoDestinationManager;
  4. import com.sap.conn.jco.JCoException;
  5. import com.sap.conn.jco.ext.Environment;
  6. import com.sap.conn.jco.ext.SessionReferenceProvider;
  7. public class DestinationProvider
  8. {
  9. public static JCoDestination getDestination() throws JCoException
  10. {
  11. // create an instance of SessionReferenceProvider
  12. // and register in environment
  13. SessionReferenceProvider provider = new SessionReferencProviderImpl();
  14. Environment.registerSessionReferenceProvider(provider);
  15. JCoDestination destination = JCoDestinationManager.getDestination("ECC");
  16. return destination;
  17. }
  18. }

定义WorkingThread, 从Thread类继承,在这个线程中执行函数ZINCREMENT_COUNTER 5次。

  1. package jco3.demo6;
  2. import com.sap.conn.jco.JCoDestination;
  3. import com.sap.conn.jco.JCoException;
  4. public class WorkingThread extends Thread
  5. {
  6. private boolean doneSignal;
  7. private JCoDestination destination;
  8. // constructor
  9. public WorkingThread(JCoDestination destination, boolean doneSignal)
  10. {
  11. this.destination = destination;
  12. this.doneSignal = doneSignal;
  13. }
  14. public boolean hasDone()
  15. {
  16. return doneSignal;
  17. }
  18. @Override
  19. public void run()
  20. {
  21. /**
  22. * run method of runIncrement() for five times
  23. */
  24. for (int i = 0; i < 5; i++){
  25. try {
  26. RfcFunctions.runIncrement(this.destination);
  27. System.out.println("Run " + (i+1) + " times.");
  28. } catch (JCoException e) {
  29. e.printStackTrace();
  30. }
  31. }
  32. this.doneSignal = true;
  33. }
  34. }

doneSignal用于标识该线程是否结束。线程本身结束,是run()方法运行完毕。

好了,最后来测试在多线程中函数调用:

  1. package jco3.demo6;
  2. import com.sap.conn.jco.JCoContext;
  3. import com.sap.conn.jco.JCoDestination;
  4. import com.sap.conn.jco.JCoException;
  5. import jco3.session.DestinationProvider;
  6. public class TestSAPSessionMultiThread
  7. {
  8. public static void main(String[] args) throws JCoException, InterruptedException
  9. {
  10. /**
  11. * Run ZINCREMENT_COUNTER & ZGET_COUNTER functions in
  12. * different threads in a stateful way.
  13. *
  14. * The SAP will keep a session id which was created in
  15. * JCoSessionReferenceImpl class
  16. * and used in SessionReferenceProviderImpl class.
  17. *
  18. * Before using, SessionReferenceProviderImpl class should be
  19. * registered using Environment.registerSessionReferenceProvider() method.
  20. */
  21. // get JCoDestination object instance
  22. JCoDestination destination = DestinationProvider.getDestination();
  23. // make sure the two functions will be executed in the same session
  24. JCoContext.begin(destination);
  25. // Before increment
  26. System.out.println("Before execution of ZINCREMENT_COUNTER:");
  27. System.out.println("Counter:" + RfcFunctions.runGetCounter(destination));
  28. // start a new Thread in which function ZINCREMENT_COUNTER
  29. // will be executed for five times
  30. WorkingThread workingThread = new WorkingThread(destination, false);
  31. workingThread.start();
  32. // wait and switch thread
  33. Thread.sleep(1000);
  34. // After increment
  35. if (workingThread.hasDone() == true){
  36. System.out.println("After execution of ZINCREMENT_COUNTER:");
  37. System.out.println("Counter:" + RfcFunctions.runGetCounter(destination));
  38. }
  39. // release the connection
  40. JCoContext.end(destination);
  41. }
  42. }

与前面同一个线程中代码的主要区别是:
定义一个WorkingThread类的实例,然后启动线程:

  1. WorkingThread workingThread = new WorkingThread(destination, false);
  2. workingThread.start();

然后通过Thread.sleep(), 将线程切换到workingThread中执行,执行完毕再回到主线程显示结果。

文/StoneWM(简书作者)
原文链接:http://www.jianshu.com/p/2ce28196483c
著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”。

版权声明:本文为zfswff原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://www.cnblogs.com/zfswff/p/5671160.html