7

我之前有一个关于将 Mathematica 与用 C++ 编写的函数集成的问题。

这是一个后续问题:

如果计算时间太长,我希望能够使用Evaluation > Abort Evaluation中止它。答案中建议的哪些技术使具有可中断的基于 C 的扩展功能成为可能?C端如何实现“可中断性”?

我需要以一种既不会破坏它也不会破坏 Mathematica 内核的方式使我的函数可中断(即应该可以在 Mathematica 被中断后再次调用该函数)

4

3 回答 3

5

对于基于 MathLink 的函数,您必须做两件事(在 Windows 上):用于MLAbort检查中止,并调用MLCallYieldFunction, 临时让出处理器。两者都在 Todd Gayley 的 MathLink 教程中进行了描述,可在此处获得。

使用我之前的答案中的位,这是一个计算素数的示例代码(以一种低效的方式,但这是我们在这里需要的说明):

code = 
"
#include <stdlib.h>

extern void primes(int n);

static void yield(){
    MLCallYieldFunction(
        MLYieldFunction(stdlink), 
        stdlink,
       (MLYieldParameters)0 );
}

static void abort(){
    MLPutFunction(stdlink,\" Abort \",0);
}

void primes(int n){
    int i = 0, j=0,prime = 1, *d = (int *)malloc(n*sizeof(int)),ctr = 0;    
    if(!d) {
       abort();
       return;
    }
    for(i=2;!MLAbort && i<=n;i++){
        j=2;
        prime = 1;      
        while (!MLAbort && j*j <=i){
            if(i % j == 0){
                prime = 0;
                break;
            }
            j++;
        }
        if(prime) d[ctr++] = i;
        yield();
    }
    if(MLAbort){
        abort();
        goto R1;
    }

    MLPutFunction(stdlink,\"List\",ctr);
    for(i=0; !MLAbort && i < ctr; i++ ){
        MLPutInteger(stdlink,d[i]);
        yield();        
    }
    if(MLAbort) abort();

 R1: free(d);
 }
 ";

和模板:

template = 
"
void primes P((int ));

:Begin:
:Function:       primes
:Pattern:        primes[n_Integer]
:Arguments:      { n }
:ArgumentTypes:  { Integer }
:ReturnType:     Manual
:End:
";

这是创建程序的代码(取自上一个答案,稍作修改):

Needs["CCompilerDriver`"];
fullCCode = makeMLinkCodeF[code];
projectDir = "C:\\Temp\\MLProject1";
If[! FileExistsQ[projectDir], CreateDirectory[projectDir]]
pname = "primes";
files = MapThread[
   Export[FileNameJoin[{projectDir, pname <> #2}], #1, 
     "String"] &, {{fullCCode, template}, {".c", ".tm"}}];

现在,我们在这里创建它:

In[461]:= exe=CreateExecutable[files,pname];
Install[exe]

Out[462]= LinkObject["C:\Users\Archie\AppData\Roaming\Mathematica\SystemFiles\LibraryResources\
Windows-x86-64\primes.exe",161,10]

并使用它:

In[464]:= primes[20]
Out[464]= {2,3,5,7,11,13,17,19}

In[465]:= primes[10000000]
Out[465]= $Aborted

在后一种情况下,我使用了 Alt+"。" 中止计算。请注意,如果您不调用yield.

一般的思想是,您必须检查MLAbort并调用MLCallYieldFunction每一个昂贵的计算,例如大循环等。也许,像我上面所做的那样对内部循环这样做是一种矫枉过正的做法。您可以尝试做的一件事是使用 C 预处理器(宏)将样板代码分解。

于 2011-11-26T20:03:33.097 回答
3

如果您使用LibraryLink将外部 C 代码链接到 Mathematica 内核,您可以使用库回调函数AbortQ检查是否正在进行中止。

于 2011-11-27T09:44:54.857 回答
3

没有尝试过,看起来表达式包功能可能以这种方式工作 - 如果您的 C 代码返回并要求mathematica 定期执行更多工作,那么希望在mathematica 端中止执行将告诉C 代码没有更多的工作要做。

于 2011-11-26T15:48:15.697 回答