Skip to content

Latest commit

 

History

History
500 lines (362 loc) · 24.8 KB

README.md

File metadata and controls

500 lines (362 loc) · 24.8 KB

Generalidades

JavaScript es un lenguaje de programación ideado en 1995 en Netscape a partir de los lenguajes C, C++ y Java.

Este resumen presenta las principales características de la variante de JavaScript denominada JavaScript-PDL que es la que hay que utilizar para la práctica de la asignatura. No hay que considerar los elementos de JavaScript no mencionados en este resumen. Entre corchetes se dan indicaciones sobre la obligatoriedad u opcionalidad de algunas partes del lenguaje en cuanto a su implementación, aunque no se recomienda que el grupo de prácticas intente abarcar muchos elementos opcionales. Con el fin de facilitar la implementación de la Práctica, las características mostradas en esta página pueden no coincidir al 100% con el estándar del lenguaje JavaScript, por lo que, en caso de duda, se deberá implementar el comportamiento aquí descrito.

JavaScript-PDL es un lenguaje en el que se diferencian las minúsculas y las mayúsculas (es case sensitive).

JavaScript-PDL es un lenguaje de formato libre, es decir, que se admiten espacios, tabuladores, saltos de línea y comentarios en cualquier parte del código. Las sentencias finalizan en punto y coma.

Las palabras clave que tiene el lenguaje son reservadas, aunque cada grupo de alumnos sólo ha de tener en cuenta las palabras asignadas a su grupo.

JavaScript-PDL es un lenguaje con estructura de bloques que se definen mediante la utilización de las llaves { } y, por tanto, maneja los conceptos de identificadores globales y locales. Los identificadores declarados fuera de cualquier función son globales y pueden ser utilizados desde cualquier función. Los declarados en el interior de una función son locales a dicha función.

En JavaScript-PDL no es obligatorio declarar todos los identificadores antes de que se utilicen; en este caso, un uso de un identificador no declarado se considera una variable global entera. Además, hay que realizar la implementación considerando que es un lenguaje con recursividad, por lo que cualquier función puede ser recursiva. El lenguaje no permite la definición de funciones anidadas.

Estructura de un Programa

Debe considerarse que un programa en JavaScript-PDL estará compuesto por un único fichero que puede tener declaraciones de variables globales, sentencias y declaración de funciones, en cualquier orden.

En este enlace se muestra un ejemplo de un fichero válido para el trabajo de esta asignatura.

Programa Principal

El programa principal (lo que se ejecutaría al arrancar el programa) estará formado por todas las sentencias ubicadas fuera de las funciones.

Por tanto, la ejecución comenzaría por la primera sentencia que se encuentre en el fuente (fuera de una función) y proseguiría secuencialmente hasta el final del fichero ejecutando todas las sentencias situadas fuera de las funciones. Hay que tener en cuenta que una función se ejecuta únicamente cuando sea invocada.

Comentarios

En JavaScript-PDL hay dos tipos de comentarios [cada grupo deberá implementar obligatoriamente el que le corresponda]:

  1. Comentario de bloque: Se utilizan los caracteres /* para abrir el comentario, y */ para cerrarlo. No se admiten comentarios anidados. Los comentarios pueden ocupar más de una línea y pueden ir colocados en cualquier parte del código:

    /* Comentario con apertura y cierre */
    
  2. Comentario de línea: Los comentarios comienzan por los caracteres // y finalizan al acabar la línea. Este tipo de comentario sólo ocupa una línea y puede ir colocado en cualquier parte del código:

    // Comentario de línea
    

Constantes

El lenguaje dispone de varios tipos de constantes [implementación obligatoria de las constantes enteras y cadenas]:

Enteras

Para representar las constantes enteras se utilizan los dígitos decimales. Por ejemplo: 159.

Los números enteros se tienen que poder representar con una palabra (16 bits, incluido el signo), por lo que el máximo entero válido será el 32767.

Cadenas de Caracteres

Las constantes cadena van encerradas entre comillas dobles ("Hola, mundo") o entre comillas simples ('Hola, mundo') [cada grupo deberá implementar obligatoriamente la que le corresponda]. Se utiliza internamente el carácter nulo (cuyo código ASCII es 0) como carácter de fin de cadena. Puede aparecer cualquier carácter imprimible en la cadena (por tanto, no es válido un salto de línea como tal; para usarlo, se pueden emplear las secuencias de escape).

Para representar caracteres especiales dentro de una cadena se utiliza una secuencia de escape. Una secuencia de escape se representa mediante el carácter barra inversa seguido de un determinado carácter. Algunos de estos caracteres son: el salto de línea (\n), las comillas dobles (\"), las comillas simples (\') o el carácter de barra inversa (\\) [la implementación de estos caracteres especiales es opcional].

Una cadena no puede contener más de 64 caracteres.

Lógicas

En JavaScript-PDL existen dos constantes lógicas: true y false [es opcional implementar las constantes lógicas, que son las palabras reservadas true y false].

Operadores

Este lenguaje presenta un conjunto de operadores con los que escribir distintas expresiones. Además, se pueden utilizar los paréntesis para agrupar subexpresiones [es obligatorio implementar los paréntesis]. Las expresiones pueden tener varios operadores, varios operandos, paréntesis... Las expresiones se pueden utilizar en multitud de construcciones del lenguaje: asignaciones, condiciones de un if, condiciones de un bucle, parámetros de una función, instrucciones de salida, instrucción de retorno...

Operadores Aritméticos

Son los operadores que permiten realizar la suma, resta, producto, división y módulo: +-*/ y % [obligatorio implementar al menos uno y dos como máximo]. Se aplican sobre datos enteros, proporcionando un resultado entero (en el caso de la división, redondeando el valor si es necesario).

También existen los operadores más y menos unarios: +- [implementación opcional]. Estos operadores se pueden utilizar delante de una constante entera, una variable o una expresión.

Operadores de Relación

Son los operadores que permiten realizar las comparaciones de igual, distinto, menor, mayor, menor o igual, mayor o igual: ==!=<><= y >= [obligatorio implementar al menos uno de los operadores y dos como máximo]. Se aplican sobre datos enteros y proporcionan un resultado lógico.

Operadores Lógicos

Representan las operaciones de conjunción, disyunción y negación: &&|| y ! [obligatorio implementar al menos uno y dos como máximo]. Se aplican sobre datos lógicos y devuelven un resultado lógico.

Operadores de Incremento y Decremento

Permiten auto-incrementar o auto-decrementar el valor de una variable entera: ++ y -- (pueden actuar como prefijos o como sufijos) [algunos grupos tienen que implementar uno de estos operadores]. Se aplican sobre variables enteras y devuelven un resultado entero modificando también el valor de la variable. Ejemplo:

a = j++ /* si j valía 5, ahora a == 5 y j == 6 */
a = ++j; /* si j valía 5, ahora a == 6 y j == 6 */

Operadores de asignación

Permiten realizar asignaciones simples o realizando simultáneamente una operación: = (asignación), += (asignación con suma), -= (asignación con resta), *= (asignación con producto), /= (asignación con división), %= (asignación con módulo), &= (asignación con y lógico) y |= (asignación con o lógico) [todos los grupos tienen que implementar la asignación simple (=) y algunos grupos deberán implementar uno de los operadores de asignación con operación]. Ejemplo:

n += m;    /* es equivalente a n = n + m */
b1 &= b2;  /* es equivalente a b1 = b1 && b2 */

Precedencia de Operadores

En la tabla siguiente se muestra la precedencia de los operadores con el siguiente significado: los operadores del mismo grupo tienen la misma precedencia y, conforme se desciende por la tabla, la precedencia aumenta. En cualquier caso, el uso de paréntesis permite alterar el orden de evaluación de las expresiones [es obligatorio para todos los grupos tener en cuenta las precedencias de los operadores utilizados].

Precedencias de los OperadoresMás información

Operadores Significado Asociatividad
OR (formato barra) O lógico Izquierda a derecha
&& Y lógico Izquierda a derecha
==, != IgualDistinto Izquierda a derecha
>, >=, <, <= MayorMayor o igualMenorMenor o igual Izquierda a derecha
+, - SumaResta Izquierda a derecha
*, /, % ProductoDivisiónMódulo Izquierda a derecha
!, ++, --, +, - Negación lógica, Autoincremento, Autodecremento, Más unario, Menos unario Derecha a izquierda

Identificadores

Los nombres de identificadores están formados por cualquier cantidad de letras,dígitos y subrayados (_), siendo el primero siempre una letra. Ejemplos: aa3A3Sueldo_de_Trabajadorz_9_9__...

Como ya se ha dicho, el lenguaje es dependiente de minúsculas o mayúsculas, por lo que los nombres a3 y A3 son identificadores distintos.

Declaraciones

El lenguaje JavaScript-PDL no exige declaración de las variables que se utilicen. En el caso de que se use un nombre de variable que no ha sido declarado previamente, se considera que dicha variable es global y entera.

Para realizar una declaración de una variable, se coloca la palabra let seguida del tipo y del nombre de la variable.

let Tipo var0;

Pueden realizarse declaraciones en cualquier lugar de un bloque de una función; en este caso, la variable será visible desde ese punto hasta el final de la función. También pueden realizarse declaraciones fuera de las funciones en cualquier parte del código (variables globales), siendo solo visibles desde ese punto hasta el final del fichero.

Opcionalmente, puede inicializarse una variable en la misma instrucción de la declaración, colocando detrás del nombre de la variable el operador de asignación (=) seguido de una expresión [es opcional implementar la inicialización de variables].

let Tipo var4 = expresión4;

Si una variable no se inicializa cuando se declara se realiza una inicialización por omisión basándose en su tipo: 0 si es entera, falso si es lógica y la cadena vacía ("") si es cadena.

El ámbito de una variable será global si no se ha declarado o si se declara fuera de cualquier función, y será local si se declara dentro del cuerpo de una función. No se admite la redeclaración del mismo identificador en un mismo ámbito.

Tipos de Datos

El lenguaje dispone de distintos tipos de datos básicos.

Se deben considerar sólo los siguientes tipos de datos básicos: entero, lógico y cadena.

El tipo entero se refiere a un número entero que debe representarse con un tamaño de 1 palabra (16 bits). Se representa con la palabra number.

El tipo lógico permite representar valores lógicos. Se representa también con un tamaño de 1 palabra (16 bits). Las expresiones relacionales y lógicas devuelven un valor lógico. Se representa con la palabra boolean.

El tipo cadena permite representar secuencias de caracteres. Se representa con la palabra string y una variable de tipo cadena ocupa 64 palabras (128 bytes).

El lenguaje no tiene conversiones automáticas entre tipos.

Ejemplos:

let number i = 11;    // variable entera
let string st;         // variable cadena
let boolean b;         // variable lógica
let number c = 66+i;  // variable entera
b = i != c + 1;        // i y c+1 son enteros; b valdrá verdadero
c = c + i;             // i y c son enteras; c valdrá 88
i = b + i;             // Error: no se puede sumar un lógico con un entero
b = ! i;               // Error: el operador de negación solo puede aplicarse a lógicos

Instrucciones de Entrada/Salida

Las instrucciones de entrada/salida disponibles en el lenguaje son dos. Ambas tienen la sintaxis de una llamada a función, pero no retornan ningún valor. Por tanto, su uso equivale al uso de una sentencia.

La instrucción alert (expresión) evalúa la expresión e imprime el resultado por pantalla. La expresión puede ser de tipo cadena o entera. Por ejemplo:

c= 50; alert(c * 2 + 16); /* imprime: 116 */
a= 'Adiós';
alert('Hola'); alert(a); /* imprime HolaAdiós */

La instrucción input (variable) lee un número o una cadena del teclado y lo almacena en la variable variable, que tiene que ser, respectivamente, de tipo entero o cadena. Por ejemplo:

let number a;
let string c;
input (a);    // lee un número
alert(a * a); /* imprime el cuadrado del número leído */
alert("Pon tu nombre");
input (c);    // lee una cadena
alert("Hola, ");
alert(c); /* imprime las cadenas */

Sentencias

De todo el grupo de sentencias del lenguaje JavaScript, se han seleccionado para ser implementadas las que aparecen a continuación [opcional u obligatoriamente, según se indique en cada caso]. Además de las sentencias aquí indicadas, también se consideran sentencias en este lenguaje las instrucciones de entrada/salida, así como las declaraciones de variables.

Sentencias de Asignación

Existe una sentencia de asignación en JavaScript-PDL, que se construye mediante el símbolo de asignación = [es obligatorio implementar la sentencia de asignación por todos los grupos; los grupos que tengan el operador de asignación con operación deberán implementar también la sentencia de asignación con operación con el operador asignado]. Su sintaxis general es la siguiente: identificador, igual y expresión. Esta sentencia asigna al identificador el resultado de evaluar la expresión:

i= 8 + 6;

Como ya se ha indicado, no hay conversiones entre tipos, por lo que tanto el identificador como la expresión han de ser del mismo tipo.

let number i = 123;	// i es una variable entera
let string cad;
alert (i);   // imprime el valor entero 123
cad= 'hola';
alert (cad); // imprime el valor cadena "hola"
i = i > 88;  // Error: no se puede asignar un lógico a un entero

Sentencia de Llamada a una Función

Esta sentencia permite invocar la ejecución de una función que debe estar previamente definida [implementación obligatoria].

La llamada a una función se realiza mediante el nombre de la función seguido de los argumentos actuales (separados por comas) entre paréntesis (si no tiene argumentos, hay que poner los paréntesis vacíos). Los argumentos pueden ser cualquier expresión válida en el lenguaje:

p1 (5);        /* llamada a una función con un argumento entero */
p2 ();         /* llamada sin argumentos a una función */
p3 (b, i - 8); /* llamada con dos argumentos a una función */

Los parámetros actuales en la llamada tienen que coincidir en número y tipo con los parámetros formales de la declaración de la función.

Si una función devuelve un valor, podrá incluirse una llamada a dicha función dentro de cualquier expresión. Si la llamada se realiza como una sentencia (no se realiza en una expresión), se invocará a la función pero el valor devuelto se perderá:

b= fun1 (9); /* llamada a una función con un argumento entero */
c= b + fun2 (b, fun3() - 8); /* llamada con dos argumentos a una función,
               siendo fun3, una llamada a otra función sin argumentos */
fun2 (5, c); /* el valor devuelto por fun2 se pierde */

Sentencia de Retorno de una Función

JavaScript-PDL dispone de la sentencia return para finalizar la ejecución de una función y volver al punto desde el que fue llamada [implementación obligatoria]. Si no se desea que una función devuelva un valor, ésta terminará cuando se ejecute la instrucción return (sin expresión) o al llegar al final del cuerpo de la función. Si se desea que la función devuelva algún dato, deberá incluirse una expresión en la sentencia return. Si se indica, el tipo de la expresión retornada deberá coincidir con el tipo de la función. Si no se incluye una expresión, la función debe haber sido declarada sin tipo. No es necesario que todas las funciones tengan instrucción de retorno.

function number SumaAlCuadrado (number a, number b)
{
  j= a + b;
  return j * j;
  /* La función finaliza y devuelve el valor entero de la expresión */
}
function pro (number x)
{
  x= SumaAlCuadrado (x - 1, x);
   /* x contendrá el valor devuelto por la función: (x+x-1)^2 */
  if (x > (194/2)) return; /* finaliza la ejecución si se ejecuta */
  alert (SumaAlCuadrado (x, x));
} /* finaliza la ejecución si antes no se ejecutó el return */

Sentencia Condicional simple

Selecciona la ejecución de una sentencia, dependiendo del valor correspondiente de una condición de tipo lógico [implementación obligatoria para todos los grupos]:

if (condición) sentencia

Si la condición lógica se evalúa como cierta se ejecuta la sentencia que puede ser cualquier sentencia simple del lenguaje, es decir, asignación, instrucción de entrada/salida, llamada a función o retorno (también break o sentencias de auto-incremento o auto-decremento para los grupos que tengan dichas opciones); en caso contrario, se finaliza su ejecución:

if (a > b) c= b;
if (fin) alert("adiós");

Sentencia Condicional compuesta

Selecciona la ejecución de una de las secuencias de sentencias que encierra, dependiendo del valor correspondiente de una condición de tipo lógico. Tiene dos formatos [implementación obligatoria de ambos para los grupos que les corresponda]:

  • if (condición)
    {
       cuerpo1
    }
    
  • if (condición)
    {
       cuerpo1
    }
    else
    {
       cuerpo2
    }
    

Si la condición lógica se evalúa como cierta se ejecuta el cuerpo1; en caso contrario, se ejecuta el cuerpo2 (si el programador lo ha escrito). Cada uno de estos cuerpos serán cero, una o más sentencias o declaraciones, y van siempre entre llaves.

if (a > b)
{
  c= b;
}
else
{
  c= a;
  if (fin);
  {
    alert("adiós");
  }
}

Sentencia Repetitiva while

Esta sentencia permite repetir la ejecución de unas sentencias basándose en el resultado de una expresión lógica [implementación obligatoria para algunos grupos]. La sintaxis es:

while (condición)
{
   cuerpo
}

Se evalúa la condición lógica y, si resulta ser cierta, se ejecuta el cuerpo (que será un bloque de sentencias o declaraciones de variables entre llaves). Este proceso se repite hasta que la condición sea falsa:

while (n <= 10)
{
    n= n + 1;
    alert (n);
} /* mientras que n sea menor o igual que 10... */

Sentencia Repetitiva do while

Esta sentencia permite repetir la ejecución de las sentencias del bucle mientras se cumpla una condición [implementación obligatoria para algunos grupos]. La sintaxis es:

do {
   cuerpo
} while (condición);

En esta instrucción se ha de colocar un bloque de sentencias o declaraciones de variables encerradas entre llaves. Se ejecuta el cuerpo; seguidamente se evalúa la condición lógica y, si resulta ser cierta, se ejecuta de nuevo el cuerpo. Este proceso se repite hasta que la condición sea falsa:

do {
  a++;
  c *= b;
} while (a < b);

Sentencia Repetitiva for

Esta sentencia for permite ejecutar un bucle según una condición compleja [implementación obligatoria para algunos grupos]. La sintaxis es:

for (inicialización; condición; actualización)
{
   cuerpo
}

La inicialización debe ser una sentencia de asignación o nada; la condición debe ser una expresión lógica; y la actualización puede ser una asignación (sencilla o con operación [para los grupos que tengan esta opción]), un autoincremento o autodecremento [para los grupos que tengan esta opción] o estar vacía. El cuerpo (sentencias o declaraciones de variables) está en un bloque delimitado por llaves.

El funcionamiento de este bucle es como sigue:

  1. Se ejecuta la inicialización
  2. Se evalúa la condición
  3. Si la condición lógica es falsa, se abandona la ejecución del bucle
  4. Se ejecuta el cuerpo
  5. Se ejecuta la actualización
  6. Se vuelve al paso 2.
for (i = 1; i < 10; i++)
{
 f *= i;
}

Sentencia de Selección Múltiple

Esta sentencia selecciona y ejecuta unas sentencias basándose en el resultado de una expresión [implementación obligatoria para algunos grupos]. La sintaxis de la sentencia de selección múltiple es (el cuerpo de una etiqueta case o default puede omitirse):

switch (expresión)
{
  case valor1: cuerpo1
  case valor2: cuerpo2
  /* . . . */
  default: cuerpon
}

Su funcionamiento es como sigue: Se evalúa la expresión (que debe ser de tipo entero), se busca el valor que coincida con el resultado (los valores tienen que ser constantes enteras) y se ejecuta su cuerpo asociado (cada cuerpo puede estar formado por sentencias o declaraciones de variables). Una vez ejecutado, se continúa la ejecución de todos los cuerpos asociados a todos los valores que se encuentren a continuación hasta el final del switch. Si no se encuentra el valor, se ejecutan las sentencias asociadas a default, si el programador la ha incluido [el default es opcional para la implementación]:

switch (dia)
{
   case 1: alert('lunes');
   case 2: alert('martes');
   case 3: alert('miércoles');
   case 4: alert('jueves');
   case 5:
           alert('viernes');
   default: alert('fiesta');
} /* si dia == 4, se imprimirá: jueves viernes fiesta */

Sentencia break

Esta sentencia aborta la ejecución de un bucle o un switch [implementación obligatoria para los grupos que tengan switch]:

switch (dia)
{
   case 1: alert("lunes"); break;
   case 2: alert("martes");
   case 3: alert("miércoles");
           break;
   case 4: alert("jueves"); break;
   case 5: alert("viernes"); break;
   case 6:
   default: alert("fiesta");
} /* si dia == 4, se imprimirá: jueves */

Funciones

Es necesario definir cada función antes de poder utilizarla. La definición de una función se realiza indicando la palabra function, el tipo de retorno (si la función devuelve algo), el nombre y, entre paréntesis, los argumentos (si existen) con sus tipos. Tras esta cabecera va un bloque (delimitado por llaves) con el cuerpo de la función (que puede tener cualquier cantidad de sentencias o declaraciones de variables):

function [Tipo] nombre (lista de argumentos)
{
   sentencias | declaración de variables
}

La lista de argumentos (que puede estar vacía y, en este caso, se ponen los paréntesis vacíos) consta del tipo y del nombre de cada parámetro formal. Si hay más de un argumento, éstos se separan por comas. Los argumentos se pasan siempre por valor.

Las funciones pueden recibir como parámetros cualquiera de los tipos básicos del lenguaje (entero, lógico o cadena).

Las funciones pueden devolver un valor de uno de los tipos básicos del lenguaje (numberboolean o string). El tipo de retorno de la función se determina según el tipo que aparezca en su declaración. Si se omite el tipo en la declaración, se entiende que la función no devolverá ningún valor. En caso de que las instrucciones return de una función tengan expresiones de un tipo distinto al declarado, será un error.

JavaScript-PDL admite recursividad. Todos los grupos de prácticas han de considerarla en su implementación. Cualquier función puede ser recursiva, es decir, puede llamarse a sí misma.

El lenguaje JavaScript-PDL no permite la definición de funciones anidadas. Esto implica que dentro de una función no se puede declarar ni definir otra función.

Dentro de una función se tiene acceso a las variables locales, a sus argumentos y a las variables globales. Si en una función se declara una variable local o un argumento con el mismo nombre que un identificador global, éste último no es accesible desde dicha función.

let number x;  // global
function number factorial (number x)
   /* se define la función recursiva con un parámetro,
      que oculta a la variable global de igual nombre */
{
  if (x > 1)
  {
    return x * factorial (x - 1);
  }
  else
  {
    return 1;
  }
}	// la función devuelve un entero
function boolean Suma (number aux, number fin)
  /* se define la función Suma que recibe
     dos enteros por valor */
  /* usa la variable global x */
{
    for (x= 1; x < fin; x= x + 2)
    {
      aux += factorial (aux-1);
    }
    return aux > 10000;
}	// la función devuelve un lógico
function Imprime (number a)
{
    alert (a);
    return;	// esta instrucción se podría omitir
}	// la función no devuelve nada
Imprime (factorial (Suma (5, 3)));	// se llama a las tres funciones