Benvinguts al introducció de Sage

Sage és un programari matemàtic gratuït i de codi obert que dóna suport a la recerca i l’ensenyament en àlgebra, geometria, teoria de nombres, criptografia, càlcul numèric, i àrees relacionades. Tant el model de desenvolupament de Sage com la tecnologia pròpia de Sage es distingeixen per un èmfasi extremadament fort en el fet de ser lliure, en la comunitat, la cooperació i la col.laboració: estem construint el cotxe, no pas reinventant la roda. L’objectiu global de Sage és el de crear una alternativa viable, lliure, i de codi obert als paquets de Maple, Mathematica, Magma, i MATLAB.

Aquest tutorial és una introducció al Sage basada en un manual escrit per Maria Bras-Amorós. Es pot llegir bé sigui en HTML o en PDF.

Aquest tutorial està publicat amb una llicència Creative Commons Attribution-Share Alike 3.0 License.

Indicacions per començar

  • La pàgina oficial de Sage és http://sagemath.org. Des d’allà podeu descarregar-vos el programa.

  • Es comença amb sage i s’acaba amb quit.

  • Totes les ordres han d’acabar en salt de línia.

  • Per interrompre un càlcul: Ctrl + C.

  • Es pot fer servir la tecla \(\uparrow\) per recuperar codi escrit anteriorment.

  • Per escriure ordres de sistema: !ordre.

  • Per llegir el fitxer fin.sage: load  fin.sage.

  • Per llegir el fitxer fin.sage i escriure els resultats al fitxer fout: sage fin.sage > fout.

  • Per comentar una línia utilitzem # .... Tot el que hi hagi després de # no serà llegit per Sage.

  • Per comentar tot un paràgraf o part d’un fitxer escrivim tres vegades cometes al principi i al final i el que quedi entremig no serà llegit per Sage:

    """
    bla
    bla
    bla
    """
    
  • Per buscar ordres, escrivim les primeres lletres i utilizem el tabulador per veure les possibles complecions.

Accions simples i expressions

Assignació de variables i escriptura

  • Per assignar variables: nom_variable = valor.

  • Per mostrar per pantalla:

    • El valor de les variables var1, var2: var1, var2.

    • Text: print("Un text").

Operadors relacionals

  • < (menor que), > (major que), <= (menor o igual que), >= (major o igual que),

  • == (igual que), != (diferent de),

  • in (pertany), not in (no pertany).

Operadors booleans

  • x and y (\(x\) i \(y\)),

  • x or y (\(x\) o \(y\)),

  • not x (no \(x\)).

Funcions definides pel Sage

Sage té definides les seves pròpies funcions que depenen d’una o vàries variables (o cap). Per cridar-les escrivim el nom de la funció seguit de les variables entre parèntesis. Per exemple:

sage: floor(3.141592)
3
sage: gcd(12,8)
4
sage: sum([3,2,5])
10
>>> from sage.all import *
>>> floor(RealNumber('3.141592'))
3
>>> gcd(Integer(12),Integer(8))
4
>>> sum([Integer(3),Integer(2),Integer(5)])
10

Hi ha funcions que retornen més d’un valor. Per exemple la funció divmod(a,b) ens retorna el quocient i el residu de dividir \(a\) entre \(b\):

sage: divmod(19,7)
(2, 5)
>>> from sage.all import *
>>> divmod(Integer(19),Integer(7))
(2, 5)

Per assignar aquests valors a variables ho fem de la manera següent:

sage: q,r = divmod(19,7)
sage: q
2
sage: r
5
>>> from sage.all import *
>>> q,r = divmod(Integer(19),Integer(7))
>>> q
2
>>> r
5

També podríem assignar el parell de valors a una sola variable:

sage: D = divmod(19,7)
sage: D
(2, 5)
sage: D[0]
2
sage: D[1]
5
>>> from sage.all import *
>>> D = divmod(Integer(19),Integer(7))
>>> D
(2, 5)
>>> D[Integer(0)]
2
>>> D[Integer(1)]
5

Composició d’accions

Composició condicional

L’estructura següent executarà bloc1 només si condicio1 és certa. El bloc bloc2 només s’executarà si condicio1 és falsa, i condicio2 és certa. Si ambdues condicions són falses, aleshores s’executarà bloc3.

Cal tenir en compte que la indentació determina l’inici i fi dels blocs i que, per tant, és obligatòria.

sage: if condicio1:
....:     bloc1
....: elif condicio2:
....:     bloc2
....: else:
....:     bloc3
>>> from sage.all import *
>>> if condicio1:
...     bloc1
... elif condicio2:
...     bloc2
... else:
...     bloc3

Composició iterativa

Les estructures següents executaran bloc repetides vegades. La primera ho farà \(i_0 - i_1 + 1\) vegades, i la variable i prendrà valors consecutius \(i_0,\ldots,i_1\). El segon bloc és similar, però i prendrà els valors que apareguin a llista, en el mateix ordre en què hi apareguin. Finalment, el tercer bloc s’executarà mentre condicio sigui certa. Si condicio és falsa al començament, bloc no s’executarà cap vegada.

sage: for i in [i0..i1]:
....:     bloc

sage: for i in llista:
....:     bloc

sage: while condicio:
....:     bloc
>>> from sage.all import *
>>> for i in (ellipsis_range(i0,Ellipsis,i1)):
...     bloc

>>> for i in llista:
...     bloc

>>> while condicio:
...     bloc

Exemples:

sage: for i in [2..5]:
....:     print(i + 1)
3
4
5
6
>>> from sage.all import *
>>> for i in (ellipsis_range(Integer(2),Ellipsis,Integer(5))):
...     print(i + Integer(1))
3
4
5
6

sage: for i in [-1,"foo",3.4]:
....:     print(i)
-1
foo
3.40000000000000
>>> from sage.all import *
>>> for i in [-Integer(1),"foo",RealNumber('3.4')]:
...     print(i)
-1
foo
3.40000000000000

sage: i = 1
sage: while i < 4:
....:     print(i)
....:     i += 1
1
2
3
>>> from sage.all import *
>>> i = Integer(1)
>>> while i < Integer(4):
...     print(i)
...     i += Integer(1)
1
2
3

Conjunts i seqüències

Conjunts i seqüències

Tant els conjunts (set) com les seqüències (list) són col.leccions d’objectes. Un conjunt no és ordenat, per tant, un element pot ser en un conjunt com a molt una vegada. Una seqüència, en canvi, és ordenada i, per tant, la repetició és possible. Les seqüències s’escriuen entre [ i ]. Els conjunts es construeixen a partir de seqüències fent servir Set(sequencia).

Per exemple:

sage: S = [ (-11)^2, (-7)^2, (-5)^2, (-3)^2, 3^2, 5^2, 7^2, 11^2 ]
sage: C = Set(S)
sage: S
[121, 49, 25, 9, 9, 25, 49, 121]
sage: C    # random
{121, 9, 49, 25}
sage: sorted(C)
[9, 25, 49, 121]
>>> from sage.all import *
>>> S = [ (-Integer(11))**Integer(2), (-Integer(7))**Integer(2), (-Integer(5))**Integer(2), (-Integer(3))**Integer(2), Integer(3)**Integer(2), Integer(5)**Integer(2), Integer(7)**Integer(2), Integer(11)**Integer(2) ]
>>> C = Set(S)
>>> S
[121, 49, 25, 9, 9, 25, 49, 121]
>>> C    # random
{121, 9, 49, 25}
>>> sorted(C)
[9, 25, 49, 121]

La \(i\)-èssima entrada d’una seqüència (o d’un conjunt) \(C\) és C[i]. Però compte perquè Sage enumera les posicions des de \(0\)!:

sage: S[1] = 1000
sage: S
[121, 1000, 25, 9, 9, 25, 49, 121]
sage: S[3]
9
>>> from sage.all import *
>>> S[Integer(1)] = Integer(1000)
>>> S
[121, 1000, 25, 9, 9, 25, 49, 121]
>>> S[Integer(3)]
9

Sage té constructors especials per a seqüències. Les expressions [a..b] i [a,a+k..b] designen respectivament les progressions [a,a+1,a+2,...,b] i [a,a+k,a+2k,...,b'], on \(b'\) és el màxim enter de la forma \(a+ik\) menor o igual que \(b\).

D’altra banda,

[ expressio(x) for x in D ]
[ expressio(x,y) for x in D for y in E ]

denota la seqüència de valors expressio(x) (resp. expressió(x,y)) avaluada per tot \(x\in D\) (resp. \(x\in D, y\in E\)).

Així mateix,

[ expressio(x) for x in D if condicio ]

denota la seqüència de valors expressio(x) avaluada per tot \(x\in D\) tals que el booleà condicio és cert. Per exemple, hauríem pogut crear S de la manera següent:

sage: S = [ n^2 for n in [-11,-9..11] if is_prime(abs(n)) ]
sage: print(S)
[121, 49, 25, 9, 9, 25, 49, 121]
>>> from sage.all import *
>>> S = [ n**Integer(2) for n in (ellipsis_range(-Integer(11),-Integer(9),Ellipsis,Integer(11))) if is_prime(abs(n)) ]
>>> print(S)
[121, 49, 25, 9, 9, 25, 49, 121]

Podem aplicar una funció a tots els elements d’una llista de la manera següent:

sage: [cos(t) for t in [0,pi..6*pi]]
[1, -1, 1, -1, 1, -1, 1]
>>> from sage.all import *
>>> [cos(t) for t in (ellipsis_range(Integer(0),pi,Ellipsis,Integer(6)*pi))]
[1, -1, 1, -1, 1, -1, 1]

Operacions per conjunts i seqüències

  • len(S) retorna el cardinal de \(S\).

  • sum(S) retorna la suma dels elements de \(S\).

  • prod(S) retorna el producte dels elements de \(S\).

  • S + T retorna \(S\) concatenat amb \(T\).

  • min(S), max(S) retorna el mínim i el màxim de S, que ha de contenir elements amb un ordre.

Operacions només per conjunts

  • A.union(B) retorna \(A\cup B\).

  • A.intersection(B) retorna \(A\cap B\).

  • A.difference(B) retorna \(A\setminus B\).

Operacions només per seqüències

  • S.append(x) afegeig \(x\) al final de \(S\).

  • S.remove(x) esborra \(x\) de \(S\).

  • S.index(x) retorna la posició de l’element \(x\) dins de \(S\).

  • S.insert(i,x) mou una posició els elements amb índex igual o superior a \(i\) i afegeix \(x\) a la posició \(i\).

  • S.reverse() capgira la seqüència \(S\).

  • S.sort() ordena la seqüència \(S\).

  • S[randint(0,len(S))] retorna un element aleatori de \(S\).

Booleans

  • x in C, x not in C determina si \(x\) pertany o no a \(C\).

  • A.issubset(B), A.issuperset(B) determina si \(A\) és un subconjunt de \(B\), o si \(A\) conté a \(B\) com a subconjunt (només per conjunts).

  • D == C, D != C determina si \(D\) és igual a \(C\) o no.

Funcions

La declaració general d’una funció de \(n\) arguments per la que s’hagi de fer una sèrie d’operacions és:

def f(x1, x2, ..., xn):
    ...
    return (val1, val2, ...)

Per exemple:

sage: def solucions_reals_equacio_segon_grau(a,b,c):
....:     discriminant = b^2 - 4*a*c
....:     arrel_discr = sqrt(discriminant)
....:     if discriminant > 0:
....:         print("hi ha dues solucions reals")
....:         sol1 = (-b + arrel_discr) / (2*a)
....:         sol2 = (-b - arrel_discr) / (2*a)
....:         return (sol1, sol2)
....:     elif discriminant == 0:
....:         print("hi ha una solucio real")
....:         sol = -b / (2*a)
....:         return (sol)
....:     else:
....:         print("no hi ha solucions reals")
....:         return ()
>>> from sage.all import *
>>> def solucions_reals_equacio_segon_grau(a,b,c):
...     discriminant = b**Integer(2) - Integer(4)*a*c
...     arrel_discr = sqrt(discriminant)
...     if discriminant > Integer(0):
...         print("hi ha dues solucions reals")
...         sol1 = (-b + arrel_discr) / (Integer(2)*a)
...         sol2 = (-b - arrel_discr) / (Integer(2)*a)
...         return (sol1, sol2)
...     elif discriminant == Integer(0):
...         print("hi ha una solucio real")
...         sol = -b / (Integer(2)*a)
...         return (sol)
...     else:
...         print("no hi ha solucions reals")
...         return ()

Observem que l’indentat és important. Ara podem cridar-la:

sage: solucions_reals_equacio_segon_grau(1,0,-1)
hi ha dues solucions reals
(1, -1)
>>> from sage.all import *
>>> solucions_reals_equacio_segon_grau(Integer(1),Integer(0),-Integer(1))
hi ha dues solucions reals
(1, -1)

Com que en l’exemple donat la funció retorna dos valors, podem assignar-los a dues variables:

sage: a,b = solucions_reals_equacio_segon_grau(1,0,-1)
hi ha dues solucions reals
sage: a
1
sage: b
-1
>>> from sage.all import *
>>> a,b = solucions_reals_equacio_segon_grau(Integer(1),Integer(0),-Integer(1))
hi ha dues solucions reals
>>> a
1
>>> b
-1

Aquesta crida, però, ens donaria un error si la solució no existís o fos única.

Observem que totes les variables utilitzades són per defecte locals i que no ha calgut declarar-les. Una variable externa a la funció amb nom igual a una de les variables locals no es modificarà a causa de la funció. Per exemple, considerem la funció \(f\):

sage: def f():
....:     a = 5
....:     return a
>>> from sage.all import *
>>> def f():
...     a = Integer(5)
...     return a

Observem el resultat del codi següent:

sage: a = 3
sage: print(f())
5
sage: print(a)
3
>>> from sage.all import *
>>> a = Integer(3)
>>> print(f())
5
>>> print(a)
3

Enters, anells \(\mathbb{Z}/n\mathbb{Z}\), racionals i reals

Enters

L’anell dels enters el cridem o bé amb Integers() o bé amb ZZ. A continuació llistem algunes funcions pels enters:

  • Operacions bàsiques: +, -, *, /, ^ (suma, resta, producte, quocient (possiblement racional) i potència).

  • n // m (quocient de dividir \(n\) entre \(m\))

  • n % m (\(n\) mòdul \(m\), el residu de dividir \(n\) entre \(m\)).

  • divmod(a,b) (quocient i residu de dividir \(a\) entre \(b\)).

  • abs(n) (valor absolut).

  • randint(a,b) (un enter aleatori entre a i b, ambdós inclosos).

  • random_prime(a) (un primer aleatori menor o igual que \(a\)).

  • divisors(n), prime_divisors(n), number_of_divisors(n) (la seqüència de divisors de \(n\), o només els divisors primers, o el nombre total de divisors).

  • gcd(m,n), gcd(S), lcm(m,n), lcm(S) (el màxim comú divisor i el mínim comú múltiple de \(m\) i \(n\) o bé de la seqüència d’enters \(S\)).

  • xgcd(m,n) (retorna tres valors \(d\), \(a\) i \(b\) on \(d\) és el màxim comú divisor de \(m\) i \(n\) i on \(am+bn=d\)).

  • euler_phi(n) (el nombre d’unitats de l’anell \(\mathbb{Z}/n\mathbb{Z}\)).

  • factorial(n) (el factorial de l’enter \(n\)).

Booleans

  • is_odd(i), is_even(i) retornen si l’enter \(i\) és senar (o parell).

  • is_prime(i), is_prime_power(i) retornen si l’enter \(i\) és primer (o una potència d’un primer).

  • is_square(n) retorna si l’enter \(n\) és un quadrat perfecte.

Anells \(\mathbb{Z}/n\mathbb{Z}\)

L’anell de residus mòdul \(n\) el cridem amb Integers(n) o amb Zmod(n). Si posem R = Zmod(n) per algun \(n\) aleshores els elements de \(R\) els cridem utilitzant R(1), R(2), etc. Algunes funcions que ens poden ser útils són

  • Operacions bàsiques: +, -, *, ^.

  • inverse_mod(x,m) retorna l’invers de \(x \pmod{m}\).

  • solve_mod(expr1 == expr2,m) resol equacions amb congruències; és important que les incògnites s’hagin alliberat abans, per exemple amb var('x'). Per exemple:

sage: x, y = var('x','y')
sage: solve_mod(3*x + 2*y == 1, 5)
[(0, 3), (1, 4), (2, 0), (3, 1), (4, 2)]
>>> from sage.all import *
>>> x, y = var('x','y')
>>> solve_mod(Integer(3)*x + Integer(2)*y == Integer(1), Integer(5))
[(0, 3), (1, 4), (2, 0), (3, 1), (4, 2)]
  • primitive_root(n) retorna un enter que genera el grup multiplicatiu dels enters mòdul \(n\), si existeix.

  • multiplicative_order(n), additive_order(n) retornen l’ordre multiplicatiu i additiu de l’enter \(n\).

Booleans

  • is_field() determina si un conjunt és un cos.

  • is_unit() determina si un element és invertible.

L’exemple que segueix pot ser il.lustratiu:

sage: is_prime(19)
True
sage: primitive_root(19)
2
sage: Z19 = Zmod(19)
sage: [x for x in Z19]
[0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18]
sage: Z19(2).additive_order()
19
sage: Z19(2).multiplicative_order()
18
sage: Z19(5).multiplicative_order()
9
sage: Set([2^i % 19 for i in [1..18]])
{1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18}
sage: Set([5^i % 19 for i in [1..18]])
{1, 4, 5, 6, 7, 9, 11, 16, 17}
>>> from sage.all import *
>>> is_prime(Integer(19))
True
>>> primitive_root(Integer(19))
2
>>> Z19 = Zmod(Integer(19))
>>> [x for x in Z19]
[0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18]
>>> Z19(Integer(2)).additive_order()
19
>>> Z19(Integer(2)).multiplicative_order()
18
>>> Z19(Integer(5)).multiplicative_order()
9
>>> Set([Integer(2)**i % Integer(19) for i in (ellipsis_range(Integer(1),Ellipsis,Integer(18)))])
{1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18}
>>> Set([Integer(5)**i % Integer(19) for i in (ellipsis_range(Integer(1),Ellipsis,Integer(18)))])
{1, 4, 5, 6, 7, 9, 11, 16, 17}

Racionals

El cos dels nombres racionals es denota amb RationalField() o QQ. Algunes operacions que podem fer amb els racionals són

  • Operacions bàsiques: +, -, *, /, ^.

  • numerator(q), denominator(q) (el numerador i denominador de \(q\)).

Reals

El cos dels reals el denotem amb RealField() o RR. Algunes funcions per als reals són:

  • Operacions bàsiques: +, -, *, /, ^.

  • ceil(r), floor(r) retorna la part entera superior o inferior de \(r\).

  • abs(r) retorna el valor absolut de \(r\).

  • sqrt(r) retorna l’arrel quadrada de \(r\).

Equacions

Es poden resoldre equacions utilitzant l’ordre solve(). Escrivint ?solve el sage ens dóna una explicació molt extensa. Aquí en repetim els primers exemples:

sage: x, y = var('x, y')
sage: solve([x + y == 6, x - y == 4], x, y)
[[x == 5, y == 1]]
sage: solve([x^2 + y^2 == 1, y^2 == x^3 + x + 1], x, y)
[[x == -1/2*I*sqrt(3) - 1/2, y == -sqrt(-1/2*I*sqrt(3) + 3/2)], [x == -1/2*I*sqrt(3) - 1/2, y == sqrt(-1/2*I*sqrt(3) + 3/2)], [x == 1/2*I*sqrt(3) - 1/2, y == -sqrt(1/2*I*sqrt(3) + 3/2)], [x == 1/2*I*sqrt(3) - 1/2, y == sqrt(1/2*I*sqrt(3) + 3/2)], [x == 0, y == -1], [x == 0, y == 1]]
sage: solutions = solve([sqrt(x) + sqrt(y) == 5, x + y == 10], x, y,solution_dict = True)
sage: len(solutions)
2
sage: type(solutions[0])
<... 'dict'>
sage: for sol in solutions: print((sol[x].n(digits=3), sol[y].n(digits=3)))
(5.00 - 5.59*I, 5.00 + 5.59*I)
(5.00 + 5.59*I, 5.00 - 5.59*I)
>>> from sage.all import *
>>> x, y = var('x, y')
>>> solve([x + y == Integer(6), x - y == Integer(4)], x, y)
[[x == 5, y == 1]]
>>> solve([x**Integer(2) + y**Integer(2) == Integer(1), y**Integer(2) == x**Integer(3) + x + Integer(1)], x, y)
[[x == -1/2*I*sqrt(3) - 1/2, y == -sqrt(-1/2*I*sqrt(3) + 3/2)], [x == -1/2*I*sqrt(3) - 1/2, y == sqrt(-1/2*I*sqrt(3) + 3/2)], [x == 1/2*I*sqrt(3) - 1/2, y == -sqrt(1/2*I*sqrt(3) + 3/2)], [x == 1/2*I*sqrt(3) - 1/2, y == sqrt(1/2*I*sqrt(3) + 3/2)], [x == 0, y == -1], [x == 0, y == 1]]
>>> solutions = solve([sqrt(x) + sqrt(y) == Integer(5), x + y == Integer(10)], x, y,solution_dict = True)
>>> len(solutions)
2
>>> type(solutions[Integer(0)])
<... 'dict'>
>>> for sol in solutions: print((sol[x].n(digits=Integer(3)), sol[y].n(digits=Integer(3))))
(5.00 - 5.59*I, 5.00 + 5.59*I)
(5.00 + 5.59*I, 5.00 - 5.59*I)

Anell de polinomis

Definició de polinomis

Per generar l’anell de polinomis sobre un anell \(R\) ho fem així:

sage: P.<x> = PolynomialRing(R)
>>> from sage.all import *
>>> P = PolynomialRing(R, names=('x',)); (x,) = P._first_ngens(1)

A més, així queda definida \(x\) com la indeterminada dels polinomis de \(P\). Per escriure un polinomi de \(P\) ho podem fer de dues maneres:

sage: P.<x> = PolynomialRing(QQ)
sage: x^3-7*x^2+5
x^3 - 7*x^2 + 5
sage: P([5,0,-7,1])
x^3 - 7*x^2 + 5
>>> from sage.all import *
>>> P = PolynomialRing(QQ, names=('x',)); (x,) = P._first_ngens(1)
>>> x**Integer(3)-Integer(7)*x**Integer(2)+Integer(5)
x^3 - 7*x^2 + 5
>>> P([Integer(5),Integer(0),-Integer(7),Integer(1)])
x^3 - 7*x^2 + 5

Operacions

  • Operacions bàsiques: +, -, *, /, ^.

  • f(a) per avaluar un polinomi \(f\) en \(a\).

  • f.degree() (el grau del polinomi).

  • f.coeffs() retorna tots els coeficients de \(f\) mentre que f.coefficients() només retorna els coeficients no-nuls.

  • f.leading_coefficient(f) retorna el coeficient principal de \(f\).

  • parent(f) ens diu on pertany el polinomi \(f\).

  • divmod(f,g) (quocient i residu de dividir \(f\) entre \(g\)), que es poden obtenir per separat amb f // g i f % g.

  • gcd(f,g), xgcd(f,g), lcm(f,g) (vegeu l’apartat d’enters).

  • factor(f) (la factorització de \(f\)).

  • f.roots(f).

Booleans

  • f.is_irreducible() retorna si \(f\) és irreductible.

  • f.is_primitive() retorna si \(f\) és primitiu.

Vegem un exemple:

sage: P.<x> = PolynomialRing(GF(49,'a'))
sage: f = 4*x^5+5*x+2
sage: g = x^8 + 6*x^7 + x^2
sage: d,a,b = xgcd(f,g)
sage: d
1
sage: d == a*f + b*g
True
>>> from sage.all import *
>>> P = PolynomialRing(GF(Integer(49),'a'), names=('x',)); (x,) = P._first_ngens(1)
>>> f = Integer(4)*x**Integer(5)+Integer(5)*x+Integer(2)
>>> g = x**Integer(8) + Integer(6)*x**Integer(7) + x**Integer(2)
>>> d,a,b = xgcd(f,g)
>>> d
1
>>> d == a*f + b*g
True

Cossos finits

Construcció de cossos finits i extensions

Per crear el cos finit de \(p^n\) elements escrivim:

sage: K = GF(p^n,'a')
>>> from sage.all import *
>>> K = GF(p**n,'a')

o bé:

sage: K.<a> = GF(p^n)
>>> from sage.all import *
>>> K = GF(p**n, names=('a',)); (a,) = K._first_ngens(1)

Queda així definida també \(a\) com la classe de la indeterminada dels polinomis sobre \(F = \mathbb{Z}/p\mathbb{Z}\) tal que \(K = F[x]/(f(x))\). Només podem obviar la variable \(a\) quan \(n=1\).

També podem definir cossos finits forçant un determinat polinomi \(f\) de l’anell de polinomis sobre \(\mathbb{Z}/p\mathbb{Z}\) i de grau \(n\) utilitzant:

sage: F = GF(q, modulus = f)
>>> from sage.all import *
>>> F = GF(q, modulus = f)

Donat un cos finit sempre podem conèixer-ne el cos primer escrivint F.prime_subfield(). Si hem definit \(F\) com a extensió d’un cos \(K\) direm que \(K\) és el cos base de \(F\) i el podrem conèixer escrivint F.base_ring(). Per a un cos que haguem definit directament es considera que el cos base és el cos primer.

Operacions

  • Operacions bàsiques: +, -, *, /, ^.

  • K.polynomial() retorna el polinomi que defineix \(K\).

  • b.minimal_polynomial()` (o ``b.minpoly()) retorna el polinomi mínim de \(b\).

  • b.multiplicative_order() retorna l’ordre multiplicatiu, el mínim enter positiu \(n\) tal que \(b^n = 1\).

  • len(K) és equivalent a K.cardinality() retorna el nombre d’elements del cos \(K\).

  • K.random_element() retorna un element aleatori de \(K\).

Booleans

  • f.is_primitive() retorna si \(f\) és un polinomi primitiu.

Algebra lineal

Construcció de vectors

Donat un cos \(K\), creem l’espai vectorial \(K^n\) mitjançant VectorSpace(K,n), o escrivint K^n. Per crear vectors podem fer una coerció dins l’espai dels vectors corresponent o bé els podem definir directament:

sage: K = GF(9,'a')
sage: V3 = VectorSpace(K,3)
sage: v = V3([1,0,1])
sage: v
(1, 0, 1)
sage: v[1]
0

sage: w = vector(K,[1,2,3])
sage: w
(1, 2, 0)
sage: w[2]
0
>>> from sage.all import *
>>> K = GF(Integer(9),'a')
>>> V3 = VectorSpace(K,Integer(3))
>>> v = V3([Integer(1),Integer(0),Integer(1)])
>>> v
(1, 0, 1)
>>> v[Integer(1)]
0

>>> w = vector(K,[Integer(1),Integer(2),Integer(3)])
>>> w
(1, 2, 0)
>>> w[Integer(2)]
0

Operacions

  • Operacions bàsiques: +, -, *.

  • v.inner_product(w) (producte escalar),

  • v.pairwise_product(w) (producte vectorial a \(K^3\)).

Construcció de matrius

Donat un anell \(R\), creem el conjunt de matrius de \(m\) files i \(n\) columnes mitjançant MatrixSpace(R,m,n). Per crear matrius podem fer una coerció dins l’espai de les matrius corresponent o bé les podem definir directament:

sage: F9.<alpha> = GF(9)
sage: M = MatrixSpace(F9,2,3)
sage: M([alpha, 2*alpha, 3*alpha, alpha, alpha^2, alpha^3])
[      alpha     2*alpha           0]
[      alpha   alpha + 1 2*alpha + 1]
sage: matrix(2,3,[alpha,2*alpha,3*alpha,alpha,alpha^2,alpha^3])
[      alpha     2*alpha           0]
[      alpha   alpha + 1 2*alpha + 1]
sage: matrix(3,2,[alpha,2*alpha,3*alpha,alpha,alpha^2,alpha^3])
[      alpha     2*alpha]
[          0       alpha]
[  alpha + 1 2*alpha + 1]
>>> from sage.all import *
>>> F9 = GF(Integer(9), names=('alpha',)); (alpha,) = F9._first_ngens(1)
>>> M = MatrixSpace(F9,Integer(2),Integer(3))
>>> M([alpha, Integer(2)*alpha, Integer(3)*alpha, alpha, alpha**Integer(2), alpha**Integer(3)])
[      alpha     2*alpha           0]
[      alpha   alpha + 1 2*alpha + 1]
>>> matrix(Integer(2),Integer(3),[alpha,Integer(2)*alpha,Integer(3)*alpha,alpha,alpha**Integer(2),alpha**Integer(3)])
[      alpha     2*alpha           0]
[      alpha   alpha + 1 2*alpha + 1]
>>> matrix(Integer(3),Integer(2),[alpha,Integer(2)*alpha,Integer(3)*alpha,alpha,alpha**Integer(2),alpha**Integer(3)])
[      alpha     2*alpha]
[          0       alpha]
[  alpha + 1 2*alpha + 1]

Si volem, podem especificar l’anell sobre el qual està definida una matriu. Així, les dues ordres següents ens donarien matrius diferents:

sage: m1 = matrix(Zmod(5),[[1,2],[3,4]]); m1
[1 2]
[3 4]
sage: m2 = matrix(Zmod(7),[[1,2],[3,4]]); m2
[1 2]
[3 4]
sage: m1 == m2
False
>>> from sage.all import *
>>> m1 = matrix(Zmod(Integer(5)),[[Integer(1),Integer(2)],[Integer(3),Integer(4)]]); m1
[1 2]
[3 4]
>>> m2 = matrix(Zmod(Integer(7)),[[Integer(1),Integer(2)],[Integer(3),Integer(4)]]); m2
[1 2]
[3 4]
>>> m1 == m2
False

Per obtenir els elements d’una matriu \(m\) utilitzarem m[i,j] i per obtenir les seves files utilitzarem m[i]. Seguint l’exemple anterior,

sage: m = matrix(F9,[[alpha,2*alpha,3*alpha],[alpha,alpha^2,alpha^3]])
sage: m[0,1]
2*alpha
sage: m[1]
(alpha, alpha + 1, 2*alpha + 1)
>>> from sage.all import *
>>> m = matrix(F9,[[alpha,Integer(2)*alpha,Integer(3)*alpha],[alpha,alpha**Integer(2),alpha**Integer(3)]])
>>> m[Integer(0),Integer(1)]
2*alpha
>>> m[Integer(1)]
(alpha, alpha + 1, 2*alpha + 1)

Per sumar, restar i multiplicar per escalars es fa amb la notació usual. Per trobar la matriu inversa d’una matriu invertible \(m\) escriurem m^-1.

Per trobar les solucions d’un sistema lineal \(x m=v\) on \(m\) és una matriu i \(v\) és un vector tenim m.solve_left() mentre que per resoldre el sistema \(x m=v\) tenim m.solve_right().

sage: m = matrix(Integers(),[[0,1],[2,0]])
sage: m
[0 1]
[2 0]
sage: v = vector(Integers(),[2,2])
sage: v
(2, 2)
sage: m.solve_left(v)
(2, 1)
sage: m.solve_right(v)
(1, 2)
>>> from sage.all import *
>>> m = matrix(Integers(),[[Integer(0),Integer(1)],[Integer(2),Integer(0)]])
>>> m
[0 1]
[2 0]
>>> v = vector(Integers(),[Integer(2),Integer(2)])
>>> v
(2, 2)
>>> m.solve_left(v)
(2, 1)
>>> m.solve_right(v)
(1, 2)

També podem calcular submatrius. Per obtenir la submatriu \(3 \times 3\) començant a l’entrada \((1,1)\) d’una matriu \(4 \times 4\):

sage: m = matrix(4, [1..16])
sage: m.submatrix(1, 1)
[ 6  7  8]
[10 11 12]
[14 15 16]
>>> from sage.all import *
>>> m = matrix(Integer(4), (ellipsis_range(Integer(1),Ellipsis,Integer(16))))
>>> m.submatrix(Integer(1), Integer(1))
[ 6  7  8]
[10 11 12]
[14 15 16]

Ara només n’agafem dues files:

sage: m.submatrix(1, 1, 2)
[ 6  7  8]
[10 11 12]
>>> from sage.all import *
>>> m.submatrix(Integer(1), Integer(1), Integer(2))
[ 6  7  8]
[10 11 12]

I ara només una columna:

sage: m.submatrix(1, 1, 2, 1)
[ 6]
[10]
>>> from sage.all import *
>>> m.submatrix(Integer(1), Integer(1), Integer(2), Integer(1))
[ 6]
[10]

També es poden escollir \(0\) files o columnes:

sage: m.submatrix(1, 1, 0)
[]
>>> from sage.all import *
>>> m.submatrix(Integer(1), Integer(1), Integer(0))
[]

Operacions amb matrius en general

  • Operacions bàsiques: +, -, *.

  • A.nrows() i A.ncols() retorna el nombre de files i columnes de la matriu.

  • m.transpose() retorna la matriu transposada.

  • m.rank(), m.kernel(), m.image() retorna el rang, nucli i imatge de la matriu.

Booleans

  • m.is_square() (per saber si la matriu és quadrada).

Operacions amb matrius quadrades

  • Operacions bàsiques: +, -, *, ^.

  • m.determinant(), m.trace() i m.inverse() retorna el determinant, la traça i la inversa de la matriu (aquesta última, només en cas que existeixi).

Booleans

  • m.is_symmetric(), m.is_invertible(), m.is_singular() (si la matriu és simètrica, invertible o singular (és a dir, no invertible)).

  • m.is_zero(), m.is_one() (si la matriu és zero, o la matriu identitat).

Càlcul diferencial

Amb Sage es poden calcular les derivades i integrals de moltes funcions. Per exemple, per calcular la derivada de \(\sin(u)\) respecte \(u\), fem:

sage: u = var('u')
sage: diff(sin(u), u)
cos(u)
>>> from sage.all import *
>>> u = var('u')
>>> diff(sin(u), u)
cos(u)

La quarta derivada de \(\sin(x^2)\) s’obté mitjançant:

sage: diff(sin(x^2), x, 4)
16*x^4*sin(x^2) - 48*x^2*cos(x^2) - 12*sin(x^2)
>>> from sage.all import *
>>> diff(sin(x**Integer(2)), x, Integer(4))
16*x^4*sin(x^2) - 48*x^2*cos(x^2) - 12*sin(x^2)

Per calcular les derivades parcials de \(x^2+17y^2\) respecte \(x\) i \(y\) fem:

sage: x, y = var('x,y')
sage: f = x^2 + 17*y^2
sage: f.diff(x)
2*x
sage: f.diff(y)
34*y
>>> from sage.all import *
>>> x, y = var('x,y')
>>> f = x**Integer(2) + Integer(17)*y**Integer(2)
>>> f.diff(x)
2*x
>>> f.diff(y)
34*y

També podem calcular integrals, tant definides com indefinides. Per calcular \(\int x\sin(x^2)\, dx\) i \(\int_0^1 \frac{x}{x^2+1}\, dx\) fem:

sage: integral(x*sin(x^2), x)
-1/2*cos(x^2)
sage: integral(x/(x^2+1), x, 0, 1)
1/2*log(2)
>>> from sage.all import *
>>> integral(x*sin(x**Integer(2)), x)
-1/2*cos(x^2)
>>> integral(x/(x**Integer(2)+Integer(1)), x, Integer(0), Integer(1))
1/2*log(2)

També podem aconseguir una descomposició en fraccions parcials, per exemple de \(\frac{1}{x^2-1}\):

sage: f = 1/((1+x)*(x-1))
sage: f.partial_fraction(x)
-1/2/(x + 1) + 1/2/(x - 1)
>>> from sage.all import *
>>> f = Integer(1)/((Integer(1)+x)*(x-Integer(1)))
>>> f.partial_fraction(x)
-1/2/(x + 1) + 1/2/(x - 1)

Gràfics

Amb Sage podem produir gràfics 2-D i 3-D.

Gràfics bidimensionals

Podem dibuixar cercles, línies, polígons; gràfics de funcions en coordenades cartesianes; i també en coordenades polars, corbes de nivell i gràfics de camps vectorials. Podem trobar més exemples de les funcions gràfiques del Sage a la documentació Sage Constructions

Per dibuixar un cercle groc de radi 1, centrat a l’origen, fem:

sage: circle((0,0), 1, rgbcolor=(1,1,0))
Graphics object consisting of 1 graphics primitive
>>> from sage.all import *
>>> circle((Integer(0),Integer(0)), Integer(1), rgbcolor=(Integer(1),Integer(1),Integer(0)))
Graphics object consisting of 1 graphics primitive

I un cercle sòlid:

sage: circle((0,0), 1, rgbcolor=(1,1,0), fill=True)
Graphics object consisting of 1 graphics primitive
>>> from sage.all import *
>>> circle((Integer(0),Integer(0)), Integer(1), rgbcolor=(Integer(1),Integer(1),Integer(0)), fill=True)
Graphics object consisting of 1 graphics primitive

També podem assignar el cercle a una variable. Llavors no obtindrem cap dibuix:

sage: c = circle((0,0), 1, rgbcolor=(1,1,0))
>>> from sage.all import *
>>> c = circle((Integer(0),Integer(0)), Integer(1), rgbcolor=(Integer(1),Integer(1),Integer(0)))

I el podem visualitzar mitjançant c.show() o show(c):

sage: c.show()
>>> from sage.all import *
>>> c.show()

L’ordre c.save('fitxer.png') desa el dibuix al disc.

Amb l’opció aspect_ratio = 1 podem aconseguir que els dos eixos estiguin a la mateixa escala:

sage: c.show(aspect_ratio=1)
>>> from sage.all import *
>>> c.show(aspect_ratio=Integer(1))

També hauriem pogut utilitzar show(c, aspect_ratio=1), o desar-ho amb l’ordre c.save('fitxer.png', aspect_ratio=1).

Així obtenim el gràfic d’una funció bàsica:

sage: plot(cos, (-5,5))
Graphics object consisting of 1 graphics primitive
>>> from sage.all import *
>>> plot(cos, (-Integer(5),Integer(5)))
Graphics object consisting of 1 graphics primitive

Si primer alliberem una variable, podem obtenir gràfics de funcions paramètriques:

sage: x = var('x')
sage: parametric_plot((cos(x),sin(x)^3),(x,0,2*pi),rgbcolor=hue(0.6))
Graphics object consisting of 1 graphics primitive
>>> from sage.all import *
>>> x = var('x')
>>> parametric_plot((cos(x),sin(x)**Integer(3)),(x,Integer(0),Integer(2)*pi),rgbcolor=hue(RealNumber('0.6')))
Graphics object consisting of 1 graphics primitive

Podem combinar diversos gràfics en una de sol:

sage: x = var('x')
sage: p1 = parametric_plot((cos(x),sin(x)),(x,0,2*pi),rgbcolor=hue(0.2))
sage: p2 = parametric_plot((cos(x),sin(x)^2),(x,0,2*pi),rgbcolor=hue(0.4))
sage: p3 = parametric_plot((cos(x),sin(x)^3),(x,0,2*pi),rgbcolor=hue(0.6))
sage: show(p1+p2+p3, axes=false)
>>> from sage.all import *
>>> x = var('x')
>>> p1 = parametric_plot((cos(x),sin(x)),(x,Integer(0),Integer(2)*pi),rgbcolor=hue(RealNumber('0.2')))
>>> p2 = parametric_plot((cos(x),sin(x)**Integer(2)),(x,Integer(0),Integer(2)*pi),rgbcolor=hue(RealNumber('0.4')))
>>> p3 = parametric_plot((cos(x),sin(x)**Integer(3)),(x,Integer(0),Integer(2)*pi),rgbcolor=hue(RealNumber('0.6')))
>>> show(p1+p2+p3, axes=false)

També podem crear polígons. Comencem amb una llista L de punts, i després utilitzem la funció polygon per dibuixar el polígon que té aquests punts com a frontera. Per exemple, un deltoide verd:

sage: L = [[-1+cos(pi*i/100)*(1+cos(pi*i/100)),\
....: 2*sin(pi*i/100)*(1-cos(pi*i/100))] for i in range(200)]
sage: p = polygon(L, rgbcolor=(1/8,3/4,1/2))
sage: p
Graphics object consisting of 1 graphics primitive
>>> from sage.all import *
>>> L = [[-Integer(1)+cos(pi*i/Integer(100))*(Integer(1)+cos(pi*i/Integer(100))),Integer(2)*sin(pi*i/Integer(100))*(Integer(1)-cos(pi*i/Integer(100)))] for i in range(Integer(200))]
>>> p = polygon(L, rgbcolor=(Integer(1)/Integer(8),Integer(3)/Integer(4),Integer(1)/Integer(2)))
>>> p
Graphics object consisting of 1 graphics primitive

Si no volem veure els eixos, utilitzem show(p, axes=false).

També podem afegir text a un gràfic:

sage: L = [[6*cos(pi*i/100)+5*cos((6/2)*pi*i/100),\
....: 6*sin(pi*i/100)-5*sin((6/2)*pi*i/100)] for i in range(200)]
sage: p = polygon(L, rgbcolor=(1/8,1/4,1/2))
sage: t = text("hipotrocoide", (5,4), rgbcolor=(1,0,0))
sage: show(p+t)
>>> from sage.all import *
>>> L = [[Integer(6)*cos(pi*i/Integer(100))+Integer(5)*cos((Integer(6)/Integer(2))*pi*i/Integer(100)),Integer(6)*sin(pi*i/Integer(100))-Integer(5)*sin((Integer(6)/Integer(2))*pi*i/Integer(100))] for i in range(Integer(200))]
>>> p = polygon(L, rgbcolor=(Integer(1)/Integer(8),Integer(1)/Integer(4),Integer(1)/Integer(2)))
>>> t = text("hipotrocoide", (Integer(5),Integer(4)), rgbcolor=(Integer(1),Integer(0),Integer(0)))
>>> show(p+t)

Un gràfic més complicat és el que mostra múltiples branques de la funció arcsinus: és a dir, el gràfic de \(y=\sin(x)\) on \(x\) entre \(-2\pi\) i \(2\pi\), i rotat un angle de 45 graus. Així és com ho construirem en Sage:

sage: v = [(sin(x),x) for x in srange(-2*float(pi),2*float(pi),0.1)]
sage: line(v)
Graphics object consisting of 1 graphics primitive
>>> from sage.all import *
>>> v = [(sin(x),x) for x in srange(-Integer(2)*float(pi),Integer(2)*float(pi),RealNumber('0.1'))]
>>> line(v)
Graphics object consisting of 1 graphics primitive

Finalment, un exemple d’un gràfic de corbes de nivell:

sage: f = lambda x,y: cos(x*y)
sage: contour_plot(f, (-4, 4), (-4, 4))
Graphics object consisting of 1 graphics primitive
>>> from sage.all import *
>>> f = lambda x,y: cos(x*y)
>>> contour_plot(f, (-Integer(4), Integer(4)), (-Integer(4), Integer(4)))
Graphics object consisting of 1 graphics primitive

Gràfics tridimensionals

També podem aconseguir gràfics 3-D amb Sage. Per defecte, aquests gràfics es visualitzen amb el paquet [Jmol], que permet rotar i ampliar l’objecte de manera interactiva, utilitzant el ratolí.

Per visualitzar una funció de la forma \(f(x,y) = z\) utilitzem plot3d:

sage: x, y = var('x,y')
sage: plot3d(x^2 + y^2, (x,-2,2), (y,-2,2))
Graphics3d Object
>>> from sage.all import *
>>> x, y = var('x,y')
>>> plot3d(x**Integer(2) + y**Integer(2), (x,-Integer(2),Integer(2)), (y,-Integer(2),Integer(2)))
Graphics3d Object

També podem utilitzar parametric_plot3d per visualitzar una superfície paramètrica on cadascuna de les coordenades \(x, y, z\) ve determinada per una funció d’una o dues variables (normalment denotades per \(u\) i \(v\)). Així, el gràfic anterior també es pot visualitzar paramètricament de la manera següent:

sage: u, v = var('u, v')
sage: f_x(u, v) = u
sage: f_y(u, v) = v
sage: f_z(u, v) = u^2 + v^2
sage: parametric_plot3d([f_x, f_y, f_z], (u, -2, 2), (v, -2, 2))
Graphics3d Object
>>> from sage.all import *
>>> u, v = var('u, v')
>>> __tmp__=var("u,v"); f_x = symbolic_expression(u).function(u,v)
>>> __tmp__=var("u,v"); f_y = symbolic_expression(v).function(u,v)
>>> __tmp__=var("u,v"); f_z = symbolic_expression(u**Integer(2) + v**Integer(2)).function(u,v)
>>> parametric_plot3d([f_x, f_y, f_z], (u, -Integer(2), Integer(2)), (v, -Integer(2), Integer(2)))
Graphics3d Object

Finalment, podem utilitzar implicit_plot3d, per visualitzar les corbes de nivell d’una funció com \(f(x, y, z)\). Aquesta ordre visualitza una esfera utilitzant la fórmula clàssica:

sage: x, y, z = var('x, y, z')
sage: implicit_plot3d(x^2 + y^2 + z^2 - 4, (x,-2, 2), (y,-2, 2), (z,-2, 2))
Graphics3d Object
>>> from sage.all import *
>>> x, y, z = var('x, y, z')
>>> implicit_plot3d(x**Integer(2) + y**Integer(2) + z**Integer(2) - Integer(4), (x,-Integer(2), Integer(2)), (y,-Integer(2), Integer(2)), (z,-Integer(2), Integer(2)))
Graphics3d Object

Índexs i taules

[Jmol]

Jmol: un visualitzador d’estructures químiques en 3D, de codi obert i escrit en Java i Javascript. http://www.jmol.org/.