Skip to content

Latest commit

 

History

History
774 lines (484 loc) · 29.4 KB

README.md

File metadata and controls

774 lines (484 loc) · 29.4 KB

Curso de Programación Orientada a Objetos

Apuntes y código del Curso de Programación Orientada a Objetos de Platzi

📚 Módulo 1. Bienvenida e Introducción

  • ¿Por qué aprender Programación Orientada a Objetos?

    • Porque permite programar más rápido, esto se debe a que se hace un análisis previo de lo que se esta realizando.
    • Al saber analizar problemas y entender la programación orientada a objetos se puede avanzar y dejar de ser un programador Junior. Esto se debe a que varias de las preguntas más frecuentes de los reclutadores son ¿Qué es encapsulamiento, abstracción, herencia, polimorfismo?
    • Saber POO permite dejar ala mala práctica de copiar y pegar código y tomar el control del proyecto y el código.

    En el curso se abordan 3 aspectos fundamentales:

    • Analizar:
      • Observación
      • Entendimiento
      • Lectura: "leer" muy bien la situación del problema
    • Plasmar:
      • Diagramas: plasmar el análisis de una forma gráfica
    • Programar:
      • Desarrollar lo que se acaba de diagramar usando algún lenguaje de programación.
  • ¿Qué resuelve la Programación Orientada a Objetos?

    Permite resolver muchos de los "huecos" que tiene la programación estructurada. Entre los problemas que resuelve están:

    • El código muy largo.
    • Evita que si algo falla todo se rompa.
    • Reduce la dificultad de mantener el código.
    • Evita el código espagueti (demasiadas sentencias de control anidadas).
  • Paradigma Orientado a Objetos

    La filosofía de Orientación a Objetos surge a partir de la necesitad de los programadores de plasmar soluciones a problemas en el código. Se basa en un análisis del problema que se quiere abordar para posteriormente empezar a programar basándose a las conclusiones llegadas en dicho análisis. Esto ahorra el problema de no saber por donde empezar a resolver un problema con software.

    En dicho análisis se observan los problemas en forma de objetos para después llevar todo a una solución en código.

    Paradigma: teoría que suministra la base y modelo para resolver problemas.

    El Paradigma de Programación Orientada a Objetos se compone de 4 elementos:

    • Clases
    • Propiedades
    • Métodos
    • Objetos

    Además de esos 4 elementos , la POO también tiene 4 "Pilares":

    • Encapsulamiento
    • Abstracción
    • Herencia
    • Polimorfismo
  • Lenguajes Orientados a Objetos

    Algunos lenguajes orientados a objetos son:

    • Java: orientado a objetos naturalmente, de hecho POO es su paradigma principal. Se usa principalmente en desarrollo de Apps para Android y Backend en servidores, APIs, etc. Extensión de archivo → .java
    • PHP: es un leguaje interpretado creado para programación Web. Extensión de archivo → .php
    • Python: esta diseñado para ser fácil de usar, tiene una sintaxis amigable e intuitiva. Se puede usar en diversos campos como Backend, Data Science e Inteligencia Artificial. Extensión de archivo → .py
    • JavaScript: es un lenguaje interpretado. Esta orientado a objetos pero basándose en prototipos. Es un leguaje pensado para Web y actualmente puede usarse tanto en el Frontend como en el Backend. Extensión de archivo → .js
    • C#
    • Ruby
    • Kotlin
  • Diagramas de Modelado

    Los diagramas de modelado son una especie de gráficos que sirven de intermediarios entre el análisis que se hace del problema que se quiere solucionar y el código que se va a generar para solucionar dicho problema.

    Hay 2 opciones de diagramas de modelado disponibles:

    • OMT: (Object Modeling Techniques). Creado en 1991. Es una metodología para el análisis orientado a objetos. Lo que propone es que una vez se hace el análisis y se tienen identificados los objetos, atributos, métodos, etc; se pasa a plasmar cada objeto en un recuadro, se ponen todos los atributos y métodos dentro de dicho recuadro y luego se usa una serie de conectores para establecer las relaciones entre los diferentes objetos.

      OMT

      Actualmente esta en desuso y no se recomienda emplearlo en ningún proyecto.

    • UML: (Unified Modeling Language - Lenguaje de modelado unificado). Creado en 1997. Se podría resumir como la versión moderna de OMT, pues unifica las bases y técnicas de este y las mejora. Es el método para crear diagramas de modelado que se usa en la actualidad. Tiene muchas más capacidades que OMT, entre las cuáles están:

      • Clases
      • Casos de uso
      • Objetos
      • Actividades
      • Iteración
      • Estados
      • Implementación
      • Etc

      Aprender más de UML

      UML

      Aprender UML es importante pues permite plasmar de forma gráfica lo que se va a implementar en el código y como esta construido el proyecto.

      Algunas herramientas para trabajar con UML:

📚 Módulo 2. Orientación a Objetos

  • Objetos

    Cuando hay un problema que se quiere resolver con Software lo primero que hay que hacer es identificar los Objetos.

    • ¿Cómo identificar a los Objetos? Los objetos son aquellos que tienen propiedades y comportamientos. Siempre son sustantivos. Pueden ser físicos (un Usuario por ejemplo) o conceptuales, es decir, que no existen físicamente, son simbologías de procesos que lleva a cabo el software que se esta construyendo (una de Sesión de Usuario por ejemplo).
    • ¿Qué son las Propiedades? También llamadas atributos. Son características o propiedades que describen a un objeto (por ejemplo: nombre, tamaño, forma, estado). Los tributos también son sustantivos.
    • ¿Qué son los comportamientos? También se les conoce como métodos. Son todas las operaciones/acciones del objeto (por ejemplo: login(), logout(), makeReport()). Suelen ser verbos o sustantivo y verbo.
  • Abstracción y Clases

    Una Clase es el modelo sobre el cuál se construye un objeto. Para definirlo de una forma más sencilla de entender, se puede pensar en un una clase como un molde que se usa para crear objetos.

    En este punto se analizan Objetos, se establecen todos sus atributos y métodos y en base a eso se crean Clases. Este proceso de abstraer los datos de un objeto para generar una clase (molde) se le conoce como Abstracción. En resumidas cuentas, la abstracción es ese proceso de analizar los objetos para identificar sus atributos y métodos comunes para en base a eso crear un molde llamando Clase.

  • Modularidad

    Consiste en dividir un sistema en partes más pequeñas, cada una de esas partes se llaman módulos. Cada uno de esos módulos funciona de manera independiente, pero juntos conforman el sistema completo.

    La modularidad permite hacer más fácil el mantenimiento y la escalabilidad de un proyecto, pues mientras en las programación estructurada se tiene un sólo archivo muy grande en el que esta toda la lógica del sistema y en el que un error puede detener la ejecución de todo el programa, en la programación orientada a objetos las diferentes funciones de un sistema se dividen en módulos independientes, por lo que a la hora de solucionar bugs sólo hay que ubicar el módulo que esta fallando y repararlo sin tener que afectar al resto del código; y si se quieren añadir más funcionalidades al proyecto solo hay que crear nuevos módulos e integrarlos dentro del sistema.

    Ventajas de la modularidad:

    • Reutilizar código.
    • Se evitan colapsos.
    • Código más mantenible.
    • Mejor legibilidad en el código
    • Rápida resolución de problemas

    Las clases son el elemento que permite la modularidad en la POO, pues permiten dividir el problema en distintas clases/módulos y a su vez separar dichas clases en diferentes archivos.

  • Analizando Uber en Objetos

    • Análisis del problema:

      Análisis del funcionamiento de Uber

    • Análisis de los Objetos:

      User
      Route
      Driver
      UberX
      UberPool
      UberBlack
      UberVan
      Trip
      Cash
      Card
      Paypal

📚 Módulo 3. Programación Orientada a Objetos. Análisis

  • Clases en UML y su sintaxis en código

    Las clases en UML son representadas por un rectángulo. En la parte superior se pone el nombre de la clase (verde), después los atributos (morado) y por último los comportamientos/métodos (azul).

    Definir Clases enUML

    • Definir Clases en Java:

      class Person {
          String name = ""; /* Atributo */
          void walk() {} /* Comportamiento / método */
      }
    • Definir Clases en Python:

      class Person:
          name = ""; # Atributo
          def walk(): # Comportamiento / método
    • Definir Clases en JavaScript:

      function Person() {} /* "Clase" */
      
      Person.prototype.walk = function() { 
           /* Comportamiento / método */
      }
    • Definir Clases en PHP:

      class Person {
          $name = ""; # Atributo
          function walk() {} # Comportamiento / método
      }
  • Modelando nuestros objetos Uber

    Este es el diagrama de los diferentes Objetos vistos en clase junto con sus respectivos atributos

    Diagrama UML Objetos Uber

  • ¿Qué es la herencia?

    "Don´t repeat yourself"

    Es una filosofía que promueve la reducción de la duplicación en programación.

    Toda pieza de información nunca debería ser duplicada pues esto incrementa la dificultad en los cambios y la evolución del proyecto. En cambio se debe procurar reutilizar código.

    La herencia es una pieza fundamental para la reutilización de código, permite crear nuevas clases a partir de otras. Cuando se detecten elementos duplicados, se hace una abstracción para generar una Clase mas general de la que otras clases hereden todas esas características duplicadas.

    "Cuando se detectan características y comportamientos iguales se debe realizar una abstracción."

    En la herencia se establece una relación padre e hijo. A la clase padre se le conoce como súper clase y a sus clases hijas se les llama subclases.

    Súperclases y subclases

    En el siguiente ejemplo hay varios atributos(rojo) y comportamientos(verde) que se están repitiendo en varias clases.

    Ejemplo de herencia parte 1

    A partir de esto se puede crear una súperclase(SeleccionFutbol) que agrupe esos atributos y comportamientos que se repiten y hacer que las clases Futbolista, Entrenador y Masajista los hereden.

    Ejemplo de herencia parte 2

    De esta forma todos los atributos y métodos de la clase padre serán heredados a las clases hijas, lo que ahorra el tener que escribir en código cada uno de esos atributos y comportamientos para cada Clase.

    Otra razón por la que se puede aplicar herencia es por lógica de negocio, es decir clases que si bien no comparten elementos, debido a la forma en la que funcione el software se crea un Clase padre que herede a un grupo de Clases hijas que están relacionadas entre si.

  • Aplicando Herencia a nuestro proyecto Uber

    Diagrama UML Uber

📚 Módulo 4. Clases, Objetos y Método Constructor

  • Definiendo clases en Java y Python

    Main.java, main.py, index.js → Es el punto de entrada del programa, aquí iniciará la ejecución del mismo, se importarán otras clases/módulos etc. Si el programa fuera una una especie de árbol este archivo sería como la punta de la raíz desde la cuál empezará a ejecutarse todo.

  • Definiendo Clases en JavaScript

    Hasta el estándar EcmaScript 6 no existía el concepto de clases, por lo que estas se creaban a partir de prototipos que son una forma de representar clases a partir de la declaración de una función.

    function Account() { /* Account → nombre "Clase" */
        this.id; /* Atributo */
        this.name;
        this.document;
        this.email;
        this.password;
    }
  • Objetos, método constructor y su sintaxis en código

    Los Objetos son instancias de la clase. Son el resultado de lo que se "moldeo" en la clase.

    • Declarar Objetos en Java

      Person person = new Person();
    • Declarar Objetos en Python

      person = Person()
    • Declarar Objetos en JavaScript

      var person = new Person();
    • Declarar Objetos en PHP

      $person = new Person();

      Método Constructor: es un método que le da un estado inicial al objeto. Debe llamarse igual que la clase seguido de paréntesis. Dentro de esos paréntesis se ponen los parámetros mínimos para inicializar el objeto.

      Crear métodos constructores en diferentes lenguajes:

      • Java

        public Person(String name) { /* String name → Parámetro */
            this.name =  name;
        }
      • JavaScript

        function Person (name) { /* name → Parámetro */
            this.name = name
        }
      • Python

        def __init__(self, name): # name → Parámetro
            self.name = name
      • PHP

        public function _construct($name) { # $name → Parámetro
            $this -> name = name;
        }

      Llamar a un Objeto en diferentes lenguajes

      • Java

        Person person = new Person("Andrés");
      • JavaScript

        var person = new Person("Andrés");
      • Python

        person = Person("Andrés")
      • PHP

        $person = new Person("Andrés);
  • Objetos. Dando vida a nuestras clases en Java y Python

    Java, entre otros lenguajes, provee un método constructor por defecto para los objetos que se creen.

    Car car = new Car();

    Una vez se ha instanciado el objeto se puede acceder a sus atributos usando notación de punto.

    car.license = "AMQ258";

    También pueden crearse métodos dentro de la clase y acceder a ellos usando notación de punto.

    public class Car {
    	/* Atributos */
            
    	void printDataCar() { /* Método */
            
            System.out.println("Car License: " + license);
            System.out.println("Car Driver: " + driver);
            System.out.println("Car Passengers: " + passenger);
    
        }
    }

    Lllamar al método printDataCar() con la sintaxis de notación de punto

    car.printDataCar();

    En el caso de Python, primero hay que importar el módulo con la Clase

    from car import Car # car → nombre de archivo sin la extención. Car → Nombre de la clase

    Luego se instancia el objeto, y una vez hecho eso ya se puede acceder a los atributos del objeto usando la notación de punto.

    car = car()
    car.licese = "AMS286"
    car.driver = "Felipe López"
    car.passengers = 4
    
    # Para imprimir los atributos del objeto
    print(vars(car))
  • Declarando un Método Constructor en Java y JavaScript

    Método constructor en Java:

    public Car(String license, Account driver) { /* String license, Account driver → Parámetros obligatorios para crear el objeto. */
    	this.license = license; /* this.license → Hace resferencia a la variable de la clase. license → Hace referencia a la variable del método */ 
    }

    Los parámetros del método constructor pueden tener cualquier nombre pero es una buena práctica nombrarlos igual que los atributos de la clase.

    Los parámetros que se le pasan al método son variables locales, para que el valor que se les pase se aplique a los atributos de la Clase hay que llamar al atributo con this.

    Para instanciar un Objeto, en Java la sintaxis sería la siguiente:

    Car car = new Car("AMQ258", new Account("Andrés López", "AND123"));

    Se pone el tipo de la Clase seguido del nombre de la instancia, y luego con new se crea dicha instancia y se pasan los parámetros necesarios para crear el objeto deseado.

    Método Constructor en JavaScript

    function Car(license, driver) { /* license, driver → Parámetros para crear el objeto */
    
        this.id;
        this.license = license;
        this.driver = driver;
        this.passenger;
        
    }

    En el caso de JavaScript la función que define la Clase también sirve como método constructor, por lo que sólo hay que pasarle a dicha función los parámetros necesarios para crear el objeto.

    El funcionamiento de this en JavaScript es igual que en Java, pues este también hace referencia a la Clase para así poder acceder a sus atributos usando la notación de punto.

    Los métodos en JavaScript se crean fuera de la Clase y usando Prototipos.

    Car.prototype.printDataCar = function () { /* Método de la Clase Car */
        
        console.log(this.driver)
        console.log(this.driver.name)
        console.log(this.driver.document)
         
    }

    Para instanciar un Objeto, en JavaScript la sintaxis sería la siguiente:

    var car = new Car("AW572", new Account("Andrés López", "ANDL580"))

    Se usa una variable donde se va crear el Objeto, y luego con new se crea el objeto y se pasan los parámetros necesarios para su creación.

  • JavaScript orientado a objetos, lo más nuevo

    Desde EcmaScript 6 existen en JavaScript nuevas formas de trabajar con POO, entre ellas una nueva forma de crear "Clases" y el Método Constructor, no obstante estos no dejan de ser prototipos.

    La nueva sintaxis para crear una "Clase" con su respectivo método constructor es:

    class Car { /* Crea la Clase */
        
        constructor(license, driver) { /* Crea el método constructor */
            
            this.id
            this.license = license
            this.driver = driver
            this.passengers
            
        }
        
        printDataCar() { /* Crea un método para la Clase Car */
            
            console.log(this.driver)
            console.log(this.driver.name)
            console.log(this.driver.document)
            
        }
        
    }

    A la hora de instanciar un objeto no hay cambios, se sigue usando la misma sintaxis de antes.

  • Declarando un Método Constructor en Python

    En Python existe un concepto llamando Métodos Mágicos, estos son básicamente métodos que se auto-ejecutan bajo ciertas condiciones. Dentro de esta familia de métodos esta el método constructor.

    __init__ es uno de estos métodos mágicos y lo que hace es personalizar la instanciación de la Clase o en palabras más sencillas, lo que este dentro de este objeto será lo primero que se ejecute cuando se instancia un objeto; esto es muy útil para por ejemplo, inicializar atributos de la Clase cuando se crea un Objeto.

    class Account:
        id       = int
        name     = str
        document = str
        email    = str
        password = str
    
        def __init__(self, name, document):
            self.name     = name
            self.document = document

    Otro de estos métodos mágicos es __new__ que es el método que construye el objeto como tal.

    Otra peculiaridad de Python es que en lugar de this se usa la palabra reservada self, aunque esta hace los mismo que this y hace referencia a la Clase para así poder acceder a sus atributos/métodos.

📚 Módulo 5. Herencia

  • Aplicando herencia en lenguaje Java y PHP

    Implementando la Herencia en diferentes lenguajes:

    • Java: la herencia se define de forma explicita usando la palabra reservada extends seguida de la Clase padre.

      class CarBasic extends Car

      Luego de esto hay que crear un método constructor dentro de la Clase hija que coincida con el de la Clase padre. Luego se usa super() para llamar al método constructor de la Clase padre (súper Clase).

      super hace referencia a la Clase padre, de hecho con super se puede acceder a los atributos/métodos de la Clase padre usando notación de punto.

      Luego se usa this para pasar el valor de los parámetros a los atributos

      public CarBasic(String license, Account driver, String brand; String model) { /* Los parámetros incluyen los del constructor de la clase padre más los parámetros propios de la clase hija */
      	super(license, driver);
          this.brand = brand;
          this.model = model;
      }
    • Python: la herencia se define pasando la Clase padre como parámetro al instanciar la Clase Hija.

      class Student(Person):
    • JavaScript: aquí se toma la "clase" hija (prototipo) y la instancia cómo si fuera la "Clase" padre.

      student.prototype = new Person();
    • PHP: se usa la misma sintaxis de Java para definir la herencia.

      <?php
      
      require_once('car.php'); /* Importa el archivo que contiene la case padre */
      
      class CarBasic extends Car {
      
          public $brand;
          public $model;    
      
      }
      
      ?>

      Luego se declaran los atributos de la Clase hija, después se crea el método constructor. En el caso de PHP en lugar de usar super() como en Java se usa parent::__construct() para acceder al constructor de la Clase padre.

      Después de esto se puede usar $this para acceder a los atributos de la Clase y asignarles el valor de los parámetros del método constructor.

      public function __construct($license, $driver, $brand, $model) { /* Método Constructor, recibe los parámetros d ela Clase padre junto con los de la Clase hija */
      
        parent::__construct($license, $driver, $brand. $model);
        $this->brand = $brand;
        $this->model = $model;    
      }

      Para importar e instanciar un objeto la sintaxis sería la siguiente:

      <?php
      
      /* Se importa el archivo con la Clase requerida */
      require_once('uberX.php');
      
      $uberX = new UberX("DFJ159", new Account("Andrés López", "AND741G779"), "Chrevrolet", "Spark");
      $uberX->printDataCar();
      
      ?>
  • Aplicando herencia en lenguaje Python y JavaScript

    En el caso de Python para definir la herencia entre clases primero hay que importar la súper clase.

    from car import Car # car → el nombre del archivo que contiene la Clase pero sin extención. Car → El nombre de la Clase que se quiere importar.

    Luego se define la herencia usando la siguiente sintaxis:

    class UberX(Car): #UberX → (Sub Clase) La clase sobre la que se va a aplicar la herencia. Car → (Súper Clase) La Clase que se va a heredar.

    Después se crea el método constructor con los parámetros requeridos para crear el objeto, incluidos los parámetros de la súper Clase. Luego se usa super para acceder al Método Constructor de la súper Clase y finalmente con self se acceden a los atributos de sub Clase para asignarles el valor que se pase por parámetro en el método constructor.

    def __init__(self, license, driver, brand, model): # Método Constructor con los parámetros para crear el objeto
        super.__init__(license, driver)
        self.brand = brand
        self.model = model

    Nota: si al ejecutar el programa aparece un mensaje de error como este: __int__ requires a super object but received a str hay que hacer el siguiente cambio al acceder al constructor de la súper Clase:

    super(UberX, self).__init__(license, driver)

    En el caso de JavaScript, si se usa la nueva sintaxis que ofrece EcmaScript 6 se usa extends para definir la herencia. Luego se crea el constructor y se definen los atributos para crear el objeto.

    class UberX extends Car {
        constructor(license, driver, brand, model) {
            this.brand = brand;
            this.model = model;
        }
    }

    Para crear una instancia de un objeto hay que asegurarse de tener el archivo con la súper Clase vinculado en el HTML, después se crea una instancia del objeto en JavaScript con la siguiente sintaxis:

    var uberX = new UberX("SXC53", new Account("Felipe Molina", "UISD6JASD6T832"), "Ford", "Focus")
  • Otros tipos de Herencia

    Se les puede llamar familias a las Clases que estén siendo heredadas. De hecho una forma en la que durante el análisis del proyecto se pueden creen ciertas Clases es por el hecho de hay Clases que lógicamente deben estar en una familia, a pesar de que no compartan código ni existan entre ellas atributos/métodos que se repitan.

📚 Módulo 6. Encapsulamiento

  • Encapsulamiento

    Permite hacer que un dato permanezca inviolable, inalterable; esto se consigue asignándole un modificador de acceso. Al encapsulamiento también se le conoce como ocultación de información. Los modificadores de acceso son los siguientes:

    • public: es el más permisivo de todos. Puede ser accedido por cualquier elemento.
    • protected: puede ser accedido a nivel de la Clase, los paquetes de la Clase y las subclases.
    • default: permite el acceso a nivel de Clases internas y paquetes. No puede ser accedido a nivel de herencia.
    • private: es el más restrictivo de todos. Sólo puede ser accedido a nivel de Clases. Sólo puede ser modificado y accedido dentro de la misma Clase.

    Estos modificadores se pueden aplicar a Clases, atributos y métodos.

    Con el encapsulamiento se consigue no sólo ocultar un dato sino también protegerlo.

    Entre los beneficios del encapsulamiento estas los siguientes:

    • Permite controlar como se acceden y modifican los datos.
    • Se puede crear un código más flexible y fácil de cambiar frente a futuros requerimientos.
    • Se pueden modificar partes del código sin que esto afecte al resto del software.
    • Permite mantener la integridad de los datos.
  • Encapsulando atributos en Java

    Cuando no se define un modificador de acceso, Java internamente asigna el modificador default.

    Añadir un modificador de acceso en Java:

    private Integer passenger;

    Getters y Setters: son métodos que permiten acceder y modificar un dato que esta protegido por un modificador de acceso.

    • Creando un gettter:

      public Integer getPassenger() {   // Integer → Tipo de dato que va a retornar el método
          return passenger;
      }
    • Creando un setter:

      public void setPassenger(Integer passenger) { // void → Indica que el método no va a retornar nada.
          this.passenger = passenger;
      }

      El getter permite acceder al dato protegido, mientras que el setter permite modificar dicho dato. Al ser ambos métodos de tipo public estarán disponibles desde cualquier lugar. A la hora de llamar estos métodos, no cambia nada, el proceso es el mismo que al llamar a cualquier otro método.

      Usar getters y setter es particularmente útil no sólo para proteger datos, sino también para validarlos:

      public void setPassenger(Integer passenger) {
      
          if (passenger == 4) {
      
              this.passenger = passenger;
      
          } else {
      
              System.err.println("⚠  Necesitas asignar por lo menos 4 pasajeros");
      
          }
      
      }

📚 Módulo 7. Polimorfismo

  • Generando polimorfismo en Java

    Poli → muchas, morfismo → formas.

    En el polimorfismo se tiene un método que se comparte entre varias Clases y cada una de ellas le da el comportamiento que necesite.

    Ejemplo de Polimorfismo

    Si se quiere una definición más especifica, se podría decir que el polimorfismo consiste en construir métodos con el mismo nombre pero con un comportamiento diferente.

    Aplicar polimorfismo en Java:

    Para sobrescribir un método que se hereda de una Súper Clase se usa la siguiente sintaxis:

    @Override // Indica que se quiere sobrescribir el método.
    public void setPassenger(Integer passenger) {
        if (passenger == 6) {
            this.passenger = passenger;
        }
    }