Lenguaje de Especificacion
Lenguaje de Especificacion
Lenguaje de Especificacion
-1-
Lenguaje de Especificacin
Contenido
....................................................................................................................................... 1
Lenguaje de Especificacin............................................................................................ 1
Contenido....................................................................................................................... 1
Tipos de datos................................................................................................................ 2
Tipos de Datos Bsicos................................................................................................... 2
Trminos......................................................................................................................... 4
Funciones Auxiliares....................................................................................................... 4
Tipos Enumerados y Sinnimos de Tipo.........................................................................5
Secuencias..................................................................................................................... 6
Especificacin de Problemas........................................................................................ 10
Tipos Compuestos........................................................................................................ 14
Funcin ifThenElse........................................................................................................ 20
Lenguaje de especificacin Algoritmos y Estructura de Datos I 1er Cuatrimestre 2014
-2-
Tipos de datos
Recordemos que un tipo de datos es un conjunto de valores con ciertas
operaciones en comn. Vamos a empezar viendo unos tipos de datos bsicos para
despus introducir otros ms complejos.
En nuestro lenguaje de especificacin, para hablar de un elemento de un tipo,
escribimos un trmino o expresin. Un trmino puede ser
una variable,
una constante del tipo
o una funcin (operacin) aplicada a otros trminos.
Todos los tipos tienen un elemento distinguido (un valor especial, distinto de los
dems) llamado indefinido. En el lenguaje de especificacin, podemos representarlo
por la constante Indef o por el smbolo .
-3-
Las constantes del tipo son los literales que representan nmeros enteros en
notacin decimal: 0; 1; 1; 2; 2; 3; 3; ...
Sus funciones son las operaciones ms comunes en Z:
a + b,
a b,
a * b (multiplicacin, tambin se puede escribir a b o a b),
a,
abs(a) (valor absoluto),
a div b (divisin entera, b 0),
a mod b (resto de la divisin entera, b 0),
pot(a,b) (potenciacin, tambin se puede escribir ab).
Las comparaciones entre enteros son trminos de tipo Bool:
a < b,
a > b,
a <= b (o a b),
a >= b (o a b).
Las comparaciones se pueden encadenar: si escribimos a1 1 a2 2 a3, donde los i son
comparaciones (<, >, <=, >=, ==), es como si hubiramos escrito (a1 1 a2) (a2 2 a3), y
as sucesivamente para cadenas ms largas.
La operacin beta(a) se aplica a un trmino de tipo Bool y devuelve su valor
numrico de verdad (0 o 1). Por ejemplo beta(1 < 2) == 1, y beta(2 < 1) == 0. Otra
forma de escribirla es (a).
Caracteres (Char)
Cada elemento de este tipo es una letra o smbolo. Sus constantes se escriben
entre comillas simples: 'A', 'B', ..., 'Z', ..., 'a', 'b', ..., 'z', ..., '0', '1', ..., '9', ...
Lenguaje de especificacin Algoritmos y Estructura de Datos I 1er Cuatrimestre 2014
-4-
La funcin ord(a) numera todos los caracteres (devuelve un nmero entero distinto
para cada carcter). No es importante qu nmero corresponde a cada carcter, pero
s se sabe que la numeracin es coherente con el orden alfabtico y numrico:
ord(A) + 1 == ord (B);
ord(a) + 1 == ord (b);
ord(1) + 1 == ord (2).
La funcin inversa es char(a): dado un entero, devuelve el carcter
correspondiente, y tambin se puede escribir ord1(a).
Las comparaciones entre caracteres se interpretan como comparaciones entre sus
rdenes: (a < b) == (ord(a) < ord(b)).
Trminos
Los trminos de un tipo pueden ser trminos simples (variables o constantes del
tipo) o trminos compuestos (combinaciones de funciones aplicadas a trminos de
los tipos correspondientes).
Por ejemplo, los siguientes son trminos de tipo Int:
0+1
pot(((3 + 4) * 7),2)
1
2 * (1 + 1 == 2)
1 + ord(A)
Y estos otros tambin lo son, siempre y cuando x sea una variable de tipo Int; y, de
tipo Float; z de tipo Bool:
2*x+1
(pot(y,2) > ) + x
(x mod 3) * (z)
En cuanto a su semntica, los trminos representan elementos del tipo
correspondiente. Su valor es Indef cuando no se puede realizar alguna de las
operaciones que contiene. Por ejemplo,
1 div 0
pot(-1, 0.5)
Todas las operaciones se consideran estrictas (si uno de sus argumentos es
indefinido, el resultado tambin lo ser); excepto los conectivos lgicos, como ya
hemos visto. Por ejemplo, el trmino 0 * (1/0) es indefinido. En particular, intentar
comparar la igualdad de un trmino con otro indefinido es una expresin indefinida.
Esto vale tambin para los casos en que ambos trminos son indefinidos; por ejemplo,
Indef == Indef es Indef y no True.
Funciones Auxiliares
Para simplificar la escritura y la lectura de especificaciones, muchas veces vamos a
definir funciones auxiliares. Se trata de funciones que se pueden usar (nicamente) en
la especificacin, a diferencia de las que especificamos como solucin a un problema.
El mecanismo para definirlas simplemente le asigna un nombre a una expresin.
Lenguaje de especificacin Algoritmos y Estructura de Datos I 1er Cuatrimestre 2014
-5-
La sintaxis es la siguiente:
aux f(parmetros): tipo = e;
Donde f es el nombre que se le quiere dar a la funcin auxiliar, que podr usarse
en el resto de la especificacin en lugar de la expresin e. Los parmetros (opcionales)
pueden aparecer en e y se reemplazan por las expresiones correspondientes cada vez
que se utiliza f.
Por ejemplo, si definimos aux suc(i: Int): Int = i + 1; podemos despus usar la funcin
suc en la poscondicin de un problema.
tipo es el tipo resultante de la funcin definida y debe coincidir con el tipo de la
expresin que la define (e).
Es importante tener clara la diferencia entre definir una funcin auxiliar y
especificar un problema. Cuando definimos una funcin auxiliar, damos una
expresin del lenguaje a la que la funcin es equivalente. Y esto nos permite usar la
funcin dentro de las especificaciones. Cuando especificamos un problema, en
cambio, estamos dando las condiciones (el contrato) que debera cumplir alguna
funcin para ser solucin del problema. Eso no quiere decir que exista esa funcin
(podra no haber ningn algoritmo que sirviera como solucin) o que sepamos cmo
escribirla. Si alguna vez damos una funcin que solucione el problema planteado,
vamos a hacerlo en otro lenguaje (probablemente, en un lenguaje de programacin),
no en el lenguaje de especificacin. Por todo esto no podemos usar, en la
especificacin de un problema o de un tipo, otra funcin que hayamos especificado.
-6-
Secuencias
Las secuencias (tambin se las llama listas) son una familia de tipos que tiene
un lugar muy importante en el lenguaje de especificacin. Son una familia porque para
cada tipo de datos existe un tipo secuencia correspondiente, que tiene como
elementos las secuencias de elementos de ese tipo.
Las secuencias estn formadas por varios elementos del mismo tipo, posiblemente
repetidos, ubicados en un cierto orden.
El tipo de las secuencias con elementos de tipo T se llama [T] (se lee secuencia de
T). Y una forma de escribir un elemento de este tipo es escribiendo entre corchetes
varios trminos de tipo T separados por comas. Por ejemplo, el siguiente es un trmino
de tipo [Int] (secuencia de Ints):
[1,1+1,3,2*2,3,5]
La secuencia vaca (de elementos de cualquier tipo) se representa con dos
corchetes enfrentados: [].
Se puede formar secuencias de elementos de cualquier tipo. Como las secuencias
son tipos, en particular se puede usar secuencias de secuencias. Por ejemplo, el
siguiente es un trmino del tipo [[Float]] (secuencia de secuencias de Floats):
[[2.3, 7.1], [23.6, 9, 0], [5], [], [], [4.3]]
[expresin1..expresin2]
Define una secuencia mediante un intervalo. Las expresiones son del mismo tipo,
discreto y totalmente ordenado (por ejemplo, Int, Char, enumerados). El resultado
son todos los valores del tipo entre el de la expresin1 y el de la expresin2 (ambas
inclusive).
Si no vale expresin1 <= expresin2, la secuencia es vaca.
Usando parntesis en lugar de corchetes, se puede excluir uno o los dos extremos.
Ejemplos:
[5..9] == [5,6,7,8,9]
[5..9) == [5,6,7,8]
Lenguaje de especificacin Algoritmos y Estructura de Datos I 1er Cuatrimestre 2014
-7-
(5..9] == [6,7,8,9]
(5..9) == [6,7,8]
Operaciones frecuentes
En esta seccin presentamos algunas operaciones con secuencias que usaremos
frecuentemente en las especificaciones. Ms adelante examinaremos las secuencias
con ms detalle. Muchas de estas operaciones tienen una precondicin, porque no
pueden aplicarse a cualquier combinacin de argumentos, si la precondicin no se
cumple el resultado es indefinido. Esto no significa que estn especificadas mediante
contratos: luego veremos que se trata de observadores y funciones auxiliares.
Longitud: long(a: [T]): Int
Es la longitud de la secuencia a.
Notacin: long(a) se puede escribir |a|.
Indexacin: ndice(a: [T], i: Int): T
requiere 0 <= i < |a|;
Es el elemento en la i-sima posicin de a. La primera posicin es la 0.
Notacin: ndice(a,i) se puede escribir a[i] y tambin ai.
Cabeza: cab(a: [T]): T
requiere |a| > 0;
Es el primer elemento de la secuencia.
Cola: cola(a: [T]): [T]
requiere |a| > 0;
Es la secuencia sin su primer elemento.
Pertenencia: en(t: T, a: [T]): Bool
Indica si el elemento aparece (al menos una vez) en la secuencia.
Notacin: en(t,a) se puede escribir t en a y tambin t a. Otra forma de escribir no(t
en a) es t a.
Agregar cabeza: cons(t: T, a: [T]): [T]
Construye una secuencia como la recibida, a la que se le ha agregado t como
primer elemento.
Notacin: cons(t,a) se puede escribir t:a.
Concatenacin: conc(a, b: [T]): [T]
Construye una secuencia con los elementos de a, seguidos de los de b.
Notacin: conc(a,b) se puede escribir a ++ b.
Subsecuencia: sub(a: [T], d, h: Int): [T]
La subsecuencia de a formada por los elementos ubicados en las posiciones entre d
y h (ambos inclusive).
Todos los casos en que no se cumple 0 <= d <= h < |a| equivalen a la secuencia
vaca.
Notacin para obtener subsecuencias mediante intervalos.
a[d..h] == sub(a,d,h)
a[d..h) == sub(a,d,h-1)
a(d..h] == sub(a,d+1,h)
a(d..h) == sub(a,d+1,h-1)
a[d..] == sub(a,d,|a|-1)
Lenguaje de especificacin Algoritmos y Estructura de Datos I 1er Cuatrimestre 2014
-8-
a(d..] == sub(a,d+1,|a|-1)
a[..h] == sub(a,0,h)
a[..h) == sub(a,0,h-1)
En las especificaciones resulta muy frecuente usar intervalos abiertos a derecha,
ya que |a[d..h)| == h - d, y |a[..h)| == h.
Asignacin a una posicin: cambiar(a: [T], i: Int, val: T): [T]
requiere 0 <= i < |a|;
Es una secuencia igual a la secuencia a, excepto porque el valor en la posicin i es
val.
Operaciones de combinacin
Estas operaciones combinan los valores de todos los elementos de una secuencia
de valores de verdad o de nmeros.
Todos verdaderos: todos(sec: [Bool]): Bool
Es verdadero solamente si todos los elementos de la secuencia son True (o la
secuencia es vaca).
Alguno verdadero: alguno(sec: [Bool]): Bool
Es verdadero solamente si algn elemento de la secuencia es True (y ninguno es
Indef).
Sumatoria: sum(sec: [T]): T
T debe ser un tipo numrico (Float, Int).
Calcula la suma de todos los elementos de la secuencia. Si la secuencia es vaca, el
resultado es 0.
Notacin: sum(sec) se puede escribir sec.
Ejemplo:
aux PotenciasNegativasDeDosHastan(n: Int): Float = [2-m | m < [1..n]];
Productoria: prod(sec: [T]): T
T debe ser un tipo numrico (Float, Int).
Calcula el producto de todos los elementos de la secuencia. Si la secuencia es
vaca, el resultado es 1.
Notacin: prod(sec) se puede escribir sec.
Para todo: ( selectores, condiciones) expresin
Es un trmino de tipo Bool que permite afirmar que todos los elementos de una
lista definida por comprensin cumplen una propiedad (representada por la
expresin, que tambin debe ser Bool). Es simplemente una forma equivalente de
escribir lo siguiente:
todos([expresin | selectores, condiciones]).
Ejemplo (los elementos en posiciones pares son mayores que 5):
aux par(n: Int): Bool = n mod 2 == 0;
aux posParM5(a: [Int]): Bool = ( i < [0..|a|), par(i)) a[i]>5;
La expresin que define esta funcin es equivalente a esta otra:
todos([a[i]>5 | i < [0..|a|), par(i)]);
Notacin: en lugar de se puede escribir paratodo.
Existe: ( selectores, condiciones) expresin
Lenguaje de especificacin Algoritmos y Estructura de Datos I 1er Cuatrimestre 2014
-9-
La diferencia con la notacin anterior es que, en lugar de afirmar que todos los
elementos de la lista determinada por los selectores y condiciones cumplen la
expresin, aqu se dice que hay alguno que la cumple. Es, por lo tanto, equivalente
a escribir lo siguiente:
alguno([expresin | selectores, condiciones]).
Ejemplo (hay algn elemento de la lista que es par y mayor que 5):
aux hayParM5(a: [Int]): Bool = ( x < a, par(x)) x>5;
Y es equivalente a definir
aux hayParM5(a: [Int]): Bool = alguno([x>5 | x < a, par(x)]);
Notacin: en lugar de se puede escribir existe o existen.
Cantidades
A veces queremos saber cuntos elementos de una secuencia cumplen una
condicin. Una forma de hacerlo es contando la cantidad de elementos de una
secuencia construida por comprensin. Por ejemplo, para ver cuntas veces aparece el
elemento x en la secuencia a, podemos definir esta funcin:
aux cuenta(x: T, a: [T]): Int = long([y | y <- a, y == x]);
Y podemos usarla para ver si dos secuencias tienen los mismos elementos, aunque
en distinto orden:
aux mismos(a, b: [T]): Bool = |a| == |b| ( c < a) cuenta(c,a) == cuenta(c,b);
Otro ejemplo:
aux cantPrimosMenores(n: Int): Int = long([y | y <- [0..n), primo(y)]);
aux primo(n: Int): Bool = n >= 2 && ( m < [2..n)) n mod m == 0;
Acumulacin
La notacin de acumulacin provee un mecanismo similar al de construccin de
secuencias por comprensin, aumentando su poder expresivo. Construye un valor a
partir de una o ms secuencias. La forma general de una expresin de acumulacin es
la siguiente:
acum(expresin | inicializacin, selectores, condicin)
La diferencia entre esta sintaxis y la de secuencias por comprensin est en la
inicializacin, que es de la forma acumulador: tipoAcum = init. Donde acumulador es
una variable e init es una expresin de tipo tipoAcum.
El resto de los componentes coinciden con los de las secuencias por comprensin,
excepto porque en expresin, adems de las variables de los selectores, puede
aparecer el acumulador. El acumulador solo puede aparecer en la expresin pero no
en la condicin.
El resultado de la acumulacin es de tipo tipoAcum.
El acumulador tiene como valor inicial el de init. A medida que las variables de los
selectores toman cada uno de los valores de la secuencia correspondiente, se
vuelve a calcular el valor de la expresin y ese es el valor que adquiere el
acumulador. El resultado de la acumulacin es el valor final del acumulador.
Veamos algunos ejemplos:
aux sum(l: [Float]): Float = acum(s + i | s: Float = 0, i <- l);
aux prod(l: [Float]): Float = acum(p * i | p: Float = 1, i <- l);
La siguiente funcin auxiliar construye, dado n 1, la sucesin de los n+1 primeros
nmeros de la de Fibonacci:
Lenguaje de especificacin Algoritmos y Estructura de Datos I 1er Cuatrimestre 2014
- 10 -
Especificacin de Problemas
Recordemos que vamos a especificar problemas que se resuelvan mediante
funciones. La funcin que sirva como solucin va a tener por conjunto de partida los
valores posibles de sus parmetros. Siempre vamos a considerar que los parmetros
tienen valores definidos y tamao finito. La especificacin determina el contrato que
debera cumplir la funcin para ser considerada solucin del problema.
Como tambin mencionamos, es importante distinguir el qu (la especificacin, el
contrato a cumplir) del cmo (la implementacin de la funcin, que se escribir luego
en un lenguaje de programacin). Es decir que en la especificacin solamente vamos a
decir qu condiciones tiene que cumplir la solucin, pero vamos a evitar referencias a
posibles mtodos para resolver el problema.
Esta tcnica de especificacin deja abierta la posibilidad a dar distintas soluciones
a un mismo problema. Una diferencia obvia entre una solucin y otra puede ser el
lenguaje de programacin elegido para escribir la funcin. En la materia vamos a
trabajar con dos; y se van a dar cuenta de que no solamente difieren en su sintaxis,
sino tambin en la manera en que se piensan las soluciones para implementarlas en
cada uno de ellos.
Incluso si se elige un nico lenguaje de programacin, generalmente va a haber
distintas formas de programar una funcin que cumpla con la especificacin (si es que
hay alguna, como veremos ms adelante). Cada algoritmo posible constituye una
solucin distinta para el problema. En consecuencia, para cada especificacin existe
un conjunto (tal vez vaco) de algoritmos que la cumplen.
Ejemplos
Veamos algunos ejemplos de especificacin de problemas.
Si el problema que tenemos es calcular el cociente de dos enteros dados,
podramos escribir la siguiente especificacin:
problema divisin(a, b: Int) = result: Int {
requiere b != 0;
asegura result == a div b;
}
Extendamos ahora el problema a calcular el cociente y el resto de la divisin,
siempre y cuando el divisor sea positivo. Se nos presenta entonces la necesidad de
que la funcin devuelva dos valores. Una forma de lograrlo es usando un par
Lenguaje de especificacin Algoritmos y Estructura de Datos I 1er Cuatrimestre 2014
- 11 -
ordenado. Las tuplas son tipos de datos que veremos en detalle ms adelante. Por
ahora, alcanza con saber que el tipo (Int, Int) representa los pares ordenados de
enteros y que las funciones prm y sgd devuelven la primera y la segunda
componente, respectivamente.
problema cocienteResto(a, b: Int) = result: (Int,Int) {
requiere b > 0;
asegura a == q * b + r && 0 <= r < b;
aux q: Int = prm(result), r: Int = sgd(result);
}
Como la poscondicin se haca un poco larga, les pusimos nombre a las dos
componentes del resultado. Tambin noten que definimos dos nombres en una
nica sentencia aux.
Una alternativa hubiera sido dejar el cociente como nico resultado del problema, y
agregarle un parmetro modificable para representar el resto. En matemtica, la
aplicacin de funciones da un resultado, sin modificar el valor de los parmetros;
sin embargo nuestro lenguaje permite especificar que la solucin a un problema s
lo hace.
problema cocienteResto2(a, b, r: Int) = q: Int {
requiere b > 0;
modifica r;
asegura a == q * b + r && 0 <= r < b;
}
Podramos haber usado tambin un parmetro para devolver el cociente; en cuyo
caso el problema no tiene ningn resultado:
problema cocienteResto3(a, b, q, r: Int) {
requiere b > 0;
modifica q, r;
asegura a == q * b + r && 0 <= r < b;
}
Plantemosnos ahora el problema de calcular la suma de los inversos
multiplicativos de varios nmeros reales. Como no sabemos cuntos nmeros van
a ser (la cantidad va a variar en cada instancia del problema), no podemos usar
tuplas, que tienen una cantidad fija de componentes. Vamos a trabajar entonces
con secuencias:
problema sumarInvertidos(a: [Float]) = result: Float {
requiere 0 a;
asegura result == [1/x | x <- a];
}
En la precondicin estamos pidiendo que el argumento no contenga ningn 0
(porque ese nmero no puede invertirse). Si no lo pidiramos, la poscondicin
podra indefinirse, y esta es una situacin que tenemos que evitar a toda costa: las
precondiciones y las poscondiciones deben estar definidas para cualquier
valor de los parmetros. Podramos haber escrito esta misma precondicin de
distintas formas, las siguientes son todas equivalentes:
requiere ( x < a ) x 0;
requiere ( x < a) x == 0;
requiere ( i < [0..|a|)) a[i] == 0;
Lenguaje de especificacin Algoritmos y Estructura de Datos I 1er Cuatrimestre 2014
- 12 -
Sintaxis
La especificacin de un problema tiene esta forma:
problema nombre (parmetros) = nombreRes: tipoRes contrato
Lenguaje de especificacin Algoritmos y Estructura de Datos I 1er Cuatrimestre 2014
- 13 -
parmetros es una lista opcional, separada por comas, donde cada elemento es de
la forma variable: tipo, o simplemente una variable. En este ltimo caso, el tipo ser
el del parmetro siguiente (el ltimo elemento debe incluir su tipo).
tipoRes es el tipo del resultado y nombreRes es el nombre con el que vamos a
referirnos a ese resultado en el contrato. Cuando el problema modifica sus
parmetros en lugar de dar un resultado, se omiten ambos (y el =).
El contrato est formado por varias sentencias encerradas entre llaves ({ y }).
Cada sentencia est encabezada por una palabra clave ( requiere, asegura, modifica o
aux), como se explica a continuacin.
Todas las sentencias terminan con un punto y coma que es opcional.
requiere nombre: P;
Introduce una precondicin en la especificacin de una funcin (problema). El
predicado P debe cumplirse antes de la invocacin de la funcin. Equivale a la
obligacin del usuario en el contrato especificado.
Si hay ms de una precondicin, se las supone unidas por el operador &&, con su
evaluacin de cortocircuito (es decir que si una es falsa, no importa que las
siguientes se indefinan).
Cuando se resuelva el problema se har mediante una funcin, que es lo que se
llama una implementacin de la especificacin. Todas las precondiciones pueden
suponerse ciertas en la implementacin, porque si no valen cuando se la invoca, la
funcin no est obligada a cumplir el contrato. Esta semntica es de eximicin de
responsabilidad; difiere de una semntica de condicin de activacin, que
significara que la funcin no se va a ejecutar si se la llama en un estado que no
satisfaga la precondicin.
Por ejemplo, considrese la especificacin de la raz cuadrada que pide la siguiente
precondicin: requiere x >= 0;
Si el valor del argumento es negativo, la especificacin no indica qu debera hacer
la implementacin, por lo que esa posibilidad no necesita considerarse al escribir
esta ltima.
Si una precondicin es el predicado constante True, se puede omitir la clusula
requiere completa. Estas funciones pueden ser llamadas siempre. Su
implementacin no puede hacer suposiciones sobre la llamada, excepto la que
consiste en que los parmetros van a tener los tipos declarados (y que tienen
valores definidos, de tamao finito).
El nombre es optativo y puede servir tanto para aclarar el objetivo de la
precondicin, como para hacer referencia a ella en otros predicados.
asegura nombre: P;
Introduce una poscondicin en la especificacin de un problema. P es
generalmente un predicado en el que se mencionan los parmetros y el resultado
del problema. Representa las obligaciones, en el contrato especificado, de una
funcin que resuelva el problema. Cuando se razona sobre un llamado a esa
funcin, puede suponerse cierta luego de la invocacin.
Si hay ms de una poscondicin, se las evala en orden como si estuvieran unidas
con el operador &&.
La interpretacin de una especificacin es que si las precondiciones se cumplen;
entonces la funcin que resuelva el problema debe terminar normalmente, en un
estado que cumpla las poscondiciones.
Lenguaje de especificacin Algoritmos y Estructura de Datos I 1er Cuatrimestre 2014
- 14 -
modifica variables;
Introduce la lista de variables que pueden ser modificadas por la ejecucin de la
funcin. Una variable se modifica si su valor cambia como efecto de la ejecucin de
la funcin especificada (tiene un valor distinto en el pre-estado y en el
post-estado). Ntese que la clusula modifica no significa que la ejecucin deba
cambiar los valores de las variables; simplemente le otorga a la funcin permiso
para cambiarlas. Por ejemplo, si una funcin que intercambia los valores de dos de
sus parmetros, recibe dos variables con el mismo valor, no debera haber ningn
cambio.
La clusula modifica es un atajo sintctico para evitar escribir un predicado ms
largo en la poscondicin. Como solamente los objetos listados en la clusula
modifica pueden cambiar de valor en la ejecucin, los dems deben mantenerlo.
Cuando la especificacin permite modificaciones, a veces es necesario hacer
referencia al valor de una varialbe en el pre-estado (estado previo a la ejecucin de
la funcin). Se hace mediante la sintaxis pre(e).
aux definiciones;
Las definiciones son de la forma F(parmetros): tipo = e, separadas por comas.
Cada una define un nombre de funcin auxiliar ( F) que puede ser usado en el resto
de la especificacin para reemplazar a la expresin e. Los parmetros (opcionales,
con la sintaxis ya vista) pueden aparecer libres en e y se reemplazan por las
expresiones correspondientes cada vez que se utiliza F. Los parmetros del
problema especificado tambin pueden aparecer en e. En e no se puede mencionar
F. El nombre puede ser usado en el bloque donde aparece o en toda la
especificacin si no est dentro de un bloque. El tipo puede omitirse, porque se
deduce del de la expresin, pero es buena prctica incluirlo.
Tipos Compuestos
Cada valor de un tipo bsico representa un elemento atmico, indivisible. En
cambio, un valor de un tipo compuesto contiene informacin que puede ser dividida
en componentes de otros tipos. Ya vimos dos ejemplos de tipos compuestos: las
secuencias y las tuplas. Un valor de tipo secuencia de enteros tiene varios
componentes, cada uno es un entero. Un valor de tipo par ordenado de enteros tiene
dos componentes.
Para definir un tipo compuesto, tenemos que darle un nombre y uno o ms
observadores. Los observadores son funciones que se aplican a valores del tipo
compuesto (pueden tener ms parmetros) y devuelven el valor de sus componentes.
Se usan en el resto de la especificacin (precondiciones, poscondiciones, funciones
auxiliares).
Los observadores son los que definen el tipo compuesto. Si dos trminos del tipo
dan el mismo resultado para todos los observadores, se los considera iguales (esta es
la definicin implcita de la igualdad, ==, para los tipos compuestos).
Si un trmino del tipo tiene valor definido, entonces ninguna operacin (ni siquiera
los observadores) puede dar resultado indefinido al aplicrsele. Para garantizarlo, los
observadores pueden tener precondiciones. Por el contrario, los observadores no
tienen poscondiciones. Si hay condiciones generales que deban cumplir los elementos
del tipo, se las presenta como invariantes de tipo.
Lenguaje de especificacin Algoritmos y Estructura de Datos I 1er Cuatrimestre 2014
- 15 -
Ejemplos
Veamos algunos ejemplos de tipos compuestos:
El tipo Punto representa un punto en el plano. Sus componentes son las dos
coordenadas (X e Y); tiene un observador para obtener cada una de ellas.
tipo Punto {
observador X(p: Punto): Float;
observador Y(p: Punto): Float;
}
Especifiquemos ahora un problema que reciba dos nmeros reales y construya un
punto con esas coordenadas:
problema nuevoPunto(a, b: Float) = result: Punto {
asegura X(result) == a
asegura Y(result) == b;
}
En este ejemplo se ve cmo especificar un problema que tiene un tipo compuesto
como resultado: indicando qu se obtiene al aplicarle cada uno de los observadores.
Si queremos definir una funcin auxiliar para calcular la distancia entre dos puntos,
no vamos a usar esta misma tcnica; porque el resultado no es de tipo compuesto.
Pero s vamos a aplicar los observadores a los parmetros, para obtener los
componentes de los argumentos:
aux dist(p, q: Punto): Float = ((X(p)-X(q))2+(Y(p)-Y(q))2)1/2;
Ahora vamos a aprovechar el tipo compuesto que acabamos de definir para definir
un tipo nuevo:
aux tri(a, b, c: Punto): Bool = dist(a,b) < dist(a,c) + dist(b,c);
tipo Tringulo {
observador V1(t: Tringulo): Punto;
observador V2(t: Tringulo): Punto;
observador V3(t: Tringulo): Punto;
invariante tri(V1(t),V2(t),V3(t));
invariante tri(V1(t),V3(t),V2(t));
invariante tri(V2(t),V3(t),V1(t));
}
Los invariantes de tipo son la garanta de que los elementos estn bien
construidos, deben cumplirse para cualquier elemento del tipo. En este caso, se
piden las desigualdades triangulares (representadas por la funcin auxiliar tri) para
que no haya un par de puntos iguales y los puntos no estn alineados.
Todos los problemas que reciban valores del tipo compuesto como argumentos
pueden suponer, adems de su precondicin, que se cumplen los invariantes. Y los
que calculen resultados de ese tipo tienen que asegurar que los invariantes no se
van a violar.
Por ejemplo, especifiquemos el problema de construir un tringulo a partir de tres
puntos, garantizando las desigualdades triangulares:
Lenguaje de especificacin Algoritmos y Estructura de Datos I 1er Cuatrimestre 2014
- 16 -
Tipos genricos
En los ejemplos de tipos compuestos que vimos recin, cada uno de los
componentes es de un tipo predeterminado. Hay otros tipos compuestos que
representan estructuras cuyo contenido van a ser valores, no siempre del mismo tipo.
Un ejemplo que ya hemos visto son las secuencias. Cada secuencia representa una
coleccin ordenada de elementos que pueden ser de cualquier tipo (aunque todos
deben ser del mismo). Las secuencias estructuran elementos de un tipo siempre de la
misma forma, sin importar de qu tipo se trate. Existen secuencias de enteros, de
reales, de crculos y de todos los tipos que definamos. Incluso secuencias de secuencias
de enteros. Y todas las secuencias tienen el mismo comportamiento: las
especificaciones y definiciones de funciones bsicas valen para secuencias de todo tipo.
Estos tipos se llaman tipos genricos o tipos paramtricos, porque el nombre
del tipo tiene parmetros, variables que representan a los tipos de los componentes.
Los parmetros de tipo se escriben entre los smbolos < y > (o y ) despus del
nombre del tipo. Por ejemplo, Matriz<T> es el tipo genrico de las matrices cuyos
elementos pertenecen a un tipo al que, en la especificacin de las matrices, nos
vamos a referir como T.
Vimos que los problemas tenan distintas instancias, una para cada combinacin de
valores de sus parmetros. Lo mismo ocurre con los tipos genricos. Por ejemplo, una
Lenguaje de especificacin Algoritmos y Estructura de Datos I 1er Cuatrimestre 2014
- 17 -
instancia del tipo genrico Matriz<T> es Matriz<Char>, el tipo de las matrices cuyos
elementos son caracteres. Un tipo genrico define entonces, en realidad, una familia
de tipos; cada elemento de la familia es una instancia del tipo genrico. Por ejemplo,
Matriz<Char> es uno de los miembros de la familia definida por Matriz<T>; otros
miembros son Matriz<Int>, Matriz<Da>, Matriz<Tringulo>.
Ejemplos
Veamos algunos ejemplos de tipos compuestos genricos:
Empezamos por el que mencionamos en la explicacin, las matrices. Las
operaciones que definen una matriz son sus dimensiones (nmero de filas y
columnas) y la obtencin del contenido en una posicin:
tipo Matriz<T> {
observador filas(m: Matriz<T>): Int;
observador columnas(m: Matriz<T>): Int;
observador val(m: Matriz<T>, f, c: Int): T {
requiere 0 <= f < filas(m);
requiere 0 <= c < columnas(m);
}
invariante filas(m) > 0;
invariante columnas(m) > 0;
}
Tambin podemos definir y especificar operaciones genricas, funciones cuya
descripcin dependa nicamente de la estructura del tipo genrico y, por lo tanto,
no necesiten definirse sobre una instancia en particular. Un ejemplo es la funcin
elementos, que cuenta la cantidad de elementos de una matriz:
aux elementos(m: Matriz<T>): Int = filas(m) * columnas(m);
Especifiquemos ahora el problema de cambiar el valor de una posicin de una
matriz:
problema cambiarM(m: Matriz<T>, f, c: Int, v: T) {
requiere 0 <= f < filas(m);
requiere 0 <= c < columnas(m);
modifica m;
asegura val(m,f,c) == v;
asegura ( i<-[0.. filas(m)), j<-[0.. columnas(m)),if || jc)
val(m,i,j) == val(pre(m),i,j);
}
En la poscondicin hizo falta asegurar que los valores de las dems posiciones de
la matriz no se modificaban, porque la clusula modifica solo avisa que m puede
cambiar, pero no qu posiciones van a cambiar (recordar que en la clusula
modifica slo se pueden listar variables). Es por esto que en la poscondicin
tenemos que escribir explcitamente que las posiciones distintas a (f,c) no cambian
de valor.
Y, por supuesto, podemos tambin especificar problemas para instancias
particulares del tipo genrico. Por ejemplo, el que construye la matriz identidad de
n n:
problema matId(n: Int) = result: Matriz<Int> {
requiere n > 0;
asegura filas(result) == columnas(result) == n;
Lenguaje de especificacin Algoritmos y Estructura de Datos I 1er Cuatrimestre 2014
- 18 -
- 19 -
Como sus nombres lo indican, las dos precondiciones limitan el uso de la funcin a
los casos en que cada palabra tenga al menos una letra y no contenga ningn
espacio (dejara de ser una palabra). Tambin podramos haber reemplazado la
precondicin SinEspacios por una que pidiera que las palabras no contuvieran
ningn smbolo que no fuera una letra, de la siguiente manera:
requiere SinSmbolos: ( p <- palabras) soloLetras(p);
aux letras: [Char] = [A..Z] ++ [a..z];
aux soloLetras(s: [Char]): Bool = ( l <- s) l en letras;
Otros tipos genricos que ya mencionamos son las tuplas. Se trata de secuencias
de tamao fijo, cada uno de cuyos elementos puede pertenecer a un tipo distinto.
Como ejemplo, veamos los pares y las ternas:
tipo Par<A,B> {
observador prm(p: Par<A,B>): A;
observador sgd(p: Par<A,B>): B;
}
tipo Terna<A,B,C> {
observador prm3(t: Terna<A,B,C>): A;
observador sgd3(t: Terna<A,B,C>): B;
observador trc3(t: Terna<A,B,C>): C;
}
Ya habamos visto que Par<A,B> tambin se poda escribir (A,B); y Terna<A,B,C> es
(A,B,C).
La siguiente especificacin describe una funcin (intil) que toma un par formado
por un entero y un carcter, y devuelve otro par del mismo tipo, con el doble del
entero y el siguiente carcter:
problema otroPar(p: (Int,Char)) = result: (Int,Char) {
asegura prm(result) == 2*prm(p);
asegura sgd(result) == char(ord(sgd(p))+1);
}
Cul es la diferencia entre escribir esta especificacin y definir una funcin
auxiliar que devuelva un par con estas mismas caractersticas? Al haberlo
planteado como un problema, nos estamos proponiendo construir una funcin (en
un lenguaje de programacin) que cumpla la especificacin. Si hubiramos definido
una funcin auxiliar, podramos usarla en el resto de la especificacin.
Para construir una tupla, se escriben los trminos correspondientes a cada
componente como una lista separada por comas y encerrada entre parntesis. Por
ejemplo, las poscondiciones de la funcin anterior podran haberse escrito as:
asegura result == (2*prm(p), char(ord(sgd(p))+1));
Sintaxis
La definicin de un tipo compuesto tiene esta forma:
tipo nombre<parTipo> declaracin
parTipo es una lista opcional de variables de tipo separadas por coma. Las variables
aparecen en la declaracin representando tipos de datos.
La declaracin son sentencias de tipo entre llaves, encabezadas por una palabra
clave (observador, invariante, aux):
Lenguaje de especificacin Algoritmos y Estructura de Datos I 1er Cuatrimestre 2014
- 20 -
invariante nombre: P;
El nombre es opcional. P es un predicado que puede tener una nica variable libre,
la cual se interpreta como cualquier valor del tipo especificado. Esta expresin
debe ser vlida antes y despus de la invocacin a todas las funciones, por lo que
cualquier especificacin o definicin de funcin puede suponerla vlida (sin
necesidad de tenerlo como parte de su precondicin). Los problemas tienen la
obligacin de cumplirla en su poscondicin (sin que figure explcitamente).
aux definiciones;
Es la misma sentencia que hemos visto para las especificaciones. Cuando figura
dentro de una declaracin de un tipo, los nombres definidos pueden usarse en toda
la declaracin. Si se quiere que estn disponibles en la especificacin de otras
funciones, la sentencia debe colocarse fuera de la declaracin.
Funcin ifThenElse
La funcin IfThenElse elige entre dos elementos del mismo tipo, de acuerdo a una
guarda. Si la guarda es verdadera, elige el primero de los dos; si no, elige el segundo.
Por ejemplo, ifThenElse(x!=0, 1/x, 8) es el valor de 1/x cuando x es distinto de cero y es 8 en
caso contrario. Tambin se puede escribir if x!=0 Then 1/x Else 8. Observar que IfThenElse es
una funcin no estricta.
Con esta funcin podemos definir tambin algunas otras de las funciones auxiliares
que estamos usando para las secuencias:
aux cons(t: T, a: [T]): [T] = [if i==-1 then t else a[i] | i < [-1..|a|)];
aux conc(a, b: [T]): [T] = [if i<|a| then ai else bi-|a| | i < [0..|a|+|b|)];
aux cambiar(a: [T], i: Int, v: T): [T] = [if ij then ai else v | j < [0..|a|)];