Произвольное приближение Python Sympy к произвольному выражению Sympy?

Я обнаружил, что хочу использовать аппроксимации, предоставляемые как часть пакета mpmath, но не понимаю, что именно они должны делать:

http://docs.sympy.org/dev/modules/mpmath/calculus/closed.html

В чем именно разница между выражением sympy и выражением sympy.mpmath?

Если мне нужно тейлоровское приближение к символьному выражению без понимания того, что делает пакет mpmath, я могу сделать следующее:

#Imports
import sympy
import sympy.parsing
import sympy.parsing.sympy_parser
import Library_TaylorApproximation

#Create a sympy expression to approximate
ExampleStringExpression = 'sin(x)'
ExampleSympyExpression = sympy.parsing.sympy_parser.parse_expr(ExampleStringExpression)


#Create a taylor expantion sympy expression around the point x=0
SympyTaylorApproximation = sympy.series( 
    ExampleSympyExpression,
    sympy.Symbol('x'),
    1, 
    4,
    ).removeO()

#Cast the sympy expressions to python functions which can be evaluated:
VariableNames = [str(var) for var in SympyTaylorApproximation.free_symbols]
PythonFunctionOriginal =  sympy.lambdify(VariableNames, ExampleSympyExpression)
PythonFunctionApproximation = sympy.lambdify(VariableNames, SympyTaylorApproximation)

#Evaluate the approximation and the original at a point:
print PythonFunctionOriginal(2)
print PythonFunctionApproximation(2)

#>>> 0.909297426826
#>>> 0.870987413961

Однако, если я попытаюсь сделать то же самое с mpmath на основе документации:

TaylorCoefficients = sympy.mpmath.taylor(ExampleSympyExpression, 1, 4 )
print 'TaylorCoefficients', TaylorCoefficients

#>>> TypeError: 'sin' object is not callable

Я могу попытаться втиснуть туда функцию python (которую можно вызвать):

TaylorCoefficients = sympy.mpmath.taylor(PythonFunctionOriginal, 1, 4 )
print 'TaylorCoefficients', TaylorCoefficients

#>>> TaylorCoefficients [mpf('0.8414709848078965'), mpf('0.0'), mpf('0.0'), mpf('0.0'), mpf('-8.3694689805155739e+57')]

Но вышеизложенное не имеет никакого смысла, потому что я знаю, что нельзя брать производные от функции python.

Я могу вызвать функцию mpmath sin:

TaylorCoefficients = sympy.mpmath.taylor(sympy.mpmath.sin, 1, 4 )
print 'TaylorCoefficients', TaylorCoefficients
#>>> TaylorCoefficients [mpf('0.8414709848078965'), mpf('0.54030230586813977'), mpf('-0.42073549240394825'), mpf('-0.090050384311356632'), mpf('0.035061291033662352')]

Но тогда я не могу делать над ним манипуляции так, как мне бы тоже хотелось -> как Если бы я хотел

SinTimesCos = sympy.mpmath.sin*sympy.mpmath.cos
TaylorCoefficients = sympy.mpmath.taylor(SinTimesCos, 1, 4 )
print 'TaylorCoefficients', TaylorCoefficients
#>>> TypeError: unsupported operand type(s) for *: 'function' and 'function'

ЧТО такое функция mpmath?

Это не выражение sympy, и это также не функция python. Как выполнять манипуляции с произвольными выражениями?

Казалось бы, я не могу аппроксимировать произвольные выражения sympy в документации. http://docs.sympy.org/dev/modules/mpmath/calculus/closed.html

Как я могу использовать произвольные приближения (Паде/Чеби-Шев/Фурье) к произвольным выражениям sympy?

РЕДАКТИРОВАТЬ:

Итак, примером того, что я ищу, является следующее приближение:

#Start with a sympy expression of (a, b, x)
expressionString = 'cos(a*x)*sin(b*x)*(x**2)'
expressionSympy = sympy.parsing.sympy_parser.parse_expr(expressionString)

#Do not want to decide on value of `a or b` in advance.
#Do want approximation with respect to x:

wantedSympyExpression = SympyChebyChev( expressionSympy, sympy.Symbol('x') ) 

Результатом может быть либо список выражений коэффициентов, являющихся функциями a и b:

wantedSympyExpressionCoefficients = [ Coef0Expression(a,b), Coef1Expression(a,b), ... , CoefNExpression(a,b)]

ИЛИ результатом может быть само выражение sympy целиком (которое само по себе является функцией a, b):

wantedSympyExpression = Coef0Expression(a,b) + Coef1Expression(a,b) *(x**2) + ... + CoefNExpression(a,b) (x**N)

Обратите внимание, что a и b не выбираются до выполнения аппроксимации.


person D Adams    schedule 12.11.2015    source источник


Ответы (3)


Функции mpmath — это обычные функции Python. Они просто делают свою арифметику с произвольной точностью.

Но вышеизложенное не имеет никакого смысла, потому что я знаю, что нельзя брать производные от функции python.

Вы не можете взять производную символически, но вы можете приблизительно вычислить производную, вычислив функцию несколько раз и используя методы численного дифференцирования. Это то, что делает sympy.mpmath.taylor. Цитирую документы:

Коэффициенты вычисляются с использованием численного дифференцирования высокого порядка. Функция должна иметь возможность вычисляться с произвольной точностью.

person user2357112 supports Monica    schedule 12.11.2015
comment
Итак, мой PythonFunctionOriginal нельзя оценить с произвольной точностью? - person D Adams; 12.11.2015
comment
@DAdams: Правильно. Скорее всего, поэтому высшие члены ряда Тейлора оказались равными нулю. - person user2357112 supports Monica; 12.11.2015
comment
Этот ответ был чрезвычайно проницательным в отношении объяснения того, что делают функции npmath. Возможно, мы могли бы перенести этот ответ и вопрос What exactly is the difference between a sympy expression and a sympy.mpmath expression ? на другую страницу? - person D Adams; 16.11.2015

Если у вас есть выражение SymPy и вы хотите оценить его с произвольной точностью, используйте evalf, например

sympy.sin(1).evalf(100)

Вы можете использовать sin(x).evalf(100, subs={x:1}), чтобы заменить x на 1 перед оценкой. evalf использует mpmath под капотом, так что это даст вам тот же результат, что и mpmath, но без необходимости использовать mpmath напрямую.

person asmeurer    schedule 12.11.2015
comment
Если это сработает: это будет принятый ответ - и я должен вам дать пять. - person D Adams; 12.11.2015
comment
Итак, это работает, если я заменяю каждую переменную, но я не могу принять это как ответ, потому что для моей версии Тейлора я могу решить использовать переменную, сделать приближение, а затем мое приближение может сгенерировать другое симпатичное выражение. Этот материал npmath не позволяет мне получить выражение sympy как приближение к другому выражению sympy. Вы все еще можете получить высокие пять, хотя! - person D Adams; 13.11.2015
comment
Я не понимаю, в чем твоя проблема. Можете ли вы показать пример? Если это не помещается в комментарий, вы можете открыть новый вопрос. - person asmeurer; 16.11.2015
comment
Я надеюсь сгенерировать набор символьных коэффициентов для чебычева/паде/фурье произвольного символьного выражения. Я отредактировал последние несколько строк вопроса для ясности. - person D Adams; 16.11.2015
comment
Было бы лучше написать это как новый вопрос. Это отвечает на ваш первоначальный вопрос (в чем разница между выражением sympy и выражением mpmath?). - person asmeurer; 16.11.2015
comment
Но это был не мой первоначальный вопрос - я признаю, что был сбит с толку, когда спросил, потому что предполагал, что функции mpmath будут содержать ответ на мой первоначальный вопрос. - person D Adams; 17.11.2015

РЕДАКТИРОВАТЬ: перечитывая мой ответ -> я подумал, что восполню несколько недостающих частей в качестве услуги для кого-то, кто когда-нибудь действительно воспользуется этим. Ниже я указал, как я назвал свои библиотеки и какие импорты потребовались. У меня нет времени, чтобы вносить реальный вклад в sympy в данный момент, но чувствую, что эта функциональность наверняка будет использоваться другими профессорами/студентами математики/физики.

Обратите внимание, что по причинам нехватки места следующие две библиотеки опущены, и я добавлю ссылку на свой репозиторий в будущем.

import Library_SympyExpressionToPythonFunction

Создает объект вызываемой функции Python с теми же аргументами (число и имена) свободных переменных в выражении sympy.

import Library_SympyExpressionToStringExpression

Буквально просто делает str(SympyExpression)

#-------------------------------------------------------------------------------

Library_GenerateChebyShevPolynomial:::

#-------------------------------------------------------------------------------


import pprint
import Library_SympyExpressionToPythonFunction
import Library_SympyExpressionToStringExpression
import sympy
import sympy.core

def Main(
    ApproximationSymbol = sympy.Symbol('x'),
    ResultType = 'sympy',
    Kind= None,
    Order= None,
    ReturnAll = False,
    CheckArguments = True,
    PrintExtra = False,
    ):

    Result = None

    if (CheckArguments):
        ArgumentErrorMessage = ""

        if (len(ArgumentErrorMessage) > 0 ):
            if(PrintExtra):
                print "ArgumentErrorMessage:\n", ArgumentErrorMessage
            raise Exception(ArgumentErrorMessage)

    ChebyChevPolynomials = []
    ChebyChevPolynomials.append(sympy.sympify(1.))
    ChebyChevPolynomials.append(ApproximationSymbol)

    #Generate the polynomial with sympy:
    for Term in range(Order + 1)[2:]:
        Tn = ChebyChevPolynomials[Term - 1]
        Tnminus1 = ChebyChevPolynomials[Term - 2]
        Tnplus1 = 2*ApproximationSymbol*Tn - Tnminus1

        ChebyChevPolynomials.append(Tnplus1.simplify().expand().trigsimp())

    if(PrintExtra): print 'ChebyChevPolynomials'
    if(PrintExtra): pprint.pprint(ChebyChevPolynomials)


    if (ReturnAll):
        Result = []
        for SympyChebyChevPolynomial in ChebyChevPolynomials:
            if (ResultType == 'python'):
                Result.append(Library_SympyExpressionToPythonFunction.Main(SympyChebyChevPolynomial))
            elif (ResultType == 'string'):
                Result.append(Library_SympyExpressionToStringExpression.Main(SympyChebyChevPolynomial))
            else:
                Result.append(SympyChebyChevPolynomial)

    else:
        SympyExpression = ChebyChevPolynomials[Order] #the last one

        #If the result type is something other than sympy, we can cast it into that type here:
        if (ResultType == 'python'):
            Result = Library_SympyExpressionToPythonFunction.Main(SympyExpression)
        elif (ResultType == 'string'):
            Result = Library_SympyExpressionToStringExpression.Main(SympyExpression)
        else:
            Result = SympyExpression



    return Result 


#-------------------------------------------------------------------------------

Library_SympyChebyShevApproximationOneDimension

#-------------------------------------------------------------------------------


import numpy
import sympy
import sympy.mpmath
import pprint
import Library_SympyExpressionToPythonFunction
import Library_GenerateChebyShevPolynomial

def Main(
    SympyExpression= None,
    DomainMinimumPoint= None,
    DomainMaximumPoint= None,
    ApproximationOrder= None,
    CheckArguments = True,
    PrintExtra = False,
    ):

    #Tsymb = sympy.Symbol('t')
    Xsymb = sympy.Symbol('x')
    DomainStart = DomainMinimumPoint[0]
    print 'DomainStart', DomainStart
    DomainEnd = DomainMaximumPoint[0]
    print 'DomainEnd', DomainEnd

    #Transform the coefficients and the result to be on arbitrary inverval instead of from 0 to 1
    DomainWidth = DomainEnd - DomainStart
    DomainCenter = (DomainEnd - DomainStart) / 2.
    t = (Xsymb*(DomainWidth) + DomainStart + DomainEnd) / 2.
    x = (2.*Xsymb - DomainStart - DomainEnd) / (DomainWidth)
    SympyExpression = SympyExpression.subs(Xsymb, t)

    #GET THE COEFFICIENTS:
    Coefficients = []
    for CoefficientNumber in range(ApproximationOrder):
        if(PrintExtra): print 'CoefficientNumber', CoefficientNumber

        Coefficient = 0.0
        for k in range(1, ApproximationOrder + 1):
            if(PrintExtra): print '  k', k

            CoefficientFunctionPart = SympyExpression.subs(Xsymb, sympy.cos( sympy.pi*( float(k) - .5 )/ float(ApproximationOrder) )  )
            if(PrintExtra): print '  CoefficientFunctionPart', CoefficientFunctionPart

            CeofficientCosArg = float(CoefficientNumber)*( float(k) - .5 )/ float( ApproximationOrder)
            if(PrintExtra): print '  ',CoefficientNumber,'*','(',k,'-.5)/(', ApproximationOrder ,') == ', CeofficientCosArg

            CoefficientCosPart      =   sympy.cos( sympy.pi*CeofficientCosArg )
            if(PrintExtra): print '  CoefficientCosPart', CoefficientCosPart

            Coefficient += CoefficientFunctionPart*CoefficientCosPart

        if(PrintExtra): print 'Coefficient==', Coefficient

        Coefficient = (2./ApproximationOrder)*Coefficient.evalf(10)

        if(PrintExtra): print 'Coefficient==', Coefficient

        Coefficients.append(Coefficient)

    print '\n\nCoefficients'
    pprint.pprint( Coefficients )


    #GET THE POLYNOMIALS:
    ChebyShevPolynomials = Library_GenerateChebyShevPolynomial.Main(
        ResultType = 'sympy',
        Kind= 1,
        Order= ApproximationOrder-1,
        ReturnAll = True,
        )

    print '\nChebyShevPolynomials'
    pprint.pprint( ChebyShevPolynomials )


    Result = 0.0 -.5*(Coefficients[0])
    for Coefficient, ChebyShevPolynomial in zip(Coefficients, ChebyShevPolynomials):
        Result += Coefficient*ChebyShevPolynomial

    #Transform the coefficients and the result to be on arbitrary inverval instead of from 0 to 1
    Result = Result.subs(Xsymb, x)

    return Result

------------------------------------------------------------------------------

Example_SympyChebyShevApproximationOneDimension:

#------------------------------------------------------------------------------
import sympy
import sympy.mpmath
import matplotlib.pyplot as plt
import json
import pprint




import Library_GenerateBesselFunction
import Library_SympyChebyShevApproximationOneDimension
import Library_SympyExpressionToPythonFunction
import Library_GraphOneDimensionalFunction


ApproximationOrder = 10

#CREATE THE EXAMPLE EXRESSION:
Kind = 1
Order = 2
ExampleSympyExpression = sympy.sin(sympy.Symbol('x'))

"""
Library_GenerateBesselFunction.Main(
    ResultType =  'sympy',
    Kind =  Kind,
    Order =  Order,
    VariableNames = ['x'],
    ) 
"""
PythonOriginalFunction = Library_SympyExpressionToPythonFunction.Main( 
    ExampleSympyExpression ,
    FloatPrecision = 100,
    )

#CREATE THE NATIVE CHEBY APPROXIMATION

ChebyDomainMin = 5.
ChebyDomainMax = 10.
ChebyDomain = [ChebyDomainMin, ChebyDomainMax]
ChebyExpandedPolynomialCoefficients, ChebyError = sympy.mpmath.chebyfit(
    PythonOriginalFunction, 
    ChebyDomain, 
    ApproximationOrder, 
    error=True
    )
print 'ChebyExpandedPolynomialCoefficients'
pprint.pprint( ChebyExpandedPolynomialCoefficients )
def PythonChebyChevApproximation(Point):
    Result = sympy.mpmath.polyval(ChebyExpandedPolynomialCoefficients, Point)
    return Result


#CREATE THE GENERIC ONE DIMENSIONAL CHEBY APPROXIMATION:
SympyChebyApproximation = Library_SympyChebyShevApproximationOneDimension.Main(
    SympyExpression = ExampleSympyExpression*sympy.cos( sympy.Symbol('a') ),
    ApproximationSymbol = sympy.Symbol('x'),
    DomainMinimumPoint = [ChebyDomainMin],
    DomainMaximumPoint = [ChebyDomainMax],
    ApproximationOrder = ApproximationOrder
    )


print 'SympyChebyApproximation', SympyChebyApproximation

SympyChebyApproximation = SympyChebyApproximation.subs(sympy.Symbol('a'), 0.0)

print 'SympyChebyApproximation', SympyChebyApproximation

PythonCastedChebyChevApproximationGeneric = Library_SympyExpressionToPythonFunction.Main( 
    SympyChebyApproximation ,
    FloatPrecision = 100,
    )

print 'PythonCastedChebyChevApproximationGeneric(1)', PythonCastedChebyChevApproximationGeneric(1.)
person D Adams    schedule 20.11.2015
comment
Извините за путаницу вопроса ^^ это то, о чем я просил. - person D Adams; 20.11.2015