Error, cannot split rhs for multiple assignment
|
|
|
||||||||||||||||||||||
к режиму параллельного сервера, когда все загруженные документы являются независимыми. Режим параллельного сервера детально рассмотрен в [9-12]. При этом, определение считается сделанным только после его реального вычисления. После перезагрузки пакета все определения переменных и пользовательских функций/процедур (отсутствующих в библиотеках,
логически сцепленных с главной библиотекой пакета) теряются, требуя нового переопределения. Без перезагрузки пакета этого можно добиться по restart-предложению, приводящего все установки ядра пакета в исходное состояние (очистка РОП, отмена всех сделанных ранее определений, выгрузка всех загруженных модулей и т.д.), либо присвоением идентификаторам переменных невычисленных значений вида Id:= ‘Id’. Следующий простой фрагмент иллюстрирует вышесказанное:
> x:= 19.42: y:= 30.175: Grodno:= sin(x) + cos(y); Grodno := 0.8639257079 > restart; Grodno:= sin(x) + cos(y); Grodno := sin(x) + cos(y)
> G:= proc() nargs end proc: G(42, 47, 67, 62, 89, 96); 6 > G:= ‘G’: G(42, 47, 67, 62, 89, 96); G(42, 47, 67, 62, 89, 96)
Из фрагмента хорошо видно, что ранее сделанное определение Grodno-переменной отменяется после выполнения restart-предложения. Выполнение restart-предложения следует лишь на внешнем уровне ТД и не рекомендуется внутри ряда его конструкций (процедуры, функ— ции и др.) во избежание возникновения особых и аварийных ситуаций, включая “зависание” пакета, требующее перезагрузки ПК. При этом, следует иметь в виду, что освобождаемая в этом случае память не возвращается операционной среде, а присоединяется к собственному пулу свободной памяти пакета. Поэтому при необходимости получения максимально возможной памяти для решения больших задач пользователю рекомендуется все же производить перезагрузку пакета в Windows-среде. Тогда как второй способ отмены определенности для переменных более универсален. В книге [103] и прилагаемой к ней библиотеке представлены средства, обеспечивающие возможность восстановления из процедур исходного состояния объектов. Например, вызов процедуры prestart() очищает все переменные, определенные в текущем сеансе, исключая пакетные переменные.
Предложение присвоения. Идентификатору может быть присвоено любое допустимое Maple— выражение, делающее его определенным; в противном случае идентификатор называется неопределенным, результатом вычисления которого является символьное представление самого идентификатора, что весьма прозрачно иллюстрирует следующий простой пример:
>macro(Salcombe = Vasco): Eesti:= 19.95: Vasco:= Noosphere: Tallinn:= 20.06:
>TRG:= sqrt(Lasnamae*(Eesti + Tallinn)/(Salcombe + Vasco)) + `Baltic Branch`;
TRG := 4.472694937 |
Lasnamae |
+ Baltic Branch |
|
Noosphere |
|||
Присвоение идентификатору определенного или неопределенного значения производится посредством традиционного (:=)-оператора присвоения вида А:= В, присваивающего левой А-части В-значение. При этом, в качестве левой А-части могут выступать простой идентификатор, индексированный идентификатор или идентификатор функции с аргументами. Точнее, присвоение А-части В-значения корректно, если A имеет assignable-тип, т.е. type(A, ‘assignable’); true. Вычисленное (или упрощенное) значение В-части присваивается идентификатору А-части.
Оператор присваивания допускает возможность множественного присваивания и определяется конструкциями следующего простого вида:
Id1, Id2, …, Idn:= <Выражение_1>, <Выражение_2>, …, <Выражение_n>
При этом, при равном количестве компонент правой и левой частей присвоения производятся на основе взаимно-однозначного соответствия. В противном случае инициируются ошибочные ситуации «Error, ambiguous multiple assignment» либо «Error, cannot split rhs for multiple assignment«. Следующий фрагмент иллюстрирует случаи множественного присваивания:
31
> A, B, C:= 64, 59:
Error, ambiguous multiple assignment
> V, G, S, Art, Kr, Arn:= 64, 59, 39, 17, 10, 44: [V, G, S, Art, Kr, Arn];
[64, 59, 39, 17, 10, 44]
> x, y, z, t, h:= 2006: [x, y, z, t, h];
Error, cannot split rhs for multiple assignment
[x, y, z, t, h]
Примеры фрагмента достаточно прозрачны и особых пояснений не требуют. Сам принцип реализации множественного присваивания также достаточно прост. Вместе с тем, языком не поддерживаются уже достаточно простые конструкции множественного присваивания, что иллюстрируют первый и последний примеры фрагмента.
Между тем, в ряде случаев возникает необходимость назначения того же самого выражения достаточно длинной последовательности имен или запросов функций. Данная проблема решается оператором &ma, который имеет идентичный с оператором `:=` приоритет. Оператор &ma имеет два формата кодирования, а именно: процедурный и операторный форматы:
&ma(‘x’, ‘y’, ‘z’, …, <rhs>) |
и |
(‘x’, ‘y’, ‘z’, …) &ma (<rhs>) |
В общем случае, в обоих случаях в конструкции lhs &ma rhs элементы lhs должны быть закодированы в невычисленном формате, т.е. в апострофах (‘). Исключение составляет лишь первый случай присвоения. Кроме того, в операторном формате, левая часть lhs должна быть закодирована в скобках. Кроме того, если правая часть rhs удовлетворяет условию type(rhs, {‘..’, ‘<‘, ` <= `, ‘.’, ‘*’, ‘^’, ‘+’, ‘=’}) = true, то правая часть должна также кодироваться в скобках. Наконец, если необходимо присвоить NULL-значение (т. е. ничего) элементам левой части lhs, то в качестве правой части rhs кодируется _NULL-значение. Успешный вызов процедуры &ma или применения оператора &ma возвращает NULL-значение, т. е. ничего с выполнением указанных присвоений. В целом ряде приложений оператор/процедура &ma представляются достаточно полезными [103]. Ниже приведен ряд примеров на применение оператора &ma:
Процедурный формат
> &ma(h(x), g(y), v(z), r(g), w(h), (a + b)/(c — d)); h(x), g(y), v(z), r(g), w(h);
a + b a + b a + b a + b a + b |
|||||||||
, |
, |
, |
, |
||||||
c − d |
c − d |
c − d |
c − d |
c − d |
|||||
> &ma(‘x’, ‘y’, ‘z’, ‘g’, ‘h’, «(a + b)/(c — d)»); |
x, y, z, g, h; |
«(a+b)/(c-d)», «(a+b)/(c-d)», «(a+b)/(c-d)», «(a+b)/(c-d)», «(a+b)/(c-d)»
> &ma(‘x’, ‘y’, ‘z’, ‘g’, ‘h’, _NULL); x, y, z, g, h;
> &ma’x’, ‘y’, ‘z’, ‘g’, ‘h’, 2006); x, y, z, g, h; |
2006, 2006, 2006, 2006, 2006 |
|||||||||||
> &ma(‘x’, ‘y’, ‘z’, ‘g’, ‘h’, sin(a)*cos(b)); x, y, z, g, h; |
||||||||||||
sin(a) cos(b), sin(a) cos(b), sin(a) cos(b), sin(a) cos(b), sin(a) cos(b) |
||||||||||||
Операторный формат |
||||||||||||
> (‘x’, ‘y’, ‘z’, ‘g’, ‘h’) |
&ma |
_NULL; x, y, z, g, h; |
||||||||||
> (‘x’, ‘y’, ‘z’, ‘g’, ‘h’) |
&ma |
2006; x, y, z, g, h; 2006, 2006, 2006, 2006, 2006 |
||||||||||
> (‘x’, ‘y’, ‘z’, ‘g’, ‘h’) |
&ma |
(sin(a)*cos(b)); |
x, y, z, g, h; |
|||||||||
sin(a) cos(b), sin(a) cos(b), sin(a) cos(b), sin(a) cos(b), sin(a) cos(b) |
||||||||||||
> (‘x’, ‘y’, ‘z’, ‘g’, ‘h’) |
&ma |
((a + b)/(c — d)); |
x, y, z, g, h; |
|||||||||
a + b a + b a + b a + b a + b |
||||||||||||
, |
, |
, |
, |
|||||||||
c − d |
c − d |
c − d |
c − d |
c − d |
Для проверки идентификатора на предмет его определенности используется встроенная функция assigned языка, кодируемая в виде assigned(Идентификатор) и возвращающая значение true в случае определенности идентификатора (простого, индексированного или вызова функции/процедуры), и false-значение в противном случае. При этом, следует иметь в виду, что определенным идентификатор полагается тогда, когда он был использован в качестве левой части (:=)-оператора присвоения, даже если его правая часть являлась неопределенной. Ли-
32
бо он получил присвоение по аssign-процедуре. Следующий простой фрагмент иллюстрирует вышесказанное:
> agn:= 1947: avz:= grodno: assign(vsv=1967, art=kr): seq(assigned(k), k= [agn, avz, vsv, art]); true, true, true, true
> map(type,[agn, avz, vsv, art],’assignable’);
[false, true, false, true]
> map(assigned, [agn, avz, vsv, art]);
Error, illegal use of an object as a name
С другой стороны, по конструкции type(Id, ‘assignable’) можно тестировать допустимость присвоения Id–переменной (простой, индексированной или вызова функции/процедуры) выражения: возврат true-значения говорит о такой возможности, false – нет. Следует обратить внимание на последний пример фрагмента, иллюстрирующий некорректность использования встроенной функции map при попытке организации простого цикла.
Вызов функции indets(W {, <Тип>}) возвращает множество всех идентификаторов заданного его первым фактическим W-аргументом Maple-выражения. При этом, W-выражение рассматривается функцией рациональным, т.е. образованным посредством {+, —, *, /}-операций. Поэтому в качестве результата могут возвращаться не только простые идентификаторы W-выра- жения, но и неопределенные его подвыражения. В случае кодирования второго необязательного аргумента, он определяет тип возвращаемых идентификаторов, являясь своего рода фильтром. В качестве второго фактического аргумента могут выступать как отдельный тип, так и их множество в соответствии с типами, распознаваемыми тестирующей type-функцией языка, рассматриваемой детально ниже. Следующий фрагмент иллюстрирует применение indets-функции для выделения идентификаторов переменных:
> indets(x^3 + 57*y — 99*x*y + (z + sin(t))/(a + b + c) = G);
{sin(t), x, y, z, t, a, b, c, G}
> indets(x^3 + 57*y — 99*x*y + (z + sin(t))/(a + b + c) = G, function);
{sin(t)}
> indets(x^3 + z/y — 99*x*y + sin(t), {integer, name});
{-99, -1, 3, x, y, z, t}
> indets(x^3 + z/y — 99*x*y + sin(t), {integer, name, `*`, `+`});
{-99, -1, 3, x3 + yz − 99 x y + sin(t ), yz , −99 x y, x, y, z, t }
> indets(x^3 + z/y — 99*x*y + sin(t), {algnum, trig});
{-99, -1, 3, sin(t)}
Из приведенного фрагмента, в частности, следует, что по indets-функции можно получать не только неопределенные идентификаторы, но и числовые компоненты тестируемого выражения, а также совокупности комбинаций составляющих его компонент. Таким образом, indets-функция несет существенно более развитую смысловую нагрузку, чем определение простых идентификаторов. По своим возможностям она представляется достаточно эффективным средством при решении задач символьных обработки и анализа выражений, а также в целом ряде других важных задач.
Для возможности использования произвольного А-выражения в качестве объекта, которому могут присваиваться значения, Maple-язык располагает evaln-функцией, кодируемой в виде evaln(A) и возвращающей имя выражения. В качестве А-выражения допускаются: простой идентификатор, индексированный идентификатор, вызов функции/процедуры или конкатенация символов. В результате применения к А-выражению evaln-функции оно становится доступным для присвоения ему значений, однако если ему не было сделано реального присвоения, то применение к нему assigned-функции возвращает значение false, т.е. А-выраже- ние остается неопределенным. Таким образом, по конструкции Id:= evaln(Id) производится
33
отмена присвоенного Id-идентификатору значения, делая его неопределенным, как это иллюстрирует следующий простой фрагмент:
> Asv:= 32: assigned(Asv); Asv:= evaln(Asv): Asv, assigned(Asv); true
Asv, false
> Asv:= 67: assigned(Asv); Asv:= ‘Asv’: Asv, assigned(Asv); true
Asv, false
Из приведенного фрагмента непосредственно следует, что первоначальное присвоение Asv— идентификатору значения делает его определенным, на что указывает и результат вызова assigned-функции. Тогда как последующее вычисление предложения Asv:=evaln(Asv) делает Asv-идентификатор вновь неопределенным. Вторым способом приведения идентификатора к неопределенному состоянию является использование конструкции вида Id:= ‘Id’, как это иллюстрирует второй пример фрагмента.
Для выполнения присвоений можно воспользоваться и assign-процедурой, допускающей в общем случае три формата кодирования, а именно: assign({A, B|A = B|C}), где А – идентификатор, В — любое допустимое выражение и С — список, множество или последовательность уравнений. В первых двух случаях применение assign-процедуры эквивалентно применению (:=)-оператора присвоения, тогда как третий случай применяется для обеспечения присвоения левой части каждого элемента списка, множества или последовательности уравнений. Простой фрагмент иллюстрирует вышесказанное:
>assign(AGn, 59); assign(AVz=64); assign([xx=1, yy=2, zz=3, h=cos(y)*tan(z)]);
>assign({xx1=4, yy1=5, zz1=6, y=x*sin(x)}); [AGn, AVz, xx, yy, zz, xx1, yy1, zz1, y, h];
[59, 64, 1, 2, 3, 4, 5, 6, x sin(x), cos(x sin(x)) tan(z)]
> assign(‘x’, assign(‘y’, assign(‘z’, 64))); [x, y, z]; |
# Maple 7 и выше |
[64] |
|
> assign(‘x’, assign(‘y’, assign(‘z’, 64))); [x, y, z]; |
# Maple 6 и ниже |
Error, (in assign) invalid arguments
> `if`(type(AV, ‘symbol’) = true, assign(AV, 64), false); AV;
64
> GS:= op([57*sin(19.95), assign(‘VG’, 52)]) + VG*cos(19.99);
GS := 72.50422598
Успешное выполнение assign-процедуры производит указанные присвоения и возвращает NULL-значение, в противном случае инициируется соответствующая ошибочная ситуация. Данная ситуация, в частности, возникает при попытке рекурсивного вызова assign-процеду- ры для релизов 6 и ниже, тогда как в более старших релизах подобного ограничения нет. По отношению к процедуре assign релизы пакета характеризуются весьма существенной несовместимостью, что стимулировало нас к созданию аналога стандартной процедуры, который не только устраняет указанную несовместимость, но и расширяет функциональные возможности [31,39,41-43,45,46,103]. Это и другие наши средства рассмотрены детально в книге [103] и представлены в прилагаемой к ней библиотеке программных средств.
При этом, следует отметить, что в целом ряде случаев assign-процедура является единственно возможным способом присвоения значений, например, внутри выражений, как это иллюстрирует последний пример фрагмента, который содержит структуры и функции, рассматриваемые ниже. Механизм assign-процедуры достаточно эффективен в различных вычислительных констукциях, многочисленные примеры применения которого приводятся ниже при рассмотрении различных аспектов Maple-языка, а также в нашей библиотеке [103].
Обратной к assign-процедуре является unassign-процедура с форматом кодирования:
unassign(<Идентификатор_1>, <Идентификатор_2>, …)
34
отменяющая определения для указанных последовательностью ее фактических аргументов идентификаторов. Успешный вызов процедуры unassign выполняет отмену назначений, возвращая NULL-значение. Однако, процедура не действует на идентификаторы с protected-ат- рибутом, инициируя ошибочную ситуацию с выводом соответствующей диагностики. Приведем простые примеры на использование unassign-процедуры.
> AS:= 39: AV:= 64: AG:= 59: Kr:= 10: Art:= 17: AS, AV, AG, Kr, Art;
39, 64, 59, 10, 17
> unassign(AS, AV, AG, Kr, Art);
Error, (in unassign) cannot unassign ’39’ (argument must be assignable)
> unassign(‘AS’, ‘AV’, ‘AG’, ‘Kr’, ‘Art’); AS, AV, AG, Kr, Art;
AS, AV, AG, Kr, Art
> `if`(type(AV, ‘symbol’)=true, assign(AV, 64), unassign(‘AV’)); AV;
64
Вприведенном фрагменте пяти переменным присваиваются целочисленные значения, а затем по unassign-процедуре делается попытка отменить сделанные назначения. Попытка вызывает ошибочную ситуацию, обусловленную тем, что в точке вызова unassign-процедуре передаются не сами идентификаторы, а их значения (кстати, именно данная ситуация одна из наиболее типичных при ошибочных вызовах assign—процедуры). Для устранения ее идентификаторы следует кодировать в невычисленном формате (кодируя в апострофах), что и иллюстрирует повторный вызов unassign-процедуры. Последний пример иллюстрирует применение процедур assign и unassign в условном if-предложении языка, по которому AV-переменной присваивается целочисленное значение, если она была неопределенной, и отменяется ее определение в противном случае.
Вцелях защиты идентификаторов от возможных модификаций их определений (назначений) им присваивается protected-атрибут, делающий невозможной какую-либо модификацию указанного типа. Большинство пакетных идентификаторов имеют protected-атрибут, в чем легко убедиться, применяя к ним attributes-функцию, кодируемую в следующем формате:
attributes(<Идентификатор>)
и возвращающую значения атрибутов заданного идентификатора, в частности protected-ат- рибута. Если идентификатору не приписано атрибутов, то вызов на нем attributes-функции возвращает NULL-значение, т.е. ничего. Для защиты от модификации либо снятия защиты используются процедуры protect и unprotect языка Maple соответственно, кодируемые в следующем простом формате:
{protect|unprotect}(<Идентификатор_1>, <Идентификатор_2>, …)
Следующий весьма простой фрагмент Maple-документа иллюстрирует вышесказанное:
>protect(AV_42); attributes(AV_42); protected
>unassign(AV_42);
Error, (in assign) attempting to assign to `AV_42` which is protected
> AV_42:= 64:
Error, attempting to assign to `AV_42` which is protected
> unprotect(AV_42); attributes(AV_42); AV_42:= 64;
AV_42 := 64
Следует при этом отметить, что действие protect-процедуры не распространяется на глобаль— ные предопределенные переменные Maple, значения которых можно модифицировать согласно условиям пользователя. Такая попытка вызывает ошибочную ситуацию:
> map(attributes,[Digits,Order,printlevel]);protect(‘Digits’);protect(‘Order’);protect(‘printlevel’);
[]
Error, (in protect) an Environment variable cannot be protected Error, (in protect) an Environment variable cannot be protected Error, (in protect) an Environment variable cannot be protected
35
Хотя по unprotect-процедуре отменяется protected-атрибут любого идентификатора, однако для пакетных идентификаторов этого (по целому ряду причин, здесь не рассматриваемых) не рекомендуется делать.
В целом ряде случаев в качестве весьма полезных средств могут выступать две встроенные функции со следующими форматами кодирования: unames() и anames({ |<Тип>}), возвращающие последовательности соответственно неопределенных и определенных идентификаторов (как пользовательских, так и пакетных), приписанных текущему Maple-сеансу. При этом, для случая anames-функции можно получать выборку определенных идентификаторов, значения которых имеют указанный Тип. Следующий фрагмент иллюстрирует результат вызова указанных выше функций:
> restart; unames();
identical, anyfunc, equation, positive, Integer, restart, radical, And, gamma, neg_infinity, none, default, nonposint, relation, odd, infolevel, indexable, algebraic, SFloat, RootOf, TABLE, float, real_to_complex, embedded_real, vector, _syslib, realcons, name, assign, INTERFACE_GET, …
> restart: SV:= 39: GS:= 82: Art:= sin(17): Kr:= sqrt(10): AV:= 64: anames();
sqrt/primes, type/interfaceargs, GS, sqrt, AV, csgn, interface, type/SymbolicInfinity, Art, sin, SV, Kr
> anames(‘integer’); # Maple 8
sqrt/primes, GS, Digits, printlevel, Order, AV, SV
> anames(‘environment’);
Testzero, UseHardwareFloats, Rounding, %, %%%, Digits, index/newtable, mod, %%, Order, printlevel, Normalizer, NumericEventHandlers
> SV:= 39: GS:= 82: `Art/Kr`:= sin(17): Kr:= sqrt(10): _AV:= 64: anames(‘user’); # Maple 10
Kr, SV, GS
> anames(‘alluser’);
Kr, SV, GS, _AV, Art/Kr
При этом, unames-функция была вызвана в самом начале сеанса работы с пакетом и возвращаемый ею результат представлен только начальным отрезком достаточно длинной последовательности пакетных идентификаторов. Что касается anames-функции, то она в качестве второго необязательного аргумента допускает тип, идентификаторы с которым будут ею возвращаться. При этом, дополнительно к типу и в зависимости от релиза в качестве второго аргумента допускается использование таких ключевых слов как environment, user, alluser, с назначением которых можно ознакомиться по справочной базе пакета.
36
1.3. Средства Maple-языка для определения свойств
переменных
Важным средством управления вычислениями и преобразованиями в среде Maple-языка является assume-процедура и ряд сопутствующих ей средств. Процедура имеет следующие три формата кодирования:
assume(x1, p1, x2, p2, …) assume(x1::p1, x2::p2, …) assume(x1p1, x2p2, …)
и позволяет наделять идентификаторы (переменные) или допустимые Maple-выражения xj заданными свойствами pj, т.е. устанавливать определенные свойства и соотношения между ними. Третий формат процедуры определяет соотношения, налагающие свойство pj на выражение xj. Например, простейшие, но весьма часто используемые вызовы assume(x >= 0) процедуры, определяют для некоторой х-переменной свойство быть неотрицательной действительной константой. Наделяемое по assume-процедуре свойство не является пассивным и соответствующим образом обрабатывается Maple-языком пакета при выполнении вычислений либо преобразований выражений, содержащих переменные, наделенные подобными свойствами. Тестировать наличие приписанного х-переменной свойства можно по вызову процедуры about(x), тогда как наделять х-переменную дополнительными свойствами можно по вызову процедуры additionally(x, Свойство). Следующий фрагмент иллюстрирует сказанное:
> assume(x >= 0): map(about, [x, y, z]); [ ]
Originally x, renamed x~: is assumed to be: RealRange(0, infinity) y: nothing known about this object
z: nothing known about this object |
||
> assume(a >= 0): A:= sqrt(-a): assume(a <= 0): B:= sqrt(-a): [A, B]; |
[ a~ I, −a~ ] |
|
> simplify(map(sqrt, [x^2*a, y^2*a, z^2*a])); [x~ |
−a~ I, y2 a~ , |
z2 a~ ] |
> additionally(x, ‘odd’): about(x); |
||
Originally x, renamed x~: is assumed to be: |
||
AndProp(RealRange(0, infinity), LinearProp(2, integer, 1)) |
||
> map(is, [x, y, z], ‘odd’, ‘posint’); [true, false, false] |
||
> assume(y, ‘natural’): map(is, [x, y], ‘nonnegative’); |
[true, true] |
|
> unassign(‘x’): about(x); |
||
x: nothing known about this object |
> assume(y, {y >= 0, y < 64, ‘odd’}): about(y);
Originally y, renamed y~: is assumed to be: {LinearProp(2,integer,1), 0 <= y, y < 64}
> hasassumptions(y); true
Из примеров данного фрагмента, в частности, следует, что переменная с приписанным ей свойством выводится помеченной символом тильды ( ), а по about-процедуре выводится информация о всех приписанных переменной свойствах либо об их отсутствии. Один из примеров фрагмента иллюстрирует влияние наличия свойства положительной определенности переменной на результат упрощения содержащего ее выражения. В другом примере фрагмента иллюстрируется возможность определения для переменной множественности свойств, определяемых как поддерживаемыми языком стандартными свойствами, так и допустимыми отношениями для переменной. Вызов процедуры hasassumptions(x) возвращает true, если на x-выражение было наложено какое-либо соотношение, и false в противном случае.
Режим идентификации assume-переменных определяется showassumed-параметром interface— процедуры, принимающим значение {0|1 (по умолчанию)|2}: 0 — отсутствует идентификация, 1 — переменные сопровождаются знаком тильды и 2 — все такие переменные перечисляются в конце выражений, как это иллюстрирует следующий весьма простой фрагмент:
> assume(V >= 64): assume(G >= 59): S:= V+G; interface(showassumed=0); V + G;
S := V~ + G~
37
V + G
> assume(Art >= 17, Kr >= 10): interface(showassumed=2): S^2 + (Art + Kr)^2;
S2 + (Art + Kr)2
with assumptions on Art and Kr
> assume(x, ‘odd’, y::integer, z >= 0), is(x + y, ‘integer’); true
> x:= ‘x’: unassign(‘y’), map(is, [x, y], ‘integer’), is(z, ‘nonnegative’); [false, false], true
До 7-го релиза включительно оперативно переопределять режим идентификации assume-пе- ременных можно было переключателями функции Assumed Variables группы Options GUI.
По тестирующей процедуре is(x, Свойство) возвращается true-значение, если х-переменная обладает указанным вторым фактическим аргументом свойством, и значение false в противном случае. При невозможности идентифицировать для х-переменной свойство (например,
если она по assume—процедуре свойствами не наделялась) возвращается FAIL-значение. Наконец, отменять приписанные х-переменной свойства можно посредством выполнения простой конструкции x:= ‘x’ либо вызовом unassign(‘x’); сказанное иллюстрируют последние примеры предыдущего фрагмента. При этом, проверять можно как конкретную отдельную переменную, так и выражение по нескольким ведущим переменным и набору искомых свойств.
Maple-язык поддерживает работу со свойствами шести основных групп, а именно:
1) имя свойства, например, continuous, unary;
2) большинство имен типов, например, integer, float, odd, even;
3) числовые диапазоны, например, RealRange(a, b), RealRange(-infinity, b), RealRange( a, infinity), где a и b могут быть или числовыми значениями или Open(a), где a – числовое значение
4) AndProp(a, b, …) – and-выражение свойств <a and b and …>, где a, b, … – свойства, определенные выше
5) OrProp(a, b, …) – or- выражение свойств, где объект может удовлетворять любому из a, b, …
свойств
6) диапазон свойств p1 .. p2, где p1 и p2 свойства. Данное свойство означает, что объект удовлетворяет по меньшей мере p2, но не более, чем p1; например, integer .. rational удовлетворяется integers/2.
За более детальной информацией по поддерживаемым Maple-языком свойствам остальных групп можно обращаться либо к справке по пакету, либо к книгам [8-14,78-86,88,103,105].
Механизм свойств, определяемый assume и сопутствующей ей группой процедур coulditbe, additionally, is, about, hasassumptions и addproperty, использует специальную глобальную
_EnvTry-переменную для определения режима как идентификации у переменных приписанных им свойств, так и их обработки. При этом, данная переменная допускает только два значения: normal (по умолчанию) и hard, из которых указание второго значения может потребовать при вычислениях существенных временных затрат. В текущих реализациях пакета значение глобальной _EnvTry-переменной, определяющей режим обработки переменных, наделенных assume-свойствами, не определено, что иллюстрирует следующий достаточно прозрачный фрагмент:
>_EnvTry, about(_EnvTry); _EnvTry
_EnvTry: nothing known about this object
>assume(V >= 64): about(V);
Originally V, renamed V~: is assumed to be: RealRange(64, infinity)
> `if`(is(V, RealRange(64, infinity)), ln(V) + 42, sqrt(Art + Kr)); ln(V~) + 42
> assume(47 <= G, G <= 59): about(G); |
|
Originally G, renamed G~: is assumed to be: RealRange(47, 59) |
|
> `if`(is(G, RealRange(47, 59)), [10, 17, Sv, Art, Kr], Family(x, y, z)); |
[10, 17, Sv, Art, Kr] |
> assume(x >= 0), simplify(sqrt(x^2)), simplify(sqrt(y^2)); x~, csgn(y) y |
|
> sqrt(a*b), sqrt(a^2), assume(a >= 0, b <= 0), sqrt(a*b), sqrt(a^2); |
a b, a2 , −a~ b~ I, a~ |
38
Механизм приписанных свойств является достаточно развитым и мощным средством как чи— словых вычислений, так и символьных вычислений и преобразований. Использование его оказывается весьма эффективным при программировании целого ряда важных задач во многих приложениях. Последние примеры предыдущего фрагмента иллюстрируют некоторые простые элементы его использования в конкретном программировании. Тогда как конкретный assume-механизм базируется на алгоритмах Т. Вейбеля и Г. Гоннета. С интересным обсуждением принципов его применения, реализации и ограничений можно довольно детально ознакомиться в интересных работах указанных авторов, цитируемых в конце справки по пакету (см. ?assume), и цитируемых в них многочисленных источниках различного назначения.
39
1.4. Типы числовых и символьных данных Maple-языка
Средства Maple-языка поддерживают работу как с простыми, так и сложными типами данных числового или символьного (алгебраического) характера. В первую очередь рассмотрим типы простых данных числового характера, предварив краткой информацией по очень важным встроенным функциям nops и op, непосредственно связанных со структурной организацией Maple-выражений. Первая функция возвращает число операндов выражения, заданного ее единственным фактическим аргументом. Тогда как вторая имеет более сложный формат кодирования следующего вида:
op({ |n,|n..m,|<Список>,} <Выражение>)
где первый необязательный фактический аргумент определяет возврат соответственно: n-го операнда, с n-го по m-й операнды либо операнды согласно Списка их позиций в порядке возрастания уровней вложенности Выражения. При этом, при n=0 возвращается тип самого выражения, а в случае отсутствия указанного первым аргументом операнда инициируется ошибочная ситуация. Для случая n < 0 выполняется соотношение op(n,V) ≡ op(nops(V)+n+1,V), где V — выражение, а для неопределенного Id-идентификатора имеют место соотношения: op(0, Id) symbol и op(1, Id) Id. Отсутствие первого аргумента op-функции аналогично вызову вида op(1..nops(V),V). Для вывода структурной организации произвольного выражения может оказаться полезной конструкция вида:
op(‘k’, <Выражение>) $’k’=0 .. nops(<Выражение>)
вычисление которой возвращает последовательность типа и всех операндов первого уровня вложенности указанного выражения, как это иллюстрирует следующий простой пример:
> Art:= 3*sin(x) + 10*cos(y)/AV + sqrt(AG^2 + AS^2)*TRG: op(‘k’, Art)$’k’=0 .. nops(Art);
+, 3 sin(x), |
5 |
cos(y), AG2 + AS2 |
TRG |
||
32 |
|||||
Ниже будет рассмотрено достаточно средств Maple-языка, ориентированных на задачи символьной обработки выражений, включая и те, которые базируются на их структурном анализе. Целый ряд средств для решения подобных задач предоставляет и наша Библиотека [103]. Нам же для дальнейшего будет пока вполне достаточно информации по nops и op.
• Целые (integer); представляют собой цепочки из одной или более цифр, максимальная длина которых определяется используемой платформой ЭВМ: для 32-битной она не превышает 524280 цифр, а для 64-битной — 38654705646 цифр. Целые могут быть со знаком и без: 1999, — 57, 140642. На числах данного типа функции op и nops возвращают соответственно значение числа и значение 1, тогда как функция type идентифицирует их тип как integer, например:
> op(429957123456789), nops(429957123456789); 429957123456789, 1 > type(429957123456789, ‘integer’); true
• Действительные (float) с плавающей точкой; представляют собой цепочки из десятичных цифр с десятичной точкой в {начале|середине|конце} цепочки; числа данного типа допускают следующие два основных формата кодирования:
[<знак>]{<целое>.<целое>|.<целое>|<целое>.} Float([<знак>]<мантисса>, [<знак>]<экспонента>) ≡ [<знак>]<мантисса>.{E|e}[<знак>]<экспонента>
В качестве мантиссы и экспоненты используются целые со знаком или без; при этом, мантисса может иметь любую длину, но экспонента ограничивается длиной машинного слова: для 32битной платформы значение экспоненты не превышает целого 2147483647, а для 64-битной платформы – целого значения 9223372036854775807. Тогда как максимальное допустимое число цифр мантиссы аналогично максимальному допустимому числу цифр целого (integer) числа. Число цифр мантиссы, участвующих в операциях арифметики с плавающей точкой,
40
Соседние файлы в папке Books
- #
- #
18.03.201519.23 Mб35Дьяконов_maple_9.5-10.djvu
Hi there!
The first time I compile the following code, I get the error message
«Error, cannot split rhs into multiple assignment.»
when trying to solve an issue with the procedure. I then have to compile the procedure over and over again, until it finally works (which it does eventually, without changing the code.) The problematic line is
Knoten, Eigenvektoren := Eigenvectors(evalf[15](M));
it is one of the last lines within the code below. Is it possible to get rid of that issue? It is annoying and unprofessional to have to compile a code over and over again until it finally works.
GaußKronrodQuadraturKurz:= proc(Unten, Oben, f,G,n)::real;
#Unten:= Untere Intervallgrenze; Oben:= Obere Intervallgrenze; G:= Gewicht;
#f:= zu untersuchende Funktion; n:= Berechnung der Knotenanzahl mittels 2*n+1
local
A,B,P,S,T, #Listen
a,b,p,s,t, #Listenelemente
i,j,k, #Laufvariablen
M, #werdende Gauss-Kronrod-Jacobi-Matrix
m, #Matrixeinträge
u,l, #Hilfsvariablen Gemischte Momente
RekursivesZwischenergebnis,Gewichte,Knoten,Eigenvektoren,AktuellerNormierterVektor,Hilfsvariable,Endergebnis;
with(LinearAlgebra):
A := [seq(a[i], i = 0 .. n)];
B := [seq(b[i], i = 0 .. n)];
P := [seq(p[i], i = -1 .. ceil(3*n/2)+1)];
S := [seq(s[i], i = -1 .. floor(n/2))];
T := [seq(t[i], i = -1 .. floor(n/2))];
p[-1]:= 0;
p[0]:=1;
for i from -1 to floor(n/2) do
s[i]:=0;
t[i]:=0
end do;
for j from 1 to 2*n+1 do
RekursivesZwischenergebnis:= x^j;
for i from 0 to j-1 do
RekursivesZwischenergebnis:= RekursivesZwischenergebnis —
(int(x^j*p[i],x=Unten..Oben)/int(p[i]*p[i],x=Unten..Oben))*p[i] #Gram-Schmidt algorithm
end do;
p[j]:=RekursivesZwischenergebnis;
end do;
a[0]:=-coeff(p[1],x,0);
#p[0+1]=(x-a[0])*p[0]-b[0]*p[0-1] -> p[1]=x*p[0]-a[0]*p[0]-b[0]*p[-1] ->
#p[1]=x*1-a[0]*1-0 -> a[0]=x-p[1] -> a[0]= -coeff(p[1],x,0), da p[1] monisch ist und von Grad 1 #ist
b[0]:=int(p[0]^2, x=Unten..Oben); #by definition
for j from 1 to ceil(3*n/2) do
#Genau genommen muss a nur bis floor(3/(2*n)) initialisiert werden, allerdings wird der Wert #ohnehin für die Berechnung von b gebraucht. Die Initialisierung schadet nicht.
a[j]:= coeff(p[j],x,j-1)- coeff(p[j+1],x,j);
#p[j+1]=(x-a[j])*p[j]-b[j]*p[j-1] -> p[j+1]=x*p[j]-a[j]*p[j]-b[j]p[j-1] ->
#coeff(p[j+1],x,j)=coeff(x*p[j],x,j)-coeff(a[j]*p[j],x,j)
#(da b[j]*p[j-1] vom Grad j-1 ist) ->
#coeff(p[j+1],x,j)=coeff(x*p[j],x,j)-a[j], da p[j] monisch ist ->
#coeff(p[j+1],x,j)=coeff(p[j],x,j-1)-a[j]->
#a[j]=coeff(p[j],x,j-1)-coeff(p[j+1],x,j)
b[j]:= quo((x-a[j])*p[j]-p[j+1],p[j-1],x);
#p[j+1]=(x-a[j])*p[j]-b[j]*p[j-1] -> -p[j+1]+(x-a[j])*p[j]= b[j]*p[j-1]
#b[j]=((x-a[j])*p[j]-p[j+1])/p[j-1]
end do;
t[0]:=b[n+1]; #t[0]:= /hat{b}[0], Beginn der ostwärtigen Phase
for i from 0 to n-2 do # n-2 ist die Anzahl der zu berechnenden Diagonalen
u:=0;
for k from floor((i+1)/2) to 0 by -1 do # aufgrund des diagonalen Vorgehens ist nur bei jedem
# zweiten Schleifendurchlauf eine Inkrementierung
# vorzunehmen
l:=i-k;
u:=u+(a[k+n+1]-a[l])*t[k]+b[k+n+1]*s[k-1]-b[l]*s[k]; # Ausrechnen gemischter Momente über die
# fünfgliedrige Rekursion
s[k]:=u
end do;
for j from -1 to floor(n/2) do # Durchrotieren der Werte der gemischten Momente, da ein # jeweiliges gemischtes Moment beim zweiten auf die Generierung # folgenden
# Schleifendurchlauf das letzte mal benötigt und danach über-
# schrieben wird. Die am Ende vorliegenden Werte sind gerade
# die, die bei der südwärtigen Phase benötigt werden.
Hilfsvariable:=s[j];
s[j]:=t[j];
t[j]:=Hilfsvariable
end do;
end do;
for j from floor(n/2) to 0 by -1 do
s[j]:=s[j-1]
end do;
for i from n-1 to 2*n-3 do #entspricht der Anzahl der restlichen Diagonalen
u:=0;
for k from i+1-n to floor((i-1)/2) do #berechnet die gemischten Momente innerhalb einer
#Diagonalen, von oben rechts nach unten links.
l:=i-k;
j:=n-1-l;
u:=u-(a[k+n+1]-a[l])*t[j]-b[k+n+1]*s[j]+b[l]*s[j+1];
s[j]:=u
end do;
if i mod 2 = 0 then #Ausrechnen eines fehlenden Koeffizienzen über die fünfgliedrige Rekursion #am Eintrag (k,k)
k:= i/2;
a[k+n+1]:=a[k]+(s[j]-b[k+n+1]*s[j+1])/t[j+1]
else #Ausrechnen eines fehlenden Koeffizienzen über die fünfgliedrige Rekursion #am Eintrag (k,k-1)
k:=(i+1)/2;
b[k+n+1]:=s[j]/s[j+1]
end if;
for j from -1 to floor(n/2) do #Erneutes Durchrotieren der Werte der gemischten Momente
Hilfsvariable:=s[j];
s[j]:=t[j];
t[j]:=Hilfsvariable
end do;
end do;
a[2*n]:=a[n-1]-b[2*n]*s[0]/t[0]; #Berechnung des letzten fehlenden Koeffizienten über die #fünfgliedrige Rekursion am Eintrag (n-1,n-1)
M:=Matrix(2*n+1, shape=symmetric);#definieren der werdenden Gauß-Krondrod-Matrix
M(1,1):=a[0];
for m from 2 to (2*n+1) do #generieren der Gauss-Kronrod-Matrix
M(m-1,m):= sqrt(b[m-1]);
M(m,m-1):= sqrt(b[m-1]);
M(m,m):= a[m-1];
end do;
Knoten, Eigenvektoren := Eigenvectors(evalf[15](M));# «Die gesuchten Knoten sind die Eigenwerte #dieser Matrix, und die Gewichte sind proportional zu den ersten Komponenten der normalisierten #Eigenvektoren»
for m from 1 to 2*n+1 do
AktuellerNormierterVektor:= Normalize(Column(Eigenvektoren,m),Euclidean);
Gewichte[m]:=AktuellerNormierterVektor[1]^2*b[0]
end do;
Endergebnis:=Re(add(Gewichte[i]*eval(f*diff(G,x),x=Knoten[i]),i=1..2*n+1));
end proc
An example of an application of the procedure is
GaußKronrodQuadraturKurz(-2, 1, 3*x*3*x^2*sin(x),x,3)
Thank you very much!
If you do either of the following then the call will return an expression sequence of containing three Matrices, and it will work for input A
a Matrix
.
with(LinearAlgebra):
P, L, U := LUDecomposition(A);
Or,
P, L, U := LinearAlgebra:-LUDecomposition(A);
But if you don’t load the LinearAlgebra
package and you also don’t use the long-form name of that package’s export then you’re simply making a function call to an undefined name.
And, if you are in fact accidentally simply calling the undefined name LUDecomposition
(because you have not used one of the mentioned methods for calling the package export of that name), then the result will be an unevaluated function call. And you cannot assign that single thing to three names on the left-hand-side of an assignment statement.
restart;
A := Matrix([[1,2],[3,5]]):
# Look, this next one does no computation.
# The return value is a single thing, an unevaluated
# function call.
LUDecomposition(A);
/[1 2]
LUDecomposition|[ ]|
[3 5]/
P, L, U := LUDecomposition(A);
Error, cannot split rhs for multiple assignment
But this would return the actual result,
LinearAlgebra:-LUDecomposition(A);
[1 0] [1 0] [1 2]
[ ], [ ], [ ]
[0 1] [3 1] [0 -1]
And so those three Matrices could be assigned to three names.
P, L, U := LinearAlgebra:-LUDecomposition(A);
[1 0] [1 0] [1 2]
P, L, U := [ ], [ ], [ ]
[0 1] [3 1] [0 -1]
Another way to do it is to first load the whole package, so that you can utilize the short-form command name,
with(LinearAlgebra):
LUDecomposition(A);
[1 0] [1 0] [1 2]
[ ], [ ], [ ]
[0 1] [3 1] [0 -1]
P, L, U := LUDecomposition(A);
[1 0] [1 0] [1 2]
P, L, U := [ ], [ ], [ ]
[0 1] [3 1] [0 -1]
Я пытаюсь использовать метод LU для решения Ax=B
, и когда я начинаю это делать, используя:
(P, L, U := LUDecomposition(A))
Для создания трех разных матриц (P, L, U)
я получаю сообщение об ошибке
Ошибка, невозможно разделить правую часть для множественного назначения что не имеет смысла, так как
LUDecomposition
создает 3 матрицы, которые соответствуют моемуP, L, U
.
Если вы выполните одно из следующих действий, вызов вернет последовательность выражений, содержащую три матрицы, и она будет работать для ввода A
a Matrix
.
with(LinearAlgebra):
P, L, U := LUDecomposition(A);
Или,
P, L, U := LinearAlgebra:-LUDecomposition(A);
Но если вы не загружаете пакет LinearAlgebra
и не используете полное имя экспорта этого пакета, то вы просто вызываете функцию с неопределенным именем.
И, если вы на самом деле случайно просто вызываете неопределенное имя LUDecomposition
(поскольку вы не использовали один из упомянутых методов для вызова экспорта пакета с этим именем), то результатом будет неоцененный вызов функции. И вы не можете присвоить эту единственную вещь трем именам в левой части оператора присваивания.
restart;
A := Matrix([[1,2],[3,5]]):
# Look, this next one does no computation.
# The return value is a single thing, an unevaluated
# function call.
LUDecomposition(A);
/[1 2]
LUDecomposition|[ ]|
[3 5]/
P, L, U := LUDecomposition(A);
Error, cannot split rhs for multiple assignment
Но это вернет фактический результат,
LinearAlgebra:-LUDecomposition(A);
[1 0] [1 0] [1 2]
[ ], [ ], [ ]
[0 1] [3 1] [0 -1]
Таким образом, этим трем Матрицам можно было присвоить три имени.
P, L, U := LinearAlgebra:-LUDecomposition(A);
[1 0] [1 0] [1 2]
P, L, U := [ ], [ ], [ ]
[0 1] [3 1] [0 -1]
Другой способ сделать это — сначала загрузить весь пакет, чтобы вы могли использовать короткое имя команды,
with(LinearAlgebra):
LUDecomposition(A);
[1 0] [1 0] [1 2]
[ ], [ ], [ ]
[0 1] [3 1] [0 -1]
P, L, U := LUDecomposition(A);
[1 0] [1 0] [1 2]
P, L, U := [ ], [ ], [ ]
[0 1] [3 1] [0 -1]
0
acer
24 Ноя 2019 в 05:32