Python orienté objet - Guide rapide

Les langages de programmation émergent constamment, de même que les différentes méthodologies. La programmation orientée objet est l'une de ces méthodes qui est devenue très populaire ces dernières années.

Ce chapitre décrit les fonctionnalités du langage de programmation Python qui en font un langage de programmation orienté objet.

Schéma de classification de la programmation linguistique

Python peut être caractérisé sous des méthodologies de programmation orientées objet. L'image suivante montre les caractéristiques de divers langages de programmation. Observez les fonctionnalités de Python qui le rendent orienté objet.

Cours de langue Catégories Langues
Paradigme de programmation De procédure C, C ++, C #, Objective-C, java, Go
Scripting CoffeeScript, JavaScript, Python, Perl, Php, Ruby
Fonctionnel Clojure, Eralang, Haskell, Scala
Classe de compilation Statique C, C ++, C #, Objective-C, java, Go, Haskell, Scala
Dynamique CoffeeScript, JavaScript, Python, Perl, Php, Ruby, Clojure, Erlang
Classe de type Fort C #, java, Go, Python, Ruby, Clojure, Erlang, Haskell, Scala
Faible C, C ++, C #, Objective-C, CoffeeScript, JavaScript, Perl, Php
Classe de mémoire Géré Autres
Non géré C, C ++, C #, Objective-C

Qu'est-ce que la programmation orientée objet?

Object Orientedsignifie dirigé vers des objets. En d'autres termes, cela signifie fonctionnellement orienté vers la modélisation d'objets. C'est l'une des nombreuses techniques utilisées pour modéliser des systèmes complexes en décrivant une collection d'objets en interaction via leurs données et leur comportement.

Python, une programmation orientée objet (POO), est un moyen de programmation qui se concentre sur l'utilisation d'objets et de classes pour concevoir et créer des applications. Les principaux piliers de la programmation orientée objet (POO) sont Inheritance, Polymorphism, Abstraction, un d Encapsulation.

L'analyse orientée objet (OOA) est le processus qui consiste à examiner un problème, un système ou une tâche et à identifier les objets et les interactions entre eux.

Pourquoi choisir la programmation orientée objet?

Python a été conçu avec une approche orientée objet. La POO offre les avantages suivants -

  • Fournit une structure de programme claire, ce qui facilite la cartographie des problèmes du monde réel et de leurs solutions.

  • Facilite la maintenance et la modification faciles du code existant.

  • Améliore la modularité du programme car chaque objet existe indépendamment et de nouvelles fonctionnalités peuvent être ajoutées facilement sans perturber celles existantes.

  • Présente un bon cadre pour les bibliothèques de code où les composants fournis peuvent être facilement adaptés et modifiés par le programmeur.

  • Donne la réutilisabilité du code

Programmation procédurale ou orientée objet

La programmation procédurale est dérivée de la programmation structurelle basée sur les concepts de functions/procedure/routines. Il est facile d'accéder et de modifier les données dans la programmation orientée procédurale. D'autre part, la programmation orientée objet (POO) permet la décomposition d'un problème en un certain nombre d'unités appeléesobjectspuis créez les données et les fonctions autour de ces objets. Il met davantage l'accent sur les données que sur la procédure ou les fonctions. Toujours dans la POO, les données sont masquées et ne sont pas accessibles par une procédure externe.

Le tableau de l'image suivante montre les principales différences entre les approches POP et POO.

Différence entre la programmation orientée procédure (POP) vs. Programmation orientée objet (POO).

Programmation orientée procédure Programmation orientée objet
Basé sur Dans Pop, tout se concentre sur les données et les fonctions Oops est basé sur un scénario du monde réel.Le programme entier est divisé en petites parties appelées objet
Réutilisabilité Réutilisation du code limité Réutilisation du code
Approche Approche descendante Conception orientée objet
Accéder aux prescripteurs Aucun Public, privé et protégé
Mouvement de données Les données peuvent passer librement des fonctions aux fonctions du système Dans Oups, les données peuvent se déplacer et communiquer entre elles via les fonctions membres
Accès aux données Dans pop, la plupart des fonctions utilisent des données globales pour le partage qui sont accessibles librement d'une fonction à l'autre dans le système Dans Oups, les données ne peuvent pas se déplacer librement d'une méthode à l'autre, elles peuvent être conservées en public ou en privé afin que nous puissions contrôler l'accès aux données
Masquage des données En pop, façon si spécifique de masquer les données, donc un peu moins sécurisée Il fournit un masquage des données, tellement plus sécurisé
Surcharge Pas possible Fonctions et surcharge de l'opérateur
Exemple-Langues C, VB, Fortran, Pascal C ++, Python, Java, C #
Abstraction Utilise l'abstraction au niveau de la procédure Utilise l'abstraction au niveau de la classe et de l'objet

Principes de la programmation orientée objet

La programmation orientée objet (POO) est basée sur le concept de objects plutôt que des actions, et dataplutôt que logique. Pour qu'un langage de programmation soit orienté objet, il doit avoir un mécanisme permettant de travailler avec des classes et des objets ainsi que la mise en œuvre et l'utilisation des principes et concepts fondamentaux orientés objet, à savoir l'héritage, l'abstraction, l'encapsulation et le polymorphisme.

Laissez-nous comprendre chacun des piliers de la programmation orientée objet en bref -

Encapsulation

Cette propriété masque les détails inutiles et facilite la gestion de la structure du programme. L'implémentation et l'état de chaque objet sont cachés derrière des limites bien définies et qui fournissent une interface claire et simple pour travailler avec eux. Une façon d'y parvenir est de rendre les données privées.

Héritage

L'héritage, également appelé généralisation, nous permet de capturer une relation hiérarchique entre classes et objets. Par exemple, un «fruit» est une généralisation de «orange». L'héritage est très utile du point de vue de la réutilisation du code.

Abstraction

Cette propriété nous permet de masquer les détails et d'exposer uniquement les caractéristiques essentielles d'un concept ou d'un objet. Par exemple, une personne conduisant un scooter sait qu'en appuyant sur un klaxon, un son est émis, mais il n'a aucune idée de la façon dont le son est réellement généré en appuyant sur le klaxon.

Polymorphisme

Le poly-morphisme signifie plusieurs formes. Autrement dit, une chose ou une action est présente sous différentes formes ou manières. Un bon exemple de polymorphisme est la surcharge des constructeurs dans les classes.

Python orienté objet

Le cœur de la programmation Python est object et OOP, cependant vous n'avez pas besoin de vous limiter à utiliser la POO en organisant votre code en classes. La POO ajoute à toute la philosophie de conception de Python et encourage une manière propre et pragmatique de programmer. La POO permet également d'écrire des programmes plus volumineux et complexes.

Modules vs classes et objets

Les modules sont comme des «dictionnaires»

Lorsque vous travaillez sur des modules, notez les points suivants -

  • Un module Python est un package pour encapsuler du code réutilisable.

  • Les modules résident dans un dossier avec un __init__.py fichier dessus.

  • Les modules contiennent des fonctions et des classes.

  • Les modules sont importés à l'aide du import mot-clé.

Rappelez-vous qu'un dictionnaire est un key-valuepaire. Cela signifie que si vous avez un dictionnaire avec une cléEmployeID et vous voulez le récupérer, vous devrez alors utiliser les lignes de code suivantes -

employee = {“EmployeID”: “Employee Unique Identity!”}
print (employee [‘EmployeID])

Vous devrez travailler sur des modules avec le processus suivant -

  • Un module est un fichier Python contenant des fonctions ou des variables.

  • Importez le fichier dont vous avez besoin.

  • Maintenant, vous pouvez accéder aux fonctions ou variables de ce module avec le '.' (dot) Opérateur.

Considérez un module nommé employee.py avec une fonction appelée employee. Le code de la fonction est donné ci-dessous -

# this goes in employee.py
def EmployeID():
   print (“Employee Unique Identity!”)

Importez maintenant le module puis accédez à la fonction EmployeID -

import employee
employee. EmployeID()

Vous pouvez y insérer une variable nommée Age, comme indiqué -

def EmployeID():
   print (“Employee Unique Identity!”)
# just a variable
Age = “Employee age is **”

Maintenant, accédez à cette variable de la manière suivante -

import employee
employee.EmployeID()
print(employee.Age)

Maintenant, comparons cela au dictionnaire -

Employee[‘EmployeID’] # get EmployeID from employee
Employee.employeID() # get employeID from the module
Employee.Age # get access to variable

Notez qu'il existe un modèle commun en Python -

  • Prenez un key = value conteneur de style

  • Obtenez quelque chose par le nom de la clé

Lors de la comparaison d'un module avec un dictionnaire, les deux sont similaires, sauf avec ce qui suit -

  • Dans le cas du dictionary, la clé est une chaîne et la syntaxe est [clé].

  • Dans le cas du module, la clé est un identifiant et la syntaxe est .key.

Les classes sont comme des modules

Module est un dictionnaire spécialisé qui peut stocker du code Python afin que vous puissiez y accéder avec le '.' Opérateur. Une classe est un moyen de prendre un regroupement de fonctions et de données et de les placer dans un conteneur afin que vous puissiez y accéder avec l'opérateur '.'.

Si vous devez créer une classe similaire au module employé, vous pouvez le faire en utilisant le code suivant -

class employee(object):
   def __init__(self):
      self. Age = “Employee Age is ##”
   def EmployeID(self):
      print (“This is just employee unique identity”)

Note- Les classes sont préférées aux modules car vous pouvez les réutiliser telles quelles et sans trop d'interférences. Alors qu'avec les modules, vous n'en avez qu'un avec l'ensemble du programme.

Les objets sont comme des mini-importations

Une classe est comme un mini-module et vous pouvez importer de la même manière que vous le faites pour les classes, en utilisant le concept appelé instantiate. Notez que lorsque vous instanciez une classe, vous obtenez unobject.

Vous pouvez instancier un objet, comme pour appeler une classe comme une fonction, comme indiqué -

this_obj = employee() # Instantiatethis_obj.EmployeID() # get EmployeId from the class
print(this_obj.Age) # get variable Age

Vous pouvez le faire de l'une des trois manières suivantes -

# dictionary style
Employee[‘EmployeID’]
# module style
Employee.EmployeID()
Print(employee.Age)
# Class style
this_obj = employee()
this_obj.employeID()
Print(this_obj.Age)

Ce chapitre explique en détail la configuration de l'environnement Python sur votre ordinateur local.

Prérequis et boîtes à outils

Avant de poursuivre votre apprentissage de Python, nous vous suggérons de vérifier si les conditions préalables suivantes sont remplies -

  • La dernière version de Python est installée sur votre ordinateur

  • Un IDE ou un éditeur de texte est installé

  • Vous avez une connaissance de base pour écrire et déboguer en Python, c'est-à-dire que vous pouvez faire ce qui suit en Python -

    • Capable d'écrire et d'exécuter des programmes Python.

    • Déboguez les programmes et diagnostiquez les erreurs.

    • Travaillez avec des types de données de base.

    • Écrire for boucles, while boucles, et if déclarations

    • Code functions

Si vous n'avez aucune expérience en langage de programmation, vous pouvez trouver de nombreux tutoriels pour débutants en Python sur

https://www.tutorialpoints.com/

Installer Python

Les étapes suivantes vous montrent en détail comment installer Python sur votre ordinateur local -

Step 1 - Accédez au site officiel de Python https://www.python.org/, clique sur le Downloads et choisissez la dernière version ou toute version stable de votre choix.

Step 2- Enregistrez le fichier exe du programme d'installation Python que vous téléchargez et une fois que vous l'avez téléchargé, ouvrez-le. Cliquer surRun et choisissez Next option par défaut et terminez l'installation.

Step 3- Après avoir installé, vous devriez maintenant voir le menu Python comme indiqué dans l'image ci-dessous. Démarrez le programme en choisissant IDLE (Python GUI).

Cela démarrera le shell Python. Tapez des commandes simples pour vérifier l'installation.

Choisir un IDE

Un environnement de développement intégré est un éditeur de texte orienté vers le développement de logiciels. Vous devrez installer un IDE pour contrôler le flux de votre programmation et regrouper les projets lorsque vous travaillez sur Python. Voici quelques IDE disponibles en ligne. Vous pouvez en choisir un à votre convenance.

  • IDE Pycharm
  • IDE de Komodo
  • Eric Python IDE

Note - Eclipse IDE est principalement utilisé en Java, mais il dispose d'un plugin Python.

Pycharm

Pycharm, l'IDE multiplateforme est l'un des IDE les plus populaires actuellement disponibles. Il fournit une assistance et une analyse de codage avec l'achèvement du code, la navigation de projet et de code, les tests unitaires intégrés, l'intégration du contrôle de version, le débogage et bien plus encore

Lien de téléchargement

https://www.jetbrains.com/pycharm/download/#section=windows

Languages Supported - Python, HTML, CSS, JavaScript, Coffee Script, TypeScript, Cython, AngularJS, Node.js, langages de modèles.

Capture d'écran

Pourquoi choisir?

PyCharm offre les fonctionnalités et avantages suivants à ses utilisateurs -

  • IDE multiplateforme compatible avec Windows, Linux et Mac OS
  • Inclut Django IDE, plus le support CSS et JavaScript
  • Comprend des milliers de plugins, un terminal intégré et un contrôle de version
  • S'intègre à Git, SVN et Mercurial
  • Propose des outils d'édition intelligents pour Python
  • Intégration facile avec Virtualenv, Docker et Vagrant
  • Fonctions de navigation et de recherche simples
  • Analyse et refactoring de code
  • Injections configurables
  • Prend en charge des tonnes de bibliothèques Python
  • Contient des modèles et des débogueurs JavaScript
  • Inclut les débogueurs Python / Django
  • Fonctionne avec Google App Engine, des frameworks supplémentaires et des bibliothèques.
  • A une interface utilisateur personnalisable, une émulation VIM disponible

IDE de Komodo

C'est un IDE polyglotte qui prend en charge plus de 100 langues et essentiellement des langages dynamiques tels que Python, PHP et Ruby. Il s'agit d'un IDE commercial disponible pour un essai gratuit de 21 jours avec toutes les fonctionnalités. ActiveState est la société de logiciels qui gère le développement de l'IDE Komodo. Il propose également une version réduite de Komodo connue sous le nom de Komodo Edit pour des tâches de programmation simples.

Cet IDE contient toutes sortes de fonctionnalités du niveau le plus basique au niveau avancé. Si vous êtes étudiant ou indépendant, vous pouvez l'acheter près de la moitié du prix réel. Cependant, il est entièrement gratuit pour les enseignants et les professeurs d'institutions et d'universités reconnues.

Il dispose de toutes les fonctionnalités dont vous avez besoin pour le développement Web et mobile, y compris la prise en charge de tous vos langages et frameworks.

Lien de téléchargement

Les liens de téléchargement pour Komodo Edit (version gratuite) et Komodo IDE (version payante) sont indiqués ici -

Komodo Edit (free)

https://www.activestate.com/komodo-edit

Komodo IDE (paid)

https://www.activestate.com/komodo-ide/downloads/ide

Capture d'écran

Pourquoi choisir?

  • IDE puissant avec prise en charge de Perl, PHP, Python, Ruby et bien d'autres.
  • IDE multiplateforme.

Il comprend des fonctionnalités de base telles que la prise en charge du débogueur intégré, la saisie semi-automatique, le visualiseur de modèle d'objet de document (DOM), le navigateur de code, les shells interactifs, la configuration des points d'arrêt, le profilage de code, les tests unitaires intégrés. En bref, c'est un IDE professionnel avec une multitude de fonctionnalités améliorant la productivité.

Eric Python IDE

C'est un IDE open source pour Python et Ruby. Eric est un éditeur et IDE complet, écrit en Python. Il est basé sur la boîte à outils d'interface graphique Qt multiplateforme, intégrant le contrôle de l'éditeur Scintilla très flexible. L'IDE est très configurable et on peut choisir ce qu'il faut utiliser ou non. Vous pouvez télécharger Eric IDE à partir du lien ci-dessous:

https://eric-ide.python-projects.org/eric-download.html

Pourquoi choisir

  • Grande indentation, mise en évidence des erreurs.
  • Aide au code
  • Achèvement du code
  • Nettoyage de code avec PyLint
  • Recherche rapide
  • Débogueur Python intégré.

Capture d'écran

Choisir un éditeur de texte

Vous n'avez peut-être pas toujours besoin d'un IDE. Pour des tâches telles que l'apprentissage du code avec Python ou Arduino, ou lorsque vous travaillez sur un script rapide dans un script shell pour vous aider à automatiser certaines tâches, un éditeur de texte simple et léger centré sur le code fera l'affaire. De plus, de nombreux éditeurs de texte offrent des fonctionnalités telles que la coloration syntaxique et l'exécution de scripts dans le programme, similaires aux IDE. Certains des éditeurs de texte sont donnés ici -

  • Atom
  • Texte sublime
  • Notepad++

Éditeur de texte Atom

Atom est un éditeur de texte piratable construit par l'équipe de GitHub. C'est un éditeur de texte et de code gratuit et open source, ce qui signifie que tout le code est disponible pour que vous puissiez le lire, le modifier pour votre propre usage et même apporter des améliorations. Il s'agit d'un éditeur de texte multiplateforme compatible pour macOS, Linux et Microsoft Windows avec prise en charge des plug-ins écrits en Node.js et Git Control intégré.

Lien de téléchargement

https://atom.io/

Capture d'écran

Langues prises en charge

C / C ++, C #, CSS, CoffeeScript, HTML, JavaScript, Java, JSON, Julia, Objective-C, PHP, Perl, Python, Ruby on Rails, Ruby, script Shell, Scala, SQL, XML, YAML et bien d'autres.

Éditeur de texte sublime

Sublime text est un logiciel propriétaire et il vous propose une version d'essai gratuite pour le tester avant de l'acheter. Selon stackoverflow.com , il s'agit du quatrième environnement de développement le plus populaire.

Certains des avantages qu'il offre sont sa vitesse incroyable, sa facilité d'utilisation et son soutien communautaire. Il prend également en charge de nombreux langages de programmation et langages de balisage, et des fonctions peuvent être ajoutées par les utilisateurs avec des plugins, généralement construits par la communauté et maintenus sous des licences de logiciels libres.

Capture d'écran

Langue prise en charge

  • Python, Ruby, JavaScript etc.

Pourquoi choisir?

  • Personnalisez les raccourcis clavier, les menus, les extraits, les macros, les complétions et plus encore.

  • Fonction de saisie semi-automatique

  • Insérez rapidement du texte et du code avec des extraits de texte sublimes à l'aide d'extraits de code, de marqueurs de champ et d'espaces réservés
  • S'ouvre rapidement

  • Prise en charge multiplateforme pour Mac, Linux et Windows.

  • Faites passer le curseur là où vous voulez aller

  • Sélectionnez plusieurs lignes, mots et colonnes

Bloc-notes ++

C'est un éditeur de code source gratuit et un remplacement du bloc-notes qui prend en charge plusieurs langages, de l'assembly au XML, en passant par Python. Fonctionnant dans l'environnement MS Windows, son utilisation est régie par la licence GPL. En plus de la coloration syntaxique, Notepad ++ possède certaines fonctionnalités qui sont particulièrement utiles aux codeurs.

Capture d'écran

Principales caractéristiques

  • Mise en évidence de la syntaxe et pliage de la syntaxe
  • Recherche / remplacement PCRE (Expression régulière compatible Perl)
  • Interface graphique entièrement personnalisable
  • Achèvement automatique
  • Modification par onglets
  • Multi-View
  • Environnement multilingue
  • Lancable avec différents arguments

Langue prise en charge

  • Presque tous les langages (plus de 60 langues) comme Python, C, C ++, C #, Java, etc.

Les structures de données Python sont très intuitives d'un point de vue syntaxique et offrent un large choix d'opérations. Vous devez choisir la structure de données Python en fonction de ce que les données impliquent, si elles doivent être modifiées, ou s'il s'agit d'une donnée fixe et du type d'accès requis, comme au début / à la fin / aléatoire, etc.

Listes

Une liste représente le type de structure de données le plus polyvalent en Python. Une liste est un conteneur qui contient des valeurs séparées par des virgules (éléments ou éléments) entre crochets. Les listes sont utiles lorsque nous voulons travailler avec plusieurs valeurs associées. Comme les listes conservent les données ensemble, nous pouvons effectuer les mêmes méthodes et opérations sur plusieurs valeurs à la fois. Les index des listes commencent à zéro et contrairement aux chaînes, les listes sont modifiables.

Structure des données - Liste

>>>
>>> # Any Empty List
>>> empty_list = []
>>>
>>> # A list of String
>>> str_list = ['Life', 'Is', 'Beautiful']
>>> # A list of Integers
>>> int_list = [1, 4, 5, 9, 18]
>>>
>>> #Mixed items list
>>> mixed_list = ['This', 9, 'is', 18, 45.9, 'a', 54, 'mixed', 99, 'list']
>>> # To print the list
>>>
>>> print(empty_list)
[]
>>> print(str_list)
['Life', 'Is', 'Beautiful']
>>> print(type(str_list))
<class 'list'>
>>> print(int_list)
[1, 4, 5, 9, 18]
>>> print(mixed_list)
['This', 9, 'is', 18, 45.9, 'a', 54, 'mixed', 99, 'list']

Accéder aux éléments de la liste Python

Chaque élément d'une liste se voit attribuer un numéro - c'est-à-dire l'index ou la position de ce numéro. L'indexation commence toujours à zéro, le deuxième index est un et ainsi de suite. Pour accéder aux éléments d'une liste, nous pouvons utiliser ces numéros d'index entre crochets. Observez le code suivant par exemple -

>>> mixed_list = ['This', 9, 'is', 18, 45.9, 'a', 54, 'mixed', 99, 'list']
>>>
>>> # To access the First Item of the list
>>> mixed_list[0]
'This'
>>> # To access the 4th item
>>> mixed_list[3]
18
>>> # To access the last item of the list
>>> mixed_list[-1]
'list'

Objets vides

Les objets vides sont les types intégrés Python les plus simples et les plus élémentaires. Nous les avons utilisés plusieurs fois sans nous en rendre compte et l'avons étendu à toutes les classes que nous avons créées. Le but principal d'écrire une classe vide est de bloquer quelque chose pour le moment, puis de l'étendre et d'y ajouter un comportement.

Ajouter un comportement à une classe signifie remplacer une structure de données par un objet et modifier toutes les références à celui-ci. Il est donc important de vérifier les données, s'il s'agit d'un objet déguisé, avant de créer quoi que ce soit. Observez le code suivant pour une meilleure compréhension:

>>> #Empty objects
>>>
>>> obj = object()
>>> obj.x = 9
Traceback (most recent call last):
File "<pyshell#3>", line 1, in <module>
obj.x = 9
AttributeError: 'object' object has no attribute 'x'

Donc, d'en haut, nous pouvons voir qu'il n'est pas possible de définir des attributs sur un objet qui a été instancié directement. Lorsque Python permet à un objet d'avoir des attributs arbitraires, il faut une certaine quantité de mémoire système pour garder une trace des attributs de chaque objet, pour stocker à la fois le nom d'attribut et sa valeur. Même si aucun attribut n'est stocké, une certaine quantité de mémoire est allouée pour de nouveaux attributs potentiels.

Ainsi, Python désactive les propriétés arbitraires sur l'objet et plusieurs autres éléments intégrés, par défaut.

>>> # Empty Objects
>>>
>>> class EmpObject:
    pass
>>> obj = EmpObject()
>>> obj.x = 'Hello, World!'
>>> obj.x
'Hello, World!'

Par conséquent, si nous voulons regrouper les propriétés, nous pourrions les stocker dans un objet vide comme indiqué dans le code ci-dessus. Cependant, cette méthode n'est pas toujours suggérée. N'oubliez pas que les classes et les objets ne doivent être utilisés que lorsque vous souhaitez spécifier à la fois des données et des comportements.

Tuples

Les tuples sont similaires aux listes et peuvent stocker des éléments. Cependant, ils sont immuables, nous ne pouvons donc pas ajouter, supprimer ou remplacer des objets. Le principal avantage que le tuple offre en raison de son immuabilité est que nous pouvons les utiliser comme clés dans des dictionnaires, ou dans d'autres emplacements où un objet nécessite une valeur de hachage.

Les tuples sont utilisés pour stocker des données et non des comportements. Si vous avez besoin d'un comportement pour manipuler un tuple, vous devez le transmettre à une fonction (ou à une méthode sur un autre objet) qui exécute l'action.

Comme le tuple peut agir comme une clé de dictionnaire, les valeurs stockées sont différentes les unes des autres. Nous pouvons créer un tuple en séparant les valeurs par une virgule. Les tuples sont placés entre parenthèses mais pas obligatoires. Le code suivant montre deux affectations identiques.

>>> stock1 = 'MSFT', 95.00, 97.45, 92.45
>>> stock2 = ('MSFT', 95.00, 97.45, 92.45)
>>> type (stock1)
<class 'tuple'>
>>> type(stock2)
<class 'tuple'>
>>> stock1 == stock2
True
>>>

Définition d'un tuple

Les tuples sont très similaires à list sauf que l'ensemble des éléments est placé entre parenthèses au lieu de crochets.

Tout comme lorsque vous découpez une liste, vous obtenez une nouvelle liste et lorsque vous découpez un tuple, vous obtenez un nouveau tuple.

>>> tupl = ('Tuple','is', 'an','IMMUTABLE', 'list')
>>> tupl
('Tuple', 'is', 'an', 'IMMUTABLE', 'list')
>>> tupl[0]
'Tuple'
>>> tupl[-1]
'list'
>>> tupl[1:3]
('is', 'an')

Méthodes de tuple Python

Le code suivant montre les méthodes dans les tuples Python -

>>> tupl
('Tuple', 'is', 'an', 'IMMUTABLE', 'list')
>>> tupl.append('new')
Traceback (most recent call last):
   File "<pyshell#148>", line 1, in <module>
      tupl.append('new')
AttributeError: 'tuple' object has no attribute 'append'
>>> tupl.remove('is')
Traceback (most recent call last):
   File "<pyshell#149>", line 1, in <module>
      tupl.remove('is')
AttributeError: 'tuple' object has no attribute 'remove'
>>> tupl.index('list')
4
>>> tupl.index('new')
Traceback (most recent call last):
   File "<pyshell#151>", line 1, in <module>
      tupl.index('new')
ValueError: tuple.index(x): x not in tuple
>>> "is" in tupl
True
>>> tupl.count('is')
1

À partir du code ci-dessus, nous pouvons comprendre que les tuples sont immuables et donc -

  • Toi cannot ajouter des éléments à un tuple.

  • Toi cannot ajouter ou étendre une méthode.

  • Toi cannot supprime les éléments d'un tuple.

  • Les tuples ont no supprimer ou pop méthode.

  • Count et index sont les méthodes disponibles dans un tuple.

dictionnaire

Dictionary est l'un des types de données intégrés de Python et définit des relations un à un entre les clés et les valeurs.

Définition de dictionnaires

Observez le code suivant pour comprendre la définition d'un dictionnaire -

>>> # empty dictionary
>>> my_dict = {}
>>>
>>> # dictionary with integer keys
>>> my_dict = { 1:'msft', 2: 'IT'}
>>>
>>> # dictionary with mixed keys
>>> my_dict = {'name': 'Aarav', 1: [ 2, 4, 10]}
>>>
>>> # using built-in function dict()
>>> my_dict = dict({1:'msft', 2:'IT'})
>>>
>>> # From sequence having each item as a pair
>>> my_dict = dict([(1,'msft'), (2,'IT')])
>>>
>>> # Accessing elements of a dictionary
>>> my_dict[1]
'msft'
>>> my_dict[2]
'IT'
>>> my_dict['IT']
Traceback (most recent call last):
   File "<pyshell#177>", line 1, in <module>
   my_dict['IT']
KeyError: 'IT'
>>>

À partir du code ci-dessus, nous pouvons observer que:

  • Nous créons d'abord un dictionnaire avec deux éléments et l'affectons à la variable my_dict. Chaque élément est une paire clé-valeur, et l'ensemble des éléments est entouré d'accolades.

  • Le nombre 1 est la clé et msftest sa valeur. De même,2 est la clé et IT est sa valeur.

  • Vous pouvez obtenir des valeurs par clé, mais pas l'inverse. Ainsi quand on essaiemy_dict[‘IT’] , cela soulève une exception, car IT n'est pas une clé.

Modifier les dictionnaires

Observez le code suivant pour comprendre la modification d'un dictionnaire -

>>> # Modifying a Dictionary
>>>
>>> my_dict
{1: 'msft', 2: 'IT'}
>>> my_dict[2] = 'Software'
>>> my_dict
{1: 'msft', 2: 'Software'}
>>>
>>> my_dict[3] = 'Microsoft Technologies'
>>> my_dict
{1: 'msft', 2: 'Software', 3: 'Microsoft Technologies'}

À partir du code ci-dessus, nous pouvons observer que -

  • Vous ne pouvez pas avoir de clés en double dans un dictionnaire. La modification de la valeur d'une clé existante supprimera l'ancienne valeur.

  • Vous pouvez ajouter de nouvelles paires clé-valeur à tout moment.

  • Les dictionnaires n'ont aucune notion d'ordre parmi les éléments. Ce sont de simples collections non ordonnées.

Mélanger les types de données dans un dictionnaire

Observez le code suivant pour comprendre le mélange de types de données dans un dictionnaire -

>>> # Mixing Data Types in a Dictionary
>>>
>>> my_dict
{1: 'msft', 2: 'Software', 3: 'Microsoft Technologies'}
>>> my_dict[4] = 'Operating System'
>>> my_dict
{1: 'msft', 2: 'Software', 3: 'Microsoft Technologies', 4: 'Operating System'}
>>> my_dict['Bill Gates'] = 'Owner'
>>> my_dict
{1: 'msft', 2: 'Software', 3: 'Microsoft Technologies', 4: 'Operating System',
'Bill Gates': 'Owner'}

À partir du code ci-dessus, nous pouvons observer que -

  • Non seulement les chaînes, mais la valeur du dictionnaire peut être de n'importe quel type de données, y compris des chaînes, des entiers, y compris le dictionnaire lui-même.

  • Contrairement aux valeurs de dictionnaire, les clés de dictionnaire sont plus restreintes, mais peuvent être de n'importe quel type comme des chaînes, des entiers ou tout autre.

Suppression d'éléments des dictionnaires

Observez le code suivant pour comprendre comment supprimer des éléments d'un dictionnaire -

>>> # Deleting Items from a Dictionary
>>>
>>> my_dict
{1: 'msft', 2: 'Software', 3: 'Microsoft Technologies', 4: 'Operating System',
'Bill Gates': 'Owner'}
>>>
>>> del my_dict['Bill Gates']
>>> my_dict
{1: 'msft', 2: 'Software', 3: 'Microsoft Technologies', 4: 'Operating System'}
>>>
>>> my_dict.clear()
>>> my_dict
{}

À partir du code ci-dessus, nous pouvons observer que -

  • del - vous permet de supprimer des éléments individuels d'un dictionnaire par clé.

  • clear - supprime tous les éléments d'un dictionnaire.

Ensembles

Set () est une collection non ordonnée sans éléments en double. Bien que les éléments individuels soient immuables, l'ensemble lui-même est modifiable, c'est-à-dire que nous pouvons ajouter ou supprimer des éléments / éléments de l'ensemble. Nous pouvons effectuer des opérations mathématiques comme l'union, l'intersection, etc. avec un ensemble.

Bien que les ensembles en général puissent être implémentés à l'aide d'arbres, les ensembles en Python peuvent être implémentés à l'aide d'une table de hachage. Cela lui permet une méthode hautement optimisée pour vérifier si un élément spécifique est contenu dans l'ensemble

Créer un ensemble

Un ensemble est créé en plaçant tous les éléments (éléments) à l'intérieur d'accolades {}, séparés par des virgules ou en utilisant la fonction intégrée set(). Observez les lignes de code suivantes -

>>> #set of integers
>>> my_set = {1,2,4,8}
>>> print(my_set)
{8, 1, 2, 4}
>>>
>>> #set of mixed datatypes
>>> my_set = {1.0, "Hello World!", (2, 4, 6)}
>>> print(my_set)
{1.0, (2, 4, 6), 'Hello World!'}
>>>

Méthodes pour les ensembles

Observez le code suivant pour comprendre les méthodes pour les ensembles -

>>> >>> #METHODS FOR SETS
>>>
>>> #add(x) Method
>>> topics = {'Python', 'Java', 'C#'}
>>> topics.add('C++')
>>> topics
{'C#', 'C++', 'Java', 'Python'}
>>>
>>> #union(s) Method, returns a union of two set.
>>> topics
{'C#', 'C++', 'Java', 'Python'}
>>> team = {'Developer', 'Content Writer', 'Editor','Tester'}
>>> group = topics.union(team)
>>> group
{'Tester', 'C#', 'Python', 'Editor', 'Developer', 'C++', 'Java', 'Content
Writer'}
>>> # intersets(s) method, returns an intersection of two sets
>>> inters = topics.intersection(team)
>>> inters
set()
>>>
>>> # difference(s) Method, returns a set containing all the elements of
invoking set but not of the second set.
>>>
>>> safe = topics.difference(team)
>>> safe
{'Python', 'C++', 'Java', 'C#'}
>>>
>>> diff = topics.difference(group)
>>> diff
set()
>>> #clear() Method, Empties the whole set.
>>> group.clear()
>>> group
set()
>>>

Opérateurs pour ensembles

Observez le code suivant pour comprendre les opérateurs pour les ensembles -

>>> # PYTHON SET OPERATIONS
>>>
>>> #Creating two sets
>>> set1 = set()
>>> set2 = set()
>>>
>>> # Adding elements to set
>>> for i in range(1,5):
   set1.add(i)
>>> for j in range(4,9):
   set2.add(j)
>>> set1
{1, 2, 3, 4}
>>> set2
{4, 5, 6, 7, 8}
>>>
>>> #Union of set1 and set2
>>> set3 = set1 | set2 # same as set1.union(set2)
>>> print('Union of set1 & set2: set3 = ', set3)
Union of set1 & set2: set3 = {1, 2, 3, 4, 5, 6, 7, 8}
>>>
>>> #Intersection of set1 & set2
>>> set4 = set1 & set2 # same as set1.intersection(set2)
>>> print('Intersection of set1 and set2: set4 = ', set4)
Intersection of set1 and set2: set4 = {4}
>>>
>>> # Checking relation between set3 and set4
>>> if set3 > set4: # set3.issuperset(set4)
   print('Set3 is superset of set4')
elif set3 < set4: #set3.issubset(set4)
   print('Set3 is subset of set4')
else: #set3 == set4
   print('Set 3 is same as set4')
Set3 is superset of set4
>>>
>>> # Difference between set3 and set4
>>> set5 = set3 - set4
>>> print('Elements in set3 and not in set4: set5 = ', set5)
Elements in set3 and not in set4: set5 = {1, 2, 3, 5, 6, 7, 8}
>>>
>>> # Check if set4 and set5 are disjoint sets
>>> if set4.isdisjoint(set5):
   print('Set4 and set5 have nothing in common\n')
Set4 and set5 have nothing in common
>>> # Removing all the values of set5
>>> set5.clear()
>>> set5 set()

Dans ce chapitre, nous discuterons en détail des termes orientés objet et des concepts de programmation. La classe est juste une fabrique pour une instance. Cette fabrique contient le plan directeur qui décrit comment créer les instances. Une instance ou un objet sont construits à partir de la classe. Dans la plupart des cas, nous pouvons avoir plus d'une instance d'une classe. Chaque instance a un ensemble d'attributs et ces attributs sont définis dans une classe, de sorte que chaque instance d'une classe particulière est censée avoir les mêmes attributs.

Packs de classes: comportement et état

Une classe vous permettra de regrouper le comportement et l'état d'un objet. Observez le schéma suivant pour une meilleure compréhension -

Les points suivants méritent d'être soulignés lors de la discussion des groupes de classes -

  • Le mot behavior est identique à function - c'est un morceau de code qui fait quelque chose (ou implémente un comportement)

  • Le mot state est identique à variables - c'est un endroit pour stocker des valeurs dans une classe.

  • Lorsque nous affirmons ensemble un comportement et un état de classe, cela signifie qu'une classe regroupe des fonctions et des variables.

Les classes ont des méthodes et des attributs

En Python, la création d'une méthode définit un comportement de classe. Le mot méthode est le nom POO donné à une fonction définie dans une classe. Pour résumer -

  • Class functions - est synonyme de methods

  • Class variables - est synonyme de name attributes.

  • Class - un plan pour une instance avec un comportement exact.

  • Object - l'une des instances de la classe, exécute la fonctionnalité définie dans la classe.

  • Type - indique la classe à laquelle appartient l'instance

  • Attribute - Toute valeur d'objet: object.attribute

  • Method - un «attribut appelable» défini dans la classe

Observez le morceau de code suivant par exemple -

var = “Hello, John”
print( type (var)) # < type ‘str’> or <class 'str'>
print(var.upper()) # upper() method is called, HELLO, JOHN

Création et instanciation

Le code suivant montre comment créer notre première classe, puis son instance.

class MyClass(object):
   pass
# Create first instance of MyClass
this_obj = MyClass()
print(this_obj)
# Another instance of MyClass
that_obj = MyClass()
print (that_obj)

Ici, nous avons créé une classe appelée MyClasset qui ne fait aucune tâche. L'argumentobject dans MyClass class implique l'héritage de classe et sera discuté dans les chapitres suivants. pass dans le code ci-dessus indique que ce bloc est vide, c'est-à-dire qu'il s'agit d'une définition de classe vide.

Créons une instance this_obj de MyClass() classe et imprimez-le comme indiqué -

<__main__.MyClass object at 0x03B08E10>
<__main__.MyClass object at 0x0369D390>

Ici, nous avons créé une instance de MyClass.Le code hexadécimal fait référence à l'adresse où l'objet est stocké. Une autre instance pointe vers une autre adresse.

Définissons maintenant une variable à l'intérieur de la classe MyClass() et récupérez la variable de l'instance de cette classe comme indiqué dans le code suivant -

class MyClass(object):
   var = 9

# Create first instance of MyClass
this_obj = MyClass()
print(this_obj.var)

# Another instance of MyClass

that_obj = MyClass()
print (that_obj.var)

Production

Vous pouvez observer la sortie suivante lorsque vous exécutez le code donné ci-dessus -

9
9

Comme l'instance sait à partir de quelle classe elle est instanciée, ainsi, lorsqu'elle est demandée pour un attribut d'une instance, l'instance recherche l'attribut et la classe. C'est ce qu'on appelle leattribute lookup.

Méthodes d'instance

Une fonction définie dans une classe est appelée method.Une méthode d'instance nécessite une instance pour l'appeler et ne nécessite aucun décorateur. Lors de la création d'une méthode d'instance, le premier paramètre est toujoursself. Bien que nous puissions l'appeler (self) par n'importe quel autre nom, il est recommandé d'utiliser self, car il s'agit d'une convention de dénomination.

class MyClass(object):
   var = 9
   def firstM(self):
      print("hello, World")
obj = MyClass()
print(obj.var)
obj.firstM()

Production

Vous pouvez observer la sortie suivante lorsque vous exécutez le code donné ci-dessus -

9
hello, World

Notez que dans le programme ci-dessus, nous avons défini une méthode avec self comme argument. Mais nous ne pouvons pas appeler la méthode car nous ne lui avons déclaré aucun argument.

class MyClass(object):
   def firstM(self):
      print("hello, World")
      print(self)
obj = MyClass()
obj.firstM()
print(obj)

Production

Vous pouvez observer la sortie suivante lorsque vous exécutez le code donné ci-dessus -

hello, World
<__main__.MyClass object at 0x036A8E10>
<__main__.MyClass object at 0x036A8E10>

Encapsulation

L'encapsulation est l'un des principes fondamentaux de la POO. La POO permet de masquer la complexité du fonctionnement interne de l'objet qui est avantageuse pour le développeur de la manière suivante -

  • Simplifie et rend facile de comprendre l'utilisation d'un objet sans connaître les éléments internes.

  • Tout changement peut être facilement gérable.

La programmation orientée objet repose fortement sur l'encapsulation. Les termes encapsulation et abstraction (également appelés masquage de données) sont souvent utilisés comme synonymes. Ils sont presque synonymes, car l'abstraction est réalisée par l'encapsulation.

L'encapsulation nous fournit le mécanisme de restriction de l'accès à certains des composants de l'objet, cela signifie que la représentation interne d'un objet ne peut pas être vue de l'extérieur de la définition de l'objet. L'accès à ces données se fait généralement par des méthodes spéciales -Getters et Setters.

Ces données sont stockées dans des attributs d'instance et peuvent être manipulées de n'importe où en dehors de la classe. Pour le sécuriser, ces données ne doivent être accessibles qu'à l'aide de méthodes d'instance. L'accès direct ne devrait pas être autorisé.

class MyClass(object):
   def setAge(self, num):
      self.age = num

   def getAge(self):
      return self.age

zack = MyClass()
zack.setAge(45)
print(zack.getAge())

zack.setAge("Fourty Five")
print(zack.getAge())

Production

Vous pouvez observer la sortie suivante lorsque vous exécutez le code donné ci-dessus -

45
Fourty Five

Les données ne doivent être stockées que si elles sont correctes et valides, à l'aide de constructions de gestion des exceptions. Comme nous pouvons le voir ci-dessus, il n'y a aucune restriction sur l'entrée de l'utilisateur dans la méthode setAge (). Cela peut être une chaîne, un nombre ou une liste. Nous devons donc vérifier le code ci-dessus pour garantir l'exactitude du stockage.

class MyClass(object):
   def setAge(self, num):
      self.age = num

   def getAge(self):
      return self.age
zack = MyClass()
zack.setAge(45)
print(zack.getAge())
zack.setAge("Fourty Five")
print(zack.getAge())

Constructeur d'initiation

Le __initLa méthode __ est implicitement appelée dès qu'un objet d'une classe est instancié, ce qui initialisera l'objet.

x = MyClass()

La ligne de code ci-dessus créera une nouvelle instance et affectera cet objet à la variable locale x.

L'opération d'instanciation, c'est-à-dire calling a class object, crée un objet vide. De nombreuses classes aiment créer des objets avec des instances personnalisées à un état initial spécifique. Par conséquent, une classe peut définir une méthode spéciale nommée '__init __ ()' comme indiqué -

def __init__(self):
   self.data = []

Python appelle __init__ pendant l'instanciation pour définir un attribut supplémentaire qui doit se produire lorsqu'une classe est instanciée et qui peut être en train de configurer des valeurs de début pour cet objet ou d'exécuter une routine requise lors de l'instanciation. Ainsi, dans cet exemple, une nouvelle instance initialisée peut être obtenue par -

x = MyClass()

La méthode __init __ () peut avoir un ou plusieurs arguments pour une plus grande flexibilité. Le init signifie initialisation, car il initialise les attributs de l'instance. Il est appelé le constructeur d'une classe.

class myclass(object):
   def __init__(self,aaa, bbb):
      self.a = aaa
      self.b = bbb

x = myclass(4.5, 3)
print(x.a, x.b)

Production

4.5 3

Attributs de classe

L'attribut défini dans la classe est appelé «attributs de classe» et les attributs définis dans la fonction sont appelés «attributs d'instance». Lors de la définition, ces attributs ne sont pas préfixés par self, car ils sont la propriété de la classe et non d'une instance particulière.

Les attributs de classe sont accessibles par la classe elle-même (className.attributeName) ainsi que par les instances de la classe (inst.attributeName). Ainsi, les instances ont accès à la fois à l'attribut d'instance et aux attributs de classe.

>>> class myclass():
   age = 21
>>> myclass.age
21
>>> x = myclass()
>>> x.age
21
>>>

Un attribut de classe peut être remplacé dans une instance, même si ce n'est pas une bonne méthode pour interrompre l'encapsulation.

Il existe un chemin de recherche pour les attributs en Python. Le premier étant la méthode définie dans la classe, puis la classe au-dessus.

>>> class myclass(object):
   classy = 'class value'
>>> dd = myclass()
>>> print (dd.classy) # This should return the string 'class value'
class value
>>>
>>> dd.classy = "Instance Value"
>>> print(dd.classy) # Return the string "Instance Value"
Instance Value
>>>
>>> # This will delete the value set for 'dd.classy' in the instance.
>>> del dd.classy
>>> >>> # Since the overriding attribute was deleted, this will print 'class
value'.

>>> print(dd.classy)
class value
>>>

Nous remplaçons l'attribut de classe «classy» dans l'instance dd. Lorsqu'il est remplacé, l'interpréteur Python lit la valeur remplacée. Mais une fois que la nouvelle valeur est supprimée avec «del», la valeur remplacée n'est plus présente dans l'instance, et par conséquent, la recherche passe un niveau supérieur et l'obtient de la classe.

Utilisation des données de classe et d'instance

Dans cette section, voyons comment les données de classe sont liées aux données d'instance. Nous pouvons stocker des données dans une classe ou dans une instance. Lorsque nous concevons une classe, nous décidons quelles données appartiennent à l'instance et quelles données doivent être stockées dans la classe globale.

Une instance peut accéder aux données de classe. Si nous créons plusieurs instances, ces instances peuvent accéder à leurs valeurs d'attribut individuelles ainsi qu'aux données globales de la classe.

Ainsi, une donnée de classe est la donnée qui est partagée entre toutes les instances. Observez le code ci-dessous pour une meilleure compréhension -

class InstanceCounter(object):
   count = 0 # class attribute, will be accessible to all instances
   def __init__(self, val):
      self.val = val
      InstanceCounter.count +=1 # Increment the value of class attribute, accessible through class name
# In above line, class ('InstanceCounter') act as an object
   def set_val(self, newval):
      self.val = newval

   def get_val(self):
      return self.val

   def get_count(self):
      return InstanceCounter.count
a = InstanceCounter(9)
b = InstanceCounter(18)
c = InstanceCounter(27)

for obj in (a, b, c):
   print ('val of obj: %s' %(obj.get_val())) # Initialized value ( 9, 18, 27)
   print ('count: %s' %(obj.get_count())) # always 3

Production

val of obj: 9
count: 3
val of obj: 18
count: 3
val of obj: 27
count: 3

En bref, les attributs de classe sont les mêmes pour toutes les instances de classe tandis que les attributs d'instance sont particuliers pour chaque instance. Pour deux instances différentes, nous aurons deux attributs d'instance différents.

class myClass:
   class_attribute = 99

   def class_method(self):
      self.instance_attribute = 'I am instance attribute'

print (myClass.__dict__)

Production

Vous pouvez observer la sortie suivante lorsque vous exécutez le code donné ci-dessus -

{'__module__': '__main__', 'class_attribute': 99, 'class_method': <function myClass.class_method at 0x04128D68>, '__dict__': <attribute '__dict__' of 'myClass' objects>, '__weakref__': <attribute '__weakref__' of 'myClass' objects>, '__doc__': None}

L'attribut d'instance myClass.__dict__ comme indiqué -

>>> a = myClass()
>>> a.class_method()
>>> print(a.__dict__)
{'instance_attribute': 'I am instance attribute'}

Ce chapitre décrit en détail diverses fonctions intégrées dans Python, les opérations d'E / S sur les fichiers et les concepts de surcharge.

Fonctions intégrées Python

L'interpréteur Python a un certain nombre de fonctions appelées fonctions intégrées qui sont facilement disponibles pour une utilisation. Dans sa dernière version, Python contient 68 fonctions intégrées répertoriées dans le tableau ci-dessous -

FONCTIONS INTÉGRÉES
abdos() dict () Aidez-moi() min () setattr ()
tout() dir () hex () prochain() tranche()
tout() divmod () id () objet() trié ()
ascii () énumérer() contribution() oct() méthode statique ()
poubelle() eval () int () ouvert() str ()
booléen () exec () isinstance () ord () somme()
bytearray () filtre() issubclass () pow () super()
octets () flotte() iter () impression() tuple ()
appelable () format() len () propriété() type()
chr () frozenset () liste() intervalle() vars ()
méthode de classe () getattr () des locaux() repr () Zip *: français()
compiler() globaux () carte() renversé() __importer__()
complexe() hasattr () max () rond()
delattr () hacher() mémoire visuelle() ensemble()

Cette section décrit brièvement certaines des fonctions importantes -

fonction len ()

La fonction len () obtient la longueur des chaînes, des listes ou des collections. Il renvoie la longueur ou le nombre d'éléments d'un objet, où l'objet peut être une chaîne, une liste ou une collection.

>>> len(['hello', 9 , 45.0, 24])
4

La fonction len () fonctionne en interne comme list.__len__() ou tuple.__len__(). Ainsi, notez que len () ne fonctionne que sur les objets qui ont un __len__() méthode.

>>> set1
{1, 2, 3, 4}
>>> set1.__len__()
4

Cependant, dans la pratique, nous préférons len() à la place du __len__() fonctionner pour les raisons suivantes -

  • C'est plus efficace. Et il n'est pas nécessaire qu'une méthode particulière soit écrite pour refuser l'accès à des méthodes spéciales telles que __len__.

  • Il est facile à entretenir.

  • Il prend en charge la compatibilité descendante.

Inversé (seq)

Il renvoie l'itérateur inverse. seq doit être un objet qui a la méthode __reversed __ () ou prend en charge le protocole de séquence (la méthode __len __ () et la méthode __getitem __ ()). Il est généralement utilisé dansfor boucles lorsque nous voulons boucler des éléments de l'arrière vers l'avant.

>>> normal_list = [2, 4, 5, 7, 9]
>>>
>>> class CustomSequence():
   def __len__(self):
      return 5
   def __getitem__(self,index):
      return "x{0}".format(index)
>>> class funkyback():
   def __reversed__(self):
      return 'backwards!'
>>> for seq in normal_list, CustomSequence(), funkyback():
      print('\n{}: '.format(seq.__class__.__name__), end="")
      for item in reversed(seq):
         print(item, end=", ")

La boucle for à la fin imprime la liste inversée d'une liste normale et les instances des deux séquences personnalisées. La sortie montre quereversed() fonctionne sur tous les trois, mais a des résultats très différents lorsque nous définissons __reversed__.

Production

Vous pouvez observer la sortie suivante lorsque vous exécutez le code donné ci-dessus -

list: 9, 7, 5, 4, 2,
CustomSequence: x4, x3, x2, x1, x0,
funkyback: b, a, c, k, w, a, r, d, s, !,

Énumérer

le enumerate () ajoute un compteur à un itérable et retourne l'objet énumérer.

La syntaxe de enumerate () est -

enumerate(iterable, start = 0)

Voici le deuxième argument start est facultatif et par défaut, l'index commence par zéro (0).

>>> # Enumerate
>>> names = ['Rajesh', 'Rahul', 'Aarav', 'Sahil', 'Trevor']
>>> enumerate(names)
<enumerate object at 0x031D9F80>
>>> list(enumerate(names))
[(0, 'Rajesh'), (1, 'Rahul'), (2, 'Aarav'), (3, 'Sahil'), (4, 'Trevor')]
>>>

Alors enumerate()renvoie un itérateur qui produit un tuple qui conserve le nombre d'éléments dans la séquence passée. Puisque la valeur de retour est un itérateur, y accéder directement n'est pas très utile. Une meilleure approche pour enumerate () est de garder count dans une boucle for.

>>> for i, n in enumerate(names):
   print('Names number: ' + str(i))
   print(n)
Names number: 0
Rajesh
Names number: 1
Rahul
Names number: 2
Aarav
Names number: 3
Sahil
Names number: 4
Trevor

Il existe de nombreuses autres fonctions dans la bibliothèque standard, et voici une autre liste de certaines fonctions plus largement utilisées -

  • hasattr, getattr, setattr et delattr, qui permet aux attributs d'un objet d'être manipulés par leurs noms de chaîne.

  • all et any, qui acceptent un objet itérable et retournent True si tous les éléments, ou certains d'entre eux, sont considérés comme vrais.

  • nzip, qui prend deux séquences ou plus et renvoie une nouvelle séquence de tuples, où chaque tuple contient une seule valeur de chaque séquence.

E / S de fichier

Le concept de fichiers est associé au terme de programmation orientée objet. Python a enveloppé l'interface fournie par les systèmes d'exploitation dans l'abstraction qui nous permet de travailler avec des objets fichier.

le open()La fonction intégrée est utilisée pour ouvrir un fichier et renvoyer un objet fichier. C'est la fonction la plus couramment utilisée avec deux arguments -

open(filename, mode)

La fonction open () appelle deux arguments, le premier est le nom du fichier et le second est le mode. Ici, le mode peut être 'r' pour le mode lecture seule, 'w' pour l'écriture uniquement (un fichier existant avec le même nom sera effacé), et 'a' ouvre le fichier pour l'ajout, toutes les données écrites dans le fichier sont automatiquement ajoutées jusqu'à la fin. «r +» ouvre le fichier en lecture et en écriture. Le mode par défaut est en lecture seule.

Sous Windows, «b» ajouté au mode ouvre le fichier en mode binaire, il existe donc également des modes comme «rb», «wb» et «r + b».

>>> text = 'This is the first line'
>>> file = open('datawork','w')
>>> file.write(text)
22
>>> file.close()

Dans certains cas, nous voulons simplement ajouter au fichier existant plutôt que de l'écraser, pour cela nous pourrions fournir la valeur `` a '' comme argument de mode, à ajouter à la fin du fichier, plutôt que d'écraser complètement le fichier existant Contenu.

>>> f = open('datawork','a')
>>> text1 = ' This is second line'
>>> f.write(text1)
20
>>> f.close()

Une fois qu'un fichier est ouvert pour la lecture, nous pouvons appeler la méthode read, readline ou readlines pour obtenir le contenu du fichier. La méthode read retourne tout le contenu du fichier sous la forme d'un objet str ou bytes, selon que le deuxième argument est «b».

Pour la lisibilité, et pour éviter de lire un fichier volumineux en une seule fois, il est souvent préférable d'utiliser une boucle for directement sur un objet fichier. Pour les fichiers texte, il lira chaque ligne, une à la fois, et nous pouvons la traiter à l'intérieur du corps de la boucle. Pour les fichiers binaires, cependant, il est préférable de lire des blocs de données de taille fixe en utilisant la méthode read (), en passant un paramètre pour le nombre maximum d'octets à lire.

>>> f = open('fileone','r+')
>>> f.readline()
'This is the first line. \n'
>>> f.readline()
'This is the second line. \n'

L'écriture dans un fichier, via la méthode d'écriture sur les objets fichier, écrit un objet chaîne (octets pour les données binaires) dans le fichier. La méthode writelines accepte une séquence de chaînes et écrit chacune des valeurs itérées dans le fichier. La méthode writelines n'ajoute pas de nouvelle ligne après chaque élément de la séquence.

Enfin, la méthode close () doit être appelée lorsque nous avons fini de lire ou d'écrire le fichier, pour garantir que toutes les écritures en mémoire tampon sont écrites sur le disque, que le fichier a été correctement nettoyé et que toutes les ressources liées au fichier sont renvoyées vers le système d'exploitation. C'est une meilleure approche d'appeler la méthode close (), mais techniquement, cela se produira automatiquement lorsque le script existe.

Une alternative à la surcharge de méthode

La surcharge de méthode fait référence au fait d'avoir plusieurs méthodes avec le même nom qui acceptent différents ensembles d'arguments.

Étant donné une seule méthode ou fonction, nous pouvons spécifier nous-mêmes le nombre de paramètres. Selon la définition de la fonction, elle peut être appelée avec zéro, un, deux ou plusieurs paramètres.

class Human:
   def sayHello(self, name = None):
      if name is not None:
         print('Hello ' + name)
      else:
         print('Hello ')

#Create Instance
obj = Human()

#Call the method, else part will be executed
obj.sayHello()

#Call the method with a parameter, if part will be executed
obj.sayHello('Rahul')

Production

Hello
Hello Rahul

Arguments par défaut

Les fonctions sont aussi des objets

Un objet appelable est un objet qui peut accepter certains arguments et peut éventuellement retourner un objet. Une fonction est l'objet appelable le plus simple en Python, mais il en existe d'autres comme des classes ou certaines instances de classe.

Chaque fonction d'un Python est un objet. Les objets peuvent contenir des méthodes ou des fonctions, mais l'objet n'est pas nécessairement une fonction.

def my_func():
   print('My function was called')
my_func.description = 'A silly function'
def second_func():

   print('Second function was called')

   second_func.description = 'One more sillier function'

def another_func(func):
   print("The description:", end=" ")
   print(func.description)
   print('The name: ', end=' ')
   print(func.__name__)
   print('The class:', end=' ')
   print(func.__class__)
   print("Now I'll call the function passed in")
   func()

another_func(my_func)
another_func(second_func)

Dans le code ci-dessus, nous pouvons passer deux fonctions différentes en argument dans notre troisième fonction et obtenir une sortie différente pour chacune -

The description: A silly function
The name: my_func
The class: 
      
        Now I'll call the function passed in My function was called The description: One more sillier function The name: second_func The class: 
       
         Now I'll call the function passed in Second function was called 
       
      

callable objects

Just as functions are objects that can have attributes set on them, it is possible to create an object that can be called as though it were a function.

In Python any object with a __call__() method can be called using function-call syntax.

Inheritance and Polymorphism

Inheritance and polymorphism – this is a very important concept in Python. You must understand it better if you want to learn.

Inheritance

One of the major advantages of Object Oriented Programming is re-use. Inheritance is one of the mechanisms to achieve the same. Inheritance allows programmer to create a general or a base class first and then later extend it to more specialized class. It allows programmer to write better code.

Using inheritance you can use or inherit all the data fields and methods available in your base class. Later you can add you own methods and data fields, thus inheritance provides a way to organize code, rather than rewriting it from scratch.

In object-oriented terminology when class X extend class Y, then Y is called super/parent/base class and X is called subclass/child/derived class. One point to note here is that only data fields and method which are not private are accessible by child classes. Private data fields and methods are accessible only inside the class.

syntax to create a derived class is −

class BaseClass:
   Body of base class
class DerivedClass(BaseClass):
   Body of derived class

Inheriting Attributes

Now look at the below example −

Output

We first created a class called Date and pass the object as an argument, here-object is built-in class provided by Python. Later we created another class called time and called the Date class as an argument. Through this call we get access to all the data and attributes of Date class into the Time class. Because of that when we try to get the get_date method from the Time class object tm we created earlier possible.

Object.Attribute Lookup Hierarchy

  • The instance
  • The class
  • Any class from which this class inherits

Inheritance Examples

Let’s take a closure look into the inheritance example −

Let’s create couple of classes to participate in examples −

  • Animal − Class simulate an animal
  • Cat − Subclass of Animal
  • Dog − Subclass of Animal

In Python, constructor of class used to create an object (instance), and assign the value for the attributes.

Constructor of subclasses always called to a constructor of parent class to initialize value for the attributes in the parent class, then it start assign value for its attributes.

Output

In the above example, we see the command attributes or methods we put in the parent class so that all subclasses or child classes will inherits that property from the parent class.

If a subclass try to inherits methods or data from another subclass then it will through an error as we see when Dog class try to call swatstring() methods from that cat class, it throws an error(like AttributeError in our case).

Polymorphism (“MANY SHAPES”)

Polymorphism is an important feature of class definition in Python that is utilized when you have commonly named methods across classes or subclasses. This permits functions to use entities of different types at different times. So, it provides flexibility and loose coupling so that code can be extended and easily maintained over time.

This allows functions to use objects of any of these polymorphic classes without needing to be aware of distinctions across the classes.

Polymorphism can be carried out through inheritance, with subclasses making use of base class methods or overriding them.

Let understand the concept of polymorphism with our previous inheritance example and add one common method called show_affection in both subclasses −

From the example we can see, it refers to a design in which object of dissimilar type can be treated in the same manner or more specifically two or more classes with method of the same name or common interface because same method(show_affection in below example) is called with either type of objects.

Output

So, all animals show affections (show_affection), but they do differently. The “show_affection” behaviors is thus polymorphic in the sense that it acted differently depending on the animal. So, the abstract “animal” concept does not actually “show_affection”, but specific animals(like dogs and cats) have a concrete implementation of the action “show_affection”.

Python itself have classes that are polymorphic. Example, the len() function can be used with multiple objects and all return the correct output based on the input parameter.

Overriding

In Python, when a subclass contains a method that overrides a method of the superclass, you can also call the superclass method by calling

Super(Subclass, self).method instead of self.method.

Example

class Thought(object):
   def __init__(self):
      pass
   def message(self):
      print("Thought, always come and go")

class Advice(Thought):
   def __init__(self):
      super(Advice, self).__init__()
   def message(self):
      print('Warning: Risk is always involved when you are dealing with market!')

Inheriting the Constructor

If we see from our previous inheritance example, __init__ was located in the parent class in the up ‘cause the child class dog or cat didn’t‘ve __init__ method in it. Python used the inheritance attribute lookup to find __init__ in animal class. When we created the child class, first it will look the __init__ method in the dog class, then it didn’t find it then looked into parent class Animal and found there and called that there. So as our class design became complex we may wish to initialize a instance firstly processing it through parent class constructor and then through child class constructor.

Output

In above example- all animals have a name and all dogs a particular breed. We called parent class constructor with super. So dog has its own __init__ but the first thing that happen is we call super. Super is built in function and it is designed to relate a class to its super class or its parent class.

In this case we saying that get the super class of dog and pass the dog instance to whatever method we say here the constructor __init__. So in another words we are calling parent class Animal __init__ with the dog object. You may ask why we won’t just say Animal __init__ with the dog instance, we could do this but if the name of animal class were to change, sometime in the future. What if we wanna rearrange the class hierarchy,so the dog inherited from another class. Using super in this case allows us to keep things modular and easy to change and maintain.

So in this example we are able to combine general __init__ functionality with more specific functionality. This gives us opportunity to separate common functionality from the specific functionality which can eliminate code duplication and relate class to one another in a way that reflects the system overall design.

Conclusion

  • __init__ is like any other method; it can be inherited

  • If a class does not have a __init__ constructor, Python will check its parent class to see if it can find one.

  • As soon as it finds one, Python calls it and stops looking

  • We can use the super () function to call methods in the parent class.

  • We may want to initialize in the parent as well as our own class.

Multiple Inheritance and the Lookup Tree

As its name indicates, multiple inheritance is Python is when a class inherits from multiple classes.

For example, a child inherits personality traits from both parents (Mother and Father).

Python Multiple Inheritance Syntax

To make a class inherits from multiple parents classes, we write the the names of these classes inside the parentheses to the derived class while defining it. We separate these names with comma.

Below is an example of that −

>>> class Mother:
   pass

>>> class Father:
   pass

>>> class Child(Mother, Father):
   pass

>>> issubclass(Child, Mother) and issubclass(Child, Father)
True

Multiple inheritance refers to the ability of inheriting from two or more than two class. The complexity arises as child inherits from parent and parents inherits from the grandparent class. Python climbs an inheriting tree looking for attributes that is being requested to be read from an object. It will check the in the instance, within class then parent class and lastly from the grandparent class. Now the question arises in what order the classes will be searched - breath-first or depth-first. By default, Python goes with the depth-first.

That’s is why in the below diagram the Python searches the dothis() method first in class A. So the method resolution order in the below example will be

Mro- D→B→A→C

Look at the below multiple inheritance diagram −

Let’s go through an example to understand the “mro” feature of an Python.

Output

Example 3

Let’s take another example of “diamond shape” multiple inheritance.

Above diagram will be considered ambiguous. From our previous example understanding “method resolution order” .i.e. mro will be D→B→A→C→A but it’s not. On getting the second A from the C, Python will ignore the previous A. so the mro will be in this case will be D→B→C→A.

Let’s create an example based on above diagram −

Output

Simple rule to understand the above output is- if the same class appear in the method resolution order, the earlier appearances of this class will be remove from the method resolution order.

In conclusion −

  • Any class can inherit from multiple classes

  • Python normally uses a “depth-first” order when searching inheriting classes.

  • But when two classes inherit from the same class, Python eliminates the first appearances of that class from the mro.

Decorators, Static and Class Methods

Functions(or methods) are created by def statement.

Though methods works in exactly the same way as a function except one point where method first argument is instance object.

We can classify methods based on how they behave, like

  • Simple method − defined outside of a class. This function can access class attributes by feeding instance argument:

def outside_func(():
  • Instance method

def func(self,)
  • Class method − if we need to use class attributes

   @classmethod
def cfunc(cls,)
  • Static method − do not have any info about the class

      @staticmethod
def sfoo()

Till now we have seen the instance method, now is the time to get some insight into the other two methods,

Class Method

The @classmethod decorator, is a builtin function decorator that gets passed the class it was called on or the class of the instance it was called on as first argument. The result of that evaluation shadows your function definition.

syntax

class C(object):
   @classmethod
   def fun(cls, arg1, arg2, ...):
      ....
fun: function that needs to be converted into a class method
returns: a class method for function

They have the access to this cls argument, it can’t modify object instance state. That would require access to self.

  • It is bound to the class and not the object of the class.

  • Class methods can still modify class state that applies across all instances of the class.

Static Method

A static method takes neither a self nor a cls(class) parameter but it’s free to accept an arbitrary number of other parameters.

syntax

class C(object):
   @staticmethod
   def fun(arg1, arg2, ...):
   ...
returns: a static method for function funself.
  • A static method can neither modify object state nor class state.
  • They are restricted in what data they can access.

When to use what

  • We generally use class method to create factory methods. Factory methods return class object (similar to a constructor) for different use cases.

  • We generally use static methods to create utility functions.

Python Design Pattern

Overview

Modern software development needs to address complex business requirements. It also needs to take into account factors such as future extensibility and maintainability. A good design of a software system is vital to accomplish these goals. Design patterns play an important role in such systems.

To understand design pattern, let’s consider below example −

  • Every car’s design follows a basic design pattern, four wheels, steering wheel, the core drive system like accelerator-break-clutch, etc.

So, all things repeatedly built/ produced, shall inevitably follow a pattern in its design.. it cars, bicycle, pizza, atm machines, whatever…even your sofa bed.

Designs that have almost become standard way of coding some logic/mechanism/technique in software, hence come to be known as or studied as, Software Design Patterns.

Why is Design Pattern Important?

Benefits of using Design Patterns are −

  • Helps you to solve common design problems through a proven approach.

  • No ambiguity in the understanding as they are well documented.

  • Reduce the overall development time.

  • Helps you deal with future extensions and modifications with more ease than otherwise.

  • May reduce errors in the system since they are proven solutions to common problems.

Classification of Design Patterns

The GoF (Gang of Four) design patterns are classified into three categories namely creational, structural and behavioral.

Creational Patterns

Creational design patterns separate the object creation logic from the rest of the system. Instead of you creating objects, creational patterns creates them for you. The creational patterns include Abstract Factory, Builder, Factory Method, Prototype and Singleton.

Creational Patterns are not commonly used in Python because of the dynamic nature of the language. Also language itself provide us with all the flexibility we need to create in a sufficient elegant fashion, we rarely need to implement anything on top, like singleton or Factory.

Also these patterns provide a way to create objects while hiding the creation logic, rather than instantiating objects directly using a new operator.

Structural Patterns

Sometimes instead of starting from scratch, you need to build larger structures by using an existing set of classes. That’s where structural class patterns use inheritance to build a new structure. Structural object patterns use composition/ aggregation to obtain a new functionality. Adapter, Bridge, Composite, Decorator, Façade, Flyweight and Proxy are Structural Patterns. They offers best ways to organize class hierarchy.

Behavioral Patterns

Behavioral patterns offers best ways of handling communication between objects. Patterns comes under this categories are: Visitor, Chain of responsibility, Command, Interpreter, Iterator, Mediator, Memento, Observer, State, Strategy and Template method are Behavioral Patterns.

Because they represent the behavior of a system, they are used generally to describe the functionality of software systems.

Commonly used Design Patterns

Singleton

It is one of the most controversial and famous of all design patterns. It is used in overly object-oriented languages, and is a vital part of traditional object-oriented programming.

The Singleton pattern is used for,

  • When logging needs to be implemented. The logger instance is shared by all the components of the system.

  • The configuration files is using this because cache of information needs to be maintained and shared by all the various components in the system.

  • Managing a connection to a database.

Here is the UML diagram,

class Logger(object):
   def __new__(cls, *args, **kwargs):
      if not hasattr(cls, '_logger'):
      cls._logger = super(Logger, cls).__new__(cls, *args, **kwargs)
return cls._logger

In this example, Logger is a Singleton.

When __new__ is called, it normally constructs a new instance of that class. When we override it, we first check if our singleton instance has been created or not. If not, we create it using a super call. Thus, whenever we call the constructor on Logger, we always get the exact same instance.

>>>
>>> obj1 = Logger()
>>> obj2 = Logger()
>>> obj1 == obj2
True
>>>
>>> obj1
<__main__.Logger object at 0x03224090>
>>> obj2
<__main__.Logger object at 0x03224090>

Object Oriented Python - Advanced Features

In this we will look into some of the advanced features which Python provide

Core Syntax in our Class design

In this we will look onto, how Python allows us to take advantage of operators in our classes. Python is largely objects and methods call on objects and this even goes on even when its hidden by some convenient syntax.

>>> var1 = 'Hello'
>>> var2 = ' World!'
>>> var1 + var2
'Hello World!'
>>>
>>> var1.__add__(var2)
'Hello World!'
>>> num1 = 45
>>> num2 = 60
>>> num1.__add__(num2)
105
>>> var3 = ['a', 'b']
>>> var4 = ['hello', ' John']
>>> var3.__add__(var4)
['a', 'b', 'hello', ' John']

So if we have to add magic method __add__ to our own classes, could we do that too. Let’s try to do that.

We have a class called Sumlist which has a contructor __init__ which takes list as an argument called my_list.

class SumList(object):
   def __init__(self, my_list):
      self.mylist = my_list
   def __add__(self, other):
     new_list = [ x + y for x, y in zip(self.mylist, other.mylist)]

     return SumList(new_list)
   
   def __repr__(self):
      return str(self.mylist)

aa = SumList([3,6, 9, 12, 15])

bb = SumList([100, 200, 300, 400, 500])
cc = aa + bb # aa.__add__(bb)
print(cc) # should gives us a list ([103, 206, 309, 412, 515])

Output

[103, 206, 309, 412, 515]

But there are many methods which are internally managed by others magic methods. Below are some of them,

'abc' in var # var.__contains__('abc')
var == 'abc' # var.__eq__('abc')
var[1] # var.__getitem__(1)
var[1:3] # var.__getslice__(1, 3)
len(var) # var.__len__()
print(var) # var.__repr__()

Inheriting From built-in types

Classes can also inherit from built-in types this means inherits from any built-in and take advantage of all the functionality found there.

In below example we are inheriting from dictionary but then we are implementing one of its method __setitem__. This (setitem) is invoked when we set key and value in the dictionary. As this is a magic method, this will be called implicitly.

class MyDict(dict):

   def __setitem__(self, key, val):
      print('setting a key and value!')
      dict.__setitem__(self, key, val)

dd = MyDict()
dd['a'] = 10
dd['b'] = 20

for key in dd.keys():
   print('{0} = {1}'.format(key, dd[key]))

Output

setting a key and value!
setting a key and value!
a = 10
b = 20

Let’s extend our previous example, below we have called two magic methods called __getitem__ and __setitem__ better invoked when we deal with list index.

# Mylist inherits from 'list' object but indexes from 1 instead for 0!
class Mylist(list): # inherits from list
   def __getitem__(self, index):
      if index == 0:
         raise IndexError
      if index > 0:
         index = index - 1
         return list.__getitem__(self, index) # this method is called when

# we access a value with subscript like x[1]
   def __setitem__(self, index, value):
      if index == 0:
         raise IndexError
      if index > 0:
      index = index - 1
      list.__setitem__(self, index, value)

x = Mylist(['a', 'b', 'c']) # __init__() inherited from builtin list

print(x) # __repr__() inherited from builtin list

x.append('HELLO'); # append() inherited from builtin list

print(x[1]) # 'a' (Mylist.__getitem__ cutomizes list superclass
               # method. index is 1, but reflects 0!

print (x[4]) # 'HELLO' (index is 4 but reflects 3!

Output

['a', 'b', 'c']
a
HELLO

In above example, we set a three item list in Mylist and implicitly __init__ method is called and when we print the element x, we get the three item list ([‘a’,’b’,’c’]). Then we append another element to this list. Later we ask for index 1 and index 4. But if you see the output, we are getting element from the (index-1) what we have asked for. As we know list indexing start from 0 but here the indexing start from 1 (that’s why we are getting the first item of the list).

Naming Conventions

In this we will look into names we’ll used for variables especially private variables and conventions used by Python programmers worldwide. Although variables are designated as private but there is not privacy in Python and this by design. Like any other well documented languages, Python has naming and style conventions that it promote although it doesn’t enforce them. There is a style guide written by “Guido van Rossum” the originator of Python, that describe the best practices and use of name and is called PEP8. Here is the link for this, https://www.python.org/dev/peps/pep-0008/

PEP stands for Python enhancement proposal and is a series of documentation that distributed among the Python community to discuss proposed changes. For example it is recommended all,

  • Module names − all_lower_case
  • Class names and exception names − CamelCase
  • Global and local names − all_lower_case
  • Functions and method names − all_lower_case
  • Constants − ALL_UPPER_CASE

These are just the recommendation, you can vary if you like. But as most of the developers follows these recommendation so might me your code is less readable.

Why conform to convention?

We can follow the PEP recommendation we it allows us to get,

  • More familiar to the vast majority of developers
  • Clearer to most readers of your code.
  • Will match style of other contributers who work on same code base.
  • Mark of a professional software developers
  • Everyone will accept you.

Variable Naming − ‘Public’ and ‘Private’

In Python, when we are dealing with modules and classes, we designate some variables or attribute as private. In Python, there is no existence of “Private” instance variable which cannot be accessed except inside an object. Private simply means they are simply not intended to be used by the users of the code instead they are intended to be used internally. In general, a convention is being followed by most Python developers i.e. a name prefixed with an underscore for example. _attrval (example below) should be treated as a non-public part of the API or any Python code, whether it is a function, a method or a data member. Below is the naming convention we follow,

  • Public attributes or variables (intended to be used by the importer of this module or user of this class) −regular_lower_case

  • Private attributes or variables (internal use by the module or class) −_single_leading_underscore

  • Private attributes that shouldn’t be subclassed −__double_leading_underscore

  • Magic attributes −__double_underscores__(use them, don’t create them)

class GetSet(object):

   instance_count = 0 # public
   
   __mangled_name = 'no privacy!' # special variable

   def __init__(self, value):
      self._attrval = value # _attrval is for internal use only
      GetSet.instance_count += 1

   @property
   def var(self):
      print('Getting the "var" attribute')
      return self._attrval

   @var.setter
   def var(self, value):
      print('setting the "var" attribute')
      self._attrval = value

   @var.deleter
   def var(self):
      print('deleting the "var" attribute')
      self._attrval = None

cc = GetSet(5)
cc.var = 10 # public name
print(cc._attrval)
print(cc._GetSet__mangled_name)

Output

setting the "var" attribute
10
no privacy!

Object Oriented Python - Files and Strings

Strings

Strings are the most popular data types used in every programming language. Why? Because we, understand text better than numbers, so in writing and talking we use text and words, similarly in programming too we use strings. In string we parse text, analyse text semantics, and do data mining – and all this data is human consumed text.The string in Python is immutable.

String Manipulation

In Python, string can be marked in multiple ways, using single quote ( ‘ ), double quote( “ ) or even triple quote ( ‘’’ ) in case of multiline strings.

>>> # String Examples
>>> a = "hello"
>>> b = ''' A Multi line string,
Simple!'''
>>> e = ('Multiple' 'strings' 'togethers')

String manipulation is very useful and very widely used in every language. Often, programmers are required to break down strings and examine them closely.

Strings can be iterated over (character by character), sliced, or concatenated. The syntax is the same as for lists.

The str class has numerous methods on it to make manipulating strings easier. The dir and help commands provides guidance in the Python interpreter how to use them.

Below are some of the commonly used string methods we use.

Sr.No. Method & Description
1

isalpha()

Checks if all characters are Alphabets

2

isdigit()

Checks Digit Characters

3

isdecimal()

Checks decimal Characters

4

isnumeric()

checks Numeric Characters

5

find()

Returns the Highest Index of substrings

6

istitle()

Checks for Titlecased strings

7

join()

Returns a concatenated string

8

lower()

returns lower cased string

9

upper()

returns upper cased string

10

partion()

Returns a tuple

11

bytearray()

Returns array of given byte size

12

enumerate()

Returns an enumerate object

13

isprintable()

Checks printable character

Let’s try to run couple of string methods,

>>> str1 = 'Hello World!'
>>> str1.startswith('h')
False
>>> str1.startswith('H')
True
>>> str1.endswith('d')
False
>>> str1.endswith('d!')
True
>>> str1.find('o')
4
>>> #Above returns the index of the first occurence of the character/substring.
>>> str1.find('lo')
3
>>> str1.upper()
'HELLO WORLD!'
>>> str1.lower()
'hello world!'
>>> str1.index('b')
Traceback (most recent call last):
   File "<pyshell#19>", line 1, in <module>
      str1.index('b')
ValueError: substring not found
>>> s = ('hello How Are You')
>>> s.split(' ')
['hello', 'How', 'Are', 'You']
>>> s1 = s.split(' ')
>>> '*'.join(s1)
'hello*How*Are*You'
>>> s.partition(' ')
('hello', ' ', 'How Are You')
>>>

String Formatting

In Python 3.x formatting of strings has changed, now it more logical and is more flexible. Formatting can be done using the format() method or the % sign(old style) in format string.

The string can contain literal text or replacement fields delimited by braces {} and each replacement field may contains either the numeric index of a positional argument or the name of a keyword argument.

syntax

str.format(*args, **kwargs)

Basic Formatting

>>> '{} {}'.format('Example', 'One')
'Example One'
>>> '{} {}'.format('pie', '3.1415926')
'pie 3.1415926'

Below example allows re-arrange the order of display without changing the arguments.

>>> '{1} {0}'.format('pie', '3.1415926')
'3.1415926 pie'

Padding and aligning strings

A value can be padded to a specific length.

>>> #Padding Character, can be space or special character
>>> '{:12}'.format('PYTHON')
'PYTHON '
>>> '{:>12}'.format('PYTHON')
' PYTHON'
>>> '{:<{}s}'.format('PYTHON',12)
'PYTHON '
>>> '{:*<12}'.format('PYTHON')
'PYTHON******'
>>> '{:*^12}'.format('PYTHON')
'***PYTHON***'
>>> '{:.15}'.format('PYTHON OBJECT ORIENTED PROGRAMMING')
'PYTHON OBJECT O'
>>> #Above, truncated 15 characters from the left side of a specified string
>>> '{:.{}}'.format('PYTHON OBJECT ORIENTED',15)
'PYTHON OBJECT O'
>>> #Named Placeholders
>>> data = {'Name':'Raghu', 'Place':'Bangalore'}
>>> '{Name} {Place}'.format(**data)
'Raghu Bangalore'
>>> #Datetime
>>> from datetime import datetime
>>> '{:%Y/%m/%d.%H:%M}'.format(datetime(2018,3,26,9,57))
'2018/03/26.09:57'

Strings are Unicode

Strings as collections of immutable Unicode characters. Unicode strings provide an opportunity to create software or programs that works everywhere because the Unicode strings can represent any possible character not just the ASCII characters.

Many IO operations only know how to deal with bytes, even if the bytes object refers to textual data. It is therefore very important to know how to interchange between bytes and Unicode.

Converting text to bytes

Converting a strings to byte object is termed as encoding. There are numerous forms of encoding, most common ones are: PNG; JPEG, MP3, WAV, ASCII, UTF-8 etc. Also this(encoding) is a format to represent audio, images, text, etc. in bytes.

This conversion is possible through encode(). It take encoding technique as argument. By default, we use ‘UTF-8’ technique.

>>> # Python Code to demonstrate string encoding 
>>> 
>>> # Initialising a String 
>>> x = 'TutorialsPoint' 
>>> 
>>> #Initialising a byte object 
>>> y = b'TutorialsPoint'
>>> 
>>> # Using encode() to encode the String >>> # encoded version of x is stored in z using ASCII mapping 
>>> z = x.encode('ASCII') 
>>> 
>>> # Check if x is converted to bytes or not 
>>> 
>>> if(z==y): 
   print('Encoding Successful!') 
else: 
   print('Encoding Unsuccessful!') 
Encoding Successful!

Converting bytes to text

Converting bytes to text is called the decoding. This is implemented through decode(). We can convert a byte string to a character string if we know which encoding is used to encode it.

So Encoding and decoding are inverse processes.

>>> 
>>> # Python code to demonstrate Byte Decoding 
>>> 
>>> #Initialise a String 
>>> x = 'TutorialsPoint' 
>>> 
>>> #Initialising a byte object 
>>> y = b'TutorialsPoint' 
>>> 
>>> #using decode() to decode the Byte object 
>>> # decoded version of y is stored in z using ASCII mapping 
>>> z = y.decode('ASCII')
>>> #Check if y is converted to String or not 
>>> if (z == x): 
   print('Decoding Successful!') 
else: 
   print('Decoding Unsuccessful!') Decoding Successful! 
>>>

File I/O

Operating systems represents files as a sequence of bytes, not text.

A file is a named location on disk to store related information. It is used to permanently store data in your disk.

In Python, a file operation takes place in the following order.

  • Open a file
  • Read or write onto a file (operation).Open a file
  • Close the file.

Python wraps the incoming (or outgoing) stream of bytes with appropriate decode (or encode) calls so we can deal directly with str objects.

Opening a file

Python has a built-in function open() to open a file. This will generate a file object, also called a handle as it is used to read or modify the file accordingly.

>>> f = open(r'c:\users\rajesh\Desktop\index.webm','rb')
>>> f
<_io.BufferedReader name='c:\\users\\rajesh\\Desktop\\index.webm'>
>>> f.mode
'rb'
>>> f.name
'c:\\users\\rajesh\\Desktop\\index.webm'

For reading text from a file, we only need to pass the filename into the function. The file will be opened for reading, and the bytes will be converted to text using the platform default encoding.

Exception and Exception Classes

In general, an exception is any unusual condition. Exception usually indicates errors but sometimes they intentionally puts in the program, in cases like terminating a procedure early or recovering from a resource shortage. There are number of built-in exceptions, which indicate conditions like reading past the end of a file, or dividing by zero. We can define our own exceptions called custom exception.

Exception handling enables you handle errors gracefully and do something meaningful about it. Exception handling has two components: “throwing” and ‘catching’.

Identifying Exception (Errors)

Every error occurs in Python result an exception which will an error condition identified by its error type.

>>> #Exception
>>> 1/0
Traceback (most recent call last):
   File "<pyshell#2>", line 1, in <module>
      1/0
ZeroDivisionError: division by zero
>>>
>>> var = 20
>>> print(ver)
Traceback (most recent call last):
   File "<pyshell#5>", line 1, in <module>
      print(ver)
NameError: name 'ver' is not defined
>>> #Above as we have misspelled a variable name so we get an NameError.
>>>
>>> print('hello)

SyntaxError: EOL while scanning string literal
>>> #Above we have not closed the quote in a string, so we get SyntaxError.
>>>
>>> #Below we are asking for a key, that doen't exists.
>>> mydict = {}
>>> mydict['x']
Traceback (most recent call last):
   File "<pyshell#15>", line 1, in <module>
      mydict['x']
KeyError: 'x'
>>> #Above keyError
>>>
>>> #Below asking for a index that didn't exist in a list.
>>> mylist = [1,2,3,4]
>>> mylist[5]
Traceback (most recent call last):
   File "<pyshell#20>", line 1, in <module>
      mylist[5]
IndexError: list index out of range
>>> #Above, index out of range, raised IndexError.

Catching/Trapping Exception

When something unusual occurs in your program and you wish to handle it using the exception mechanism, you ‘throw an exception’. The keywords try and except are used to catch exceptions. Whenever an error occurs within a try block, Python looks for a matching except block to handle it. If there is one, execution jumps there.

syntax

try:
   #write some code
   #that might throw some exception
except <ExceptionType>:
   # Exception handler, alert the user

The code within the try clause will be executed statement by statement.

If an exception occurs, the rest of the try block will be skipped and the except clause will be executed.

try:
   some statement here
except:
   exception handling

Let’s write some code to see what happens when you not use any error handling mechanism in your program.

number = int(input('Please enter the number between 1 & 10: '))
print('You have entered number',number)

Above programme will work correctly as long as the user enters a number, but what happens if the users try to puts some other data type(like a string or a list).

Please enter the number between 1 > 10: 'Hi'
Traceback (most recent call last):
   File "C:/Python/Python361/exception2.py", line 1, in <module>
      number = int(input('Please enter the number between 1 & 10: '))
ValueError: invalid literal for int() with base 10: "'Hi'"

Now ValueError is an exception type. Let’s try to rewrite the above code with exception handling.

import sys

print('Previous code with exception handling')

try:
   number = int(input('Enter number between 1 > 10: '))

except(ValueError):
   print('Error..numbers only')
   sys.exit()

print('You have entered number: ',number)

If we run the program, and enter a string (instead of a number), we can see that we get a different result.

Previous code with exception handling
Enter number between 1 > 10: 'Hi'
Error..numbers only

Raising Exceptions

To raise your exceptions from your own methods you need to use raise keyword like this

raise ExceptionClass(‘Some Text Here’)

Let’s take an example

def enterAge(age):
   if age<0:
      raise ValueError('Only positive integers are allowed')
   if age % 2 ==0:
      print('Entered Age is even')
   else:
      print('Entered Age is odd')

try:
   num = int(input('Enter your age: '))
   enterAge(num)
except ValueError:
   print('Only positive integers are allowed')

Run the program and enter positive integer.

Expected Output

Enter your age: 12
Entered Age is even

But when we try to enter a negative number we get,

Expected Output

Enter your age: -2
Only positive integers are allowed

Creating Custom exception class

You can create a custom exception class by Extending BaseException class or subclass of BaseException.

From above diagram we can see most of the exception classes in Python extends from the BaseException class. You can derive your own exception class from BaseException class or from its subclass.

Create a new file called NegativeNumberException.py and write the following code.

class NegativeNumberException(RuntimeError):
   def __init__(self, age):
      super().__init__()
      self.age = age

Above code creates a new exception class named NegativeNumberException, which consists of only constructor which call parent class constructor using super()__init__() and sets the age.

Now to create your own custom exception class, will write some code and import the new exception class.

from NegativeNumberException import NegativeNumberException
def enterage(age):
   if age < 0:
      raise NegativeNumberException('Only positive integers are allowed')

   if age % 2 == 0:
      print('Age is Even')

   else:
      print('Age is Odd')

try:
   num = int(input('Enter your age: '))
   enterage(num)
except NegativeNumberException:
   print('Only positive integers are allowed')
except:
   print('Something is wrong')

Output

Enter your age: -2
Only positive integers are allowed

Another way to create a custom Exception class.

class customException(Exception):
   def __init__(self, value):
      self.parameter = value

   def __str__(self):
      return repr(self.parameter)
try:
   raise customException('My Useful Error Message!')
except customException as instance:
   print('Caught: ' + instance.parameter)

Output

Caught: My Useful Error Message!

Exception hierarchy

The class hierarchy for built-in exceptions is −

+-- SystemExit 
+-- KeyboardInterrupt 
+-- GeneratorExit 
+-- Exception 
+-- StopIteration 
+-- StopAsyncIteration 
+-- ArithmeticError 
| +-- FloatingPointError 
| +-- OverflowError 
| +-- ZeroDivisionError 
+-- AssertionError 
+-- AttributeError 
+-- BufferError 
+-- EOFError 
+-- ImportError 
+-- LookupError 
| +-- IndexError 
| +-- KeyError 
+-- MemoryError 
+-- NameError 
| +-- UnboundLocalError 
+-- OSError 
| +-- BlockingIOError 
| +-- ChildProcessError 
| +-- ConnectionError 
| | +-- BrokenPipeError 
| | +-- ConnectionAbortedError 
| | +-- ConnectionRefusedError 
| | +-- ConnectionResetError 
| +-- FileExistsError 
| +-- FileNotFoundError 
| +-- InterruptedError 
| +-- IsADirectoryError 
| +-- NotADirectoryError 
| +-- PermissionError 
| +-- ProcessLookupError 
| +-- TimeoutError 
+-- ReferenceError 
+-- RuntimeError 
| +-- NotImplementedError 
| +-- RecursionError 
+-- SyntaxError 
| +-- IndentationError
| +-- TabError 
+-- SystemError 
+-- TypeError 
+-- ValueError 
| +-- UnicodeError 
| +-- UnicodeDecodeError 
| +-- UnicodeEncodeError 
| +-- UnicodeTranslateError 
+-- Warning 
+-- DeprecationWarning 
+-- PendingDeprecationWarning 
+-- RuntimeWarning 
+-- SyntaxWarning 
+-- UserWarning 
+-- FutureWarning 
+-- ImportWarning 
+-- UnicodeWarning 
+-- BytesWarning 
+-- ResourceWarning

Object Oriented Python - Object Serialization

In the context of data storage, serialization is the process of translating data structures or object state into a format that can be stored (for example, in a file or memory buffer) or transmitted and reconstructed later.

In serialization, an object is transformed into a format that can be stored, so as to be able to deserialize it later and recreate the original object from the serialized format.

Pickle

Pickling is the process whereby a Python object hierarchy is converted into a byte stream (usually not human readable) to be written to a file, this is also known as Serialization. Unpickling is the reverse operation, whereby a byte stream is converted back into a working Python object hierarchy.

Pickle is operationally simplest way to store the object. The Python Pickle module is an object-oriented way to store objects directly in a special storage format.

What can it do?

  • Pickle can store and reproduce dictionaries and lists very easily.
  • Stores object attributes and restores them back to the same State.

What pickle can’t do?

  • It does not save an objects code. Only it’s attributes values.
  • It cannot store file handles or connection sockets.

In short we can say, pickling is a way to store and retrieve data variables into and out from files where variables can be lists, classes, etc.

To Pickle something you must −

  • import pickle
  • Write a variable to file, something like
pickle.dump(mystring, outfile, protocol),

where 3rd argument protocol is optional To unpickling something you must −

Import pickle

Write a variable to a file, something like

myString = pickle.load(inputfile)

Methods

The pickle interface provides four different methods.

  • dump() − The dump() method serializes to an open file (file-like object).

  • dumps() − Serializes to a string

  • load() − Deserializes from an open-like object.

  • loads() − Deserializes from a string.

Based on above procedure, below is an example of “pickling”.

Output

My Cat pussy is White and has 4 legs
Would you like to see her pickled? Here she is!
b'\x80\x03c__main__\nCat\nq\x00)\x81q\x01}q\x02(X\x0e\x00\x00\x00number_of_legsq\x03K\x04X\x05\x00\x00\x00colorq\x04X\x05\x00\x00\x00Whiteq\x05ub.'

So, in the example above, we have created an instance of a Cat class and then we’ve pickled it, transforming our “Cat” instance into a simple array of bytes.

This way we can easily store the bytes array on a binary file or in a database field and restore it back to its original form from our storage support in a later time.

Also if you want to create a file with a pickled object, you can use the dump() method ( instead of the dumps*()* one) passing also an opened binary file and the pickling result will be stored in the file automatically.

[….]
binary_file = open(my_pickled_Pussy.bin', mode='wb')
my_pickled_Pussy = pickle.dump(Pussy, binary_file)
binary_file.close()

Unpickling

The process that takes a binary array and converts it to an object hierarchy is called unpickling.

The unpickling process is done by using the load() function of the pickle module and returns a complete object hierarchy from a simple bytes array.

Let’s use the load function in our previous example.

Output

MeOw is black
Pussy is white

JSON

JSON(JavaScript Object Notation) has been part of the Python standard library is a lightweight data-interchange format. It is easy for humans to read and write. It is easy to parse and generate.

Because of its simplicity, JSON is a way by which we store and exchange data, which is accomplished through its JSON syntax, and is used in many web applications. As it is in human readable format, and this may be one of the reasons for using it in data transmission, in addition to its effectiveness when working with APIs.

An example of JSON-formatted data is as follow −

{"EmployID": 40203, "Name": "Zack", "Age":54, "isEmployed": True}

Python makes it simple to work with Json files. The module sused for this purpose is the JSON module. This module should be included (built-in) within your Python installation.

So let’s see how can we convert Python dictionary to JSON and write it to a text file.

JSON to Python

Reading JSON means converting JSON into a Python value (object). The json library parses JSON into a dictionary or list in Python. In order to do that, we use the loads() function (load from a string), as follow −

Output

Below is one sample json file,

data1.json
{"menu": {
   "id": "file",
   "value": "File",
   "popup": {
      "menuitem": [
         {"value": "New", "onclick": "CreateNewDoc()"},
         {"value": "Open", "onclick": "OpenDoc()"},
         {"value": "Close", "onclick": "CloseDoc()"}
      ]
   }
}}

Above content (Data1.json) looks like a conventional dictionary. We can use pickle to store this file but the output of it is not human readable form.

JSON(Java Script Object Notification) is a very simple format and that’s one of the reason for its popularity. Now let’s look into json output through below program.

Output

Above we open the json file (data1.json) for reading, obtain the file handler and pass on to json.load and getting back the object. When we try to print the output of the object, its same as the json file. Although the type of the object is dictionary, it comes out as a Python object. Writing to the json is simple as we saw this pickle. Above we load the json file, add another key value pair and writing it back to the same json file. Now if we see out data1.json, it looks different .i.e. not in the same format as we see previously.

To make our Output looks same (human readable format), add the couple of arguments into our last line of the program,

json.dump(conf, fh, indent = 4, separators = (‘,’, ‘: ‘))

Similarly like pickle, we can print the string with dumps and load with loads. Below is an example of that,

YAML

YAML may be the most human friendly data serialization standard for all programming languages.

Python yaml module is called pyaml

YAML is an alternative to JSON −

  • Human readable code − YAML is the most human readable format so much so that even its front-page content is displayed in YAML to make this point.

  • Compact code − In YAML we use whitespace indentation to denote structure not brackets.

  • Syntax for relational data − For internal references we use anchors (&) and aliases (*).

  • One of the area where it is used widely is for viewing/editing of data structures − for example configuration files, dumping during debugging and document headers.

Installing YAML

As yaml is not a built-in module, we need to install it manually. Best way to install yaml on windows machine is through pip. Run below command on your windows terminal to install yaml,

pip install pyaml (Windows machine)
sudo pip install pyaml (*nix and Mac)

On running above command, screen will display something like below based on what’s the current latest version.

Collecting pyaml
Using cached pyaml-17.12.1-py2.py3-none-any.whl
Collecting PyYAML (from pyaml)
Using cached PyYAML-3.12.tar.gz
Installing collected packages: PyYAML, pyaml
Running setup.py install for PyYAML ... done
Successfully installed PyYAML-3.12 pyaml-17.12.1

To test it, go to the Python shell and import the yaml module, import yaml, if no error is found, then we can say installation is successful.

After installing pyaml, let’s look at below code,

script_yaml1.py

Above we created three different data structure, dictionary, list and tuple. On each of the structure, we do yaml.dump. Important point is how the output is displayed on the screen.

Output

Dictionary output looks clean .ie. key: value.

White space to separate different objects.

List is notated with dash (-)

Tuple is indicated first with !!Python/tuple and then in the same format as lists.

Loading a yaml file

So let’s say I have one yaml file, which contains,

---
# An employee record
name: Raagvendra Joshi
job: Developer
skill: Oracle
employed: True
foods:
   - Apple
   - Orange
   - Strawberry
   - Mango
languages:
   Oracle: Elite
   power_builder: Elite
   Full Stack Developer: Lame
education:
   4 GCSEs
   3 A-Levels
   MCA in something called com

Now let’s write a code to load this yaml file through yaml.load function. Below is code for the same.

As the output doesn’t looks that much readable, I prettify it by using json in the end. Compare the output we got and the actual yaml file we have.

Output

One of the most important aspect of software development is debugging. In this section we’ll see different ways of Python debugging either with built-in debugger or third party debuggers.

PDB – The Python Debugger

The module PDB supports setting breakpoints. A breakpoint is an intentional pause of the program, where you can get more information about the programs state.

To set a breakpoint, insert the line

pdb.set_trace()

Example

pdb_example1.py
import pdb
x = 9
y = 7
pdb.set_trace()
total = x + y
pdb.set_trace()

We have inserted a few breakpoints in this program. The program will pause at each breakpoint (pdb.set_trace()). To view a variables contents simply type the variable name.

c:\Python\Python361>Python pdb_example1.py
> c:\Python\Python361\pdb_example1.py(8)<module>()
-> total = x + y
(Pdb) x
9
(Pdb) y
7
(Pdb) total
*** NameError: name 'total' is not defined
(Pdb)

Press c or continue to go on with the programs execution until the next breakpoint.

(Pdb) c
--Return--
> c:\Python\Python361\pdb_example1.py(8)<module>()->None
-> total = x + y
(Pdb) total
16

Eventually, you will need to debug much bigger programs – programs that use subroutines. And sometimes, the problem that you’re trying to find will lie inside a subroutine. Consider the following program.

import pdb
def squar(x, y):
   out_squared = x^2 + y^2
   return out_squared
if __name__ == "__main__":
   #pdb.set_trace()
   print (squar(4, 5))

Now on running the above program,

c:\Python\Python361>Python pdb_example2.py
> c:\Python\Python361\pdb_example2.py(10)<module>()
-> print (squar(4, 5))
(Pdb)

We can use ? to get help, but the arrow indicates the line that’s about to be executed. At this point it’s helpful to hit s to s to step into that line.

(Pdb) s
--Call--
>c:\Python\Python361\pdb_example2.py(3)squar()
-> def squar(x, y):

This is a call to a function. If you want an overview of where you are in your code, try l −

(Pdb) l
1 import pdb
2
3 def squar(x, y):
4 -> out_squared = x^2 + y^2
5
6 return out_squared
7
8 if __name__ == "__main__":
9 pdb.set_trace()
10 print (squar(4, 5))
[EOF]
(Pdb)

You can hit n to advance to the next line. At this point you are inside the out_squared method and you have access to the variable declared inside the function .i.e. x and y.

(Pdb) x
4
(Pdb) y
5
(Pdb) x^2
6
(Pdb) y^2
7
(Pdb) x**2
16
(Pdb) y**2
25
(Pdb)

So we can see the ^ operator is not what we wanted instead we need to use ** operator to do squares.

This way we can debug our program inside the functions/methods.

Logging

The logging module has been a part of Python’s Standard Library since Python version 2.3. As it’s a built-in module all Python module can participate in logging, so that our application log can include your own message integrated with messages from third party module. It provides a lot of flexibility and functionality.

Benefits of Logging

  • Diagnostic logging − It records events related to the application’s operation.

  • Audit logging − It records events for business analysis.

Messages are written and logged at levels of “severity” &minu

  • DEBUG (debug()) − diagnostic messages for development.

  • INFO (info()) − standard “progress” messages.

  • WARNING (warning()) − detected a non-serious issue.

  • ERROR (error()) − encountered an error, possibly serious.

  • CRITICAL (critical()) − usually a fatal error (program stops).

Let’s looks into below simple program,

import logging

logging.basicConfig(level=logging.INFO)

logging.debug('this message will be ignored') # This will not print
logging.info('This should be logged') # it'll print
logging.warning('And this, too') # It'll print

Above we are logging messages on severity level. First we import the module, call basicConfig and set the logging level. Level we set above is INFO. Then we have three different statement: debug statement, info statement and a warning statement.

Output of logging1.py

INFO:root:This should be logged
WARNING:root:And this, too

As the info statement is below debug statement, we are not able to see the debug message. To get the debug statement too in the Output terminal, all we need to change is the basicConfig level.

logging.basicConfig(level = logging.DEBUG)

And in the Output we can see,

DEBUG:root:this message will be ignored
INFO:root:This should be logged
WARNING:root:And this, too

Also the default behavior means if we don’t set any logging level is warning. Just comment out the second line from the above program and run the code.

#logging.basicConfig(level = logging.DEBUG)

Output

WARNING:root:And this, too

Python built in logging level are actually integers.

>>> import logging
>>>
>>> logging.DEBUG
10
>>> logging.CRITICAL
50
>>> logging.WARNING
30
>>> logging.INFO
20
>>> logging.ERROR
40
>>>

We can also save the log messages into the file.

logging.basicConfig(level = logging.DEBUG, filename = 'logging.log')

Now all log messages will go the file (logging.log) in your current working directory instead of the screen. This is a much better approach as it lets us to do post analysis of the messages we got.

We can also set the date stamp with our log message.

logging.basicConfig(level=logging.DEBUG, format = '%(asctime)s %(levelname)s:%(message)s')

Output will get something like,

2018-03-08 19:30:00,066 DEBUG:this message will be ignored
2018-03-08 19:30:00,176 INFO:This should be logged
2018-03-08 19:30:00,201 WARNING:And this, too

Benchmarking

Benchmarking or profiling is basically to test how fast is your code executes and where the bottlenecks are? The main reason to do this is for optimization.

timeit

Python comes with a in-built module called timeit. You can use it to time small code snippets. The timeit module uses platform-specific time functions so that you will get the most accurate timings possible.

So, it allows us to compare two shipment of code taken by each and then optimize the scripts to given better performance.

The timeit module has a command line interface, but it can also be imported.

There are two ways to call a script. Let’s use the script first, for that run the below code and see the Output.

import timeit
print ( 'by index: ', timeit.timeit(stmt = "mydict['c']", setup = "mydict = {'a':5, 'b':10, 'c':15}", number = 1000000))
print ( 'by get: ', timeit.timeit(stmt = 'mydict.get("c")', setup = 'mydict = {"a":5, "b":10, "c":15}', number = 1000000))

Output

by index: 0.1809192126703489
by get: 0.6088525265034692

Above we use two different method .i.e. by subscript and get to access the dictionary key value. We execute statement 1 million times as it executes too fast for a very small data. Now we can see the index access much faster as compared to the get. We can run the code multiply times and there will be slight variation in the time execution to get the better understanding.

Another way is to run the above test in the command line. Let’s do it,

c:\Python\Python361>Python -m timeit -n 1000000 -s "mydict = {'a': 5, 'b':10, 'c':15}" "mydict['c']"
1000000 loops, best of 3: 0.187 usec per loop

c:\Python\Python361>Python -m timeit -n 1000000 -s "mydict = {'a': 5, 'b':10, 'c':15}" "mydict.get('c')"
1000000 loops, best of 3: 0.659 usec per loop

Above output may vary based on your system hardware and what all applications are running currently in your system.

Below we can use the timeit module, if we want to call to a function. As we can add multiple statement inside the function to test.

import timeit

def testme(this_dict, key):
   return this_dict[key]

print (timeit.timeit("testme(mydict, key)", setup = "from __main__ import testme; mydict = {'a':9, 'b':18, 'c':27}; key = 'c'", number = 1000000))

Output

0.7713474590139164

Object Oriented Python - Libraries

Requests − Python Requests Module

Requests is a Python module which is an elegant and simple HTTP library for Python. With this you can send all kinds of HTTP requests. With this library we can add headers, form data, multipart files and parameters and access the response data.

As Requests is not a built-in module, so we need to install it first.

You can install it by running the following command in the terminal −

pip install requests

Once you have installed the module, you can verify if the installation is successful by typing below command in the Python shell.

import requests

If the installation has been successful, you won’t see any error message.

Making a GET Request

As a means of example we’ll be using the “pokeapi”

Output −

Making POST Requests

The requests library methods for all of the HTTP verbs currently in use. If you wanted to make a simple POST request to an API endpoint then you can do that like so −

req = requests.post(‘http://api/user’, data = None, json = None)

This would work in exactly the same fashion as our previous GET request, however it features two additional keyword parameters −

  • data which can be populated with say a dictionary, a file or bytes that will be passed in the HTTP body of our POST request.

  • json which can be populated with a json object that will be passed in the body of our HTTP request also.

Pandas: Python Library Pandas

Pandas is an open-source Python Library providing high-performance data manipulation and analysis tool using its powerful data structures. Pandas is one of the most widely used Python libraries in data science. It is mainly used for data munging, and with good reason: Powerful and flexible group of functionality.

Built on Numpy package and the key data structure is called the DataFrame. These dataframes allows us to store and manipulate tabular data in rows of observations and columns of variables.

There are several ways to create a DataFrame. One way is to use a dictionary. For example −

Output

From the output we can see new brics DataFrame, Pandas has assigned a key for each country as the numerical values 0 through 4.

If instead of giving indexing values from 0 to 4, we would like to have different index values, say the two letter country code, you can do that easily as well −

Adding below one lines in the above code, gives

brics.index = ['BR', 'RU', 'IN', 'CH', 'SA']

Output

Indexing DataFrames

Output

Pygame

Pygame is the open source and cross-platform library that is for making multimedia applications including games. It includes computer graphics and sound libraries designed to be used with the Python programming language. You can develop many cool games with Pygame.’

Overview

Pygame is composed of various modules, each dealing with a specific set of tasks. For example, the display module deals with the display window and screen, the draw module provides functions to draw shapes and the key module works with the keyboard. These are just some of the modules of the library.

The home of the Pygame library is at https://www.pygame.org/news

To make a Pygame application, you follow these steps −

Import the Pygame library

import pygame

Initialize the Pygame library

pygame.init()

Create a window.

screen = Pygame.display.set_mode((560,480))
Pygame.display.set_caption(‘First Pygame Game’)

Initialize game objects

In this step we load images, load sounds, do object positioning, set up some state variables, etc.

Start the game loop.

It is just a loop where we continuously handle events, checks for input, move objects, and draw them. Each iteration of the loop is called a frame.

Let’s put all the above logic into one below program,

Pygame_script.py

Output

Beautiful Soup: Web Scraping with Beautiful Soup

The general idea behind web scraping is to get the data that exists on a website, and convert it into some format that is usable for analysis.

It’s a Python library for pulling data out of HTML or XML files. With your favourite parser it provide idiomatic ways of navigating, searching and modifying the parse tree.

As BeautifulSoup is not a built-in library, we need to install it before we try to use it. To install BeautifulSoup, run the below command

$ apt-get install Python-bs4 # For Linux and Python2 
$ apt-get install Python3-bs4 # for Linux based system and Python3.

$ easy_install beautifulsoup4 # For windows machine, 
Or 
$ pip instal beatifulsoup4 # For window machine

Once the installation is done, we are ready to run few examples and explores Beautifulsoup in details,

Output

Below are some simple ways to navigate that data structure −

One common task is extracting all the URLs found within a page’s <a> tags −

Another common task is extracting all the text from a page −