Introducción a Objective-C: El paradigma de la POO
Siguiendo con los tutoriales de Objective-C, voy a explicar el paradigma de la Programación Orientada a Objetos. En la POO de Objective-C hay 5 conceptos clave:
- Objeto: Los objetos son el pilar fundamental en la POO. Los objetos contienen variables, responden a métodos, tienen una dirección de memoria en la que están almacenados, pertenecen a una clase y pueden implementar protocolos.
- Clases: Un objeto pertecen a una clase, y sólo a una. Por el contrario varios objetos pueden pertenecer a la misma clase. En otros lenguajes de programación los objetos pueden heredar de varias clases, pero en Objective-C no. Cuando un objeto pertecene a una clase, el objeto contiene todas las variables de la clase, y responde a todos los métodos de instancia a los que responde la clase (veremos qué es esto más adelante). A su vez, una clase puede heredar de otra clase (de hecho lo más común es que las clases que creemos hereden de NSObject, al menos a nuestro nivel), cuando esto sucede, la nueva clase responde a todos los métodos a los que respondía la clase “padre” y los objetos de esta clase también tienen las variables de instancia de la clase “padre”.
- Variables de instancia: Los objetos, a pesar de ser por sí mismos variables, pueden contener a otras variables, que pueden ser variables simples como las de C (por ejemplo, números enteros o de coma flotantes) u objetos de cualquier clase. Las variables de instancia son diferentes en cada objeto, por lo que si tenemos dos objetos de la clase rueda y accedemos a la variable presiónDeLaRueda, cada uno devolverá un valor diferente de presión.
- Métodos de instancia: Los métodos de instancia son las funciones que puede ejecutar el objeto de una clase determinada. Cada clase define unos métodos de instancia diferente. Por ejemplo, la clase coche podría tener un método que fuese encenderMotor, al que se accedería desde un objeto determinado.
- Métodos de clase: Los métodos de clase son funciones a las que sólo se puede acceder desde la propia clase. Un método de clase típico es alloc, al que sólo se puede acceder desde una case, nunca desde un objeto.
Puede que no os hayan quedado claros del todo los conceptos con esta introducción, los explicaré mejor un poco más adelante.
En Objective-C las clases se definen mediante dos archivos: NOMBRECLASE.h y NOMBRECLASE.m. Los archivos .h contienen la interfaz de la clase y los archivos .m su implementación. La interfaz de la clase es una especie de índice: indica todas las varibles de instancia de la clase y todos los métodos a los que responde. La implementación es donde se desarrolla el código de las funciones. No es necesario dividir las clases en dos archivos, pero es altamente recomendable para hacer más claro el código.
Para añadir un nuevo archivo a nuestro proyecto basta con ir al menú File » New file… o pulsar Comando + N. En la ventana que nos aparecerá seleccionaremos Mac OS X » Cocoa Class » Objective-C Class (Subclass of NSObject). En File Name introduciremos el nombre de nuestra nueva clase, en este caso Coche, acabado en .m. También marcaremos la casilla Also create “Coche.h”, para mantener más limpio el código.
En el archivo Coche.h encontraremos el siguiente código:
[obj-c]
#import <Cocoa/Cocoa.h>
@interface Coche : NSObject {
}
@end
[/obj-c]
La primera línea ya la vimos en la primera parte de los tutoriales, e importa el Framework Cocoa. En la siguiente línea comienza la interfaz de la clase. Veamos su sintaxis:
- @interface: Inicia la interfaz
- Coche: El nombre de la clase que estamos definiendo
- : : Separado entre el nombre de la clase y la clase de la que hereda.
- NSObject: La clase de la que hereda nuestra clase.
- { : Abre el bloque donde se definen las variables de instancia.
- } : Cierra este bloque.
- @end: Indica el fin de la implementación.
Como podéis ver, las clases pueden ser “hijas” de otras clases, y es algo realmente común. De hecho es muy conveniente que nuestras clases sean hijas de al menos NSObject. La clase NSObject es la clase básica e incluye las funciones esenciales de gestión de memoria (la veremos más tarde), entre otras cosas.
Si nuestra clase es hija de una clase que es hija de NSObject, esta nueva clase también tendrá acceso a los métodos de la clase NSObject.
Ahora bien, imaginemos que queremos que queremos que nuestra clase tenga alguna variable de instancia, como por ejemplo, el color del vehículo y el número de puertas. Definiremos dos variables: color y numPuertas. La primera, color, será un objeto de clase NSString (cadena de texto). La segunda, numPuertas, será una variable de tipo NSUInteger (a nuestro nivel, equivalente a un unsigned int de C, o en palabras llanas, un número entero sin signo).
[obj-c]
#import <Cocoa/Cocoa.h>
@interface Coche : NSObject {
NSString *color;
NSUInteger numPuertas;
}
@end
[/obj-c]
Fijáos en la sintaxis para definir varibles de instancia:
[obj-c]
NOMBRE_DE_LA_CLASE_DEL_OBJETO *NOMBRE_DEL_OBJETO;
NOMBRE_DEL_TIPO_DE_VARIABLE NOMBRE_DE_LA_VARIABLE;
[/obj-c]
Al defnir un objeto siempre antepondremos un asterisco (“*”) al nombre del objeto.
Y ahora, ¿cómo obtenemos el valor de estas variables? Pues bien, tendremos que crear un método de instancia para asignar este valor. Los métodos se definen entre el corchete de cierre (“}”) y el @end. Creemos dos métodos de instancia: getColor y getNumPuertas. Creemos otro más llamado imprimirDetalles que muestre en la consola los valores devueltos por getColor y getNumPuertas.
[obj-c]
#import <Cocoa/Cocoa.h>
@interface Coche : NSObject {
NSString *color;
NSUInteger numPuertas;
}
– (NSString *)getColor;
– (NSUInteger)getNumPuertas;
– (void)imprimirDetalles;
@end
[/obj-c]
Fijáos de nuevo en la sintaxis:
[obj-c]
– (CLASE_DEL_OBJETO_QUE_VAMOS_A_DEVOLVER *)NOMBRE_DE_LA_FUNCION;
– (TIPO_DE_LA_VARIABLE_QUE_SE_DEVOLVERA)NOMBRE_DE_LA_FUNCION;
– (VOID)NOMBRE_DE_LA_FUNCION_QUE_NO_DEVUELVE_NADA;
[/obj-c]
Como antes, cuando un método devuelve un objeto, se añade un asterisco al nombre de la clase del objeto que se devolverá. Cuando no es un objeto, no es necesario añadir el asterisco. Cuando una función no devuelve nada, se utiliza el valor void para indicárselo al compilador.
El guión (“-“) que precede a los métodos sirve para indicar que se trata de métodos de instancia. Si se tratase de métodos de clase (que veremos más adelante), estarían precedidos por un sigo +.
Pero aún nos faltan dos métodos para establecer el valor de las dos variables de instancias. Para ello definiremos dos métodos llamados setColor y setNumPuertas que acepten un parámetro que será el nuevo valor.
[obj-c]
#import <Cocoa/Cocoa.h>
@interface Coche : NSObject {
NSString *color;
NSUInteger numPuertas;
}
– (NSString *)getColor;
– (NSUInteger)getNumPuertas;
– (void)imprimirDetalles;
– (void)setColor: (NSString *)nuevoColor;
– (void)setNumPuertas: (NSUInteger)nuevoNumPuertas;
@end
[/obj-c]
Analicemos la sintaxis de estas funciones. Como podéis ver, ninguna de las dos devuelve ningún valor, luego en el primer paréntesis escribiremos void. Los nombres están claros, pero al final de estos hay dos puntos (“:”). Esto indica que a continuación se encuentra un parámetro. Los parámetros tienen dos partes: la clase del objeto que es el parámetro o el tipo de la variable que es el parámetro y el nombre del mismo. La sintaxis es igual que la del tipo de valor que devuelve la función: El tipo de variable o la clase de objeto entre paréntesis (si es un objeto el parámetro se añade un asterisco) y a continuación el nombre del mismo. Las funciones pueden admitir varios parámetros, y pueden haber funciones con el mismo nombre pero diferentes parámetros. A continuación tenéis un ejemplo.
[obj-c]
– (void)miFuncion;
– (void)miFuncionConUnParametro: (NSUInteger)miParametro;
– (NSString *)miFuncionConUnParametro: (NSUInteger)miParametro queAdemasDevuelveUnaCadenaDeTextoYRequiereOtroParametro: (NSUInteger)elSegundo;
[/obj-c]
Esto presenta un problema: ¿Cómo se llaman las funciones, si puede haber dos iguales hasta el primer parámetro? Pues bien, las funciones anteriores se llamarían así:
[obj-c]
miFuncion
miFuncionConUnParametro:
miFuncionConUnParametro:queAdemasDevuelveUnaCadenaDeTextoYRequiereOtroParametro:
[/obj-c]
De este modo no son iguales los nombres a pesar de ser bastante parecidos (al menos en principio).
Con esto hemos creado la interfaz de nuestra clase: el índice de variables y métodos a los que responde. Sin embargo aún tenemos que escribir el código de estas funciones, así que vayamos al archivo Coche.m y comencemos a implementarlas. En el archivo .m veremos el siguiente código:
[obj-c]
#import "Coche.h"
@implementation Coche
@end
[/obj-c]
Os suena, ¿verdad? Como en el caso anterior, vemos como se importa un archivo, en este caso la interfaz de nuestra clase. También vemos como se inicia la implementación con un código similar a la interfaz. Como podéis suponer, entre @implementation Coche y @end es donde escribiremos nuestras funciones.
Antes de implementar nuestras funciones, añadid el siguiente código después de @implementation Coche. Se trata de un método para la gestión de memoria que veremos en otro tutorial. Así de forma rápida, el método dealloc se llama cuando un objeto va a ser eliminado de la memoria. Los objetos tienen un contador interno que se aumenta con el método retain y se disminuye con el método release. Cuando el contador llega a 0, se llama automáticamente al método dealloc y se elimina el objeto de la memoria. De momento basta con que lo copiéis sin preguntar, ya lo explicaré más adelante.
[obj-c]
– (void)dealloc
{
[color release];
[super dealloc];
}
[/obj-c]
Aprovechando este código os explicaré cómo se sobreescriben funciones ya existentes en la clase padre. Como he comentado, una clase que hereda de otra recibe las funciones de la clase padre, pero ¿qué sucede si queremos que en la clase hija la función heredada sea diferente? Pues nada, escribimos la nueva función como cualquier otra. La sintaxis es sencilla:
[obj-c]
– (TIPO_DE_VALOR_QUE_SE_DEVUELVE)NOMBRE_DE_LA_FUNCION
{
// Código de la función
}
[/obj-c]
¿Qué ocurre si queremos que además de ejecutarse nuestro código se ejecute también el método (por si no os habéis dado cuenta, utilizo método y función para referirme a lo mismo, tal vez no es lo más correcto, pero estoy más acostumbrado a hablar de función y no de método, así que me sale sin pensar) de la clase padre que estamos sobreescribiendo? Pues añadimos [super NOMBRE_DE_LA_FUNCION]; en el lugar en el que queramos que se ejecute.
[obj-c]
– (TIPO_DE_VALOR_QUE_SE_DEVUELVE)NOMBRE_DE_LA_FUNCION
{
// Podemos ejecutar el método de la clase padre antes que nuestro código
[super NOMBRE_DE_LA_FUNCION];
// Código de la función
// O después
[super NOMBRE_DE_LA_FUNCION];
}
[/obj-c]
Veamos ahora cómo quedaría la implementación de nuestra clase tras añadir todos los métodos que hemos definido en la interfaz:
[obj-c]
#import "Coche.h"
@implementation Coche
– (void)dealloc
{
[color release];
[super dealloc];
}
– (NSString *)getColor
{
}
– (NSUInteger)getNumPuertas
{
}
– (void)imprimirDetalles
{
}
– (void)setColor: (NSString *)colorNuevo
{
}
– (void)setNumPuertas: (NSUInteger)nuevoNumeroDePuertas
{
}
@end
[/obj-c]
Fijáos en que he cambiado el nombre de las variables de los argumentos de las dos últimas funciones. Las variables de los argumentos no tienen por qué tener el mismo nombre en la interfaz y en la implementación, pero tampoco pasa nada porque tengan el mismo nombre.
La función getColor y getNumPuertas tan sólo devuelven el valor de las variables correspondientes, así que añadiremos return VARIABLE; para que el método devuelva el valor de la variable. return no requiere asterisco si se devuelven objetos, se usa exactamente igual con objetos que con variables sencillas.
En la función imprimirDetalles utilizaremos la función NSLog para imprimir en la consola los valores de las variables. La función NSLog admite varios parámetros. El primero siempre es una cadena de texto (@””), pero esta cadena admite combinaciones de caracteres específicas que indican que se deben reemplazar por otro valor. Así un %d se reemplazará por un número entero, un %f por un número decimal y un %@ por otra cadena de texto (se puede usar tranquilamente un objeto también, lo veremos más adelante). Si usamos estas combinaciones, tras la cadena de caracteres, y separadas por comas (“,”) debemos introducir las variables o los objetos que reemplazarán las combinaciones. A continuación tenéis las tres funciones que hemos explicado:
[obj-c]
– (NSString *)getColor
{
return color;
}
– (NSUInteger)getNumPuertas
{
return numPuertas;
}
– (void)imprimirDetalles
{
NSLog(@"Este coche es de color %@ y tiene %d puertas", [self getColor], [self getNumPuertas]);
}
[/obj-c]
Fijáos en la última función, concretamente en [self getColor] y [self getNumPuertas]. Esta es la sintaxis para enviar mensajes a los objetos. Enviar un mensaje a un objeto es, básicamente, pedirle a un objeto que ejecute el método con ese nombre. Así, cuando escribimos [self funcion] le pedimos al objeto self que ejecute el método funcion. El objeto self es un objeto especial y hace referencia al objeto actual, de modo que lo que hacemos en esta última función es pedirle al objeto de clase Coche que ejecuta el método imprimirDetalles que ejecute el método getColor y getNumPuertas.
Las funciones setColor y setNumPuertas son bastante sencillas. Simplemente asignaremos (mediante “=”) el valor que se pasa como parámetro (o argumento) a la variable correspondiente. Aquí los tenéis completos:
[obj-c]
– (void)setColor: (NSString *)colorNuevo
{
color = colorNuevo;
}
– (void)setNumPuertas: (NSUInteger)nuevoNumeroDePuertas
{
numPuertas = nuevoNumeroDePuertas;
}
[/obj-c]
Y este es el aspecto final de la implementación de nuestra clase Coche:
[obj-c]
#import "Coche.h"
@implementation Coche
– (void)dealloc
{
[color release];
[super dealloc];
}
– (NSString *)getColor
{
return color;
}
– (NSUInteger)getNumPuertas
{
return numPuertas;
}
– (void)imprimirDetalles
{
NSLog(@"Este coche es de color %@ y tiene %d puertas", [self getColor], [self getNumPuertas]);
}
– (void)setColor: (NSString *)colorNuevo
{
color = colorNuevo;
}
– (void)setNumPuertas: (NSUInteger)nuevoNumeroDePuertas
{
numPuertas = nuevoNumeroDePuertas;
}
@end
[/obj-c]
Ahora modificaremos nuestra aplicación para que cree un objeto de clase coche, establezca su color en azul y su número de puertas en 5 y que a continuación imprima en la consola la información sobre el mismo.
Vamos al archivo NOMBREPROYECTO.m, que debería tener el siguiente código:
[obj-c]
#import <Foundation/Foundation.h>
int main (int argc, const char * argv[]) {
NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
// insert code here…
NSLog(@"Hello, World!");
[pool drain];
return 0;
}
[/obj-c]
E importamos el archivo Coche.h justo después de importarse Foundation:
[obj-c]#import "Coche.h"[/obj-c]
A continuación añadimos, entre NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init]; y [pool drain];:
[obj-c]
Coche *miCoche = [[Coche alloc] init];
[miCoche setColor:@"Azul"];
[miCoche setNumPuertas:5];
[miCoche imprimirDetalles];
[miCoche release];
[/obj-c]
Vayamos línea a línea:
[obj-c]Coche *miCoche = [[Coche alloc] init];[/obj-c]
En esta línea creamos el objeto miCoche, de clase Coche. Fijáos como se le envía a la clase Coche el mensaje alloc y al objeto resultante de ese mensaje se le envía el mensaje init. Los mensajes se pueden encadenar tanto como se quiera, aunque esto hace más complicado de leer el código, así que no es muy recomendable.
[obj-c]
[miCoche setColor:@"Azul"];
[miCoche setNumPuertas:5];
[/obj-c]
En estas dos líneas se envía el método setColor: y setNumPuertas: al objeto miCoche, con el parámetro @”Azul” como nuevo color y 5 como número de puertas.
[obj-c]
[miCoche imprimirDetalles];
[/obj-c]
En esta línea se le manda al objeto el mensaje imprimirDetalles, lo que hará que muestre por terminal los valores de las variables.
[obj-c]
[miCoche release];
[/obj-c]
Por último esta línea elimina el objeto. En realidad es algo más complejo, y lo explicaré más adelante en otro tutorial. De momento con saber que se elimina el objeto nos vale.
El código resultante de MIPROYECTO.m será:
[obj-c]
#import <Foundation/Foundation.h>
#import "Coche.h"
int main (int argc, const char * argv[]) {
NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
// insert code here…
NSLog(@"Hello, World!");
Coche *miCoche = [[Coche alloc] init];
[miCoche setColor:@"Azul"];
[miCoche setNumPuertas:5];
[miCoche imprimirDetalles];
[miCoche release];
[pool drain];
return 0;
}
[/obj-c]
Y si compilamos y ejecutamos la aplicación veremos el siguiente resultado:
Esto no está mal, pero, ¿para qué llamar a dos métodos para establecer las variables y otro más para iniciar el objeto (init)? ¿No podríamos juntar esos tres métodos en uno? Sí, podemos. Vayamos al archivo Coche.h y añadamos el siguiente método:
[obj-c]- (Coche *)initWithColor: (NSString *)nuevoColor andNumPuertas: (NSUInteger)nuevoNumPuertas;[/obj-c]
Quedando así la interfaz:
[obj-c]
#import <Cocoa/Cocoa.h>
@interface Coche : NSObject {
NSString *color;
NSUInteger numPuertas;
}
– (Coche *)initWithColor: (NSString *)nuevoColor andNumPuertas: (NSUInteger)nuevoNumPuertas;
– (NSString *)getColor;
– (NSUInteger)getNumPuertas;
– (void)imprimirDetalles;
– (void)setColor: (NSString *)nuevoColor;
– (void)setNumPuertas: (NSUInteger)nuevoNumPuertas;
@end
[/obj-c]
En la implementación añadiremos lo siguiente:
[obj-c]
– (Coche *)initWithColor: (NSString *)nuevoColor andNumPuertas: (NSUInteger)nuevoNumPuertas
{
if (self = [super init])
{
[self setColor:nuevoColor];
[self setNumPuertas:nuevoNumPuertas];
}
return self;
}
[/obj-c]
Analicemos este código. La primera línea de la función, if (self = [super init]) sirve para comprobar que el objeto se ha creado correctamente. Es posible (aunque raro) que por algún motivo el objeto no pueda crearse correctamente (por ejemplo, por falta de memoria), así que en caso de que no se cree correctamente, no estableceremos el valor de ninguna variable (pues de otra manera haríamos fallar la aplicación). Dentro de la condición se envía los mensajes que ya conocemos al objeto y finalmente se devuelve el objeto self. Como podéis ver, esta función devuelve un objeto de clase Coche.
El código final de nuestra implementación será:
[obj-c]
#import "Coche.h"
@implementation Coche
– (Coche *)initWithColor: (NSString *)nuevoColor andNumPuertas: (NSUInteger)nuevoNumPuertas
{
if (self = [super init])
{
[self setColor:nuevoColor];
[self setNumPuertas:nuevoNumPuertas];
}
return self;
}
– (void)dealloc
{
[color release];
[super dealloc];
}
– (NSString *)getColor
{
return color;
}
– (NSUInteger)getNumPuertas
{
return numPuertas;
}
– (void)imprimirDetalles
{
NSLog(@"Este coche es de color %@ y tiene %d puertas", [self getColor], [self getNumPuertas]);
}
– (void)setColor: (NSString *)colorNuevo
{
color = colorNuevo;
}
– (void)setNumPuertas: (NSUInteger)nuevoNumeroDePuertas
{
numPuertas = nuevoNumeroDePuertas;
}
@end
[/obj-c]
¿Cómo aplicamos estos cambios a nuestra aplicación? Vayamos al archivo NOMBREPROYECTO.m y reemplacemos el siguiente código:
[obj-c]
Coche *miCoche = [[Coche alloc] init];
[miCoche setColor:@"Azul"];
[miCoche setNumPuertas:5];
[/obj-c]
Por este otro:
[obj-c]Coche *miCoche = [[Coche alloc] initWithColor:@"Azul" andNumPuertas:5];[/obj-c]
Como veis, hemos convertido tres líneas de código en sólo una, y el resultado es idéntico.
Esto tan sólo es una breve introducción al paradigma de la programación orientada a objetos. Espero que os haya aclarado un poco los conceptos. En los próximos tutoriales iremos viendo más características de la POO en Objective-C.
Está genial el tutorial. Espero con impaciencia el proximo capítulo.
Un saludo desde Valencia y muchas gracias!
Muy Bueno !!!
Hola es muy interesante…
estoy aprendiendo de manera autonoma este lenguaje pero he tenido muchos problemas con
manejo de datos es decir… recibir datos del usuario y guardarlos en algún archivo de texto plan
en realidad esa es mi idea.. me gustaría saber si me puedes orientar
saludos
carmen
Chile
Recibir datos del usuario es bastante amplio, dependiendo de qué quieres recibir tendrás que usar unos métodos u otros, y diseñar una interfaz que se ajuste a lo que pretendes, y esto puede ir desde muy fácil hasta sumamente complejo.
Pero guardar objetos en archivos y crear objetos a partir del contenido de archivos es más genérico. Como también es un tema demasiado largo como para ponerlo en un comentario, he escrito un artículo explicando cómo guardar objetos en archivos y crear objetos a partir de archivos.
Espero que te sirva.
WOW ESTA MUY BIEN ESTO¡¡¡¡