如何挂钩MATLAB来评估C代码中的符号导数?

时间:2009-06-12 13:34:18

标签: matlab symbolic-math

如何使用MATLAB的Symbolic Toolkit象征性地评估C或C ++中的衍生物?

1 个答案:

答案 0 :(得分:2)

以下是C中的一个工作示例,以及gcc \ Linux的Makefile。如果有人可以提供Windows或Mac的构建说明,请执行。

首先是C代码:

#define _GNU_SOURCE
#include <stdio.h>
#include <ctype.h>
#include <stdlib.h>
#include <string.h>
#include <engine.h>

int main (int argc, char *argv[])
{
    Engine    *engine;
    mxArray   *f;
    mxChar    *output;
    char      *input;
    char      *command;
    size_t     input_size;
    int        i;

    engine = engOpen ("matlab -nojvm");
    engEvalString (engine, "syms x");

    input = NULL;
    printf (">> "); getline (&input, &input_size, stdin);
    while (!feof (stdin)) {
        // remove the \n character
        input[strlen (input) - 1] = '\0';
        command = (char *)malloc (strlen (input) + 19);
        sprintf (command, "f=simple(diff(%s,x))", input);

        // execute the `diff` command on the user input expression
        engEvalString (engine, command);

        // the MATLAB engine does not understand the Symbolic Toolbox
        // therefore you have to convert the output expression into
        // some textual form (ccode,fortran,matlabFunction,latex)
        engEvalString (engine, "fstr=ccode(f)");
        f = engGetVariable (engine, "fstr");
        output = mxGetChars (f);

        // the expression is prefixed with some left-hand line (eg, "t0=")
        // so, discard everything to the left of the =, and display just
        // the derivative
        for (i = 0; output[i] != '='; i++);
        i++; // skip the =

        // `mxChar` is a 16-bit character, and `printf ("%ls")` is giving
        // me errors, so go through the string one character at a time, until
        // the semicolon is found (it is C-code), ignoring non-printable
        // characters along the way.
        for (; output[i] != ';'; i++) {
            if (isgraph (output[i])) {
                putchar (output[i]);
            }
        }
        putchar ('\n');
        printf (">> "); getline (&input, &input_size, stdin);
    }

    printf ("exiting...\n");
    engClose (engine);
    return EXIT_SUCCESS;
}

如果您需要使用表达式,而不是仅仅回显它,您将必须编写自己的解析器。但由于它是一个简单的C表达式(带有math.h函数),因此它非常简单。

这是我的Makefile,没有什么花哨的......只记得妥善设置你的路径。

CC=gcc
CFLAGS=-O0 -g -Wall -I$(MATLAB_DIR)/extern/include
LDFLAGS=-L$(MATLAB_DIR)/bin/glnxa64 -leng -lmx -lmat -lut -licudata -licuuc -licui18n -licuio -lhdf5
MATLAB_DIR=/opt/matlab/2008a

%.o: %.c
    $(CC) $(CFLAGS) -c $< -o $@

diff_test: diff_test.o
    $(CC) $< -o $@ $(LDFLAGS)

最后一个快速演示:

# ./diff_test 
>> sin(x)
cos(x)
>> 2*x^2*tan(x)
2.0*x*(2.0*tan(x)+x+x*pow(tan(x),2.0))
>> log(x^2)
2.0/x
>> ^D
exiting...