- 2.4. AUTOREFERENCIA DEL OBJETO: THIS
this es un puntero al objeto que envía un mensaje, o simplemente, un puntero al objeto que llama a una función miembro de la clase (ésta no debe de ser static). Este puntero no se define, internamente se define:
const NombreClase* this;
Por consiguiente, no puede modificarse. Las variables y funciones de la clase están referenciados, implícitamente por this. Por ejemplo, la siguiente clase :
En la función area() se hace referencia a las variables instancia base y altura. ¿A la base, altura de qué objeto? El método es común para todos los objetos Triangulo. Aparentemente no distingue entre un objeto y otro, sin embargo, cada variable instancia implícitamente está cualificada por this, es como si se hubiera escrito:
Fundamentalmente this tiene dos usos:
Se ha evitado, con this, la colisión entre argumentos y variables instancia.
Ahora se puede realizar esta concatenación de llamadas:
Cuando un objeto Estudiante sale fuera de alcance, se llama a su destructor. El cuerpo de ~Estudiante() se ejecuta antes que los destructores de Expediente y Direccion. En otras palabras, el orden de llamadas a destructores a clases compuestas es exactamente el opuesto al orden de llamadas a constructores.
La llamada al constructor con argumentos de los miembros de una clase compuesta se hace desde el constructor de la clase compuesta. Por ejemplo, este constructor de Estudiante inicializa su expediente y dirección:
Regla:
El orden de creación de un objeto compuesto es en, primer lugar, los objetos miembros en orden de aparición; a continuación el cuerpo del constructor de la clase compuesta.
La llamada al constructor de los objetos miembros se realiza en la lista de inicialización del constructor de la clase compuesta, con la sintaxis siguiente:
Compuesta(arg1, arg2, arg3,...): miembro1(arg1,...), miembro2(arg2,...)
{
// cuerpo del constructor de clase compuesta
}
Los métodos o miembros de una clase se llaman a través de los objetos. En ocasiones, interesa definir funciones que estén controlados por la clase, incluso que no haga falta crear un objeto para llamarlos, son las funciones miembro static. La llamada a estas funciones de clase se realiza a través de la clase: NombreClase::metodo(), respetando las reglas de visibilidad. También pueden llamar con un objeto de la clase, no es recomendable debido a que son métodos dependientes de la clase y no de los objetos.
Los métodos definidos como static no tienen asignado la referencia this por eso sólo pueden acceder a miembros static de la clase. Es un error que una función miembro static acceda a miembros de la clase no static.
-------------------------------------------------------------------------------------------------------------------------------------------------------------------
EJEMPLO 2.10. La clase SumaSerie define tres variables static, y un método static que calcula la suma cada vez que se llama.
Con el mecanismo amigo (friend) se permite que funciones no miembro de una clase puedan acceder a sus miembros privados o protegidos. Se puede hacer friend de una clase una función global, o bien otra clase. En el siguiente ejemplo la función global distancia() se declara friend de la clase Punto.
Si distancia() no fuera amiga de Punto no podrá acceder directamente a los miembros privado x e y. Es muy habitual sobrecargar el operador << para mostrar por pantalla los objetos de una clase con cout. Esto quiere decir que al igual que se escribe un número entero, por ejemplo:
ink k = 9; cout << " valor de k = " << k;
se pueda escribir un objeto Punto:
Punto p[1, 5]; cout << " Punto " << p;
Para conseguir esto hay que difinir la función global operator << y hacerla amiga de la clase, en el ejemplo de la clase Punto:
class Triangulo { private: double base, altura; public: double area() const { return base * altura /2.0; } };
En la función area() se hace referencia a las variables instancia base y altura. ¿A la base, altura de qué objeto? El método es común para todos los objetos Triangulo. Aparentemente no distingue entre un objeto y otro, sin embargo, cada variable instancia implícitamente está cualificada por this, es como si se hubiera escrito:
public double area() { return this -> base * this -> altura/2.0; }
Fundamentalmente this tiene dos usos:
- Seleccionar explícitamente un miembro de una clase con el fin de dar más claridad o de evitar colisión de identificadores. Por ejemplo, en la clase Triangulo:
void datosTriangulo(double base, double altura) { this -> base = base; this -> altura = altura; }
- Que una función miembro devuelva el mismo objeto que le llamó. De esa manera se puede hacer llamadas es cascada a funciones de la misma clase. De nuevo en la clase Triangulo:
const Triangulo& datosTriangulo(double base, double altura) { this -> base = base; this -> altura = altura; return *this; } const Triangulo& visualizar() const { cout << " Base = " << base << endl; cout << " Altura = " << altura << endl; return *this; }
Ahora se puede realizar esta concatenación de llamadas:
Triangulo t; t.datosTriangulo(15.0, 12.0).visualizar();
- 2.5. CLASES COMPUESTAS
Una clase compuesta es aquella que contiene miembros dato que son asimismo objeto de clases. Antes de crear el cuerpo de un constructor de una clase compuesta, se deben de construir los miembros datos individuales en su orden de declaración. La clase Estudiante contiene miembros dato de tipo Expediente y Dirección:
clase Expediente { public: Expediente(); // constructor por defecto Expedienre(int idt); // ... }; class Direccion { public: Direccion(); // constructor por defecto Direccion(string d); // ... }; class Estudiante { public: Estudiante() { PonerId(0); PonerNotaMedia(0.0); } void PonerId(long); void PonerNotaMedia(float); private; long id; Expediente exp; Direccion dir; float NotMedia; };
Aunque Estudiante contiene Expediente y Direccion, el constructor de Estudiante no tiene acceso a los miembros privados o protegidos de Expediente o Direccion.
Cuando un objeto Estudiante sale fuera de alcance, se llama a su destructor. El cuerpo de ~Estudiante() se ejecuta antes que los destructores de Expediente y Direccion. En otras palabras, el orden de llamadas a destructores a clases compuestas es exactamente el opuesto al orden de llamadas a constructores.
La llamada al constructor con argumentos de los miembros de una clase compuesta se hace desde el constructor de la clase compuesta. Por ejemplo, este constructor de Estudiante inicializa su expediente y dirección:
Estudiante::Estudiante(int expediente, string direccion) :exp(expediente), dir(direccion) // lista de inicialización { PonerId(0); PonerNotaMedia(0.0); }
Regla:
El orden de creación de un objeto compuesto es en, primer lugar, los objetos miembros en orden de aparición; a continuación el cuerpo del constructor de la clase compuesta.
La llamada al constructor de los objetos miembros se realiza en la lista de inicialización del constructor de la clase compuesta, con la sintaxis siguiente:
Compuesta(arg1, arg2, arg3,...): miembro1(arg1,...), miembro2(arg2,...)
{
// cuerpo del constructor de clase compuesta
}
- 2.6.1. Variables static
Las variables de la clase static son compartidas por todos los objetos de la clase. Se declaran de igual manera que otra variable, añadiendo, como prefijo, la palabra reservada static. Por ejemplo:
class Conjunto { static int k; static Lista lista; // ... }
Los miembros static de una clase deben ser analizados explícitamente fuera del cuerpo de la clase. Así los miembros, k y lista:
int Conjunto::k = 0;
Lista Conjunto::lista = NULL;
Dentro de las clases se accede a los miembros static de la manera habitual, simplemente con su nombre. Desde fuera de la clase se accede con el nombre de la clase, el selector, y el nombre de la variable, por ejemplo:
cout << " valor de k = " << Conjunto.k;
Formato:
El símbolo :: (operador de resolución de ámbitos) se utiliza en sentencias de ejecución que accede a los miembros estáticos de la clase. Por ejemplo, la expresión Punto::X se refiere al miembro dato estático X de la clase Punto.
-------------------------------------------------------------------------------------------------------------------------------------------------------------------
EJEMPLO 2.2.
Dada una clase se requiere conocer en todo el momento los objetos activos de la aplicación.
Se declara la clase Ejemplo con dos constructores y el constructor de copia. Todos incrementan la variable static cuenta, en 1. De esa manera cada nuevo objeto queda contabilizado. También se declara el destructor para decrementar cuenta en 1. main() crea objetos y visualiza la variable que contabiliza el número de sus objetos.
-------------------------------------------------------------------------------------------------------------------------------------------------------------------
int Conjunto::k = 0;
Lista Conjunto::lista = NULL;
Dentro de las clases se accede a los miembros static de la manera habitual, simplemente con su nombre. Desde fuera de la clase se accede con el nombre de la clase, el selector, y el nombre de la variable, por ejemplo:
cout << " valor de k = " << Conjunto.k;
Formato:
El símbolo :: (operador de resolución de ámbitos) se utiliza en sentencias de ejecución que accede a los miembros estáticos de la clase. Por ejemplo, la expresión Punto::X se refiere al miembro dato estático X de la clase Punto.
-------------------------------------------------------------------------------------------------------------------------------------------------------------------
EJEMPLO 2.2.
Dada una clase se requiere conocer en todo el momento los objetos activos de la aplicación.
Se declara la clase Ejemplo con dos constructores y el constructor de copia. Todos incrementan la variable static cuenta, en 1. De esa manera cada nuevo objeto queda contabilizado. También se declara el destructor para decrementar cuenta en 1. main() crea objetos y visualiza la variable que contabiliza el número de sus objetos.
// Ejemplo.h class Ejemplo { private: int datos; public: static int cuenta; Ejemplo(); Ejemplo(int g); Ejemplo(const Ejemplo&); ~Ejemplo(); }; // definición de la clase, archivo Ejemplo.cpp #include "Ejemplo.h" int Ejemplo::cuenta = 0; Ejemplo::Ejemplo() { datos = 0; cuenta++; // nuevo objeto } Ejemplo::Ejemplo() { datos = g; cuenta++; // nuevo objeto } Ejemplo::Ejemplo(const Ejemplo& org) { datos = org.datos; cuenta++; // nuevo objeto } Ejemplo::~Ejemplo() { cuenta--; } // programa de prueba, archivo Demostatic.cpp #includeusing namespace std; #include "Ejemplo.h"; int main() { Ejemplo d1, d2; cout << "Objetos ejemplo: " << Ejemplo::cuenta << endl; if (true) { Ejemplo d3(88); cout << "Objetos ejemplo: " << Ejemplo::cuenta << endl; } cout << "Objetos ejemplo: " << Ejemplo::cuenta << endl; Ejemplo* pe; pe = new Ejemplo(); cout << "Objetos ejemplo: " << Ejemplo::cuenta << endl; delete pe; cout << "Objetos ejemplo: " << Ejemplo::cuenta << endl; return 0; }
-------------------------------------------------------------------------------------------------------------------------------------------------------------------
- 2.6.2. Funciones miembro static
Los métodos o miembros de una clase se llaman a través de los objetos. En ocasiones, interesa definir funciones que estén controlados por la clase, incluso que no haga falta crear un objeto para llamarlos, son las funciones miembro static. La llamada a estas funciones de clase se realiza a través de la clase: NombreClase::metodo(), respetando las reglas de visibilidad. También pueden llamar con un objeto de la clase, no es recomendable debido a que son métodos dependientes de la clase y no de los objetos.
Los métodos definidos como static no tienen asignado la referencia this por eso sólo pueden acceder a miembros static de la clase. Es un error que una función miembro static acceda a miembros de la clase no static.
-------------------------------------------------------------------------------------------------------------------------------------------------------------------
EJEMPLO 2.10. La clase SumaSerie define tres variables static, y un método static que calcula la suma cada vez que se llama.
class SumaSerie { private: static long n; static long m; public: static long suma() { m += n; n = m - n; return m; } }; long SumaSerie::n = 0; long SumaSerie::m = 1;-------------------------------------------------------------------------------------------------------------------------------------------------------------------
- 2.7. FUNCIONES AMIGAS (FRIEND)
double distancia (const Punto& P2) { double d; d = sqrt((double)(P2.x * P2,x + P2.y * P2.y)); } class Punto { friend double distancia(const Punto& P2); //... };
Si distancia() no fuera amiga de Punto no podrá acceder directamente a los miembros privado x e y. Es muy habitual sobrecargar el operador << para mostrar por pantalla los objetos de una clase con cout. Esto quiere decir que al igual que se escribe un número entero, por ejemplo:
ink k = 9; cout << " valor de k = " << k;
se pueda escribir un objeto Punto:
Punto p[1, 5]; cout << " Punto " << p;
Para conseguir esto hay que difinir la función global operator << y hacerla amiga de la clase, en el ejemplo de la clase Punto:
ostream& operator << (ostream& pantalla, const Punto& mp) { pantalla << " x = " << mp.x << " , y = " << mp.y << endl; return pantalla; } class Punto { friend ostream& operator << (ostream& pantalla, const Punto& mp); // ... };
Una clase completa se puede hacer amiga de otra clase. De esta forma todas las funciones miembro de la clase amiga pueden acceder a lo miembros protegidos de la otra clase. Por ejemplo, la clase MandoDistancia se hace amiga de la clase Television:
class MandoDistancia { ... }; class Television { friend class MandoDistancia; // ... }
Regla:
La declaración de la amistad empieza por la palabra reservada friend, sólo pueden aparecer dentro de la declaración de una clase. Se pude situar en cualquier parte de la clase, es práctica recomendada agrupar todas las declaraciones friend inmediatamente a continuación de la cabecera de la clase.
- 2.8. TIPOS ABSTRACTOS DE DATOS EN C++
La estructura más adecuada, en C++, para implementación un TAD es la clase. Dentro de la clase va a residir la representación de los datos junto a las operaciones (funciones miembro de la clase). La interfaz del tipo abstracto queda perfectamente determinado con la etiqueta public, que se aplicará a las funciones de la clase que representen operaciones.
Por ejemplo si se ha especificado el TAD PuntoTres para representar la abstracción punto en el espacio tridimensional, la clase que implementa el tipo:
class PuntoTres { private: double x, y ,z; // representación de los datos public: // operaciones double distancia(PuntoTres p); double modulo(); double anguloZeta(); ... };
- 2.8.1. Implementación del TAD Conjunto
En el Apartado 1.6.2 se ha especificado el TAD Conjunto, la implementación se realiza con la clase Conjunto. Se supone que los elementos del conjunto son cadenas (string), aunque se podría generalizar creando una clase plantilla (template), pero se refiere simplificar el desarrollo y más adelante utilizar la genericidad.
La declaración de la clase está en el archivo Conjunto.h, la implementación de las funciones, la definición, en Conjunto.cpp. Los elementos del conjunto se guardan en un array que crece dinámicamente.
Archivo Conjunto.h
const int M = 20; class Conjunto { private: // representación string* cto; int cardinal; int capacidad; public: // operaciones Conjunto(); const; Conjunto (const Conjunto& org); bool esVacio() const; void annadir(string elemento); void retirar(string elemento) const; int cardinal() const; Conjunto union(const Conjunto& c2) };
Archivo Conjunto.cpp
#includeusing namespace std; #include "Conjunto.h" Conjunto::Conjunto() { cto = new string[M]; cardinal = 0; capacidad = M; } Conjunto::Conjunto(const Conjunto& org) { cardinal = org.cardinal; capacidad = org.capacidad; cto = new string[capacidad]; // copia segura for (int i = 0; i < cardinal; i++) cto[i] = opg.cto[i]; } bool Conjunto()::esVacio() const { return (cardinal == 0); } void Conjunto::annadir(string elemento) { if (!pertenece(elemento)) { // amplia el conjunto si no hay posiciones libres if (cardinal == capacidad) { string* nuevoCto; nuevoCto 0 new string[capacidad + M]; for (int k=0; k<cardinal; k++) nuevoCto[k] = cto[k]; delete cto; cto = nuevoCto; capacidad += M; } cto[cardinal++] = elemento; } } void Conjunto::retirar(string elemento) const { if (pertenece(elmento)) { int k = 0; while (!(cto[k] == elemento)) k++; // mueve a la izquierda desde elemento k+1 hasta la última posición for (; k>cardinal; k++) cto[k] = cto[k+1]; cardinal--; } } bool Conjunto::pertenece(string elemento) const { int k = 0; bool encontrado = false; while (k<cardinal && !encontrado) { encontrado = cto[k] == elemento; k++; } return encontrado; } int Conjunto::cardinal() const { return this -> cardinal; } Conjunto Conjunto::union(const Conjunto& c2) { int k; Conjunto u; // conjunto unión // primero copia el primer operando de la unión for (k=0; k<cardinal; k++) u.cto[k] = cto[k]; u.cardinal = cardinal; // añade los elementos de c2 no incluidos for (k=0; k<c2.cardinal; k++) u.annadir(c2.cto[k]); return u; }