The Self Taught Programmer: Definitive Guide To Programming Professionally 349441547 Programmer
User Manual:
Open the PDF directly: View PDF .
Page Count: 278 [warning: Documents this large are best viewed by clicking the View PDF Link!]
- amazon
- The Self-taught Programmer
- ____________________________
- Cory Althoff
- Part I Introduction to Programming
- Chapter 1. Introduction
- Chapter 2. Getting Started
- What is Programming
- What is Python
- Installing Python
- Troubleshooting
- The Interactive Shell
- Saving Programs
- Running Example Programs
- Vocabulary
- Challenge
- Examples
- Comments
- Printing
- Lines
- Keywords
- Spacing
- Data Types
- Constants and Variables
- Syntax
- Errors and Exceptions
- Arithmetic Operators
- Comparison Operators
- Logical Operators
- Conditional Statements
- Statements
- Vocabulary
- Challenge
- Chapter 4. Functions
- Chapter 5. Containers
- Chapter 6. String Manipulation
- Chapter 7. Loops
- Chapter 8. Modules
- Chapter 9. Files
- Chapter 10. Bringing It All Together
- Chapter 11. Practice
- Part II Introduction to Object-oriented Programming
- Part III Introduction to Programming Tools
- Part IV Introduction to Computer Science
- Part V Programming for Production
- Part VI Land a Job
TheSelf-taughtProgrammer
____________________________
CoryAlthoff
Copyright©2016byCoryAlthoff
Allrightsreserved.Thisbookoranyportionthereof
maynotbereproducedorusedinanymannerwhatsoever
withouttheexpresswrittenpermissionofthepublisher
exceptfortheuseofbriefquotationsinabookreview.
ISBN978-1-940733-01-2
LibraryofCongressNumber
www.theselftaughtprogrammer.io
ThisbookisdedicatedtomyparentsAbbyandJamesAlthoffforalwayssupportingme.
PartIIntroductiontoProgramming
Chapter1.Introduction
"Mostgoodprogrammersdoprogrammingnotbecausetheyexpecttogetpaidorget
adulationbythepublic,butbecauseitisfuntoprogram."
—LinusTorvalds
ImajoredinPoliticalScienceatClemsonUniversity.BeforeIchosethispathIconsidered
ComputerScience.IevenenrolledinanIntroductiontoProgrammingclassmyFreshman
year,butquicklydroppedit.Itwastoodifficult.WhilelivinginSiliconValleyafter
graduationIdecidedIneededtolearntoprogram.Ayearlater,Iwasworkingasasoftware
engineerIIateBay(aboveanentrylevelsoftwareengineer,butbelowaseniorsoftware
engineer).Idon’twanttogivetheimpressionthatthiswaseasy.Itwasincrediblychallenging.
Inbetweenthrowingthingsatthewall,itwasalotoffuntoo.
IstartedmyjourneylearningtoprograminPython,apopularprogramminglanguage.
Thisbook,however,isnotaboutteachingyouhowtoprograminaspecificlanguage
(althoughitdoes).Thereareplentyofamazingbooks,classesandresourcesthatdothat
already.Thefocusiseverythingelsethosestandardresourcesdonotteachyou.It’saboutthe
thingsIhadtolearnonmyowninordertobecomeasoftwareengineer.Thisbookisnot
meantforsomeonelookingforacasualintroductiontoprogrammingsotheycanwritecode
asahobby.Thisbookiswrittenspecificallyforthoselookingtoprogramprofessionally.
Whetheryourgoalistobecomeasoftwareengineer,anentrepreneurortouseyournew
programmingskillsinanotherprofession,thisbookwaswrittenforyou.
Learningaprogramminglanguageisonlypartofthebattle.Thereareotherskillsyou
needinordertospeakthelanguageofcomputerscientists.IwillteachyoueverythingI
learnedonmyjourneyfromprogrammingnovicetoprofessionalsoftwareengineer.Iwrote
thisbooktogiveaspiringprogrammersanoutlineofwhattheyneedtoknow.Asaself-taught
programmer,Ididn’tknowwhatIneededtolearn.Theintroductiontoprogrammingbooks
areallthesame.TheyteachyouthebasicsofhowtoprogramineitherPythonorRubyand
sendyouonyourway.ThefeedbackI’veheardfrompeoplefinishingthesebooksis,“What
doIdonow?Iamnotaprogrammeryet,andIdon’tknowwhattolearnnext.”Thisbookis
myanswertothatquestion.
HowThisBookIsStructured
Thisbookisdividedintosixparts,basedonmovingthroughthefollowingstages:
learningtoprogram,learningobject-orientedprogramming,learningtouseprograms(like
youroperatingsystem)thatwillmakeyouabetterprogrammer,learningComputerScience,
learningtoprogramforproductionandgettingajobandworkingonateam.
Manyofthesubjectscoveredinasinglechapterofthisbookcouldbe—andare—
coveredbyentirebooks.Mygoalisnottocovereverydetailofeverysubjectyouneedto
know.Mygoalistogiveyouamap—anoutlineofalloftheskillsyouneedtodevelopin
ordertoprogramprofessionally.
PartI:IntroductiontoProgramming.Youwillwriteyourfirstprogramasquicklyas
possible,hopefullytoday.Ifyoualreadyknowhowtoprogramyoucanusethissectionto
startlearningPython.IfyoualreadyknowPython,skipit.
PartII:IntroductiontoObject-orientedProgramming.Icoverthedifferentprogramming
paradigms—focussingonobject-orientedprogramming—andbuildagamethatwillshow
youthepowerofprogramming.Afterthissection,you’llbehookedonprogramming.
PartIII:IntroductiontoProgrammingTools.Youlearntousedifferenttoolstotakeyour
programmingproductivitytothenextlevel.Bythispointyouarehookedonprogramming
andwanttogetevenbetter.Youwilllearnmoreaboutyouroperatingsystem,howto
collaboratewithotherengineersusingversioncontrol,howtouseyourInteractive
DevelopmentEnvironmenttoboostyourproductivityandhowtoinstallandmanageother
people'sprograms.
PartIV:IntroductiontoComputerScience.Nowthatyoucanprogram,youwillhaveall
kindsofquestionsabouthoweverythingworks.Thissectioniswherealotofthosequestions
getanswered.Icoveralgorithmsanddatastructures,networkprogrammingandcomputer
architecture.
PartV:ProgrammingforProduction.Youwilllearntoprogramforproduction(create
codethatisactuallyusedbyotherpeople).Icoverthesoftwaredevelopmentprocess,testing
andbestprogrammingpractices.
PartVI:LandaJob.Thefinalsectionisaboutgettingajobasasoftwareengineer,
workingonateamandimprovingasaprogrammer.Iprovidetipsonhowtopassatechnical
interview,workonateamaswellasadviceonhowtofurtherimproveyourskills.
Ifyoudon’thaveanyprogrammingexperience,youshouldtrytopractice
programmingonyourownasmuchaspossiblebetweeneachsection.Thereareadditional
resourcestoexploreprovidedattheendofeachsection.Don’ttrytoreadthisbooktoo
quickly.Instead,useitasaguideandpracticeforaslongasyouneedinbetweensections.
EndgameFirst
ThewayIlearnedtoprogramistheoppositeofhowComputerScienceisusually
taught,andIstructuredthebooktofollowthisapproach.Traditionally,youspendalotof
timelearningtheory;somuchso,thatmanyComputerSciencegraduatescomeoutof
schoolnotknowinghowtoprogram.Inhisblog,WhyCan’tProgrammers..Program?,Jeff
Atwoodwrites:“Likeme,theauthorishavingtroublewiththefactthat199outof200
applicantsforeveryprogrammingjobcan’twritecodeatall.Irepeat:theycan’twriteany
codewhatsoever.”ThisledAtwoodtocreatetheFizzBuzzcodingchallenge,aprogramming
testusedinprogramminginterviewstoweedoutcandidates.Mostpeoplefailthechallenge,
andthat’swhywespendsomuchofthisbooklearningtheskillsyouwilluseinpractice.
Don’tworry,wealsolearnhowtopasstheFizzBuzztest.
InTheArtofLearning,JoshWaitzkinofSearchingforBobbyFischerfame,describes
howhelearnedhowtoplaychessinreverse.Insteadofstudyingopeningmoves,hestarted
learningtheendgame(wherethereareonlyafewpiecesleftontheboard)first.Thisgave
himabetterunderstandingofthegame,andhewentontowinmanychampionships.
Similarly,Ithinkitismoreeffectivetolearntoprogramfirst,thenlearntheorylater,once
youaredyingtoknowhoweverythingworksunderthehood.ThatiswhyIwaituntilthe
fourthsectionofthebooktointroduceComputerSciencetheory.Whiletheoryisimportant,it
willbeevenmorevaluableonceyoualreadyhaveprogrammingexperience.
TheSelf-taughtAdvantage
Learninghowtoprogramoutsideofschoolisincreasinglycommon.A2015Stack
Overflow(anonlinecommunityofprogrammers)surveyfound48percentofrespondents
didnothaveadegreeincomputerscience10.WhenIwashiredateBay,Iwasonateamthat
includedprogrammerswithCSdegreesfromStanford,CalandDuke,aswellastwoPhysics
PhD’s.At25,itwasintimidatingtorealizethatmy21-year-oldteammatesknew10times
moreaboutprogrammingandcomputersciencethanIdid.
Asintimidatingasitmightbetoworkwithpeoplewhohavebachelor ’s,master ’sand
PhD’sinComputerScience,neverforgetyouhavewhatIliketocallthe“self-taught
advantage.”Youarenotreadingthisbookbecauseateacherassignedittoyou,youare
readingitbecauseyouhaveadesiretolearn,andwantingtolearnisthebiggestadvantage
youcanhave.
WhyYouShouldProgram
Programmingcanhelpyourcareerregardlessofyourprofession.Ifyouarereading
thisbookIassumeyouhavealreadydecidedyouwanttolearntoprogram.ButI’mstillgoing
tocoverwhyyoushouldtogiveyouextramotivation.Learningtoprogramisempowering.I
lovecomingupwithnewideas,andIalwayshaveanewprojectIwanttoworkon.OnceI
learnedhowtoprogram,Icouldsitdownandbuildmyideaswithoutneedingtofind
someonetodoitforme.
Programmingwillalsomakeyoubebetterateverything.Seriously.Therearen’tmany
subjectsthatdon’tbenefitfromfinelytunedproblem-solvingskills.Recently,Ihadtodothe
verytedioustaskofsearchingforhousingonCraigslist,andIwasabletowriteaprogram
todotheworkformeandemailmetheresults.Learningtoprogramwillfreeyoufrom
repetitivetasksforever.
Ifyoudowanttobecomeasoftwareengineer,thereisanincreasingdemandforgood
engineersandnotenoughqualifiedpeopletofilltheavailablepositions.By2020,an
estimatedonemillionprogrammingjobswillgounfilled45.Evenifyourgoalisn’tto
becomeasoftwareengineer,jobsinfieldslikescienceandfinancearebeginningtofavor
candidateswithprogrammingexperience.
StickingWithIt
Ifyoudon’thaveanyprogrammingexperienceandarenervousaboutmakingthis
journey,Iwantyoutoknowyouareabsolutelycapableofdoingit.Therearesomecommon
misconceptionsaboutprogrammersliketheyallarereallygoodatmath.Thisisn’ttrue,butit
ishardwork.Luckily,alotofthematerialcoveredinthisbookiseasiertolearnthanyou
think.
Inordertoimproveyourprogrammingskillsyoushouldpracticeprogrammingevery
day.Theonlythingthatwillholdyoubackisnotstickingwithit,solet’sgooversomeways
tomakesureyoudo.IusedachecklisttomakesureIpracticedeverydayanditreallyhelped
mestayfocused.
Ifyouneedextrahelp,TimFerriss,aproductivityexpert,recommendsthefollowing
techniquetostaymotivated.Givemoneytofamilyorfriendswiththeinstructionsthatitis
eithertobereturnedtoyouuponcompletingyourgoalwithinatimeframeofyourchoosing,
ordonatedtoanorganizationyoudislike.
HowThisBookisFormatted
Thechaptersinthisbookbuildononeanother.Ifyouseesomethingyoudon’t
understand,itmighthavebeencoveredinapreviouschapter.Itrynottore-explainconcepts,
sokeepthisinmind.Importantwordsareitalicizedwhentheyaredefined.Thereisa
vocabularysectionattheendofeachchapterwhereeachitalicizedwordisdefined.
TechnologiesUsedInThisBook
Thisbookteachescertaintechnologiesinordertogiveyouasmuchpractical
programmingexperienceaspossible.InsomecasesIhadtochoosebetweenmanydifferent
populartechnologies.InChapter20:“VersionControl”(forthosereaderswhoare
unfamiliarwithversioncontrol,Iwillexplainlater),wegooverthebasicsofusingGit,a
popularversioncontrolsystem.IchoseGitbecauseIconsiderittheindustrystandardfor
versioncontrol.IusePythonforthemajorityofprogrammingexamplesbecauseitisa
popularfirstprogramminglanguagetolearn,anditisaveryeasylanguagetoread,evenif
youhaveneverusedit.ThereisalsoahugedemandforPythondevelopersinjustaboutevery
field.But,Itrytobeastechnologyagnosticaspossible:focusingonconceptsinsteadof
technologies.
Inordertofollowtheexamplesinthisbookyouwillneedacomputer.Youcomputer
hasanoperatingsystem—aprogramthatisthemiddlemanbetweenthephysicalcomponents
ofthecomputerandyou.Whatyouseewhenyoulookatyourscreeniscalledagraphical
userinterfaceorGUI,whichispartofbyyouroperatingsystem.
Therearethreepopularoperatingsystemsfordesktopandlaptopcomputers:Windows,
UnixandLinux.WindowsisMicrosoft’soperatingsystem.Unixisanoperatingsystem
createdinthe1970s.Currently,themostpopularUnixoperatingsystemisApple’sOSX.
Linuxisanopen-sourceoperatingsystemusedbythemajorityoftheworld’sservers.A
serverisacomputerorcomputerprogramthatperformstaskslikehostingawebsite.Open-
sourcemeansthesoftwareisnotownedbyacompanyorindividual,insteaditismaintained
byagroupofvolunteers.LinuxandUnixarebothUnix-likeoperatingsystems,whichmeans
theyareverysimilar.ThisbookassumesyouareusingacomputerrunningWindows,OSX
orUbuntu(apopularversionofLinux)asyouroperatingsystem.
Vocabulary
FizzBuzz:aprogrammingtestusedinprogramminginterviewstoweedoutcandidates.
operatingsystem:Aprogramthatisthemiddlemanbetweenthephysicalcomponentsofthe
computerandyou.
open-source:Softwarethatisnotownedbyacompanyorindividual,butisinstead
maintainedbyagroupofvolunteers.
Windows:Microsoft’soperatingsystem.
Unix:Anoperatingsystemcreatedinthe1970s.Apple’sOSXisaversionofUnix.
Linux:Anopen-sourceoperatingsystemusedbythemajorityoftheworld’sservers.
server:Acomputerorcomputerprogramthatperformstaskslikehostingawebsite.
Challenge
Createadailychecklistthatincludespracticingprogramming.
Chapter2.GettingStarted
“Agoodprogrammerissomeonewhoalwayslooksbothwaysbeforecrossingaone-way
street.”
—DougLinder
WhatisProgramming
Programmingiswritinginstructionsforacomputertoexecute.Theinstructions
mighttellthecomputertoprintHello,World!,scrapedatafromtheinternetorreadthe
contentsofafileandsavethemtoadatabase.Theseinstructionsarecalledcode.
Programmerswritecodeinmanydifferentprogramminglanguages.Inthepast,
programmingwasmuchharder,asprogrammerswereforcedtousecryptic,low-level
programminglanguageslikeassemblylanguage.Whenaprogramminglanguageislow-
levelitisclosertobeingwritteninbinary(0sand1s)thanahigh-levelprogramminglanguage
(aprogramminglanguagethatreadsmorelikeEnglish),andthusishardertounderstand.
Hereisanexampleofasimpleprogramwritteninanassemblylanguage:
global_start
section.text
_start:
movrax,1
movrdi,1
movrsi,message
movrdx,13
syscall
;exit(0)
moveax,60
xorrdi,rdi
syscall
message:
db"Hello",10
43
Hereisthesameprogramwritteninamodernprogramminglanguage:
print("Hello,World!")
Asyoucansee,programmerstodayhaveitmucheasier.Youwon’tneedtospendtime
learningcryptic,low-levelprogramminglanguagesinordertoprogram.Instead,youwill
learntouseaneasy-to-readprogramminglanguagecalledPython.
WhatisPython
Pythonisanopen-sourceprogramminglanguagecreatedbyDutchprogrammer
GuidovanRossumandnamedaftertheBritishsketchcomedygroup,MontyPython’sFlying
Circus.OneofvanRossum’skeyinsightswasthatprogrammersspendmoretimereading
codethanwritingit,sohecreatedaneasy-to-readlanguage.Pythonisoneofthemost
popularandeasiesttolearnprogramminglanguagesintheworld.Itrunsonallthemajor
operatingsystemsandcomputersandisusedineverythingfrombuildingwebserversto
creatingdesktopapplications.Becauseofitspopularity,thereisalargedemandforPython
programmers.
InstallingPython
Inordertofollowtheexamplesinthisbook,youneedtohavePython3(version3of
Python)installed.YoucandownloadPythonforWindowsandOSXat
http://python.org/downloads.IfyouareonUbuntu,Python3comesinstalledbydefault.
MakesureyoudownloadPython3,notPython2.Someoftheexamplesinthisbookwill
notworkifyouareusingPython2.
Pythonisavailableforboth32-bitand64-bitcomputers.Ifyouhaveanewcomputer,
purchasedafter2007,itismostlikelya64-bitcomputer.Ifyouaren’tsureifyourcomputer
is32-bitor64-bit,aninternetsearchshouldhelpyoufigureitout.
IfyouareonWindowsoraMac,downloadthe64-or32-bitversionofPython,open
thefileandfollowtheinstructions.Youcanalsovisit
http://theselftaughtprogrammer.io/installpythonforvideosexplaininghowtoinstallPython
oneachoperatingsystem.
Troubleshooting
Fromthispointforward,youneedtohavePythoninstalled.Ifyouhavingproblems
installingPython,pleaseskipaheadtochapter11tothesectiontitled“GettingHelp”.
TheInteractiveShell
PythoncomeswithaprogramcalledIDLE(shortforinteractivedevelopment
environment);itisalsothelastnameofEricIdle,oneofthemembersofMontyPython’s
FlyingCircus.IDLEiswhereyouwillbetypingyourPythoncode.Onceyou’vedownloaded
Python,searchforIDLEinExplorer(PC),Finder(Mac)orNautilus(Ubuntu).I
recommendcreatingadesktopshortcuttomakeiteasytofind.
ClickontheIDLEicon,andaprogramwiththefollowinglineswillopenup(althoughthis
couldchangesodon’tworryifthemessageisabsentordifferent):
Python3.5.1(v3.5.1:37a07cee5969,Dec52015,21:12:44)
[GCC4.2.1(AppleInc.build5666)(dot3)]ondarwin
Type"copyright","credits"or"license()"formoreinformation.
>>>
Thisprogramiscalledtheinteractiveshell.YoucantypePythoncodedirectlyintothe
interactiveshellandtheshellprintstheresults.Attheprompt(>>>)type:
print("Hello,World!")
andpressenter.
IDLEmightrejectcodethatiscopiedfromKindle,othereBooksorwordprocessors
likeMicrosoftWord.Ifyoucopyandpastecodeandgetanunexplainableerrormessage,try
typingthecodedirectlyintothewindow.Youmusttypethecodeexactlyasitiswrittenin
theexample,includingquotationmarks,parenthesesandanyotherpunctuation.
TheinteractiveshellwillrespondbyprintingHello,World!
Intheprogrammingworld,whenyouteachsomeoneanewprogramminglanguage,it
istraditionthatthefirstprogramyouteachthemishowtoprintHello,World!So,
congratulations!Youjustwroteyourfirstprogram.
SavingPrograms
Theinteractiveshellisusefulforquickcomputations,testingsmallbitsofcodeand
writingshortprogramsyoudon’tplanonusingagain.YoucanalsouseIDLEtosavea
programforreuse.StarttheIDLEapplication,click“File”(inthemenubaronthetopleft
oftheIDLEeditor)thenselect“NewFile.”Thiswillopenupatexteditor,whichusuallyhas
ablankwhitebackground.Youcanwriteyourcodeinthistexteditor,thensaveittorunlater.
Whenyourunyourcode,theoutputofthecodeyouwrotewillappearintheinteractiveshell.
Youneedtosaveyourchangeswhileeditingcodebeforerunningitagain.TypetheHello,
World!programintothetexteditor:
Goto“File”againandselect“SaveAs.”Nameyourfile“hello_world.py”andsaveit.
Pythonfileshavetoendwith.py.Onceyou’vesavedyourfile,click“Run”(again,inthe
menubarinthetopleftcorneroftheIDLEeditor)andselect“RunModule.”Alternatively,
youcanpresstheF5keycommand,theequivalentofselecting“RunModule”fromthemenu
bar.HelloWorld!willprintintheinteractiveshell,asifyouhadtypedthislineofcode.
However,inthiscase,sinceyousavedyourprogram,youcanrunitasmanytimesasyou
like.
Theprogramyoucreatedissimplyafilewitha.pyextension,locatedonyour
computerwhereveryousavedit.ThenameIchoseforthefile—“hello_world.py”—is
completelyarbitrary,youcannamethefileanything.Likethisexample,writingprogramsin
Pythonsimplyinvolvestypingtextintofilesandrunningthemusingtheinteractiveshell.
Easy,right?
RunningExamplePrograms
Throughoutthebook,Iwillgiveexamplesofcodeandtheresultsthatprintoutwhen
yourunit.WheneverIdothis,youshouldenterthecodeandrunityourself.
Sometimesyoushouldtypethecodeintheinteractiveshell,andsometimesyoushould
typethecodeintoanew.pyfile.Iwillexplainhowyouwillknowwheretorunyourcodein
thenextchapter.
Thereasonforthisisshortexamplesarebestrunusingtheshell,andthetexteditoris
betterforlongerprogramsyouwanttosaveandedit.Intheinteractiveshell,ifyoumakea
mistakeinyourcode—atypoforexample—andthecodedoesn’twork,youhavetotype
everythingagain.Usingthetexteditorletsyousaveyourwork,soifyoumakeamistake,you
simplyeditthemistakeandreruntheprogram.
Anotherreasonthedistinctionisimportantistheoutputofaprogramrunningfroma
fileversustheshellcanbeslightlydifferent.Ifyoutype100intotheinteractiveshelland
pressenter,theinteractiveshellwilloutput100.Ifyoutype100intoa.pyfileandrunit,
therewillbenooutput.Thisdifferencecancauseconfusion,sobemindfulofwhereyouare
runningaprogramfromifyoudonotgetthesameoutputastheexample.
Vocabulary
programming:Writinginstructionsforacomputertoexecute.
code:Theinstructionsprogrammerswriteforacomputertoexecute.
low-levelprogramminglanguage:Aprogramminglanguageclosertobeingwrittenin
binary(0sand1s)thanahigh-levelprogramminglanguage.
assemblylanguage:Atypeofdifficulttoreadprogramminglanguage.
high-levelprogramminglanguage:AprogramminglanguagethatreadsmorelikeEnglish
thanalow-levelprogramminglanguage.
Python:Theeasytoread,open-sourceprogramminglanguageyouwilllearntouseinthis
book.CreatedbyGuidovanRossumandnamedaftertheBritishsketchcomedygroup,Monty
Python’sFlyingCircus.
Challenge
TrytoprintsomethingotherthanHello,World!.
Chapter3.IntroductiontoProgramming
“It’stheonlyjobIcanthinkofwhereIgettobebothanengineerandanartist.There’san
incredible,rigorous,technicalelementtoit,whichIlikebecauseyouhavetodoveryprecise
thinking.Ontheotherhand,ithasawildlycreativesidewheretheboundariesofimagination
aretheonlyreallimitation.”
—AndyHertzfeld
OurfirstprogramprintedHello,World!.Let’sprintitahundredtimes.Typethefollowing
codeintotheinteractiveshell(printneedstobeindentedbyexactlyfourspaces):
foriinrange(100):
print("Hello,World!")
YourshellshouldprintHello,World!ahundredtimes.Eventhoughyouwillprobably
neverneedtoprintHello,World!ahundredtimes,thisexamplequicklyshowsyouhow
powerfulprogrammingis.Canyouthinkofanythingelseyoucandoahundredtimesso
easily?Ican’t.Thatisthepowerofprogramming.
Examples
Throughoutthebook,youwillsee“>>”aftereachprogrammingexample.This
representstheoutputoftheprogram(printedintheinteractiveshell).Ellipses(...)mean“and
soon.”Thepreviousprogramwillbewrittenlikethisifitismeanttobetypedintothe
interactiveshell:
foriinrange(100):
print("HelloWorld")
>>HelloWorld
>>HelloWorld
>>HelloWorld
…
Andlikethisifitshouldberunfroma.pyfile:
#https://goo.gl/rKQedq
foriinrange(100):
print("HelloWorld")
>>HelloWorld
>>HelloWorld
>>HelloWorld
…
Thedifferencebetweenthetwoexamplesistheadditionof#https://goo.gl/8eou3Zat
thetopoftheprogram.Thisiscalledacomment(explainedinthenextsection).TheURL(
https://goo.gl/8eou3Z)willtakeyoutoawebpagethatcontainsthecodefromtheexample
soyoucaneasilycopyandpasteitintotheIDLEtexteditorifyouarehavingproblems
gettingthecodetorun.Wheneveryouseeacommentlikethis,youshouldrunthecodeinthe
examplefroma.pyfile.
Inbothexamples,thecolor-codedtextrepresentscode,andallthetextthatcomesafter
“>>”representstheoutputoftheinteractiveshell.AnythingwritteninCourierNewfont
representssomeformofcodeorcodeoutput.Forexample,ifIrefertothewordforinthe
previousprogramitwillbewrittenintheCourierNewfont.
CourierNewisatypeoffix-width(non-proportional)fontoftenusedtodisplay
programmingtextbecauseeachcharacterhasthesamewidth,soindentationandotherdisplay
characteristicsofcodealignmentareeasiertoobserve.
Comments
Acommentisaline(orpartofaline)ofcodewritteninEnglish(oranotherlanguage)
precededbyaspecialsymbolthattellstheprogramminglanguageyouareusingtoignore
thatline(orpartofaline)ofcode.InPython,thepoundsymbolisusedtocreatecomments.
Acommentexplainswhatalineofcodedoes.Programmersusecommentstoclarify
whytheydidsomethingtomakethelineofcodeeasiertounderstandforwhoeverreadsit.
Youcanwritewhateveryouwantinacomment,aslongasitisonlyonelinelong.Hereisan
exampleofacommenttakingupanentirelineofcode,followedbyanexampleofacomment
takinguppartofalineofcode:
#Thisisacomment
print("Thecommentdoesnotaffectthiscode")
>>Thecommentdoesnotaffectthiscode
print("Thecommentdoesnotaffectthiscode")#Thisisacomment
>>Thecommentdoesnotaffectthiscode
Youshouldonlywriteacommentifyouaredoingsomethingunusualinyourcode,orto
explainsomethingthatisnotobviousinthecodeitself.Usecommentssparingly—donot
commentoneverylineofcodeyouwrite—savethemforspecialsituations.Hereisan
exampleofanunnecessarycomment:
print("Hello,World!")#thislineofcodeprintsHello,World!
Itisunnecessarybecauseitisveryclearwhatthislineofcodedoes.Hereisanexampleofa
goodcomment:
d=math.sqrt(l**2+w**2)#lengthofthediagonal
Evenifyouunderstoodexactlyhowthiscodeworks(andyouwillbetheendofPartI),you
stillmightnotknowhowtocalculatethelengthofadiagonalofarectangle,whichiswhythis
commentisuseful.
Printing
Wearenotlimitedtoprinting“Hello,World!”inourprograms.Wecanprintwhatever
we’dlikeaslongaswesurrounditwithquotes:
print("Python")
>>Python
print("Hola!")
>>Hola!
Lines
Pythonprogramsaredividedintolinesofcode.Takealookatthisprogram:
#line1
#line2
#line3
Therearethreelinesofcode.Itisusefultorefertoeachpieceofcodebythelineitison.In
IDLEyoucangoto“Edit”andselect“Gotoline”tojumptoaspecificlineinyourprogram.
Sometimesapieceofcodeislongandtakesupmorethanoneline.Whenthishappens,itis
okforthecodetoextendtothenextline(whenextendedaccordingtoPython'srules):
print("""Thisisareallyreallyreallyreallyreallyreally
reallyreallylonglineofcode.""")
IwillexplaintherulesforextendinglinesofPythoncodeinPartV.
Keywords
ThePythonlanguagehasalistofwordswithspecialmeaning.Thesearecalledkeywords
.forisakeywordwe’vealreadyseenthatisusedtoexecutecodemultipletimes.Wewill
learnmorekeywordsthroughoutthischapter.
Spacing
Let’stakeanotherlookatourprogramthatprintsHello,World!ahundredtimes:
#https://goo.gl/rKQedq
foriinrange(100):
print("Hello,World!")
AsInotedearlier,printisindentedbyfourspaces.Wewillcoverwhyshortly,butithas
todowithlettingPythonknowwhenblocksofcodebeginandend.Inthemeantime,pleasebe
awarethatwheneveryouseeanindentinanexampleitisanindentoffourspaces.Without
properspacing,yourprogramwillnotwork.
Spacingisnotusedlikethisinotherprogramminglanguages.Otherapproaches
includeusingkeywordsorbracketsinsteadofspacing.Pythonproponentsbelievethe
requireduseofproperspacingmakesPythonlesstedioustoreadandwritethanother
languages,butthebestapproachtothisproblemisfrequentlydebatedamongstprogrammers.
DataTypes
DifferentkindsofdatainPythonaregroupedintodifferentcategories,ordatatypes.In
Python,eachdatavalue,like2or"Hello,World!",iscalledanobject.Wewilllearnmore
aboutobjectsinPartII,butfornowthinkofanobjectasadatavalueinPythonwiththree
properties:anidentity,adatatypeandavalue.Theidentityofanobjectiswhereitisstoredin
memory,whichneverchanges(andwhichwewillbeignoringfornow).Thedatatypeofan
objectisthecategoryofdatatheobjectbelongsto,anddeterminesthepropertiestheobject
has.Thisalsoneverchanges.Thevalueofanobjectisthedataitrepresents—thenumber2,
forexample,hasavalueof2.
"Hello,World!"isanobjectwithadatatypecalledstr,shortforstring,andthevalue
"Hello,World!".Whenwerefertoanobjectwithastrdatatype,wecallitastring.Astringis
asequenceofoneormorecharacterssurroundedbyquotes.Youcanusesinglequotesor
doublequotes,butthequotesatthebeginningandendofagivenstringmustmatch:
“Hello,World!”
>>Hello,World!
‘Hello,World!’
>>Hello,World!
Acharactercanbeanysymbolfoundinaunicodecharactertableliketheinteractive
onefoundathttp://unicode-table.com/en(IexplainwhatunicodeisinPartIV).Stringsare
usedtorepresenttext,andtheyhaveuniqueproperties.
Thenumbersweusedtodomathintheprevioussectionarealsoobjects—buttheyare
notstrings,theyhaveadifferentdatatype.Wholenumbershavethedatatypeint,shortfor
integer.Numberslike2,3,4and10allhavethedatatypeint.Anotherwayofsayingthisis
theyareallintegers.Likestrings,integershavetheirownproperties.Forexample,youcan
dividetwointegers,butyoucannotdividetwostrings.
Althoughwholenumbersareintegers,fractionalnumbers(numberswithadecimal
point)haveadifferentdatatypecalledfloat.2.1,8.2and9.9999areallexamplesofobjects
withthefloatdatatype.Theyarecalledfloats.Likealldatatypes,floatshavetheirown
propertiesandbehaveinacertainway.Floatsbehavesimilarlytointegers:
2.2+2.2
>>4.4
But,therearesomeimportantdifferencesbetweenfloatsandintegersyouwilllearnabout
later.
ObjectswithabooldatatypehaveavalueofeitherTrueorFalseandarecalled
booleans:
True
>>True
False
>>False
ObjectswithadatatypeNoneTypealwayshavethevalueNone.Objectswitha
NoneTypedatatypeareusedtorepresenttheabsenceofavalue:
None
>>None
Iwillexplainhowthedifferentdatatypesareusedinprogrammingthroughoutthis
chapter.
ConstantsandVariables
YoucanusePythontodomathjustlikeyouwoulduseacalculator.Youcanadd,
subtract,divide,multiply,raiseanumbertoapowerandmuchmore.Remembertotypeallof
theexamplesinthissectionintotheshell.
2+2
>>4
2-2
>>0
4/2
>>2
2*2
>>4
Aconstantisavaluethatneverchanges.Eachofthenumbersinthepreviousexample
isaconstant:thenumbertwowillalwaysrepresentthevalue2.Avariable,ontheotherhand,
referstoavalue;butthatvaluecanchange.Avariableconsistsofanamemadeupofoneor
morecharacters.Thatnamegetsassignedtoavalueusingtheassignmentoperator(the=
sign).Unlikeaconstant,thevalueofavariablevaluecanchange.
Someprogramminglanguagesrequiretheprogrammertoincludevariable
"declarations"thattelltheprogramminglanguagewhatdatatypethevariablewillbe.Python
makesitsimpler:youcreateavariablesimplybyassigningavaluetoitusingtheassignment
operator.Hereisanexampleofcreatingavariable:
b=100
b
>>100
Hereishowwecanchangethevalueofavariable:
x=100
x
x=200
x
>>100
>>200
Wecanalsousetwovariablestoperformarithmeticoperations:
x=10
y=10
z=x+y
z
a=x-y
a
>>20
>>0
Oftenwhenprogramming,youwanttoincrement(increase)thevalueofavariable,or
decrement(decrease)thevalueofavariable.Becausethisissuchacommonoperation,
Pythonhasaspecialsyntax—ashortcut—forincrementinganddecrementingvariables.To
incrementavariable,youassignthevariabletoitself,andontheothersideoftheequalssign
youaddthevariabletothenumberyouwanttoincrementby:
x=10
x=x+1
x
>>11
Todecrementavariable,youdothesamething,butinsteadsubtractthenumberyou
wanttodecrementby:
x=10
x=10-1
x
>>9
Theseexamplesareperfectlyvalid,butthereisashortermethodforincrementingand
decrementingvariablesyoushoulduseinstead:
x=10
x+=1
x
>>11
x=10
x-=1
x
>>9
Variablesarenotlimitedtostoringintegervalues—theycanrefertoanydatatype:
my_string=“Hello,World!”
>>
my_float=2.2
>>
my_boolean=True
>>
Youcannamevariableswhateveryou’dlike,aslongasyoufollowfourrules:
0.Variablescan’thavespaces.Ifyouwanttousetwowordsinavariable,putan
underscorebetweenthem:i.e.,my_variable=“Astring!”
0.Variablenamescanonlycontainletters,numbersandtheunderscoresymbol.
0.Youcannotstartavariablenamewithanumber.Althoughyoucanstartavariable
withanunderscore,ithasaspecialmeaningthatwewillcoverlater,soavoidusingit
untilthen.
0.YoucannotusePythonkeywordsforvariablenames.Youcanfindalistofkeywords
here:http://zetcode.com/lang/python/keywords
Syntax
Syntaxisthesetofrules,principles,andprocessesthatgovernthestructureof
sentencesinagivenlanguage,specificallywordorder.70TheEnglishlanguagehasa
syntax,andsodoesPython.Forexample,inEnglishyoumustendasentencewithaperiod.
Pythonalsohasasetofrulesthatmustbefollowed,soitalsohassyntax.
InPython,stringsarealwayssurroundedbyquotes.ThisisanexampleofPython’s
syntax.ThefollowingisavalidPythonprogram:
print("Hello,World!")
ItisavalidprogrambecausewefollowedPython’ssyntaxbyusingquotesaroundourtext
whenwedefinedastring.Ifweonlyusedquotesononesideofourtextwewouldviolate
Python’ssyntax,andourcodewouldnotwork.Pythondoesnotlikewhenyouviolateits
syntax,andinthenextsectionyoufindoutwhathappenswhenyoudo.
ErrorsandExceptions
IfyouwriteaPythonprogramanddisregardPython’ssyntaxyouwillgetoneormore
errorswhenyourunyourprogram.Whenthishappens,thePythonshellwillinformyou
yourcodedidnotwork,andgivesyouinformationabouttheerrorthatoccurred.Lookat
whathappensifyoutrytodefineastringinPythonwithaquoteononlyoneside:
my_string=“HelloWorld.
>>File"/Users/coryalthoff/PycharmProjects/se.py",line1
my_string='d
^
SyntaxError:EOLwhilescanningstringliteral
Thisiscalledasyntaxerror.Syntaxerrorsarefatal.Aprogramcannotrunwithasyntax
error.Whenyourunaprogramwithasyntaxerror,Pythonletsyouknowaboutitinthe
shell.Themessagewilltellyouwhatfiletheerrorwasin,whatlineitoccurredonandwhat
kindoferroritwas.Althoughthiserrormaylookintimidating,errorslikethishappenallthe
time.
Whenthereisanerrorinyourcode,youshouldgotothelinenumbertheproblem
occurredon,andtrytofigureoutwhatyoudidwrong.Inthisexample,youwouldgotoline
1ofyourcode.Afterstaringatitforawhile,youwouldeventuallynoticethereisonlyone
quote.Tofixit,addaquoteattheendofthestringandreruntheprogram.Fromthispoint
forward,Iwillrepresenttheoutputofanerrorlikethis:
>>SyntaxError:EOLwhilescanningstringliteral
Foreasierreading,Iwillonlyshowthelastlineoftheerror.
Pythonhastwokindsoferrors:syntaxerrorsandexceptions.Anyerrorthatisnota
syntaxerrorisanexception.AZeroDivisionErrorisanexampleofanexceptionthatoccurs
ifyoutrydividingbyzero.
Thedifferencebetweenasyntaxerrorandanexceptionisexceptionsarenot
necessarilyfatal(thereisawaytomakeaprogramrunevenifthereisanexceptionwhich
youwilllearnaboutinthenextchapter).Whenanexceptionoccurs,Pythonprogrammerssay
“Python(oryourprogram)raisedanexception”.Hereisanexampleofanexception:
10/0
>>ZeroDivisionError:integerdivisionormodulobyzero
Ifyouincorrectlyindentyourcode,yougetanIndentationError:
y=2
x=1
>>IndentationError:unexpectedindent
Asyouarelearningtoprogram,youwillfrequentlygetsyntaxerrorsandexceptions
(includingoneswedidnotcover),buttheywilldecreaseovertime.Remember,whenyourun
intoasyntaxerrororexception,gotothelinewheretheproblemoccurredandstareatituntil
youfigureoutthesolution(orsearchtheinternetfortheerrororexceptionifyouare
stumped).
ArithmeticOperators
EarlierweusedPythontodosimplearithmeticcalculationslike4/2.Thesymbolswe
usedinthoseexamplesarecalledoperators.OperatorsinPythonaredividedintoseveral
categories,andtheoneswe’veseensofararecalledarithmeticoperators.Hereisalistof
someofthemostcommonarithmeticoperatorsinPython:
Operator Meaning Example Evaluatesto
** Exponent 2**2 4
% Modulo/remainder 14%4 2
// Integerdivision/floored
quotient 13//8 1
/ Division 13/8 1.625
* Multiplication 8*2 16
- Subtraction 7-1 6
+ Addition 2+2 4
Whentwonumbersaredividedthereisaquotientandaremainder.Thequotientistheresult
ofthedivisionandtheremainderiswhatisleftover.Themodulooperatorreturnsthe
remainderwhentwonumbersaredivided:
2%2
>>0
10%3
>>1
Therearetwodifferentoperatorsfordivision.Thefirstis//,whichreturnsthequotient:
14//3
>4
Thesecondis/,whichreturnstheresultofthefirstnumberdividedbythesecondasa
floatingpointnumber:
14/3
>4.666666666666667
Youcanraiseanumberbyanexponentwiththeexponentoperator:
2**2
>>4
Thevalues(inthiscasenumbers)oneithersideofanoperatorarecalledoperands.
Together,twooperandsandanoperatorformanexpression.Whenyourprogramruns,
Pythonevaluateseachexpression,andreturnsasinglevalue.Whenyoutypeanexpression
like2+2intothePythonshell,theexpressionisevaluatedto4.
Theorderofoperationsisasetofrulesusedinmathematicalcalculationstoevaluate
anexpression.RememberPleaseExcuseMyDearAuntSally?Itisanacronymusedtohelp
youremembertheorderofoperationsinmathequations:parentheses,exponents,
multiplication,division,additionandsubtraction.Parenthesesoutrankexponents,which
outrankmultiplicationanddivision,whichoutrankadditionandsubtraction.Ifthereisatie
amongoperators,likeinthecaseof15/3×2,youevaluatefromlefttoright:.Inthiscase
theansweristheresultof15dividedby3times2.Pythonfollowsthesameorderof
operationswhenitevaluatesmathematicalexpressions:
2+2*2
>>6
(2+2)*2
>>8
Inthefirstexample,2*2isevaluatedfirst,followedby2+2,becausemultiplicationtakes
precedenceoveraddition.Inthesecondexample,(2+2)isevaluatedfirst,because
expressionsinparenthesesarealwaysevaluatedfirst.
ComparisonOperators
ComparisonoperatorsareanothercategoryofoperatorsinPython.Comparison
operatorsaresimilartoarithmeticoperators;theyareoperatorsusedinexpressionswith
operandsoneitherside.Unlikeexpressionswitharithmeticoperators,expressionswith
comparisonoperatorsevaluatetoeitherTrueorFalse.
Operator Meaning Example Evaluatesto
> Greaterthan 100>10 True
< Lessthan 100<10 False
>= Greaterthanorequal 2<=2 True
<= Lessthanorequal 22/8 2.75
== Equal 3*5 15
!= Notequal 5-2 3
Anexpressionwiththe>operatorreturnsthevalueTrueifthenumberontherightisbigger
thanthenumberontheleft,andFalseifitisnot:
100>10
>>True
Anexpressionwiththe<operatorreturnsthevalueTrueifthenumberontheleftissmaller
thanthenumberontheright,andFalseifitisnot:
100<10
>>False
Anexpressionwiththe>=operatorreturnsthevalueTrueifthenumberontherightis
biggerthanorequaltothenumberontheleft.OtherwisetheexpressionreturnsFalse:
2>=2
>>True
Anexpressionwiththe<=operatorreturnsthevalueTrueifthenumberontherightis
smallerthanorequaltothenumberontheleft.OtherwisetheexpressionreturnsFalse:
2<=2
>>True
Anexpressionwiththe==operatorreturnsthevalueTrueifthetwooperandsareequal,and
Falseifnot:
2==2
>>True
1==2
>>False
Whereasanexpressionwiththe!=operatorcheckreturnsTrueifthetwooperandsarenot
equal,andFalseotherwise:
1!=2
>>True
2!=2
>>False
Earlier,weassignedvariablestonumbers,likex=100using=.Itmaybetemptingto
readthisinyourheadas“xequals100,”butdon’t.Aswesawearlier,=isusedtoassigna
valuetoavariable,nottocheckforequality.Whenyouseex=100,youshouldthink“xgets
onehundred.”Thecomparisonoperator==isusedtocheckforequality,soifyouseex==
100,thenyoushouldthink“xequals100”.
LogicalOperators
Thelastcategoryofoperatorswewillcoverarecalledlogicaloperators.Expressions
withlogicaloperatorsaresimilartoexpressionswithcomparisonoperatorsinthattheyboth
evaluatetoTrueorFalse.
Operator Meaning Example Evaluatesto
and and Trueand
True True
or or Trueand
False True
not not notTrue False
ThePythonkeywordandtakestwoexpressionsandreturnsTrueifalltheexpressions
evaluatetoTrue.IfanyoftheexpressionsareFalse,itreturnsFalse.
1==1and2==2
>>True
1==2and2==2
>>False
1==1and2==1
>>False
2==1and1==1
>>False
Youcanusetheandkeywordmultipletimesinonestatement:
1==1and10!=2and1%1!=3and2<10
>>True
WhenPythonevaluatesastatementwiththeandkeyworditstopsevaluatingthestatementas
soonasanexpressionevaluatestoFalse.Pythondoesnotevaluatetheremainingexpressions
becauseitisgoingtoreturnFalseregardlessofhowtherestoftheexpressionsevaluate
(onlyoneexpressionhastoevaluatetoFalseforFalsetogetreturnedfortheentire
statement).Understandingthisbehaviorcanhelpyouwritemoreefficientcode.For
example,ifoneofyourexpressionstakesalongtimetoevaluate,youshouldputitlast,
becauseitwillbelesslikelytobeevaluatedthere.
ThekeywordortakestwoormoreexpressionsandevaluatestoTrueifanyofthe
expressionsevaluatetoTrue:
1==1or1==2
>>True
1==1or2==2
>>True
1==2or2==1
>>False
2==1or1==2
>>False
Likeand,youcanusemultipleorkeywordsinonestatement:
1==1or1==2or1==3or1==4or1==5
>>True
TheresultoftheexampleaboveisTruebecause1==1isTrueeventhoughalltherestofthe
expressionsevaluatetoFalse.Likeand,whenevaluatingstatementswithorinthem,assoon
asoneexpressionevaluatestoTrue,Pythonstopsevaluatingtherestoftheexpressions
becauseitisgoingtoreturnTrueregardlessofhowtheyevaluate.
Youcanplacethekeywordnotinfrontofanexpression,anditwillchangetheresultof
theexpressionsevaluationtotheoppositeofwhatitwouldhaveotherwiseevaluatedto.Ifthe
expressionwouldhaveevaluatedtoTrue,itwillevaluatetoFalsewhenprecededbynot,and
ifitwouldhaveevaluatedtoFalse,itwillevaluatetoTruewhenprecededbynot.
not1==1
>>False
not1==2
>>True
ConditionalStatements
Thekeywordsif,elifandelseareusedinconditionalstatements.Conditional
statementsareatypeofcontrolstructure:ablockofcodethatcanmakedecisionsby
analyzingthevaluesofvariables.Aconditionalstatementiscodethatisabletoexecute
additionalcodecircumstantially.Hereisanexampleinpseudocode(fakecodeusedto
illustrateanexample)tohelpclarifyhowthisworks:
If(expression)Then
(code_area1)
Else
(code_area2)
Thispseudocodeexplainsthatyoucandefinetwoconditionalstatementsthatworktogether.If
theexpressiondefinedinthefirstconditionalstatementisTrue,allofthecodedefinedin
code_area1getsexecuted.IftheexpressiondefinedinthefirstconditionalstatementisFalse,
allofthecodedefinedincode_area2getsexecuted.Thefirstpartoftheexampleiscalledan
ifstatement,andthelatteriscalledanelsestatement.Together,theyformanifelse
statement:awayforprogrammerstosay“ifthishappensdothis,otherwisedothat”.Hereis
anexampleofanifelsestatementinPython:
#https://goo.gl/uYp6ha
country="America"
ifcountry=="America":
print("Hello,America!")
else:
print("Hello,Canada!")
>>Hello,America!
Lines2and3formanifstatement.Anifstatementismadeupofalineofcodestartingwith
theifkeywordfollowedbyanexpression,acolon,anindentationandoneormorelinesof
codetobeexecutediftheexpressioninthefirstlineevaluatestoTrue.Lines3and4forman
elsestatement.Anelsestatementstartswiththeelsekeyword,followedbyacolon,an
indentationandoneormorelinesofcodetoexecuteiftheexpressionintheifstatement
evaluatestoFalse.
Togethertheyformanifelsestatement.ThisexampleprintsHello,America!because
theexpressionintheifstatementevaluatestoTrue.Ifwechangethevariablecountryto
Canada,theexpressionintheifstatementwillevaluatetoFalse,theelsestatement’scodewill
executeandourprogramwillprintHelloCanada!instead.
#https://goo.gl/bd4LVW
country="Canada"
ifcountry=="America":
print("Hello,America!")
else:
print("Hello,Canada!")
>>HelloCanada!
Anifstatementcanbeusedonitsown:
#https://goo.gl/jOlzVl
country="America"
ifcountry=="America":
print("Hello,America!")
>>Hello,America!
Youcanhavemultipleifstatementsinarow:
#https://goo.gl/WBoKWA
x=2
ifx==2:
print("Thenumberis2.")
ifx%2==0:
print("Thenumberiseven.")
ifx%2!=0:
print("Thenumberisnotodd.")
>>Thenumberis2.
>>Thenumberiseven.
EachifstatementwillexecuteitscodeonlyifitsexpressionevaluatestoTrue.Inthiscase,the
firsttwoexpressionsevaluatetoTrue,sotheircodeisexecuted,butthethirdexpression
evaluatestoFalse,soitscodeisnotexecuted.
Ifyoureallywanttogetcrazy,youcanevenputanifstatementinsideofanotherif
statement.Thisiscallednesting:
#https://goo.gl/S6Z9rp
x=10
y=11
ifx==10:
ify==11:
print(x+y)
>>21
Inthiscasex+ywillonlyprintiftheexpressionsinbothifstatementsevaluatetoTrue.An
elsestatementcannotbeusedonitsown;itcanonlybeusedattheendofanifelsestatement.
Weusetheelifkeywordtocreateelifstatements.elifstandsforelseif,andelif
statementscanbeindefinitelyaddedtoanifelsestatementtoallowittomakeadditional
decisions.
Ifanifelsestatementhaselifstatementsinit,theifstatementexpressiongetsevaluated
first.IftheexpressioninthatstatementevaluatestoTrue,itscodeisexecutedandnoother
codeisexecuted.However,ifitevaluatestoFalse,eachconsecutiveelifstatementis
evaluated.AssoonasanexpressioninanelifstatementevaluatestoTrue,itscodeisexecuted
andnomorecodeexecutes.IfnoneoftheelifstatementsevaluatetoTrue,thecodeintheelse
statementisexecuted.Hereisanexampleofanifelsestatementwithelifstatementsinit:
#https://goo.gl/L0OorN
country="Thailand"
ifcountry=="Japan":
print("Hello,Japan!")
elifcountry=="Thailand":
print("Hello,Thailand!")
elifcountry=="India":
print("Hello,India!")
elifcountry=="China":
print("Hello,China!")
else:
print("Hello,world!")
>>Hello,Thailand!
HereisanexamplewherenoneoftheexpressionsintheelifstatementsevaluatetoTrue,and
thecodeintheelsestatementisexecuted:
#https://goo.gl/Qwb5OD
country="Mars"
ifcountry=="America":
print("Hello,America!")
elifcountry=="Canada":
print("Hello,Canada!")
elifcountry=="Thailand":
print("Hello,Thailand!")
elifcountry=="Mexico":
print("Hello,Mexico!")
else:
print("Hello,world!")
>>Hello,World!
Finally,youcanhavemultipleifstatementsandelifstatementsinarow:
#https://goo.gl/tgcmXN
x=100
ifx==10:
print("10!")
elifx==20:
print("20!")
else:
print("Idon’tknowwhatxis!")
ifx==100:
print("xis100!")
ifx%2==0:
print("xiseven!")
else:
print("xisodd!")
>>Idon’tknowwhatxis!
>>xis100!
>>xiseven!
Statements
AstatementisatechnicaltermthatdescribesvariouspartsofthePythonlanguage.
YoucanthinkofaPythonstatementasacommandoracalculation.Thiswillbecomemore
clearaswetakealookatexamplesofdifferentkindsofPythonstatements.Inthissectionwe
willalsotakeadetailedlookatthesyntaxofstatements.Don'tworryifsomeofthisseems
confusingatfirst.Spendasmuchtimeasnecessaryrereadingthissectionandgobacktothe
earlierexamplesandcomparethemwithwhatyouarelearninghere.Itmighttakeawhileto
understandthesyntaxofstatements,butitwillstarttomakemoresensethemoretimeyou
spendpracticingPythonandwillhelpyouunderstandseveralprogrammingconcepts.
Pythonhastwokindsofstatements:simplestatementsandcompoundstatements.
Simplestatementscanbeexpressedinonelineofcode,whereascompoundstatements
generallyspanmultiplelines(butcanbewritteninonelineinsomecircumstances).Hereare
someexamplesofsimplestatements:
print("Hello,World!")
>>‘Hello,World!”
2+2
>>4
Compoundstatementsgenerallyspanmorethanonelineofcode.You’vealreadyseen
multipleexamplesofcompoundstatements:ifstatements,ifelsestatementsandthefirst
programwewroteinthischapterthatprinted“Hello,World!”onehundredtimesareall
examplesofcompoundstatements.
Compoundstatementsaremadeupofoneormoreclause.Aclauseconsistsoftwoor
morelinesofcode:aheaderfollowedbyasuite(s).Aheaderisalineofcodeinaclause
thatcontainsakeywordfollowedbyacolonandasequenceofoneormorelinesofindented
code.Aftertheindentthereareoneormoresuites.Asuiteisjustalineofcodeinaclause.
Thesuitesarecontrolledbytheheader.Ourprogramthatprints“Hello,World!”ahundred
timesismadeupofasinglecompoundstatement:
#https://goo.gl/rKQedq
foriinrange(100):
print("Hello,World!")
>>HelloWorld
>>HelloWorld
>>HelloWorld
…
Thefirstlineoftheprogramistheheader.It’smadeupofakeyword—for—followedbya
colonandanindentedlineofcode.Aftertheindentationisasuite—print(“Hello,World!”).
Inthiscase,theheaderusesthesuitetoprintHello,World!ahundredtimes.Thisiscalleda
loop,whichyoulearnmoreaboutinChapter7.Thiscodeonlyhasoneclause.
Acompoundstatementcanalsobemadeupofmultipleclauses.Youalreadysawthis
withifelsestatements.Anytimeanifstatementisfollowedbyanelsestatement,theresultisa
compoundstatementwithmultipleclauses.Whenacompoundstatementhasmultipleclauses,
theheaderclausesworktogether.Inthecaseofanifelsecompoundstatement,whentheif
statementevaluatestoTrue,theifstatement’ssuitesexecuteandtheelsestatement’ssuitesdo
notexecute.WhentheifstatementevaluatestoFalse,theifstatement’ssuitesdonotexecute
andtheelsestatement’ssuitesexecuteinstead.Thelastexamplefromtheprevioussection
hasthreecompoundstatements:
#https://goo.gl/tgcmXN
x=100
ifx==10:
print("10!")
elifx==20:
print("20!")
else:
print("Idon’tknowwhatxis!")
ifx==100:
print("xis100!")
ifx%2==0:
print("xiseven!")
else:
print("xisodd!")
>>Idon’tknowwhatxis!
>>xis100!
>>xiseven!
Thefirstcompoundstatementhasthreeclauses,thesecondcompoundstatementhasone
clauseandthelastcompoundstatementhastwoclauses.
Onelastthingaboutstatements,statementscanhavespacesbetweenthem.Space
betweenstatementsdoesnotaffectthecode.Sometimesspacesareusedbetweenstatementsto
makecodemorereadable:
print("Michael")
print("Jordan")
>>Michael
>>Jordan
Vocabulary
comment:Aline(orpartofaline)ofcodewritteninEnglish(oranotherlanguage)preceded
byaspecialsymbolthatletstheprogramminglanguageyouareusingknowitshouldignore
thatline(orpartofaline)ofcode.
keyword:AwordwithaspecialmeaninginPython.YoucanseeallofPython’skeywordsat
http://zetcode.com/lang/python/keywords
constant:Avaluethatneverchanges.
variable:Anameassignedtoavalueusingtheassignmentoperator
assignmentoperator:The=sign.
increment:Increasingthevalueofavariable.
decrement:Decreasingthevalueofavariable.
datatype:Acategoryofdata.
object:AdatavalueinPythonwiththreeproperties:anidentity,adatatypeandavalue.
string:Anobjectwithadatatypestr.Itsvalueisasequenceofoneormorecharacters
surroundedbyquotes.
integer:Anobjectwithadatatypeint.Itsvalueisawholenumber.
float:Anobjectwithadatatypefloat.Itsvalueisafractionalnumber.
boolean:Anobjectwithadatatypebool.ItsvalueiseitherTrueorFalse.
nonetype:AnobjectwithadatatypeNoneType.ItsvalueisalwaysNone.
syntax:Thesetofrules,principles,andprocessesthatgovernthestructureofsentencesina
givenlanguage,specificallywordorder.70
syntaxerror:Afatalprogrammingerrorcausedbyviolatingaprogramminglanguage’s
syntax.
exception:Anonfatalprogrammingerror.
operator:Symbolsusedwithoperandsinanexpression.
arithmeticoperator:Acategoryofoperatorsusedinarithmeticexpressions.
operand:Avalueoneithersideofanoperator.
expression:Codewithanoperatorsurroundedbytwooperands.
orderofoperations:Asetofrulesusedinmathematicalcalculationstoevaluatean
expression.
comparisonoperator:Acategoryofoperatorsusedinexpressionthatevaluatetoeither
TrueorFalse.
logicaloperator:Acategoryofoperatorsthatevaluatetwoexpressionsandreturneither
TrueorFalse
controlstructure:Ablockofcodethatmakesdecisionsbyanalyzingthevaluesof
variables
conditionalstatement:Codethatisabletoexecuteadditionalcodecircumstantially.
ifelsestatement:Awayforprogrammerstosay“ifthishappensdothis,otherwisedo
that.”
ifstatement:Thefirstpartofanifelsestatement.
elsestatement:Thesecondpartofanifelsestatement.
elifstatement:Statementsthatcanbeindefinitelyaddedtoanifelsestatementtoallowitto
makeadditionaldecisions.
statement:Acommandoracalculation.
simplestatement:Astatementthatcanbeexpressedinonelineofcode
compoundstatement:Astatementsthatgenerallyspansmultiplelines(butcanbewrittenin
onelineinsomecircumstances).
clause:Thebuildingblocksofcompoundstatements.Aclauseismadeupoftwoormore
linesofcode:aheaderfollowedbyasuite(s).
header:Aheaderisalineofcodeinaclausecontainingakeywordfollowedbyacolonand
asequenceofoneormorelinesofindentedcode.
suite:Alineofcodeinaclausecontrolledbyaheader.
Challenge
Writeaprogramwithavariablecalledageassignedtoanintegerthatprintsdifferent
stringsdependingonwhatintegerageis.
Chapter4.Functions
“Functionsshoulddoonething.Theyshoulddoitwell.Theyshoulddoitonly.”
—RobertC.Martin
Afunctionisacompoundstatementthatcantakeinput,executeinstructions,andoptionally
returnanoutput.Callingafunctionmeansgivingthefunctiontheinputitneedstosoitcan
executeitsinstructionsandoptionallyreturnanoutput.FunctionsinPythonaresimilarto
functionsinmath.Ifyoudon’trememberfunctionsfromalgebra,hereisanexample:
#thefollowingisalgebra,notPythoncode
f(x)=x*2
Thelefthalfoftheequationdefinesafunctionf—thattakesoneparameter—x.A
parameterisdatapassedyoupassintoafunctionwhenitscalled.Afunctioncanhaveone
parameter,multipleparametersornoparameters.Afunction’sparametersareitsinput.
Therighthalfoftheequationisthedefinitionofthefunction(theinstructionsit
executeswhencalled).Therighthandsideoftheequationusesisabletousetheparameterthat
waspassedin(x)tomakeacalculationandreturntheresult(theoutput).Inthiscase,our
functiontakesaparameter(x);,multipliesitby2;andreturnstheresult.
InbothPythonandalgebra,youcallafunctionwiththefollowingsyntax:
[function_name]([parameters_seperated_by_commas]).Youcallafunctionbyputting
parenthesesafterthefunctionname.TYouputtheparametersgoinsidetheparenthesiswith
eachparameterseparatedbyacomma.Ifwecallthefunctionwejustdefinedabove(?)and
passintheparameter4,wegetthefollowingresult:
#thefollowingisalgebra,notPythoncode
f(4)
>>8
Ifwepassin10asaparameterweget20:
f(10)
>>20
Inourfirstexample,wecalledourfunctionwith4asaparameter(x).Ourfunctionexecuted
itscode,x*2,andreturnedtheresult—8.Inoursecondexample,wecalledourfunction
with10asaparameter(x).Ourfunctionexecuteditscode,x*2,andreturnedtheresult—
20.
RepresentingConcepts
Thisisabrupt,maybesomethinglike“Asyoumayhavefiguredoutbynow…”Weare
notlimitedtoprinting“Hello,World!”inourprograms.Wecanprintwhateverwe’dlike:
print(“Python!”)
>>Python!
Fromhereonout,Iwillusethefollowingconventiontoexplainconceptsliketheideathat
youcanprintanythinginPython:print(“[what_you_want_to_print]”).Thebracketsandthe
textinsideofthemrepresentthatyouneedtobereplacedbyapieceofcodesubstituteapiece
ofcodeinplaceofthem.
Whenyouaretryingtofollowanexamplewritteninthisformatlikethis,donottypethe
brackets.Thewordsinsideofthebracketsareahintforthecodeyouneedtoreplacethe
bracketswith.Everythingoutsideofthenotinbracketsrepresentsactualcodethatyou
shouldtype.Thisformatisawayofexpressingthatyoucantype
print(“[what_you_want_to_print]”)intoPython,replace[what_you_want_to_print]with
whateveryouwanttoprint,andPythonwillprintit:
print(“Idonotlikegreeneggsandham.”)
print(“TheCatintheHat”)
>>Idonotlikegreeneggsandham
>>TheCatintheHat
Programmingisfullofconventions:agreeduponwaysofdoingthings.Thisformatisan
exampleofaconventionthatisusedintheprogrammingworldandwillbeusedthroughout
thebook.
DefiningFunctions
TocreateafunctioninPythonwechooseafunctionname,defineitsparameters,define
whatthefunctionwilldo,andwecanchoosetooptionallyreturnavalue(ifwedon’treturna
valuethefunctionwillreturnNone).Weusethefollowingsyntaxtodefineafunction:
def[function_name]([parameters_seperated_by_commas]):
[function_definition]
Ourmathematicalfunctionf(x)=x*2lookslikethisinPython:
#https://github.com/calthoff/tstp/blob/master/part_I/functions/df_ex1.py
deff(x):
returnx*2
>>
ThekeyworddeftellsPythonyouareabouttodefineafunction.isakeywordusedtodefine
afunction.Whenyouuseit,Pythonknowsyouareabouttodefineone.Afterdef,youcan
nameyourfunctionanythingwhateveryou’dlike.Byconvention,youshouldneverusecapital
lettersinayourfunctionname,sandiftherearetwowordsinyourfunctionnameyoushould
separatethemwithanunderscore—like_this.Onceyou’venamedyourfunction,put
parenthesesafterit.Insidetheparentheseis,youputyourparameter(s).Inthis(theprevious?)
example,ourfunctiononlyhasoneparameter(x),butifyouwantyourfunctiontoaccept
morethanoneparameter,youmustseparateeachparameterinsidetheparentheseiswitha
comma.Aftertheparentheseisyouputacolonandindentbyfourspaces(likeanyother
compoundstatement).Anycodeindentedfourspacesafterthecolonisthefunction’s
definition.Inthiscase,ourfunction’sdefinitionisonlyoneline—returnx*2.returnis
anotherkeyword.Itisusedtodefinethevalueafunctionoutputswhenyoucallit,referredto
asthevaluethefunctionreturns.
TocallafunctioninPython,weusethesyntaxwelearnedearlier:[function_name]().
Hereisanexampleofcallingourfunctionwith2asaparameter:
f(2)
>>
Youwillnoticetheconsoledidn’tprintanything.Ourfunctionworked,butitjustdidn’tprint
theresultbecausewedidn’ttellPythonto.Ifwewanttoprintthevalueourfunctionreturned,
wecansaveourfunction’soutputinavariable:
#https://github.com/calthoff/tstp/blob/master/part_I/functions/df_ex3.py
result=f(2)
print(result)
>>4
Youcansavetheresultyourfunctionreturnsinavariablewheneveryouneedtousethevalue
laterinyourprogram.
Functionsarenotonlyjustusedtoreturnvalues.Returningavalueisoptional,asis
includingareturnstatementinyourfunction.Asidefromreturningvalues,functionsalso
encapsulatefunctionalityyouwanttoreuse.Forexample:
#addgithub
defeven_odd(x):
ifx%2==0:
print(“even”)
else:
print(“odd”)
even_odd(2)
even_odd(3)
>>even
>>odd
Wedidn’tdefineavalueforourfunctiontoreturn,butourfunctionisstilluseful.ItOur
functiontestsifx%2==0andprintswhetherxisevenorodddependingontheresult.
Remember,moduloreturnstheremainderwhenyoudividetwonumbers.Ifthereisno
remainderwhenyoudivideanumberby2(moduloreturns0)thenumberisbydefinition
even.Ifthereisaremainder,thenumberisodd.Youmayneedtousethisfunctionalityin
severaldifferentplacesinyourprogram.Itwouldbepoorprogrammingtotypethecodewe
usedinourfunction’sdefinitioneverytimeyouwanttousethisfunctionality.That’swhat
functionsarefor.Youputfunctionalityinafunction,anditletsyoueasilyreusethat
functionalitythroughoutyourprogram,withouthavingtodoanyextrawork.Thisisalittle
confusing-areyougoingtoexplainlateron?Youcouldmentionthisissomethingyou’ll
coverhowtodoinChapter__.
Parameters
Sofar,we’veonlydefinedfunctionsthatacceptoneparameter.Afunctiondoesn’thave
totakeanyparameters.Hereisanexampleofafunctionthatdoesnottakeanyparameters:
#https://github.com/calthoff/tstp/blob/master/part_I/functions/df_ex5.py
deff():
return1+1
result=f()
print(result)
>>2
Hereisanexampleofafunctionthatacceptsmultipleparameters:
#https://github.com/calthoff/tstp/blob/master/part_I/functions/df_ex4.py
deff(x,y,z):
returnx+y+z
result=f(1,2,3)
print(result)
>>6
Therearetwotypesofparametersafunctioncanaccept.Theparameterswe’veseenso
fararecalledrequiredparameters.Whenafunctioniscalled,alloftherequiredparameters
mustbepassedintothefunction,oryouwillgetanerrorinyourprogram.Thereisanother
kindofparameter—optionalparameters—thatletthecallerofthefunctionpassina
parameteriftheywantto,buttheydonothaveto.Iftheydonotpassinaparameter,adefault
valuedefinedbythefunctionwillbeusedinstead.OYoudefineoptionalparametersare
definedwiththefollowingsyntax:[function_name]([parameter_name]=[paramater_value]).
Likerequiredparameters,optionalparameterstheymustbeseparatedbycommas.Hereisan
exampleofafunctionthattakesanoptionalparameter:
#https://github.com/calthoff/tstp/blob/master/part_I/functions/op_ex1.py
deff(x=10):
ifx==10:
print(“xisten”)
else:
print(“xisnotten”)
default=f()
pass_in=f(2)
print(default)
print(pass_in)
>>‘xisten’I’mconfusedwhytherearesinglequotesoranyquotesatall.
>>‘xisnotten’I’mconfusedwhytherearesinglequotesoranyquotesatall.
First,wecallourfunctionwithoutpassinginaparameter.Becausetheparameterisoptional
wedon’thavetopassitinthisisallowed,andxisassignedthevaluewedefinedinour
optionalparameter—10.Whenourfunctioniscalled,xisequalto10,andso‘xisten’(if
thereshouldbesinglequotesabove,leaveit,ifnottakethemout)prints.
Next—wecallourfunctionagain—butthistimewepassin2asaparameter.The
defaultvalue10isignoredbecauseweprovidedavaluethistime,andsoxgets2and“xis
notten”(samethingaboutthequotes,evenifI’mjustmissingsomethingIthinktheyshould
besinglequotes)prints.Youcandefineafunctionwithbothrequiredandoptional
parameters,butthereisonerule:allofyourrequiredparametersmustbedefinedbeforeyour
optionalparameters:
defrequired_optional(x,y=10)
returnx+y
>>
PassingParameters
Whenyoudefineafunctionwithparameters,sometimesthoseparametershavetobea
specificdatatypeinorderforthefunctiontowork.Howdoyoucommunicatethistowhoever
callsyourfunction?Whenyouwriteafunction,itisgoodpracticetoleaveacommentatthe
topofthefunctionexplainingwhatdatatypeeachparameterneedstobe.Wewilldiscussthis
furtherinPpartV.Whenyougiveafunctionparameterswhenyoucallit,itisreferredtoas
“passing”thefunctionparameters.Thissentencemakessense,butisfairlyconfusing.Couldit
be:WhenyougiveafunctionparametersTHENyoucallit,itisreferredtoas“passing”the
functionparameters.
pass
Wecanusethekeywordpasstocreateafunctionthatdoesnothing:
”””https://github.com/calthoff/tstp/blob/master/part_I/functions/pass_ex1.py
”””
deff():
pass
f()
>>
passThepasskeywordisusefulwheneveryouwanttocreateafunction,butfinishthe
definitionlater.
NestedFunctions
Youcandefineafunctioninsideofafunction.Thisiscallednesting.Thefirstfunction
iscalledtheouterfunction,andthesecondfunctioniscalledtheinner(ornested)function,or
nestedfunction.Hereisanexample:
deff():
print("InnerFunction!")
defx():
print("NestedFunction!")
x()
f()
>>InnerFunction!
>>NestedFunction!
Wewillnotbecoveringwhythisisimportantinthisbookbecauseyoudonotneeditisnot
importanttousenestedfunctionswhenyouarelearningtoprogram.,butIwantedtoincluded
thisexamplesoyouknowthatitispossible.
Scope
Variableshaveanimportantpropertycalledscopewedidn’tdiscusswhenwefirst
coveredthem.Whenyoudefineavariable,thevariable’sscopereferstowhatpartofyour
programhasaccesstoit.Thisisdeterminedbywherethevariableisdefinedinyour
program.Ifyoudefineavariableoutsideofafunction(oraclass,whichwelearnaboutin
PartII)thevariablehasaglobalscope:thevariablecanbeaccessedanywhereinyour
program.Ifyoudefineavariableinsideofafunction(orclass)ithaslocalscope:thevariable
cannotbeaccessedanywhereinyourprogram—;itcanonlybeaccessedinthefunction(or
class)itwasdefinedin(oranynestedfunctionsorclasses).Herearesomeexamplesof
variableswithglobalscope:
x=1
y=2
z=3
>>
Thesevariableswerenotdefinedinsideofafunction(orclass)andthereforehaveaglobal
scope.Thismeanswecanaccessthemanywhere—includinginsideofafunction:
x=1
y=2
z=3
deff():
print(x)
print(y)
print(z)
f()
>>1
>>2
>>3
Ifwedefinethesesamevariablesinsideofafunction,wecanonlyaccesstheminsideof
thatthefunctionwedefinedthemin(orafunctionnestedinsidethefunctionwedefinedthem
in).Ifwetrytoaccessthemoutsideofthefunctiontheyweredefinedin,Pythonraisesan
exception:
deff():
x=1
y=2
z=3
print(x)
print(y)
print(z)
>>NameError:name'x'isnotdefined
Ifwesticktousingthesevariablesinsideourfunction,thereisnoproblem:
deff():
x=1
y=2
z=3
print(x)
print(y)
print(z)
f()
>>1
>>2
>>3
Ifyouwanttochangethevalueofaglobalvariableinsidealocalscope,youneedtousethe
globalkeywordfollowedbythevariableyouwanttochange:
#https://github.com/calthoff/tstp/blob/master/part_I/functions/scope_ex3.py
x=100
deff():
globalx
x+=1
print(x)
f()
>>101
Thereasonprogramminglanguageshavescopeisbecausehavingnoscope(everyvariable
canbeaccessedanywhereinaprogram)causesproblems.Ifyouhavealargeprogram,and
youwriteafunctionthatusesthevariablex,youmightaccidentlychangethevalueofa
variablecalledxthatwaspreviouslydefinedinyourprogram;whichwillchangethe
behaviorofyourprogramandmaycauseanerrororunexpectedresults.Thelargeryour
programgets,andthemorevariablesithas,themorelikelythisbecomes.However,with
scope,ifyoudefineavariablexinsideofafunction,thereisaguaranteeyouwillnot
accidentallychangethevalueofanypreviouslydefinedvariablesoutsideofyourfunction
becauseinordertochangethevalueofavariableoutsideofyourfunction,youmust
explicitlyusetheglobalkeyword.
Built-inFunctions
Pythoncomeswithalibraryofbuilt-infunctions.Theyperformallsortsofdifferent
functionalityandarereadytousewithoutanyworkonyourpart.We’vealreadyseenone
exampleofabuilt-infunction—thefirstprogramwewroteusedtheprintfunctiontoprint
“Hello,World!”.lenisanotherbuiltinfunction.Itreturnsthelengthofanobject—likea
string.Thelengthofastringisthenumberofcharactersinit.
len(“Monty”)
>>5
len(“Python”)
>>6
typeisanotherbuilt-infunction.Itreturnswhatdatatypeanobjectis:
type(“HelloWorld”)
>><type'str'>
type(100)
>><type'int'>
type(1.0)
>><type'float'>
Thebuilt-instrfunctiontakesanobjectandreturnsanewobjectwithastringdatatype.For
example,wecanusestrtoconvertanintegertoastring.
str(100)
>>'100'
inttakesanobjectandreturnsanewobjectwithanintegerdatatype:
int(“1”)
>>1
Andfloattakesanobjectandreturnsanewobjectwithanintegerdatatype:
float(100)
>>100.0
inputisabuilt-infunctionthatcollectsinformationfromthepersonusingourprogram.
“““
https://github.com/calthoff/tstp/blob/master/part_I/introduction_to_programming/input_ex1.py
”””
age=input("Howoldareyou?")
age=int(age)
ifage<21:
print("Youareyoung!")
else:
print("Wowyouareold!")
>>Howoldareyou?
Theinputfunctiontakesastringasaparameteranddisplaysthestringtothepersonusingthe
programintheshell.Theycanthentypearesponseintotheshell,andwecansavetheir
responseinavariable—inthiscasewesavetheresponseinthevariableage.
Nextweusetheintfunctiontochangeagefromastringtoanintegerbecauseinput
collectsdatafromtheuserasastring,andwewantourvariabletobeanintegersowecan
compareittootherintegers.Oncewehaveaninteger,ourifelsestatementdetermineswhich
messagegetsprintedtotheuser,dependingonwhattheytypedintotheshell.Iftheusertypes
anumbersmallerthan21,“Youareyoung!”prints.Iftheusertypesanumbergreaterthan
21,“Wowyouareold!”prints.
ExceptionHandling
Whenyourelyonuserinputfromtheinputfunction,youdonotcontroltheinputto
yourprogram—theuserdoes,andthatinputcouldcauseanerror.Forexample,saywewrite
aprogramtocollecttwonumbersfromauser,andprintouttheresultofthefirstnumber
dividedbythesecondnumber:
“““
https://github.com/calthoff/tstp/blob/master/part_I/introduction_to_programming/exception_handling_ex1.py
”””
a=input(“typeanumber”)
b=input(“typeanothernumber”)
a=int(a)
b=int(b)
print(a/b)
>>typeanumber
>>10
>>typeanothernumber
>>5
>>2
Ourprogramappearstowork.However,wewillrunintoaproblemiftheuserinputs0asthe
secondnumber:
a=input(“typeanumber”)
b=input(“typeanothernumber”)
a=int(a)
b=int(b)
print(a/b)
>>typeanumber
>>10
>>typeanothernumber
>>0
>>ZeroDivisionError:integerdivisionormodulobyzero
Ourprogramworks—untiltheuserdecidestoenter0asthesecondnumber,inwhichcase
ourprogramraisesanexception.Wecannotallowpeopletousethisprogramandhopethey
willnotenter0asthesecondnumber.Onewaytosolvethisistouseexceptionhandling,
whichallowsyouto“catch”exceptionsiftheyoccuranddecidewhattodo.
Thekeywordstryandexceptareusedforexceptionhandling.Wecanchangeour
programtouseexceptionhandlingsoifauserenters0asthesecondnumber,ourprogram
printsamessagetellingthemnottoenter0insteadofraisinganexception.
InPythonexceptionsareobjects—whichallowsustousetotheminourprograms.
EachexceptioninPythonisanobject.Youcanseethefulllistofbuilt-inexceptionshere:
https://docs.python.org/3/library/exceptions.html.Wheneveryouareinsituationwhereyou
thinkyourcodemayraiseanexception,youcanuseacompoundstatementwiththekeywords
tryandexcepttocatchtheexception.
Thetryclausedefinestheerrorthatcouldoccur.Theexceptclausedefinescodethat
willonlyexecuteiftheexceptiondefinedinyourtryclauseoccurs.Hereisanexampleof
howwecanuseexceptionhandlinginourprogramsoifauserenters0asthesecond
numberourprogramdoesn’tbreak:
“““
https://github.com/calthoff/tstp/blob/master/part_I/introduction_to_programming/exception_handling_ex2.py
”””
a=input(“typeanumber”)
b=input(“typeanothernumber”)
try:
print(a/b)
exceptZeroDivisionError:
print(“bcannotbezero.Tryagain.”)
>>typeanumber
>>10
>>typeanothernumber
>>0
>>“bcannotbezero.Tryagain.”
Iftheuserentersanythingotherthan0,thecodeinourtryblockisexecutedandourexcept
blockdoesn’tdoanything.Butiftheuserenters0,insteadofraisinganexception,thecodein
ourexceptblockisexecutedandourprogramprints“bcannotbezero.Tryagain.”.
Docstrings
Docstringsarecommentsatthetopofafunctionormethodthatexplainwhatthe
functionormethoddoes,anddocumentswhattypesoftheparametersshouldbepassedtoit.
Hereisanexample:
defadd(x,y):
"""
Returnsx+y.
:paramx:intfirstintegertobeadded.
:paramy:intsecondintegertobeadded.
:return:intsumofxandy.
"""
returnx+y
Thefirstlineofthedocstringclearlyexplainswhatourfunctiondoes.Whenotherdevelopers
reuseyourfunctionormethod,theydonotwanttohavetoreadthroughallofyourcodeto
figureoutwhatitdoes.Therestofthelinesofthedocstringliststhefunction’sparameters,its
returnvalue,andsomeadditionalinformation,includingthetypeforalloftheparameters
andthereturnvalue.Docstringswillhelpyouprogramfaster,becauseifyouforgetwhata
pieceofcodedoes,youcanquicklyfigureitoutbyreadingthedocstringinsteadofallofthe
codeinafunction,classormethod.Itwillalsomakeitmucheasierforotherprogrammersto
useyourcode.InsomecasesI’veomitteddocstringsInormallywouldhaveincludedthemto
makemycodeasconciseaspossibleforeasyreading—butwheneverIamwritingcodefor
production(codethatisactuallyusedbyotherpeople)—Iusedocstrings.
Challenge
Writeafunctionthatdoessomethinginteresting,anduseitseveraltimesinaprogram.
Chapter5.Containers
"Badprogrammersworryaboutthecode.Goodprogrammersworryaboutdatastructures
andtheirrelationships."
—LinusTorvalds
Inchapter3,welearnedhowtostoreobjectsinvariables.Inthischapterwelearntostore
objectsincontainers—specialobjectsthatcanstoreandretrieveotherobjects(likestrings).In
thischapter,wewillgooverthreecommonlyusedcontainers:lists,tuplesanddictionaries.
Lists
Alistisamutablecontainerthatstoresobjectsinaspecificorder.Whenacontaineris
mutableitmeanstheobjectsinthethecontainercanchange—objectscanbeaddedand
removedfromthecontainer.
[image]
ListsarerepresentedinPythonwithbrackets.Therearetwosyntaxestocreatealist.We
cancreateanemptylistwiththelistfunction:
new_list=list()
new_list
>>[]
Orwecancreateanemptylistwithbrackets:
new_list=[]
new_list
>>[]
Bothsyntaxescreateanewemptylist.Whenyoucreateanewlistwiththelistfunctionyou
canalsopassinobjectsyouwanttoaddtoyourlistasparameters:
my_list=list(“Apple”,“Orange”,“Pear”)
my_list
>>['Apple','Orange','Pear']
Orlikethisusingthesecondsyntax:
my_list=[“Apple”,“Orange”,“Pear”]
my_list
>>['Apple','Orange','Pear']
Eachobjectinalistiscalledaniteminthelist.Inthisexampletherearethreeitemsinour
list:‘Apple’,‘Orange’and‘Pear ’.Listskeeptheiritemsinorder—theordertheitems
enteredthelist.Unlesswechangetheorderofourlist,‘Apple’willalwaysbethefirstitem,
‘Orange’theseconditemand‘Pear ’thethirditem.‘Apple’isatthebeginningofthelist,
and‘Pear ’isattheend.Wecanaddanewitemtotheendofourlistusingtheappend
function:
my_list.append(“Banana”)
my_list.append(“Peach”)
my_list
>>['Apple','Orange','Pear',‘Banana’,’Peach’]
Listsarenotlimitedtostoringstrings—theycanstoreanydatatype:
new_list=[]
new_list.append(True)
new_list.append(100)
new_list.append(1.1)
new_list.append(‘Hello’)
>>[True,100,1.1,‘Hello’]
Everyiteminalisthasapositioninthelist—calleditsindex.Youcanfigureouttheindexof
anyiteminalistbystartingatthebeginningofthelistandcounting.Theonlytrickypartis
youhavetostartcountingatzero,becausethefirstiteminalisthasanindexofzero.Sothe
firstiteminalistisatindexzero,theseconditeminalistisindexone,andsoon.Counting
startingatzerotakessomegettingusedto,sodon’tworryifitfrustratesyouatfirst.Youcan
accesseachiteminalistwithitsindexusingthesyntax[list_name][[index]].Iputindexin
twobracketstorepresentthat[index]shouldbereplaced,butshouldbeinsidebrackets.
my_list=[“Apple”,“Orange”,“Pear”]
my_list[0]
my_list[1]
my_list[2]
>>Apple
>>Orange
>>Pear
Youcanchangeaniteminalistbysettingtheindexoftheitemtoanewobject:
color_list=[“blue”,“green”,“yellow”]
color_list
color_list[2]=“red”
color_list
>>[“blue”,“green”,“yellow”]
>>[“blue”,“green”,“red”]
Ifyoutrytoaccessanindexthatdoesn’texist,Pythonwillraiseanexception:
color_list=[“blue”,“green”,“yellow”]
color_list[4]
>>IndexError:listindexoutofrange
Youcanremovethelastitemfromalistwithpop:
color_list=[“blue”,“green”,“yellow”]
color_list
item=color_list.pop()
item
color_list
>>[“blue”,“green”,“yellow”]
>>“yellow”
>>[“blue”,“green”]
Youcannotpopfromanemptylist,ifyoudoPythonwillraiseanexception.
Youcancheckifanitemisinalistusingthekeywordin:
color_list=[“blue”,“green”,“yellow”]
“green”incolor_list
>>True
Addthekeywordnottocheckifanitemisnotinalist:
color_list=[“blue”,“green”,“yellow”]
“black”notincolor_list
>>True
Youcangetthesizeofalist(thenumberofitemsinit)withthelenfunction:
len(color_list)
>>3
Finally,youcangetarangeofitemsinalistwithslicing.Weslicealistwithastart
indexandanendindexseparatedbyacoloninbracketsoutsideofourlist.Slicingreturnsa
newlist(a“slice”oftheoldone)madeupofeverythingbetweenthestartandendindex.The
syntaxforslicingis[list_name][[start_index:end_index]].Hereisanexampleofslicingalist:
new_list=['Apple','Orange','Pear',‘Banana’,‘Peach’]
new_list[0:3]
>>['Apple','Orange','Pear']
A“gotcha”withslicingisthestartindexincludestheitematthatindex,buttheendindex
doesn’tincludetheitemattheendindex,itonlyincludestheitembeforetheendindex.This
meansifyouwanttoslicefrom“Apple”to“Pear”,youneedtoslicefromindex0,toindex3
(insteadofindex2),becausetheitemattheendindexisnotincludedintheslice.
Tuples
Atupleisanimmutablecontainerthatstoresobjectsinaspecificorder.Whena
containerisimmutableitmeansthecontentsofthecontainercannotchange.Thatmeans
unlikealist,onceyouputanobjectintoatuple,youcannolongerchangeit.Onceyoucreate
atupleyoucannotchangethevalueofanyoftheitemsinit,youcannotaddnewitemstoit,
andyoucannotremoveitemsfromit.Tuplesarerepresentedwithparenthesis.Therearetwo
syntaxestocreateatuple:
my_tuple=tuple()
my_tuple
>>()
And
my_tuple=()
my_tuple
>>()
Ifyouwantyourtupletocontainobjects,youmustaddthemtoyourtuplewhenyoucreateit.
Hereishowyouadditemstoatupleusingthefirstsyntax:
my_tuple=tuple(“brown”,“orange”,“yellow”)
my_tuple
>>(“brown”,“orange”,“yellow”)
Andthesecond:
my_tuple=(“brown”,“orange”,“yellow”)
my_tuple
>>(“brown”,“orange”,“yellow”)
Atuplewithoneiteminitstillneedsacommaaftertheitem:
(‘self_taught’,)
>>(‘self_taught’,)
Onceyou’vecreatedyourtuple,ifyoutrytoaddanobjecttoit,Pythonwillraisean
exception:
my_tuple=(“brown”,“orange”,“yellow”)
my_tuple[1]=“red”
>>TypeError:'tuple'objectdoesnotsupportitemassignment
Youcan,however,accessdatafromatuplelikealist—youcanreferenceanindexandslicea
tuple:
my_tuple=(“brown”,“orange”,“yellow”)
my_tuple[0]
my_tuple[1:2]
>>yellow
>>('yellow','orange')
Youcancheckifanitemisinatupleusingthekeywordin:
my_tuple=(“brown”,“orange”,“yellow”)
“brown”inmy_tuple
>>True
Addthekeywordnottocheckifanitemisnotinatuple:
my_tuple=(“brown”,“orange”,“yellow”)
“black”notinmy_tuple
>>True
Youmaybewonderingwhyyouwouldwanttouseadatastructurethatappearstobe
likealist,butlesshelpful.Tuplesareusefulwhenyouaredealingwithvaluesyouknowwill
neverchange,andyoudon’twantotherpartsofyourprogramtohavetheabilitytochange
thosevalues.Agoodexampleisifyouareworkingwithgeographiccoordinates.Youmay
wanttostorethelongitudeandlatitudeofNewYorkinatuplebecauseyouknowthe
longitudeandlatitudeofNewYorkisnevergoingtochange,andyouwanttomakesureother
partsofyourprogramdon’thavetheabilitytoaccidentallychangethem.
Dictionaries
Dictionariesareanotherbuilt-incontainerforstoringobjects.Theyaremutable—but
unlikelistsandtuples—theydonotstoreobjectsinaspecificorder.Instead,dictionariesare
usedtomaponeobject(calledthekey)toanotherobject(calledthevalue).Dictionariesare
representedwithcurlybrackets.Therearetwosyntaxesforcreatingdictionaries:
my_dict=dict()
my_dict
>>{}
And:
my_dict={}
my_dict
>>{}
Youaddobjectstoadictionarybymappingakeytoavalue.Eachkeymappedtoavalueina
dictionaryiscalledakeyvaluepair.Hereishowyoucreatekeyvaluepairswhenyoucreatea
dictionarywiththefirstsyntax:
my_dict=dict({“Apple”:“Red”,“Banana”:“Yellow”})
my_dict
>>{'Apple':'Red','Banana':'Yellow'}
Andthesecond:
my_dict={“Apple”:“Red”,“Banana”:“Yellow”}
my_dict
>>{'Apple':'Red','Banana':'Yellow'}
Bothsyntaxeshaveakeyseparatedfromavaluebyacolon.Eachkeyvaluepairmustbe
separatedbyacomma.Unlikeatuple,ifyouhavejustonekeyvaluepair,youdonotneeda
commaafterit.Onceyou’veaddedkeyvaluepairstoadictionary,youcanuseakeyto
lookupavalue.Youcanonlyuseakeytolookupavalue.Youcannotuseavaluetolookupa
key:
my_dict[‘Apple’]
>>Red
Dictionariesaremutable,soonceyou’vecreatedoneyoucanaddmorekeyvaluepairswith
thesyntax[my_dictionary][[key]]=[value]:
my_dictionary=dict()
my_dictionary[“programming”]=“awesome”
my_dictionary[“programming”]
my_dictionary[“BillGates”]=“rich”
my_dictionary[“BillGates”]
my_dictionary[“america_founded”]=1776
my_dictionary[“america_founded”]
>>awesome
>>rich
>>1776
Youcanusetheinkeywordtocheckifakeyisinadictionary.Youcannotusethein
keywordtocheckifavalueisinadictionary.
“BillGates”inmy_dictionary
>>True
Addthekeywordnottocheckifakeyisnotinadictionary:
“BillPlates”notinmy_dictionary
>>True
Finally,youcandeleteakeyvaluepairfromadictionarywiththekeyworddel
my_dictionary
delmy_dictionary['BillGates']
my_dictionary
>>{'america_founded':1776,'programming':'awesome','Bill
Gates':'Rich'}
>>{'america_founded':1776,'programming':'awesome'}
Challenge
Lists,tuplesanddictionariesarejustafewofthecontainersbuilt-intoPython.Take
sometimetolookupandreadaboutPythonsets.Whatisasituationwouldyouuseasetin?
Chapter6.StringManipulation
“Intheory,thereisnodifferencebetweentheoryandpractice.But,inpractice,thereis.”
—JanL.A.vandeSnepscheut
Pythonhasbuilt-infunctionalityformanipulatingstrings,suchaschangingastring’s
caseorsplittingastringintotwopartsatagivencharacter.Thisfrequentlycomesinhandy.
Sayforexample,youhaveastringINALLCAPSandyouwanttochangeittolowercase.
Luckily,withPython,wecaneasilyfixthisproblem.Inthischapterwewilllearnmoreabout
stringsandgooversomeofPython’smostusefultoolsformanipulatingstrings.
TripleStrings
Ifastringismorethanoneline,youneedtoputitintriplequotes:
“““lineone
linetwo
linethree
”””
Ifyoutrytodefineastringthatspansmorethanonelinewithsingleordoublequotes,you
willgetasyntaxerror.
Indexes
Stringsareiterable.Youcanaccesseachcharacterinastringwithitsindex,justlike
youcanaccesseachiteminatuple.Liketuples,thefirstcharacterinastringstartswithindex
0andeachsubsequentindexisincrementedby1.
my_string=“LAX”
my_string[0]
my_string[1]
my_string[2]
>>‘L’
>>‘A’
>>‘X’
Inthisexampleweusedtheindexes0,1,and2toaccesseachofthecharactersinthestring
“LAX”.Ifwetrytoaccessanelementpastthelastelement,Pythonraisesanexception:
my_string=“LAX”
my_string[3]
>>IndexError:stringindexoutofrange
StringsareImmutable
Strings,liketuples,areimmutable.Youcannotchangecharactersinastring.Ifyouwant
tochangethecharactersinastring,youneedtocreateanewstring.
Methods
Inchapter4,welearnedaboutfunctions.Inthischapter,willbeusingaconceptsimilar
tofunctions—methods—tomanipulatestrings.WelearnmoreaboutmethodsinPartIIofthis
book—butfornowyoucanthinkofmethodsasfunctionsthatobjects“come”with.
Youcanpassparameterstoamethod,whichcanexecutecodeandreturnaresultjust
likeafunction.Unlikeafunction,amethodgetscalledonanobject.Forexampleifwehavea
string“Hello”,wecouldcall“Hello”.[method_name]()onourstring.Otherthanbeing
calledonanobject,youcanthinkofamethodasthesamethingasafunction(fornow).
ChangeCase
Youcanchangeastringsoeveryletterisuppercasebycallingtheuppermethodonit:
“““Ifcomputerprogrammingwereacountry,itwouldbethethirdmostdiversefor
languagesspoken.”””.upper()
>>“““IFCOMPUTERPROGRAMMINGWEREACOUNTRY,ITWOULDBETHE
THIRDMOSTDIVERSEFORLANGUAGESSPOKEN”””
43
Similarly,youcanchangeeveryletterinastringtolowercasebycallingthelowermethod
onit:
“““AdaLovelace,thedaughteroftheEnglishpoetLordByron,isconsideredtobethe
firstcomputerprogrammer.”””.lower()
>>“adalovelace,thedaughteroftheenglishpoetlordbyron,isconsideredtobe
thefirstcomputerprogrammer”
42
Youcanalsocapitalizethefirstletterofeverywordinasentencebycallingthecapitalize
methodonastring:
”””youcanbuildacomputerusinganythingthatcanimplementaNAND-gateandthe
conceptofzero(i.e.,somethingandnothing).allTuring-completeprogramming
languagesareequallypowerful(ignoringpracticalities).lispappearedin1958andis
stillregardedasbeingamongthemorepowerfulprogramminglanguages
today.”””.capitalize()
>>”””Youcanbuildacomputerusinganythingthatcanimplement
aNAND-gateandtheconceptofzero(i.e.,something
andnothing).AllTuring-completeprogramminglanguagesare
equallypowerful(ignoringpracticalities).Lispappearedin1958
andisstillregardedasbeingamongthemorepowerful
programminglanguagestoday.”””40
format
Sometimesyouwillwanttocreateastringusingvariables.Thisisdonewiththe
formatmethod:
year_started=“1989”
“Pythonwascreatedin{}.”.format(year_started)
>>‘Pythonwascreatedin1989.’
Theformatfunctionlooksforanyoccurrencesof{}inthestringandreplacesthemwiththe
valuesyoupassintoformat.
Youarenotlimitedtousing{}once,youcanputasmanyoftheminyourstringasyou’d
like:
#https://github.com/calthoff/tstp/blob/master/part_I/string_manipulation/format.py
year_started=“1989”
creator=“GuidovanRossum”
country=“theNetherlands”
my_string=“Pythonwascreatedin{}by{}in{}.”.format(year_started,creator,
country)
print(my_string)
>>‘Pythonwascreatedin1989byGuidovanRossuminthe
Netherlands.’
split
Stringshaveamethodcalledsplitusedtoseparateonestringintotwostrings.Youpass
thesplitmethodthecharacterorcharactersyouwanttousetoseparatethestring—for
example,wecanpassinaperiodtoseparatethisquotebyDanielCoyleintotwodifferent
strings:
”””Practicedoesn’tmakeperfect.Practicemakesmyelin,andmyelinmakes
perfect.”””.split(“.”)
>>["Practicedoesn'tmakeperfect",'Practicemakesmyelin,and
myelinmakesperfect','']
Theresultisalistwithtwodifferentstringssplitattheperiodintheoriginalstring.
join
Thejoinmethodletsyouaddnewcharactersinbetweeneverycharacterinastring:
my_string=‘abc’
join_result=‘+’.join(my_string)
join_result
>>‘a+b+c’
Youcanturnalistofstringsintoasinglestringbycallingthejoinmethodonanemptystring
(“”)andpassinginalist:
the_Fox=[“The”,“fox”,“jumped”,“over”,“the”,“fence”,“.”]
one_string=“”.join(the_Fox)
one_string
>>Thefoxjumpedoverthefence.
replace
Thereplacemethodletsyoureplaceeveryoccurrenceofacharacter(s)withanother
character(s).Thefirstparameteristhecharacter(s)toreplaceandthesecondparameteristhe
character(s)toreplaceitwith.
my_string=“Thecatjumpedoverthehat.”
my_string=my_string.replace(“a”,“@”)
my_string
>>“Thec@tjumpedovertheh@t.”
index
Wecangettheindexofthefirstoccurrenceofacharacterinastringwiththeindex
method.Wepassinthecharacterwearelookingfor,andwegettheindexofthefirst
occurrenceofthecharacterinthestring:
‘cat’.index(‘a’)
>>1
in
TheinkeywordchecksifonestringisinanotherstringandreturnsTrueorFalse:
“Playboy”in”””ApicturefromPlayboymagazineisthe
mostwidelyusedforallsortsofimageprocessing
algorithms”””
>>True
42
Addthekeywordnotinfrontofintocheckifonestringisnotinanotherstring:
“hello”notin”””Thecomputerviruswasinitially
designedwithoutanyharmfulintentions”””
>>True
42
EscapingStrings
Whatifyouwanttousequotesinsideastring?Ifweusequotesinsideastringwegetasyntax
error:
”””SunTzusaid"TheSupremeartofwaristosubduetheenemywithoutfighting.""
“““
>>SyntaxError:invalidsyntax
Wecansolvethiswithescaping,whichmeansputtingaspecialsymbolinfrontofacharacter
thathasspecialmeaninginPython(inthiscasethespecialcharacterisaquote),toletPython
knowthatthisparticularcharacterismeanttobeacharacter,andnotthespecialPython
symbolitusuallyrepresents.Thespecialsymbolweusetoescapeourquoteisabackslash.
“““SunTzusaid\"TheSupremeartofwaristosubduethe
enemywithoutfighting.\"”””
>>'SunTzusaid"TheSupremeartofwaristosubduetheenemy
withoutfighting."'
Newline
Wecanuse“\n”insideastringtorepresentanewline:
print(“line1\nline2”)
>>‘line1’
>>‘line2’
Concatenation
Wecanaddtwostringstogetherusingtheadditionoperator.Theresultwillbeone
stringwiththecharactersfromthefirststringfollowedbythecharactersfromthenext
strings.Thisiscalledconcatenation:
“cat”+“in”+“the”+“hat”
>‘catinthehat’
“cat”+“in”+“the”+“hat”
>>‘catinthehat’
StringMultiplication
Wecanalsomultiplyastringbyanumberwiththemultiplicationoperator:
“cat”*3
>>‘catcatcat’
Chapter7.Loops
ThesecondprogramwelearnedhowtowriteprintedHello,World!ahundredtimes.Wedid
thisbyusingaloop.Loopsarecompoundstatementsthatletusexecutecodeoverandover
againacertainnumberoftimes,oraslongasaconditionisTrue.Inthischapterwewillgo
overtwokindsofloops—forloopsandwhileloops.
ForLoops
Forloopsexecuteasetofinstructionsacertainnumberoftimes.Yougiveaforloopa
numbertostartat(wewillcallthisnumbera),anumbertostopat(wewillcallthisnumberz
),asetofinstructions,andavariablethatkeepstrackofthenumberoftimestheinstructions
havebeenexecuted(wewillcallthisvariablei).Youcanthinkofaforloopasacircle.It
goesroundandroundexecutinginstructions.Everytimetheloopexecutesitsinstructions,i
(whichstartsatthevalueofa)getsincrementedby1.Whenibecomesequaltoz,thethe
forloopstops.
Youdefineaforloopwiththesyntaxfor[variable_name]inrange(a,z):
[code_to_execute]whereaisthenumbertostartat,zisthenumbertostopat,and
[variable_name]isavariablenameofyourchoosingthatgetsassignedtowhatevernumber
theloopison(we’vebeenreferringtoitasi)andcode_to_executeisthecodeyoudefine
thatisexecutedeachtimearoundtheloop.Thisalliseasiertounderstandwithanexample:
#https://github.com/calthoff/tstp/blob/master/part_I/loops/for_loops_ex1.py
foriinrange(0,10):
print(i)
>>0
>>1
…
>>9
Inthisexamplewechoseiasourvariable_name.Thisisthevariablenamepeopleinthe
Pythoncommunityusuallyusewhentheywriteaforloop.Yourcode_to_executehasaccess
tothevariablei.
Whenwerunthisprogram,ourprogramentersourforloop.astartsat0,zstartsat
10andistartsatthesamevalueasa—0.Thefirsttimearoundtheloopiis0,andourfor
loopexecutesitsinstructions—print(i)—whichprints0(becauseiisequalto0).Thenext
timearoundtheloop,iisincrementedby1,soiisnow1,andso1getsprinted.Thenext
timearoundtheloop,iisincrementedby1again,andsonowiis2,andso2getsprinted.
Eventually,iwillequal10.Whenthishappens,nothingwillprint,becauseiisequaltozand
sotheloopends.
Whenaloopends,itsinstructionsstopsexecuting,andPythonmovesontothenextline
ofcodeaftertheloop.Thatiswhy9isthelastnumberprinted.Ifthereweremorecodeafter
ourloop,Pythonwouldexecuteit,butinthisexamplethereisnot,andsotheprogramends.
Thereisanotherforloopsyntaxwecanuseforiteration.Iterationmeansgoingoneby
onethroughaniterable.Aniterableisanobjectthathasindexes.Someexamplesofiterables
arestrings,lists,tuplesanddictionaries.Iterationisdonewiththesyntaxforvariable_namein
iterable:[code_to_execute].Justlikethefirstsyntax,wechoseavariablenameanddefinethe
codetobeexecutedeachtimearoundtheloop.Inthissyntax,insteadofourloopexecuting
codeuntiliisequaltoz,ourloopstartsatthefirstindexintheiterable,andstopsafterthe
lastindex.Alsovariable_namegetsassignedtotheitemattheindexweareat(inthe
iterable).Hereisanexample:
#https://github.com/calthoff/tstp/blob/master/part_I/loops/for_loops_ex2.py
my_string=“Python”
forcharacterinmy_string:
print(character)
>>‘P’
>>‘y’
>>‘t’
>>‘h’
>>‘o’
>>‘n’
#https://github.com/calthoff/tstp/blob/master/part_I/loops/for_loops_ex3.py
my_list=[“a”,“b”,“c”]
foriteminmy_list:
print(character)
>>‘a’
>>‘b’
>>‘c’
#addgithub
my_tuple=(“a”,“b”,“c”)
foriteminmy_tuple:
print(character)
>>‘a’
>>‘b’
>>‘c’
#addgithub
my_dict={“self”:“taught”,“programming”:“wizard”}
forkeyinmy_dict:
print(key)
>>“self”
>>“wizard”
Eachoftheseexamplesloopsthroughaniterable,andprintseachiteminit.Youwillnotice
weusedseveraldifferentvariablenamesforvariable_name.Inthissyntax,insteadofusingi
,youwanttouseadescriptivevariablename.Inthefirstexample,weusedcharacter,because
eachitematanindexinastringiscalledacharacter.Inthesecondandthirdexamples,we
useditem,becauseeachobjectinalistortupleiscalledanitem.Inthelastexample,weused
key,becausewhenyouiteratethroughadictionarylikethis,youcanonlyaccesseachkeyin
thedictionary,notthevalue—sowechoseavariablenametomakethisclear.
Beingabletoloopthroughaniterableisveryuseful.Iterablesareusedtostoredata,
andyoucanuseforloopstoloopthroughyourdatatoeasilymakechangestoit,ormove
thedatafromoneiterabletoanother.
WhileLoops
Whileforloopsexecutecodeacertainnumberoftimes,whileloopsexecutecodeas
longastheexpressioninitsheaderevaluatestoTrue.Thesyntaxforcreatingawhileloopis
while[expression]:[code_to_execute].Likeaforloop,awhileloopgoesaroundlikea
circleexecutingcode.Thedifferenceisinsteadofexecutingcodeasetamountoftimes,a
whileloopexecutescodeaslongastheexpressionwedefineinitsheaderevaluatestoTrue.
IfweuseanexpressionthatalwaysevaluatestoTrue,ourloopwillrunforever.Thisis
calledaninfiniteloop.Writinganinfiniteloopiseasy(bepreparedtopresscontrol-con
yourkeyboardinthePythonshell.Itistheonlywaytostopaninfiniteloopfromrunning).
#https://github.com/calthoff/tstp/blob/master/part_I/loops/while_loops_ex1.py
whileTrue:
print(“Hello,World!”)
>>HelloWorld
...
BecauseawhilelooprunsaslongasitsexpressionevaluatestoTrue—andTruealways
evaluatestoTrue—thisloopwillrunforever,continuouslyexecutingthecodewedefined.In
otherwords,ourprogramwillneverstopprinting“Hello,World!”.
Nowlet’slookatawhileloopwithanexpressionthatwilleventuallyevaluatetoFalse:
#https://github.com/calthoff/tstp/blob/master/part_I/loops/while_loops_ex2.py
x=10
whilex>0:
print(‘{}’.format(x))
x-=1
print(“HappyNewYear!”)
>>10
>>9
>>8
>>7
>>6
>>5
>>4
>>3
>>2
>>1
>>‘HappyNewYear!’
OurwhileloopwillexecuteitscodeaslongastheexpressionwedefinedisTrue.Inthis
case,thatmeansitwillexecuteitscodeaslongasx>0.xstartsat10(wedefinedxbefore
wecreatedourwhileloop).Thefirsttimethroughourloop,xis10,sox>0evaluatesto
True.Ourwhileloop’scodeprintsxandthendecrementsxby1—xisnow9.Thenext
timearoundtheloopxgetsprintedandxbecomes8.Thiscontinuesuntilxbecomes0,at
whichpointx>0evaluatestoFalseandourloopends.Pythonthenexecutesthenextlineof
codeafterourloop—print(“HappyNewYear!”)—whichprints“HappyNewYear!”.
Break
Youcanprematurelyendafororwhileloopwiththekeywordbreak.Forexample,
thefollowingloopwillrunonehundredtimes:
#https://github.com/calthoff/tstp/blob/master/part_I/loops/break.py
foriinrange(0,100):
print(i)
>>0
>>1
...
Butifweaddabreakstatementtothecodetheloopexecutes,thelooponlyrunsonce:
#https://github.com/calthoff/tstp/blob/master/part_I/loops/break_ex2.py
foriinrange(0,100):
print(i)
break
>>0
Theloopgoesaroundonceandprints0.Whenthethebreakkeywordisexecuted,theloop
ends.Thisisusefulinmanysituations.Forexample,wecanwriteaprogramthataskstheuser
forinputuntiltheytype“q”toquit:
#https://github.com/calthoff/tstp/blob/master/part_I/loops/break_ex3.py
“““Ifyouareunfamiliarthereferenceinthisexample,gowatchMontyPythonandthe
Holy
Grail!”””
questions=[“Whatisyourname?”,“Whatisyourfavoritecolor?”,“Whatisyour
quest?”]
n=0
whileTrue:
print(“Typeqtoquit”)
answer=input(questions[n])
ifanswer==“q”:
break
n+=1
ifn>2:
n=0
Eachtimethroughourinfiniteloop,ourprogramwillasktheuseraquestionfromourlistof
questions.Weusethevariablentokeeptrackofanumberwhichweuseasanindextogeta
questionfromourquestionslist.Whennbecomesgreaterthan2,we’verunoutofquestions
andwesetnbackto0whichwillaskthefirstquestioninthelist.Thiswillgoon
indefinitely,unlesstheusertypesin“q”,inwhichcaseourprogramhitsthebreakkeyword
andtheloopends,whichendsourprogram.
Continue
Youcanusethekeywordcontinuetostopexecutingafororwhileloop’scode,and
jumptothetopofaloop.Sayforinstance,wewanttoprintthenumbersfrom1to5,except
forthenumber3.Wecandothisbyusingaforloopwiththecontinuekeyword:
#https://github.com/calthoff/tstp/blob/master/part_I/loops/continue.py
foriinrange(1,6):
ifi==3:
continue
print(i)
>>1
>>2
>>4
>>5
Inourloop,wheniequals3,ourprogramhitsthecontinuekeyword.Insteadofcausingour
looptoexitcompletely—likethebreakkeyword—thelooppersistsbutwegetjumpedtothe
topofourloop,whichmeansanyoftheloop’scodethatwouldhaveexecutedthattime
aroundtheloopgetsskipped.Inthiscasewheniisequalto3,everythingaftercontinueis
skipped(inthiscaseprint(i)).Theresultis3isnotprinted.Hereisthesameexamplewitha
whileloop:
#addgithub
i=1
whilei<=5:
ifi==3:
continue
print(i)
NestedLoops
Youcancombineloopsinvariousways.Forexample,youcanhaveoneloopinsideof
anotherloop.Youcanalsohavealoopinsidealoopinsidealoop.Thereisnolimittothe
amountoftimesyoucandothis,althoughinpracticeyouwanttolimitthenumberoftimes
youdo.
Whenaloopisinsideanotherloop,thesecondloopissaidtobenestedinthefirst
loop.Thefirstloopiscalledtheouterloop,andthenestedloopiscalledtheinnerloop.When
younesttwoloops,theinnerlooprunsitsfullcourseeachtimearoundtheouterloop.Hereis
anexample:
#https://github.com/calthoff/tstp/blob/master/part_I/loops/nested_loops.py
foriinrange(1,3):
print(i)
forletterin[‘a’,‘b’,‘c’]:
print(letter)
>>1
>>‘a’
>>‘b’
>>‘c’
>>2
>>‘a’
>>‘b’
>>‘c’
Thenestedforloopwilliteratethroughthelist“[a’,‘b’,‘c’]”howevermanytimesthe
outsideloopruns—inthiscasetwice.Ifwechangedourouterlooptorunthreetimes,the
innerloopwoulditeratethroughthelistthreetimes.
Ifyouhavetwolistsofnumbersandwanttocreateanewlistwithallofthenumbers
fromeachlistaddedtogetheryoucanusetwoforloops:
#https://github.com/calthoff/tstp/blob/master/part_I/loops/nested_loops_ex2.py
list1=[1,2,3,4]
list2=[5,6,7,8]
added_up=[]
foriinlist1:
forjinlist2:
added_up.append(i+j)
print(added_up)
>>[6,7,8,9,7,8,9,10,8,9,10,11,9,10,11,12]
Inthesecondforloopweusedthevariablejbecauseiisalreadyinusebythefirstloop.
Youcanalsonestaforloopinsideawhileloopandviceversa:
#https://github.com/calthoff/tstp/blob/master/part_I/loops/nested_loops_ex3.py
whileinput('Continueyorn?')!='n':
foriinrange(1,5):
print(i)
>>
Continueyorn?y
1
2
3
4
5
Continueyorn?y
1
2
3
4
5
Continueyorn?n
>>>
Thisprogramwillprintthenumbers0-4untiltheuserentersn.
Challenge
Writeaprogramthathasaninfiniteloop(withqtoquit),andeachtimethroughthe
loop,itaskstheusertoguessanumberandtellsthemwhethertheirguesswasrightorwrong.
Chapter8.Modules
Imaginewewantedtobuildalargeprojectwithtenthousandlinesofcode.Ifwekeptallof
thatcodeinonefile,ourprojectwouldbedifficulttonavigate.Everytimetherewasanerror
orexceptioninourcode,wewouldhavetoscrollthroughafilewithtenthousandlinesto
findthelineofthecodewithaproblem.Programmerssolvethisproblembydividinglarge
programsupintomultiplepieces.Eachpieceisstoredinamodule—whichisanothername
foraPythonfile.PythonhasaspecialsyntaxthatletsyouusecodefromonePythonmodule
inanotherPythonmodule.Pythonalsocomeswithbuilt-inmodulesyoucanusethatgiveyou
accesstoextrafunctionality.
ImportingBuilt-inModules
Inordertouseamodule,youmustimportitfirst,whichmeansuseaspecialsyntaxto
makethecodeinthemoduleavailabletouseinyourprogram.Thisisdonewiththesyntax
import[module_name].importisakeywordforimportingmodulesandmustbefollowed
bythemodulename.Fornow,weareonlygoingtolearnhowtoimportbuilt-inmodulesand
moduleslocatedinthesamefolderasthemoduleyouareimportingitfrom.InPartIII,we
learnhowimportmodulesfromdifferentlocations.
WecanimportPython’sbuilt-inmathmodulewiththefollowingsyntax:
importmath
>>
ThemathmoduleisamodulethatcomeswithPythonwhenyouinstallit.Itisaregular
Pythonfilewithabunchofmathrelatedfunctionality—itcontainsPythonfunctionsthatare
usefulwhenyouaredoingmath.Onceyou’veimportedamodule,youcanuseanyofthe
codefromit.Youcanaccessafunctioninthemodulewiththesyntax[path_to_module].
[function_name]().Withthissyntax,youcanuseanyofthecode(suchasafunction)fromthe
mathmoduleinyourprogram:
importmath
math.fabs(-3)
>>3
Thefabsfunctionreturnstheabsolutevalueoftheparameteryoupassin.Youmaybe
wonderinghowyouaresupposedtoknowthereisamathmodulewithafunctioncalledfabs
init.AlistofPython’sbuilt-inmodulescanbefoundathttps://docs.python.org/3/py-
modindex.html.Ifyousearchforthemathmodule,thereisalinkthattakesyoutoapagethat
listseveryfunctioninthemathmodule,whateachfunctiondoes,andwhatparametersit
takes.
Anotherbuilt-inmoduleistherandommodule.Hereisanexampleofimportingthe
randommodule,andusingafunctionfromitcalledrandintthattakestwonumbersas
parametersandreturnsarandomnumberbetweenthem.
#Theoutputofthisprogrammightnotbe52whenyourunit—it’srandom!
importrandom
random.randint(0,100)
>>52
Thereareothersyntaxesforimportingmodules,butingeneralimport[module_name]isthe
syntaxyoushoulduse,andtheonewewillbeusingthroughoutthebook.Finally,youshould
doalloftheimportsforyourprogramatthetopofyourfile.
ImportingModules
Inthissection,wearegoingtocreateamodule,andusethecodefromitinanother
module.Firstcreateanewfolderonyourcomputercalledtstp,thencreateafilecalled
hello.pyinit.Insidehello.pyaddthefollowingcode:
#https://github.com/calthoff/tstp/blob/master/part_I/modules/hello.py
defprint_hello():
print(“Hello”)
Savethefile.Insidethetstpfolder,createanotherPythonfilecalledproject.py.Inside
project.pyaddthefollowingcode:
#https://github.com/calthoff/tstp/blob/master/part_I/modules/project.py
importhello
hello.print_hello()
>>‘Hello’
Usingtheimportkeywordwecaneasilyusecodefromourfirstmoduleinoursecond
module.
Challenge
Ichallengeyoutowritethreefunctionsinamodule,andusetheminanotherPythonprogram.
Chapter9.Files
Pythonmakesiteasytoworkwithfiles.Youcaneasilyreadandwritedatafromafile.
Readingdatafromafilemeansaccessingthefilescontent.Writingdatatoafilemeansadding
newcontenttothefile.Thisisusefulwheneveryouwanttousedatafromafileinaprogram,
oroutputdatafromaprogramtoafile.Inthischapter,wewilllearnthebasicsofworking
withfiles—includinghowtoreadandwritetofiles.
WorkingWithFiles
Theopenfunctiontakesastringrepresentingthepathtoafileandastringrepresenting
themodetoopenthefileinasparameters.Thepathtoafile—alsocalledafilepath—
representsthelocationonyourcomputerafileresides,forexample:
/Users/calthoff/my_file.txtisthefilepathtoafilecalledmy_file.txt.Eachwordseparatedby
theslashthatprecedesitisthenameofafolder.Togethertheyrepresentthelocationofthat
file.Ifafilepathonlyhasthenameofthefile,Pythonwilllookforthefileinwhateverfolder
youarerunningyourprogramin.
Thesecondparameterrepresentsthemodetoopenthefilein,whichdeterminesthe
actionsyouwillbeabletoperformonthefile.Hereareafewofthemodesyoucanopena
filewith:
“r”Opensafileforreadingonly.
“w”Opensafileforwritingonly.Overwritesthefileifthefileexists.Ifthefiledoes
notexist,createsanewfileforwriting.
“w+”Opensafileforbothwritingandreading.Overwritestheexistingfileifthefile
exists.Ifthefiledoesnotexist,createsanewfileforreadingandwriting.
5
Onceyou’vepassedtheopenfunctionafilepathandmodeasparameters,opencreatesafile
object(WelearnaboutobjectsinPartII)wecanusetoreadorwritetoourfile(orboth
dependingonthemodeyouchose).Hereisanexamplewritingtoafile:
my_file=open(“my_file.txt”,“w”)
my_file.write(“HellofromPython!”)
my_file.close()
>>
opencreatesanewfilecalledmy_file.txt(becausewepassedin“w”asourmode)in
whateverdirectoryyouranourprogramin.Wesaveourfileobjectreturnedbytheopen
functionininthevariablemy_file.
Nowwecancallthewritemethodonourfileobject,whichacceptsastringasa
parameterandwritesittothenewfilewecreated.Finally,weclosedourfilebycallingthe
closemethodonthefileobject.Thisisanimportantstep.Wheneveryouopenafileusingthe
openmethod,youneedtocloseitwiththeclosemethod.Ifyouhaveaprogramwhereyou
usetheopenmethodonmultiplefiles,andyouforgettoclosethem,itcancauseproblemsin
yourprogram.
Usingwith
Becauseforgettingtoclosefilesyouopenedcancauseproblems,thereisasecond,
preferedsyntaxforopeningfiles.ThepreferredwaytoopenafileinPythonistousea
compoundstatementusingthewithkeywordandthesyntaxwithopen(‘my_file’,[mode])as
[variable_name]:[your_code].Whenyouuseacompoundstatementusingwiththefile
automaticallyclosesafterthelastsuiteexecutesinthestatement.Thefileobjectgetssavedin
avariableofyourchoosing([variable_name]).Hereistheexamplefromtheprevious
sectionwrittenusingthepreferredsyntaxforopeningafile:
#https://github.com/calthoff/tstp/blob/master/part_I/files/using_with.py
withopen(‘my_file.txt’,‘w’)asmy_file:
my_file.write(‘HellofromPython!’)
>>
Aslongasyouareinsideofthewithstatement,youcanworkwiththefileyouopenedusing
thevariableyoucreated—inthiscasemy_file.AssoonasPythonleavesthewithstatement,it
closesthefileforyou.
ReadingFiles
Ifyouwanttoaccessthecontentsofafile(readthefile)youpassin“r”asthesecond
parametertoopen.Thenyoucancallthereadmethodonyourfileobjectwhichreturnsan
iterableyoucaniteratethroughtogeteachlineofthefile.
#https://github.com/calthoff/tstp/blob/master/part_I/files/reading_files.py
withopen(“my_file.txt”,“r”)asmy_file:
forlineinmy_file.read():
print(line)
>>HellofromPython!
Youcanonlycallreadonafileonce(eachtimeyourunyourprogram)togetsitscontents,
soyoushouldsavethefilecontentsinavariableorcontainerifyouneedtousethefile
contentsagainlaterinyourprogram.Forexample,wecouldchangethepreviousexampleto
savethefilecontentsinalist:
#
my_list=list()
withopen(“my_file.txt”,“r”)asmy_file:
forlineinmy_file.read():
my_list.append(line)
print(my_list)
>>HellofromPython!
Withthefilecontentssavedinalist,wecaneasilyaccessitlaterinourprogramwheneverwe
needit.
CSVFiles
Pythoncomeswithabuilt-inmoduleforworkingwithCSVfiles.CSVstandsfor
commaseparatedvalue.ItisafileformatcommonlyusedinMicrosoftExcel:aprogramfor
creatingspreadsheets.AcommainaCSVfileiscalledadelimiter.Everypieceofdata
separatedbyacommarepresentsacellinExcel.EachlineoftheCSVfilerepresentsarowin
Excel.Hereisanexampleofthecontentsofacsvfile:
#my_file.csv
one,two,three
four,five,six
Youcouldloadthisfileintoexcelandone,twoandthreewouldeachgettheirowncellsin
thefirstrowofthespreadsheet;andfour,fiveandsixwouldeachgettheirowncellsinthe
secondrowofthespreadsheet.
WecanuseawithstatementtoopenaCSVfileliketheexamplefromtheprevious
section,butinsidethewithstatementweneedtousethecsvmoduletoconvertourfileobject
intoacsvobject.Thecsvmodulehasamethodcalledwriterthatacceptsafileobjectanda
delimiterandreturnsacsvobjectwithamethodcalledwriterowwecanusetowritetoour
CSVfile:
#https://github.com/calthoff/tstp/blob/master/part_I/files/csv_files.py
importcsv
withopen(‘my_file.csv’,‘w’)ascsvfile:
spamwriter=csv.writer(csvfile,delimiter=',’)
spamwriter.writerow([‘one’,‘two’,‘three’])
spamwriter.writerow([‘four ’,‘five’,‘six’])
>>
Thewriterowmethodacceptsalistasaparameter.Everyiteminthelistgetswrittentothe
CSVfileandeachitemisseparatedbythedelimiteryoupassedtothewritermethod(inthis
caseacomma).writerowonlywritesonerow,sowehavetocallittwicetocreatetworows.
Whenyourunthisprogram,itwillcreateanewfilecalledmy_file.csvandwhenyouopen
thefilewithatexteditor,itwilllooklikethis:
#my_file.csv
one,two,three
four,five,six
IfyouloadthisfileintoExcel(orGoogleSheetsafreeExcelalternative),thecommas
disappear,butone,twoandthreewilleachhavetheirowncellinrowone;andfour,five
andsixwilleachhavetheirowncellinrowtwo.
Wecanalsousethecsvmoduletoreadthecontentsofafile.ToreadfromaCSVfile,
firstwepassin‘r ’toopenasasthesecondparameter.Thisopensthefileforreadingonly.
Insteadofusingthewritermethodlikethepreviousexample,weusethereadermethod,but
stillpassinthefilepathandacommaasthedelineator.
readerreturnsaniterablewecaniteratethroughtoprinteachrow.Wecancallthejoin
methodonacommatoaddacommainbetweeneachvaluetoprintthecontentsofthefile
likeitappearsintheoriginalfile:
#https://github.com/calthoff/tstp/blob/master/part_I/files/csv_files_ex2.py
importcsv
withopen(‘my_file.csv’,‘r ’)ascsvfile:
spamreader=csv.reader(csvfile,delimiter=',')
forrowinspamreader:
print(','.join(row))
>>one,two,three
>>four,five,six
Challenge
Datapersistswhenitoutlivestheprogramthatcreatedit.Wecanusefilestopersistdataby
writingtheoutputofourprogramstoafile.Writeaprogramthatcollectsdatafromauser—
andsavesittoafile—sothedatapersists.
Chapter10.BringingItAllTogether
Inthischapter,wearegoingtocombinetheconceptswe’velearnedsofartobuilda
text-basedgame:theclassicgameHangman.Ifyou’veneverplayedHangman,here'showit
works.PlayerOnepicksasecretwordanddrawslinesrepresentingeachletterintheword
(wewilluseanunderscoretorepresenteachline).PlayerTwotriestoguessthewordone
letteratatime.IfPlayerTwoguessalettercorrectly,thecorrespondingunderscoreis
replacedwiththecorrectletter.IfPlayerTwoguessesincorrectly,PlayerOnedrawsapiece
ofapictureofahangman(apersonhanging).IfPlayerTwocompletesthewordbeforethe
pictureofthehangmanisdrawn,theywin,ifnottheylose.Inourprogramthecomputerwill
bePlayerOne,andthepersonplayingthegamewillbePlayerTwo.Areyoureadytobuild
Hangman?
Hangman
Hereisthebeginningofourhangmancode:
#
https://github.com/calthoff/tstp/blob/master/part_I/bringing_it_all_together/hangman.py
defhangman():
word="caat"
wrong_guesses=0
stages=["","________","||","|0
","|/|\","|/\","|"]
letters_left=list(word)
score_board=['__']*len(word)
win=False
print('WelcometoHangMan')
Firstwecreateafunctioncalledhangmanwecancalltostartthegame.Thewordtoguessis
storedinthevariableword(youcanchangethewordtheplayerhastoguessbychangingthe
variable).Weuseanothervariablewrong_guessestokeeptrackofhowmanyincorrect
letterstheplayerhasguessed.
stagesisalistfilledwithstringswewillusetodrawourHangman.Wheneachstringin
thestageslistisprintedonanewline,apictureofahangmanforms.letters_leftisalistmade
upofeachcharacterinword.Wewilluseittokeeptrackofwhichlettersarelefttoguessin
ourword.
scoreboardisalistofstringsusedtokeeptrackofthehintswedisplaytotheusere.g.,
“c__t”ifthewordis“cat”.Theinitialvalueofscore_boardiscalculatedwith[‘__’]*
len(word)whichreturnsalistmadeupofthenumberofunderscorestostartwith.For
exampleifthewordis“cat”score_boardstartsas[“__”,“__”,“__”].
Finally,wehaveawinvariablethatstartsasFalsetokeeptrackofwhethertheplayer
haswonthegameyetornot.TostartthegameoffweprintWelcometoHangman.
Whenyoubuildagame,younormallyusealoopthatcontinuesuntilthegameisover.
Hereistheloopwewilluseinourgame:
whilewrong_guesses<len(stages)-1:
print('\n')
guess=input("Guessaletter")
ifguessinletters_left:
character_index=letters_left.index(guess)
score_board[character_index]=guess
letters_left[character_index]='$'
else:
wrong_guesses+=1
print(''.join(score_board))
print('\n'.join(stages[0:wrong_guesses+1]))
if'__'notinscore_board:
print('Youwin!Thewordwas:')
print(''.join(score_board))
win=True
break
Ourloopcontinuesaslongasthevariablewrong_guessesislessthanthe
len(wrong_guesses)-1.wrong_guesseskeepstrackofthenumberofwrongletterstheuser
hasguessed,soassoonastheuserhasguessedmorewronglettersthanthenumberofstrings
thatmakeupthehangman,thegameisover.Thereasonwesubtract1isbecausethelength
functioninPythondoesnotcountfrom0,itcountsfrom1.Inordertocompensateforthis,
wehavetosubtractone,tocompensateforthefactthatlengthcountsstartingfrom1instead
of0.Thereasonwewanttocountfrom0isbecausestagesisalist,andwearegoingtobe
usingwrong_guessesasanindextogetthestringsfromstagesandindexesstartat0.
Insideourloop,weprintablankspacetomakeourgamelooknicewhenit’sprintedin
theshell.Nextwecollecttheplayer ’sguesswiththebuilt-ininputfunctionandstorethevalue
inthevariableguess.
Ifguessisinletters_left(avariablefromthebeginningofourprogramthatkeeps
trackofthelettersthathaven’tbeenguessedyet),theplayerguessedcorrectly.Iftheplayer
guessedcorrectlyweneedtoupdateourscore_boardlist,whichweuselaterinthegameto
displaythescore.Iftheuserguessed“c”,wewanttochangeourscore_boardtolooklike
this:[“c”,“__”,“__”].Weusetheindexmethodonourletters_leftlisttogetthefirstindex
oftheletterthatwasguessed.Weusethatindextoreplacetheunderscoreinscore_boardat
theindexwiththecorrectlyguessedletter.Wehaveaproblemthough.Becauseindexonly
returnsthefirstindexofthecharacterwearelookingfor,ourcodewillnotworkifword
(thewordtheplayerisguessing)hasmorethanoneofthesamecharacter.Tocompensatefor
this,wemodifyletters_leftbyreplacingthecharacterthatwasjustcorrectlyguessedwitha
dollarsignsothatthenexttimearoundtheloop,ifthereismorethanoneletterintheword
indexwillfindsecondoccurrenceofthelettersincethefirstoccurrenceoftheletterwas
replacedbyadollarsign.Ifontheotherhandtheplayerguessesanincorrectletter,wesimply
incrementwrong_guessesby1.
Nextweprintthescoreboardandprintourhangmanusingourscore_boardandstages
lists.Toprintthescoreboard,allwehavetodoisprint''.join(score_board).
Printingthehangmanistrickier.Wheneachofthestringsinourstageslistisprinted
onanewline,acompletepictureofahangmanisprinted.Wecaneasilycreatetheentire
hangmanbyprinting'\n'.join(stages).Thisconnectseachstringinthestageslistwitha
blankspace(\n).Butwewanttoprintourhangmanatthestagewearecurrentlyat,whichwe
accomplishbyslicingourstageslist.Westartatstage0andsliceupuntilwhateverstagewe
areat(representedbythevariablewrong_guesses)plusone.Thereasonweaddone,is
becausewhenyouareslicing,theendslicedoesnotgetincludedintheresults.Thisgivesus
onlythestringsweneedtoprintthestageofthehangmanwearecurrentlyonwhichwethen
print.
ThelastthingwedoinourloopischeckiftheuserhaswonthegameIfthereareno
moreunderscoresinthescore_boardlist,weknowtheuserhasguessedallthelettersand
wonthegame.Iftheuserhaswon,weprintthattheywonandweprintthewordtheycorrectly
guessed.WealsosetthevariablewintoTruewhichisusedwhenwebreakoutofourloop.
Oncewebreakoutofourloop,iftheuserwon,wedonothing,andtheprogramisover.
Iftheuserdidnotwin,thevariablewinwillbeFalse.Ifthatisthecaseweknowtheuser
lostthegameandweprintthefullhangman,print“Youlose!”followedbythewordthey
incorrectlyguessed:
ifnotwin:
print('\n'.join(wrong_guesses[0:stage]))
print('Youlose!Thewordswas{}'.format(word))
hangman()
Hereisourcompletecode:
#
https://github.com/calthoff/tstp/blob/master/part_I/bringing_it_all_together/hangman.py
defhangman():
stage=0
wrong_guesses=["","________","||","|0","|/|\","|/\
","|"]
word="cat"
score_board=['__']*len(word)
win=False
print('WelcometoHangMan')
whilestage<len(wrong_guesses)-1:
print('\n')
guess=input("Guessaletter")
ifguessinword:
score_board[word.index(guess)]=guess
else:
stage+=1
print((''.join(score_board)))
print('\n'.join(wrong_guesses[0:stage+1]))
if'__'notinscore_board:
print('Youwin!Thewordwas:')
print(''.join(score_board))
win=True
break
ifnotwin:
print('\n'.join(wrong_guesses[0:stage]))
print('Youlose!')
hangman()
Challenge
Buildingtext-basedgamesisagreatwaytoimproveyourprogrammingability.Build
anothertext-basedgamethatinterestsyou.
Chapter11.Practice
“Thefoolwonders,thewisemanasks.”
—BenjaminDisraeli
Ifthisisyourfirstprogrammingbook,Irecommendyouspendtimepracticingbefore
movingontothenextsection.Inthischapter,Iprovideexercisestohelpyougetadditional
practicebeforemovingontothenextsection,resourcestocheckout,andwecoverhowto
gethelpifyougetstuck.
Exercises
0.Createatextbasedgameofyourfavoritesport.
0.Makeupyourowntextbasedgame.WhenIwasstartingoutIbuiltafantasybased
gamebasedonHeroesofMightandMagicIII
0.Buildatextbasedmagic8ballprogramwheretheusercanshakeamagiceightball
andgetpredictionsabouttheirfuture
0.Createaprogramthataskstheuserwhatkindofmoodtheyareinandrecommendsa
song.
0.Buildaprogramthatprintsouttenbrandsandletstheusertypethemin.Whenthey
do,theprogramshouldprintthebrand’strademarki.e.,theusercouldtypeNikeandthe
programwouldprint“Justdoit.”
Read
0.http://programmers.stackexchange.com/questions/44177/what-is-the-single-most-
effective-thing-you-did-to-improve-your-programming-skil
0.https://www.codementor.io/ama/0926528143/stackoverflow-python-moderator-
martijn-pieters-zopatista
GettingHelp
Ifyougetstuck,Ihaveafewsuggestions.Thefirstispostingaquestionon
http://forum.theselftaughtprogrammer.io,itisaforumIsetupforpeoplereadingthisbook
togetanswerstoanyquestionstheyhave.
IalsorecommendcheckingoutStackExchange—anamazingresourcefor
programmers.TherearetwowebsitesonStackExchangeyoushouldexplore.Thefirstis
StackOverflow.Ifyouhaven’tusedityet,youwillsoonbefamiliarwithit.Googlealmost
anyprogrammingquestionandananswerwillpopuponStackoverflow,whichmakesita
gamechangerforlearningtoprogram.
Learningtorelyonotherpeople'shelpwasanimportantlessonforme.Strugglingto
figurethingsoutisamajorpartofthelearningprocess,butatsomepointitbecomes
counter-productive.Workingonprojectsinthepast,Iusedtocontinuetostrugglewaypast
thepointwhereitwasproductive.Todayifthathappens,Iwillpostaquestiononline,ifthe
answerisnotalreadyonthere.EverytimeI’vepostedaquestiononline,someonehasbeen
abletoanswerit.Ican’tsayenoughabouthowhelpfulandfriendlytheprogramming
communityis.
OtherResources
linktowebsiteforotherpythonresources
PartIIIntroductiontoObject-oriented
Programming
Chapter12.ProgrammingParadigms
“Thereareonlytwokindsoflanguages:theonespeoplecomplainaboutandtheonesnobody
uses”
—BjarneStroustrup
Aprogrammingparadigmisacertainwayofprogramming.Therearemanydifferent
programmingparadigms.Youwon’tneedtolearnallofthematthebeginningofyourcareer
—butit’simportanttoknowwhatsomeofthemostpopularparadigmsare.The
programmingparadigmswewillgooverinthischapterareimperativeprogramming,
functionalprogrammingandobject-orientedprogramming—withafocusonobject-oriented
programming—thesubjectoftheremainingchaptersofthissection.
State
Oneofthefundamentaldifferencesbetweendifferentprogrammingparadigmsisthe
handlingofstate.Stateisthedatayourprogramhasaccessto.Programsstoredatain
variables—sostateisthevalueofaprogram’svariablesatagiventimetheprogramis
running.
ImperativeProgramming
InPartI,welearnedtoprogramimperatively.Imperativeprogrammingcanbe
describedas“dothis,thenthat”.Animperativeprogramisasequenceofstepsmovingtoward
asolution—witheachstepchangingtheprogram’sstate.Anexampleofimperative
programmingwouldbe:
x=2
y=4
z=8
xyz=x+y+z
>>14
Eachstepoftheprogramchangestheprogram'sstate.Wegettoxyzbyfirstdefiningx,
followedbyy,followedbyzandfinallydefiningxyz.
FunctionalProgramming
Functionalprogrammingisanotherpopularprogrammingparadigm.Itoriginatesfrom
lambdacalculus.Functionalprogramminginvolveswritingfunctionsthat—giventhesame
input—alwaysreturnthesameoutput.Infunctionalprogramming,youonlyprogramwith
functions,youdonotuseclasses—afeatureofobject-orientedprogrammingwewilllearn
aboutshortly.ThereisalotofjargoninfunctionalprogrammingandMaryRoseCookdoes
agreatjobcuttingthroughitwithherdefinition,“Functionalcodeischaracterisedbyone
thing:theabsenceofsideeffects.Itdoesn’trelyondataoutsidethecurrentfunction,andit
doesn’tchangedatathatexistsoutsidethecurrentfunction.”61Shefollowsherdefinition
withanexamplewhichIwillalsosharewithyou.Hereisanunfunctionalfunction:
a=0
defincrement():
globala
a+=1
Hereisafunctionalfunction:
defincrement(a):
returna+1
Thefirstfunctionisunfunctionalbecauseitreliesondataoutsideofitself,andchangesdata
outsideofthecurrentfunctionbyincrementingaglobalvariable.Thesecondfunctionis
functionalbecauseitdoesnotrelyonanydataoutsideofitself,anditdoesnotchangeany
dataoutsideofitselfeither.Functionalprogrammerswritefunctionsthiswaytoeliminateside
effects—theunintendedconsequencesthathappenwhenyouareconstantlychangingthestate
ofyourprogram.
Object-orientedProgramming
Theobject-orientedprogrammingparadigminvolveswritingprogramswhereyou
defineandcreateobjectsthatinteractwitheachother.We’vebeenprogrammingwithobjects
thiswholetime—strings,integersandfloatsareallexamplesofobjects.Butyoucanalso
defineyourownobjectsusingclasses.Classesaretheblueprintusedtocreateobjects.You
canthinkofaclassastheideaofanobject.Thinkofanorange.Anorangeisanobject.A
fruitweighingbetween2to10ouncesistheideaofanorange—aclass.
WecanmodelorangesinPythonbydefiningaclasswecanusetocreateorange
objects.Wedefineaclassusingtheclasskeywordfollowedbythenamewewanttogiveour
class.Aclassisacompoundstatementwithaheaderfollowedbysuites.Youwritesuitesafter
theheader,whichcanbesimplestatements,aswellascompoundstatementscalledmethods.
Methodsarelikefunctions,buttheyaredefinedinsideofaclass,andcanonlybecalledonthe
objecttheclasscancreate.WesawdifferentexamplesofthisinChapter5whenwecalled
variousmethodsonstrings.Hereisanexamplehowwecanrepresentanorangeusingaclass
inPython:
#
https://github.com/calthoff/tstp/blob/master/part_II/object_oriented_programming/orange_ex1.py
classOrange:
print("Orangecreated!")
Westartedwiththeclasskeywordfollowedbythenameofourclass—inthiscaseOrange
becausewearemodelingoranges.Byconvention,classesinPythonalwaysstartwithacapital
letterandarewrittenincamelCase—whichmeansifaclassnameismadeupofmorethan
oneword,thewordsshouldnotbeseparatedbyanunderscore(likeafunctionname),instead
eachwordshouldbecapitalizedLikeThis.
Afterourclassdefinitionwehaveasimplestatement—print(“Orangecreated!”).This
codewillexecutewhenwecreateanorangeobject.Withthisclassdefinition,wecancreateas
manyOrangeobjectsaswe’dlike:
orange=Orange()
print(type(orange))
print(orange)
>>Orangecreated!
>><class'__main__.Orange'>
>><__main__.Orangeobjectat0x101a787b8>
WecreatedanewOrangeobjectusingthesamesyntaxweusetocallafunction—[
classname]().Thisiscalledinstantiatinganobject,whichmeanscreatinganewobject.
“Orangecreated!”printsassoonasweinstantiateourOrangeobject.Whenweprint
type(orange),thetypefunctiontellsusourOrangeobjectisaninstanceoftheOrangeclass
wejustcreated.WhenweprintourOrangeobject,PythonletsusknowitisanOrange
object,andthengivesusitslocationinmemory.Whenyouprintanobjectlikethis,the
locationinmemoryprintedonyourcomputerwillnotbethesameastheexample,because
theobject’slocationinmemorychangeseachtimetheprogramruns.
NowwearegoingtoaddamethodtoourOrangeclass.Youdefineamethodwiththe
samesyntaxasafunction.Thereareonlytwodifferences:amethodmustbedefinedasasuite
inaclass,andamethodhastoacceptatleastoneparameter(exceptinspecialcasesIwon’tgo
into).Youcannamethefirstparameterofamethodwhateveryou’dlike,butbyconvention
thefirstparameterinamethodisalwaysnamedself,andI’veneverseenthisconvention
broken.
Thereasoneverymethodmusthaveafirstparameterisbecausewheneveramethodis
calledonanobject,Pythonautomaticallypassesthemethodtheobjectthatcalledit.This
conceptexistsinmostprogramminglanguagesthatsupportobject-orientedprogramming,
however,Pythonmakesthepassingofselfasaparameterexplicitwhereasmanyother
languagesmakeitimplicit.WhatImeanisthatPythonmakesyouexplicitlydefineselfin
everymethodyoucreatewhereasinotherlanguagesselfisjustimpliedtohavebeenpassed
totheobject.Ifwedefineamethodandprintself,wewillseeitistheOrangeobjectwe
calledourmethodon:
classOrange:
print("Orangecreated!")
defprint_orange(self):
print(self)
Orange().print_orange()
>>Orangecreated!
>><__main__.Orangeobjectat0x101a787b8>
selfisusefulbecausewecanuseittodefineandaccessvariablesthatbelongtoourOrange
object.Wedothisbydefiningaspecialmethodcalled__init__,whichstandsforinitialize.
Whenyouinstantiateanobject,ifyou’vedefinedamethodcalled__init__,Python
automaticallycallsthe__init__methodforyouwhentheobjectiscreated.Inside__init__we
cangiveourOrangeobjectvariableswiththesyntaxself.[variable_name]=[variable_value]
.Hereisanexample:
classOrange:
print("Orangecreated!")
def__init__(self):
self.color="orange"
self.weight=10
defprint_orange(self):
print(self)
print(self.color)
print(self.weight)
orange=Orange()
orange.print_orange()
>>Orangecreated!
>><__main__.Orangeobjectat0x10564dba8>
>>orange
>>10
Using__init__wecannowcreateorangesthatgetacolorandweightwheninitializedandwe
canuseandchangethesevariablesinanyofourmethodsjustlikeregularvariables.In
Python,anymethodsurroundedonbothsidesbyunderscoresiscalledamagicmethodwhich
meansitdoessomethingspecial.Theprint_orangemethodwasusedtoillustrateanexample,
itwillnotbeamethodinourorangeaswecontinuetomodelit.
Wecanchangeourclassdefinitionsothepersoncreatingtheobjectcanpassintheir
ownvariableswhentheycreateaneworange,insteadoftheweightandcolorstartingwith
defaultvalues.Hereisournewclassdefinition:
#
https://github.com/calthoff/tstp/blob/master/part_II/object_oriented_programming/orange_ex3.py
classOrange:
def__init__(self,weight,color,mold):
"""allweightsareinoz"""
self.weight=weight
self.color=color
Nowwecancreateawidervarietyoforangesobjects:
orange=Orange(10,'orange',)
Wejustcreateda10oz(perthecomment“allweightareinoz”),orangecoloredorange.We
canaccesstheorangesvariablesusingdotnotation:
print(orange.weight)
>>10
print(orange.color)
>>“orange”
Wecanalsochangeanyofthevaluesofourorangeobject:
orange.weight=100
print(orange.weight)
>>100
Movingforwardwiththemodelingofourorange,thereismoretoanorangethanjust
itsphysicalpropertieslikecolorandweight.Orangescanalsodothings,andweneedtobe
abletomodelthataswell.Whatcananorangedo?Well,foronething,orangescangobad
frommold.WecanmodelamoldingorangebyaddingamoldvariabletoourOrangeclass,
andcreatingamethodthatincrementsthemoldvariablewhenit’scalled:
#
https://github.com/calthoff/tstp/blob/master/part_II/object_oriented_programming/orange_ex2.py
classOrange():
def__init__(self):
"""allweightsareinoz"""
self.weight=6
self.color='orange'
self.mold=0
defrot(self,days,temperature):
self.mold=days*(temperature*.1)
orange=Orange()
print(orange.mold)
orange.rot(10,98)
print(orange.mold)
>>0
>>98.0
Nowourorangeobjectswillbeabletorot.Wedefinedamethodthatacceptsthenumberof
daysit'sbeensincetheorangewaspicked,andtheaveragetemperatureduringthattimeas
parameters.Withourmade-upformula,wecanincreasetheamountofmoldtheorangehas
everytimewecalltherotmethod;andourorangenowhastheabilitytorot.
Thatwasalottotakein,solet'sgooveritallonemoretime.InObject-oriented
programming,weuseclassestomodelobjects.Theseobjectsgroupvariables(state)and
methods(foralteringstate)togetherinasingleunit—theobject.Classeshaveaspecial
methodcalled__init__thatPythoncallswhenanobjectiscreatedusingthesyntax
[class_name]().ThisisanexampleofoneofPython’smagicmethods.Thefirstargumentto
anymethodiscalledselfbyconventionandexistsbecausePythonautomaticallypassesthe
objectthatcalledthemethodintothemethod.
Challenge
Pickaprogrammingparadigmandresearchwhatproblemsitsolves.
Chapter13.TheFourPillarsofObject-oriented
Programming
"Gooddesignaddsvaluefasterthanitaddscost."
—ThomasC.Gale
Therearefourmainconceptsinobject-orientedprogramming—oftencalledthefour
pillarsofobject-orientedprogramming:inheritance,polymorphism,abstractionand
encapsulation.Theseconceptsmustallbepresentinaprogramminglanguageinorderforit
tobeconsideredanobject-orientedprogramminglanguage.Python,JavaandRubyareall
examplesofobject-orientedlanguages.Notallprogramminglanguagessupportobject-
orientedprogramming—forexampleHaskellisafunctionalprogramminglanguagethat
doesnotsupportobject-orientedprogramming.Inthischapter,wetakeadeeperlookat
object-orientedprogrammingbyexploringeachofthefourpillarsofobject-oriented
programming.
Inheritance
Inheritanceinprogrammingissimilartogeneticinheritance.Ingeneticinheritanceyou
caninheritattributesfromyourparents,likeyoureyecolor.Similarlywhenyoucreatea
class,itcaninheritfromanotherclass(whichisthencalleditsparentclass)—givingthenew
classyoucreatedtheparentclass’svariablesandmethods.Inthissectionwewillmodelakid
andadultusinginheritance.First,wedefineaclasstorepresentanadult:
“““
https://github.com/calthoff/tstp/blob/master/part_II/more_object_oriented_programming/inheritance_ex1.py
”””
classAdult():
def__init__(self,name,height,weight,eye_color):
"""heightisinfeet,weightinlbs."""
self.name=name
self.height=height
self.weight=weight
self.eye_color=eye_color
defprint_name(self):
print(self.name)
tom=Adult("Tom",6,150,"brown")
print(tom.name)
print(tom.height)
print(tom.weight)
print(tom.eye_color)
tom.print_name()
>>Tom
>>6
>>150
>>brown
>>Tom
UsingthisclasswecancreateAdultobjectswithaname,height,weightandeyecolor.In
addition,ourAdultobjectshaveamethodcalledprint_namethatprintstheparent’sname.
Wecanmodelahumanchildthatalsohasaname,height,weight,eyecolorandcan
printitsname;withanextramethodwedon’twantourAdultobjectstohavecalled
print_cartoon;usinginheritance.Youinheritfromaparentclassbyaddingparenthesistothe
classnameyouaredefining,andpassingintheclassnameyouwanttoinheritfromasa
parameter.Hereisanexample:
“““
https://github.com/calthoff/tstp/blob/master/part_II/more_object_oriented_programming/inheritance_ex2.py
”””
classAdult():
def__init__(self,name,height,weight,eye_color):
#heightisinfeet,weightinlbs.
self.name=name
self.height=height
self.weight=weight
self.eye_color=eye_color
defprint_name(self):
print(self.name)
classKid(Adult):
defprint_cartoon(self,favorite_cartoon):
print("{}'sfavoritecartoonis{}".format(self.name,
favorite_cartoon))
child=Kid("Lauren",3,50,"blue")
print(child.name)
print(child.height)
print(child.weight)
print(child.eye_color)
child.print_name()
child.print_cartoon('DuckTales')
>>brown
>>Ricky
>>DuckTales
BypassinginAdulttoourKidclass,ourKidclassinheritsthevariablesandmethodsof
ourAdultclass:whenwecreateaKidobjectwepassitaname,height,weightandeyecolor;
andourKidobjectisabletousethemethodprint_name;allofwhichwasinheritedfromits
parentclass(withouthavingtodefineanyofitinourKidclass).Thisisimportantbecause
nothavingtorepeatcodemakesourprogramsmallerandthereforemoremanageable.
AfterinheritingfromourAdultclass,allwehadtodowasdefineanewmethodcalled
print_cartooninourKidclasstocreateaKidclasswithallofthefunctionalityofourAdult
class,plusadditionalfunctionality;allwithoutaffectingourAdultclass.
Polymorphism
ThebestdefinitionI’vefoundofpolymorphismis“polymorphismistheability(in
programming)topresentthesameinterfacefordifferingunderlyingforms(datatypes)”69
Aninterfacereferstooneormorefunctionsormethods.Let’stakealookatasituation
wherethisisthecase:
print('HelloWorld')
print(200)
print(200.1)
>>“HelloWorld”
>>200
>>200.1
Inthisexample,wewereabletopresentthesameinterface(theprintfunction)forthree
differentdatatypes:astring,anintandafloat.Wedidn’tneedtocallthreeseparatefunctions
—print_string,print_int,orprint_floatinordertoprintthesethreedifferentdatatypes—
insteadPythonhasjustoneinterfaceforallofthem.
Let’stakealookatanotherexample.Saywewanttowriteaprogramthatcandraw
differentshapes:triangles,squaresandcircles.Eachoftheseshapesisdrawninadifferent
way,sothemethodstodrawthemwouldallhavedifferentimplementations.InPython,wecan
createdifferentdrawmethodsforeachshapesothatTriangle.draw()willdrawatriangle,
Square.draw()willdrawasquare,andCircle.draw()willdrawacircle.Eachoftheseshape
objectshasit’sowndrawinterfacethatknowshowtodrawitself.Whenwehaveashape
object,weknowwecancallthedrawfunctiontodrawtheshape.Thesameinterfaceis
presentedforallthedifferentshapedatatypes.
IfPythondidnotsupportpolymorphism—wewouldneedafunctionthatcreatesa
triangle,andanotherfunctioncalleddraw_triangletodrawit;afunctiontocreateacircle,
andafunctioncalleddraw_circletodrawit;etc.BecausePythonhaspolymorphism,every
shapecansimplybedrawnwithitsdrawmethod.Thismakesourshapeobjectsmucheasier
touseandexplain.Insteadofexplaining—wehavethreefunctionsrepresentingthreedifferent
shapes,andanotherthreefunctionsthatdrawseachofthem;wecansimplytellwhoeveris
usingthecode:wehavethreeshapes—ifyouwanttodrawone—callitsdrawmethod.
Abstraction
Weuseabstractioninobject-orientedprogrammingwhenwecreateaclassanddefine
itsmethods.Saywecreateaclasstorepresentaperson.Whenwedefineourpersonclass—
andthemethodsthatgowithit—wearecreatinganabstraction.Ourdefinitionofaperson
couldincludeeyecolor,haircolor,heightandethnicityaswellastheabilitytoread,write
anddraw.Wecouldhaveafivefootthreepersonwithblueeyes,blondehairunabletoread,
writeordraw.Orwecouldhaveasixfootfivepersonwithbrowneyes,brownhairthatcan
read,writeanddraw.Bothofthesefallintothecategoryofthepersonabstractionwe’ve
created.
Whenwedesignobject-orientedprograms,wecreateabstractionsofdifferentconcepts
thatallworktogethertoformourprogram.Forexample,wemaycreateanabstractionofa
person,andanabstractionofagovernment,andmodelhowmanypeopleliveundereach
governmentintheworld.Abstractionallowsustomodelobjectswithclearboundaries,and
havetheminteractwitheachother.InPart4welearnmoreaboutabstractionandhowPython
itselfisbuiltonmultiplelayersofit.
Encapsulation
Inobject-orientedprogramming,encapsulationhidesourcodesinternaldata.When
codeisencapsulated,itmeanswhenitiscalled,thecallercannotaccessthecode'sinternal
data.Takealookatthemethodget_data:
classData:
defget_data(self,index,n):
data=[1,2,3,4,5]
data.append(n)
Themethodhasavariablecalleddata.Whenwecallget_data,thereisnowayforusto
accessthisvariablebecauseofencapsulation.Iftherewasnoencapsulation,wemightbeable
toaccessthevariabledata—andappendntoit—likethis:
#warningthiscodedoesnotwork
Data.get_data.data.append(6)
Ifthiswasallowed,anyonecouldaccessthedatavariableinourget_datamethod.Insteadof
relyingonourimplementationoftheget_datamethod—theycouldappendntodata
themselves.Thisisnotaproblem—untilwechangetheimplementationofget_data.Whatif
wedecidewantthevariabledatatobeatupleinsteadofalist?Ifwemakethischange,itwill
breakanyone’scodecallingappendonthevariabledata,becausetuplesdonothavean
appendmethod.Butbecauseofencapsulation,thisscenarioisnotpossible(whichiswhythe
codedoesnotwork),andwecanbeassuredchangestotheinternalimplementationofour
codewon’tbreakourclient’scode(clientisatermforthepersonusingapieceofcode).
Composition
Whilecompositionisnotoneofthefourpillarsofobject-orientedprogramming,itis
animportantconceptrelatedtotherest.Compositionisusedtorepresent“hasa”
relationships—itoccurswhenoneobjectstoresanotherobjectasavariable.Forexample,say
wewanttorepresenttherelationshipbetweenadoganditsowner—thisisa“hasa”
relationship—adoghasanowner.Firstwedefineourdogandpeopleclasses:
classDog():
def__init__(self,name,breed,owner):
self.name=name
self.breed=breed
self.owner=owner
classPerson():
def__init__(self,name):
self.name=name
Whenwecreateourdogobject,wepassinapersonobjectastheownerparameter:
mick=Person("MickJagger")
dog=Dog("Stanley","FrenchBulldog",mick)
print(dog.owner)
>>MickJagger
NowourdogStanleyhasanowner—aPersonobjectnamedMickJagger—wecaneasily
reference.
Challenge
CreateanOrangeclassandobjectasmanytimesasyouhavetountilyoucandoit
withoutreferencingthischapter.
Chapter14.MoreObject-orientedProgramming
“Commentingyourcodeislikecleaningyourbathroom-youneverwanttodoit,butitreally
doescreateamorepleasantexperienceforyouandyourguests.”
—RyanCampbell
Inthischapterwecoveradditionalconceptsrelatedtoobject-orientedprogramming.While
someoftheseexamplesarespecifictoPython,themajorityoftheseconceptsarepresentin
otherlanguagesthatsupportobject-orientedprogramming.
HowVariablesWork
Inthissection,wearegoingtolearnmoreaboutvariables.Variable“point”toanobject.
number=100
numberpointstoanintegerobjectwiththevalue100.
[illustrationofpointtoanobject]
number=101
Whenweassignanewvaluetonumber,itpointstoanewintegerobjectwiththevalue101,
andtheoldintegerobjectwithavalueof100isdiscardedbecauseitisnolongerbeingused.
Twovariablescanpointtothesameobject:
x=100
y=x
xpointstoanintegerobjectwithavalueof100.Whenweassignytox,ynowpointstothe
sameintegerobjectxpointsto:theybothpointtoanintegerobjectwithavalueof100.
[insertillustrationfrom
http://cdn.oreillystatic.com/en/assets/1/event/95/Python%20103_%20Memory%20Model%20_%20Best%20Practices%20Presentation.pdf
]
Whatdoyouthinkthefollowingprogramwillprint?
x=10
y=x
x+=1
print(x)
print(y)
Theansweris11and10.xpointstoanintegerobjectwithavalueof10,andwhenwecreate
y,itpointstothesameintegerobject.Whenweincrementx,xpointstoanewintegerobject
—withavalueof11,buttheintegerobjectwithavalueof10isnotdiscarded,becauseitis
beingused:ystillpointstotheintegerobjectwithavalueof10.Sowhenweprintx,11
printsbecauseweassignedxtoanewintegerobject—11—butwhenweprinty—10prints
becausechangingthevalueofx,whichpointstotheintegerobject11,doesnotaffectthe
valueofy.Hereisanotherexampletoillustratethispoint.Whatdoyouthinktheoutputof
thiscodewillbe?
x=[1,2,3]
y=x
y[2]=100
print(x)
print(y)
Theoutputwillbe[1,2,100]twice.Thereasonisthatbothxandypointtothesamelist
object.Inthethirdline,wemakeachangetothatsinglelistobject,andwhenweprintboth
variables,itprintsthelistobjecttheybothpointto,withthechangesmadeinline3.
is
ThekeywordisreturnsTrueiftwoobjectsarethesameobject(theyarestoredinthesame
locationinmemory)andFalseifnot.
“““
https://github.com/calthoff/tstp/blob/master/part_II/more_object_oriented_programming/is_example.py
”””
classPerson:
def__init__(self):
self.name='Bob'
bob=Person()
the_same_bob=bob
print(bobisthe_same_bob)
another_bob=Person()
print(bobisanother_bob)
>>True
>>False
Whenweusethekeywordiswithbobandthe_same_bob,theresultisTruebecauseboth
variablespointtothesamePersonobject.WhenwecreateanewPersonobjectandcompare
ittotheoriginalbobtheresultisFalsebecausethevariablespointtodifferentPerson
objects.
None
Thebuilt-inconstantNoneisusedtorepresenttheabsenceofavalue:
x=None
x
>>None
WecantestifavariableisNoneusingconditionalstatements.
“““
https://github.com/calthoff/tstp/blob/master/part_I/introduction_to_programming/none_ex1.py
”””
x=10
ifx:
print("xisnotNone")
else:
print("xisNone:(")
>>xisnotNone
x=None
ifx:
print("xisnotNone")
else:
print("xisNone:(")
>>xisNone:(
Whilethismaynotseemusefulnow,itwillbelater.
ClassesAreObjects
InPython,classesareobjects.Thisideacomesfromtheinfluentialprogramming
languageSmallTalk.Thismeansthatwhenrunaprograminwhichyoudefineaclass—
Pythonturnsitintoanobject—whichyoucanthenuseinyourprogram:
classPterodactyl:
pass
print(Pterodactyl)
>><class'__main__.Pterodactyl'>
Withoutanyworkonourpart,Pythonturnsourclassdefinitionintoanobjectwhichwecan
thenuseinourprogram,byprintingitforinstance.
ClassVariablesvs.InstanceVariables
Classescanhavetwotypesofvariables—classvariablesandinstancevariables.Allof
thevariableswe’veseensofarhavebeeninstancevariablesdefinedwiththesyntaxself.
[variable_name]=[variable_value].Instancevariablesbelongtotheobjectthatcreatedthem.
Inotherwordswecandothis:
classLiger:
def__init__(self,name):
self.name=name
connor=Liger("Connor")
print(connor.name)
>>Connor
Classvariablesbelongtoboththeclassthatcreatedthemandtheobject.Thatmeanswecan
accessthemwiththeclassobjectPythoncreatesforeachclass:
classLiger:
interests=["swimming","eating","sleeping"]
def__init__(self,name):
self.name=name
print(Liger.interests)
>>['swimming','eating','sleeping']
InthisexamplewenevercreatedaLigerobject,yetwewereabletoaccesstheinterestsclassvariable.Thisisbecauseclass
variablescanbeaccessedbytheclassthatcreatedthem.Classvariablescanalsobeaccessedbytheobject:
classLiger:
interests=["swimming","eating","sleeping"]
def__init__(self,name):
self.name=name
larry=Liger("Larry")
print(larry.interests)
>>['swimming','eating','sleeping']
Classvariablesareusefulwhenyouwanteveryobjectinaclasstohaveaccesstoavariable.
InthiscasethenameofeachLigercanbeunique,butallofourLigerstohaveaccesstothe
samelistofinterests.
Privatevariables
Mostprogramminglanguageshavetheconceptofprivatevariablesandmethods:special
variablesandmethodsthedesignerofaclasscancreatethattheobjecthasaccessto,butthe
programmerusingtheobjectdoesnothaveaccessto.Onesituationprivatevariablesand
methodsareusefulinisifyouhavemethodorvariableinyourclasstheclassusesinternally,
butyouplantochangetheimplementationofyourcodelateron(oryouwanttopreservethe
flexibilitytohavetheoptionto),andthereforedon’twantwhoeverisusingtheclasstorely
onthosevariablesandmethodsusedinternallybecausetheymightchange(andwouldthen
breaktheircode).
UnlikeotherlanguagesPythondoesnothaveprivatevariables.Variablesthatarenot
privatearecalledpublicvariables,andallofPython’svariablesarepublic.Pythonsolvesthe
problemprivatevariablesresolveanotherway—byusingconvention.InPython,ifyouhavea
variableormethodthecallershouldnotaccess,youprecedeitsnamewithanunderscore.
Pythonprogrammersknowifamethodorvariablehasanunderscore,theyshouldn’tuseit,
althoughtheyarestillabletoattheirownrisk.Hereisanexampleofaclassthatusesthis
convention:
“““
https://github.com/calthoff/tstp/blob/master/part_II/more_object_oriented_programming/private_variables_ex1.py
”””
classPublicPrivateExample:
def__init__(self):
self.public_variable="callersknowtheycanaccessthis"
self._dontusethisvariable="callersknowtheyshouldn'taccessthis"
defpublic_method(self):
#callersknowtheycanusethismethod
pass
def_dont_use_this_method(self):
#callersknowtheyshouldn'tusethismethod
pass
Pythonprogrammersreadingthiscodewillknowself.public_variableissafetouse,butthey
shouldn’tuseself._dontusethisvariable,andiftheydo,theydosoattheirownriskbecause
thepersonmaintainingthiscodehasnoobligationtokeepself.dontusethisvariablearound
becausecallersareanotsupposedtobeaccessingit.Thesamegoesforthetwomethods:
Pythonprogrammersknowpublic_methodissafetouse,whereas_dont_use_this_methodis
not.
OverridingMethods
Whenaclassinheritsamethodfromaparent,wehavetheabilitytooverrideit.Takea
lookatthefollowingexample:
“““
https://github.com/calthoff/tstp/blob/master/part_II/more_object_oriented_programming/overriding_ex1.py
”””
classMammal:
def__init__(self,name):
self.hunger=100
self.tired=100
self.name=name
defprint_result(self,amount,action):
print("{}{}decreasedby{}.".format(self.name,action,
amount))
defeat(self,decrease):
self.hunger-=decrease
self.print_result(decrease,'hunger')
defsleep(self,decrease):
self.tired-=decrease
self.print_result(decrease,'tiredness')
classDolphin(Mammal):
pass
classTiger(Mammal):
defsleep(self,decrease):
self.tired-=decrease
print("Thetigerisreallytired!")
dolphin=Dolphin('dolphin')
dolphin.eat(10)
dolphin.sleep(10)
tiger=Tiger('tiger')
tiger.eat(10)
tiger.sleep(10)
>>dolphinhungerdecreasedby10.
>>dolphintirednessdecreasedby10.
>>tigerhungerdecreasedby10.
>>Thetigerisreallytired!
WecreatedtwoclassesthatinheritfromMammal.Thefirstclass,Dolphin,inheritsallofits
functionalityfromtheMammalparentclasswithoutmakinganychanges.Thesecondclass
Tigerdefinesamethodcalledsleep,withdifferentfunctionalitythanthesleepmethodit
inheritedfromitsparentclass.Whenwecalltiger.sleep,thenewmethodwedefinedgets
calledinsteadoftheinheritedmethod.Otherthanthis,TigerandDolphinhaveallthesame
functionalityinheritedfromtheparentclassMammal.
Super
Thebuilt-insuperfunction,letsuscallamethodaclassinheritedfromitsparent.The
superfunctionisusedwiththesyntaxsuper().[parent_method]([parameters])whereyou
replaceparent_methodwiththeparentmethodyouwanttocallandpassitanyparametersit
needs.Theparentmethodisthencalledandexecuted,andtherestofthecodeinthemethod
superwascalledfromthenfinishesexecuting.Hereisanexampleofhowwecancallthe
Mammalparentclass'ssleepmethodfromourTigerclasses’sleepmethod,inordertouse
thecodefromtheMammalclass’ssleepmethodfollowedbyadditionalfunctionality:
classTiger(Mammal):
defsleep(self,decrease):
super().sleep(decrease)
print("Thetigerisreallytired!")
tiger=Tiger('tiger')
tiger.eat(10)
tiger.sleep(10)
>>tigertirednessdecreasedby10.
>>Thetigerisreallytired!
FirstweusethesuperkeywordtocalltheMammalparentclass’ssleepmethodandpassin
thedecreasevariableasparameter.Mammal’ssleepmethodisexecutedandprints“tiger
tirednessdecreasedby10”.TheTigerclassessleepmethodthenexecutesthenew
functionalityweaddedtotheTigerclassessleepmethodand“Thetigerisreallytired!”
prints.Byusingthesuperkeywordwewereabletogiveachildclassthefunctionalityfrom
aparentclass’smethodwithouthavingtoretypethefunctionalityinthechildclass.Thisis
importantbecauseyoushouldalwaysavoidrepeatingcodewiththesamefunctionalityin
differentplacesinyourprogramwhenyoucan.
OverridingBuilt-inMethods
EveryclassinPythonautomaticallyinheritsfromaparentclasscalledObject.All
classesinPythoninheritmethodsfromthisparentclass.Python’sbuilt-infunctionsusethese
methods(whichwelearnedarecalledmagicmethods)—indifferentsituations—likewhenwe
printanobject:
classLion:
def__init__(self,name):
self.name=name
lion=Lion("Dilbert")
print(lion)
>><__main__.Lionobjectat0x101178828>
WhenweprintourLionobject,Pythoncallsthe__repr__methodonourobject,whichit
inheritedfromtheObjectparentclass.Itprintswhateverthe__repr__methodreturns.We
canoverridethisbuilt-inmethodtochangewhathappenswhentheprintfunctionprintsour
object.:
“““
https://github.com/calthoff/tstp/blob/master/part_II/more_object_oriented_programming/overriding_builtin.py
”””
classLion:
def__init__(self,name):
self.name=name
def__repr__(self):
returnself.name
lion=Lion("Dilbert")
print(lion)
>>Dilbert
Becauseweoverrodethe__repr__method,whenweprintourLionobject,theLionobject’s
name—Dilbert—getsprintedinsteadofsomethinglike<__main__.Lionobjectat0x101178828>
whichtheinherited__repr__methodwouldhavereturned.
Notallmagicmethodsareinherited.Pythonexpressionslike2+2expecttheoperandstohaveamethodtheoperator
canusetoevaluatetheexpression.Intheexample2+2,integerobjectshaveamethodcalled__add__whichiscalled
whentheexpressionisevaluated,but__add__isnotinheritedwhenyoucreateaclass.Wecancreateobjectsthatcanbe
usedasoperandsinanexpressionwiththeadditionoperatorbydefiningan__add__methodinourclass:
classAlwaysPositive:
def__init__(self,number):
self.number=number
def__add__(self,other):
returnabs(self.number+other.number)
x=AlwaysPositive(-20)
y=AlwaysPositive(10)
print(x+y)
>>10
OurAlwaysPositiveobjectscanbeusedasoperandsinanexpressionwiththeadditionoperatorbecausewedefinedamethod
called__add__.Themethodmustacceptasecondparameter,becausewhenanexpressionwithanadditionoperatoris
evaluated,__add__iscalledonthefirstoperandobject,andthesecondoperandobjectgetspassedinto__add__asa
parameter.Theexpressionthenreturnstheresultof__add__.
Inthisexample,weaddedatwist.Weusedthefunctionabstoreturntheabsolutevalueofthetwonumbersbeing
addedtogether.Becausewedefined__add__thisway,twoAlwaysPositiveobjectsaddedtogetherwillalwaysreturnthe
absolutevalueofthesumofthetwoobjects.
Challenge
Writeclassestomodelfourspeciesintheanimalkingdom.
Chapter15.BringingItAllTogether
“It'salltalkuntilthecoderuns.”
—WardCunningham
Inthischapter,wearegoingtousewhatwe’velearnedaboutclassesandobjectstocreatethe
cardgameWar.Ifyou’veneverplayed,Warisagamewhereeachplayerdrawsacardfrom
thedeck,andtheplayerwiththehighestcardwins.WewillbuildWarbydefiningaclass
representingacard,adeck,aplayer,andfinally,aclasstorepresentthegameitself.These
classeswillworktogethertocreatethegameWar.
Cards
Firstwedefineourcardclass:
#https://github.com/calthoff/tstp/blob/master/part_II/war/war.py
classCard:
suits=["spades","hearts","diamonds","clubs"]
values=[None,None,"2","3","4","5","6","7","8","9",
"10","Jack","Queen","King","Ace"]
def__init__(self,value,suit):
"""suitandvalueshouldbeintegers"""
self.value=value
self.suit=suit
def__lt__(self,other):
ifself.value<other.value:
returnTrue
ifself.value==other.value:
ifself.suit<other.suit:
returnTrue
else:
returnFalse
returnFalse
def__gt__(self,other):
ifself.value>other.value:
returnTrue
ifself.value==other.value:
ifself.suit>other.suit:
returnTrue
else:
returnFalse
returnFalse
def__repr__(self):
returnself.values[self.value]+"of"+self.suits[self.suit]
OurCardclasshastwoclassvariables:suitsandvalues.suitsisalistofstrings
representingallofthedifferentsuitsacardcouldbe.valuesisalistofstringsrepresenting
thedifferentnumericvaluesacardcouldbe.ThefirsttwoindexesofvaluesareNonesothat
thestringsinthelistmatchupwiththeindextheyrepresent—inotherwordssothatthestring
“2”invaluesispositionedatindex2.
Ourcardclassalsohastwoinstancevariables:suit,andnumber—eachrepresentedby
aninteger;andrepresentwhatkindofcardtheCardobjectis.Forexample,wecancreatea2
ofheartsbycreatingaCardobjectandpassingintheparameters2and1(1represents
heartsbecauseheartsisatindexoneinoursuitslist).LaterourCardobjectwillusethese
variablestoprintwhatkindofcarditis,byusingthemasindexesinthesuitsandvalueslists.
illustration
InordertoplayWar,weneedtocomparetwocardstoseewhichcardisbigger.We
addedtheabilitytocomparetwocardsinanexpressionwiththecomparisonoperators
greaterthan(>)andlessthan(<)toourCardclassbydefiningthetwomagicmethodsused
bythesecomparisonoperators:__lt__and__gt__.Bydefiningthesemethodswecan
determinewhathappenswhenourCardobjectsarecompared—justlikewedidwithour
AlwaysPositiveexampleinthepreviouschapterexceptwithcomparisonoperatorsinsteadof
theadditionoperator.Thecodeinthesemethodslookstoseeifthecardisgreaterthanorless
thantheothercardpassedinasaparameter.However,wemusttakeonemorethinginto
consideration—whathappensifthecardshavethesamevalue—forexampleifbothcardsare
10’s.Ifthissituationoccurs,wecomparethevalueofthesuitsandreturntheresult.This
worksbecausetheintegersweusetorepresentthesuits,areinorder—withthemostpowerful
suitgettingthehighestintegerandtheleastpowerfulsuitgettingthelowestinteger.Withthese
methodsdefined,wecancomparetocardsusingthegreaterthanandlessthanoperators:
card1=Card(10,2)
card2=Card(11,3)
print(card1<card2)
>>True
card1=Card(10,2)
card2=Card(11,3)
print(card1>card2)
>>False
Thelastthingwedoisoverridethemagicmethod__repr__,whichwelearnedinthe
previouschapterisusedbytheprintfunctiontoprintanobject.Overridingthe__repr__
methodletsusprintthecardaCardobjectrepresents:
card=Card(3,2):
printcard
>>3ofdiamonds
Deck
Nextweneedtodefineaclasstorepresentadeckofcards.Ourdeckwillcreatealistof
fiftytwocards—fourofeachsuit—whenitisinitializedanditwillhaveamethodtoremovea
cardfromthecardslist.Hereisourdeck:
fromrandomimportshuffle
classDeck:
def__init__(self):
self.cards=[]
foriinrange(2,15):
forjinrange(4):
self.cards.append(Card(i,j))
shuffle(self.cards)
defremove_card(self):
iflen(self.cards)==0:
return
returnself.cards.pop()
WhenaDeckobjectisinitialized,thetwoforloopsin__init__createallofthecardsina
fiftytwocarddeckandappendthemtoourcardslist.Thefirstloopisfrom2to15because
cardsstartwiththevalue2andendwiththevalue14(theace).Eachtimearoundtheinner
loop,anewcardwillbecreatedusingtheintegerfromtheouterloopasthevalue(i.e.,14for
anace)andtheintegerfromtheinnerloopasthesuit(i.e.a2forhearts).Thisresultsinthe
creationoffiftytwocards—oneforeverysuit.
Finally,weusetheshufflemethodfromtherandommoduletorandomlyrearrangethe
itemsinourcardslisttomimicshufflingadeckofcards.Ourdeckhasjustonemethod,
remove_card,whichreturnsNoneifourlistofcardsisempty,andotherwiseremovesand
returnsacardfromthecardslist.UsingourDeckclass,wecancreateanewdeckofcards
andprintoutallthecardsinthedeck:
deck=Deck()
forcardindeck.cards:
print(card)
>>4ofspades
>>8ofhearts
...
Player
Weneedaclassrepresentingeachplayerinthegamesowecankeeptrackoftheir
cardsandhowmanyroundsthey’vewon.Hereisourplayerclass:
classPlayer:
def__init__(self,name):
self.wins=0
self.card=None
self.name=name
OurPlayerclasshasthreeinstancevariables:winstokeeptrackofhowmanyrounds
they’vewon,cardtorepresentthecurrentcardtheplayerisholding,andnametokeeptrack
oftheplayer ’sname.
Game
Finally,wecancreatetheclassrepresentingthegameitself:
classGame:
def__init__(self):
name1=input("player1name")
name2=input("player2name")
self.deck=Deck()
self.player1=Player(name1)
self.player2=Player(name2)
defplay_game(self):
cards=self.deck.cards
print("beginningWar!")
response=None
whilelen(cards)>=2andresponse!='q':
response=input('qtoquit.Anyotherkeytoplay.')
player1_card=self.deck.remove_card()
player2_card=self.deck.remove_card()
print("{}drew{}{}drew{}".format(self.player1.name,
player1_card,self.player2.name,player2_card))
ifplayer1_card>player2_card:
self.player1.wins+=1
print("{}winsthisround".format(self.player1.name))
else:
self.player2.wins+=1
print("{}winsthisround".format(self.player2.name))
print("TheWarisover.{}wins".format(self.winner(self.player1,
self.player2)))
defwinner(self,player1,player2):
ifplayer1.wins>player2.wins:
returnplayer1.name
ifplayer1.wins<player2.wins:
returnplayer2.name
return"Itwasatie!"
Whenourgameobjectisinitialized,the__init__methodiscalledandweusetheinput
functiontocollectthenamesofthetwoplayersinthegamewhichwesaveinthevariables
name1andname2.WecreateanewDeckobjectandstoreitintheinstancevariabledeck,
andcreatetwoPlayerobjectsusingthenameswecollectedfromtheuserandstorethemin
theinstancevariablesplayer1andplayer2.
OurGameclasshasamethodcalledplay_gameusedtostartthegame.Theheartof
themethodistheloopthatkeepsthegamegoingaslongastherearetwoormorecardsleft
inthedeck,andaslongasthevariableresponseisnotequalto“q”.Beforeourloopwe
defineresponseasNone,andlaterwesetittotheinputoftheuser,whichistheonlysituation
itcanbecome“q”andendthegame.Otherwise,thegameendswhentherearelessthantwo
cardsleftinthedeck.
Eachtimethroughtheloop,twocardsaredrawn.Thefirstcardisassignedtoplayer1
andthesecondcardisassignedtoplayer2.Nextweprintthenameofeachplayerandwhat
cardtheydrew.Wecomparethetwocardstoseewhichcardisgreater,incrementthewins
instancevariablefortheplayerthatwon,andprintamessagethatsayswhichplayerwon.
OurGameclassalsohasamethodcalledwinnerwhichtakestwoplayerobjects,looks
atthenumberofroundstheywon,andreturnstheplayerthatwonthemostround.Whenour
deckrunsoutofcards,weprintamessagethatthewarisover,callthewinnermethod
(passinginbothplayer1andplayer2)andprintthenameoftheresult—theplayerthatwon.
War
Whenweputitalltogether,wehavethecardgameWar:
fromrandomimportshuffle
classCard:
suits=["spades","hearts","diamonds","clubs"]
values=[None,None,"2","3","4","5","6","7","8","9",
"10","Jack","Queen","King","Ace"]
def__init__(self,value,suit):
"""suitandvalueshouldbeintegers"""
self.value=value
self.suit=suit
def__lt__(self,other):
ifself.value<other.value:
returnTrue
ifself.value==other.value:
ifself.suit<other.suit:
returnTrue
else:
returnFalse
returnFalse
def__gt__(self,other):
ifself.value>other.value:
returnTrue
ifself.value==other.value:
ifself.suit>other.suit:
returnTrue
else:
returnFalse
returnFalse
def__repr__(self):
returnself.values[self.value]+"of"+self.suits[self.suit]
classDeck:
def__init__(self):
self.cards=[]
foriinrange(2,15):
forjinrange(4):
self.cards.append(Card(i,j))
shuffle(self.cards)
defremove_card(self):
iflen(self.cards)==0:
return
returnself.cards.pop()
classPlayer:
def__init__(self,name):
self.wins=0
self.card=None
self.name=name
classGame:
def__init__(self):
name1=input("player1name")
name2=input("player2name")
self.deck=Deck()
self.player1=Player(name1)
self.player2=Player(name2)
defplay_game(self):
cards=self.deck.cards
print("beginningWar!")
response=None
whilelen(cards)>=2andresponse!='q':
response=input('qtoquit.Anyotherkeytoplay.')
player1_card=self.deck.remove_card()
player2_card=self.deck.remove_card()
print("{}drew{}{}drew{}".format(self.player1.name,player1_card,self.player2.name,
player2_card))
ifplayer1_card>player2_card:
self.player1.wins+=1
print("{}winsthisround".format(self.player1.name))
else:
self.player2.wins+=1
print("{}winsthisround".format(self.player2.name))
print("TheWarisover.{}wins".format(self.winner(self.player1,self.player2)))
defwinner(self,player1,player2):
ifplayer1.wins>player2.wins:
returnplayer1.name
ifplayer1.wins<player2.wins:
returnplayer2.name
return"Itwasatie!"
game=Game()
game.play_game()
>>"player1name"
…
Chapter16.Practice
Exercises
0.Buildanobject-orientedtext-basedversionofBlackjack.
0.Buildawebscrapertocollectdatafromanotherwebsite.
0.FindaPythonprojectyouareinterestedinhostedonpip,(hintlookonGitHub),
downloadit,anduseitinaprogram.
Read
0.https://julien.danjou.info/blog/2013/guide-python-static-class-abstract-
methods
0.http://stackoverflow.com/questions/2573135/python-progression-path-
from-apprentice-to-guru
PartIIIIntroductiontoProgrammingTools
Chapter17.Bash
Inthissectionofthebookwelearntousetoolsthatwillmakeusmoreeffective
programmers.InthischapterwelearntousethecommandlineshellBashthatcomeswith
manyversionsofLinuxandOSX(whichareUnix-likeoperatingsystems).Acommandline
shellisaprogramthatletsyoutypeinstructionsintoitthatyourcomputerexecuteswhenyou
pressenter.
OnceyouareabletouseBash,youwillbeabletouseUnix-likeoperatingsystems
moreeffectively.ThisisimportantbecauseUnix-likeoperatingsystemsaresowidelyusedin
theprogrammingworld:Linuxrunsonthemajorityoftheworld’swebservers,andmany
programmingjobswillrequireyoutobefamiliarwithUnix-likeoperatingsystems.
TheexamplesinthischapterassumeyouareusingeitherUbuntuorOSX.Ifyouare
usingWindows,youcanfollowalongwiththeexamplesbygoingto
theselftaughtprogrammer.io/bashwhereyouwillfindalinktoawebsitethatemulatesBash—
youjusttypethecommandsintothegreenboxinthewebsite.
FindingBash
YoucanfindBashbysearchingforterminalfromtheicontitledSearchyourcomputer
andonlineresourcesifyouareusingUbuntuorfromSpotlightsearchifyouareusinga
Mac.
UsingBash
BashissimilartothePythonShell.TheBashcommandlineshelltakesitsown
programminglanguagecalledBashasinput.TheprogramminglanguageBashhas
commands—whicharelikefunctionsinPython.Westartwithakeyword;typeaspace;type
theparameterwewanttogivethecommand(ifany);hittheenterkey;andBashreturnsthe
result.OneofBash’scommandsisecho,whichissimilartotheprintfunctioninPython.
Hereisanexample:
$echoHello,World!
>>Hello,World!
WetypedthecommandechointoBash,followedbyaspaceandHello,World!asa
parameter.Whenwepressenter,Hello,World!printsintheBashcommandlineshell.
Youcanalsouseprogramsyou’veinstalled—likePython—fromtheBashcommandline
shell.Enterthecommandpython3tousePythonfromtheshell:
$python3
>>
NowyoucanexecutePythoncode:
print(“Hello,World!”)
>>Hello,World!
Enterexit()toexitPython.
RelativevsAbsolutePaths
Anoperatingsystemismadeupofdirectoriesandfiles.Adirectoryisanotherwordfor
afolderonyourcomputer.Alldirectoriesandfileshaveapath.Apathisasanaddresswhere
adirectoryorfileexistsinyouroperatingsystem.WhenyouareusingtheBashcommand
lineshell,youarealwaysinadirectory,locatedataspecificpath.Youcanseethepathyou
areatbyenteringthecommandpwdintheBashcommandlineshell:
pwd
>>/Users/bernie
pwdstandsforprintworkingdirectory(aworkingdirectoryisthedirectoryyouare
currentlyin),andthepaththatprintsonyourcomputerwillbethenameofwhateverdirectory
youarein.
Thefoldersinyouroperatingsystemareatree.Atreeisanimportantconceptin
ComputerSciencecalledadatastructure(coveredinPartIV).Inatree,thereisarootatthe
top.Therootcanhavebranches,andeachoneofthebranchescanhavemorebranches,and
thosebranchescanhavebranches.Thisgoesonindefinitely.
[addillustrationofatreedatastructure]
root
/\
homeetc
/\
berniebin
/\
testprojects
Everybranchonthetreeisadirectory,includingtheroot.Thetreeshowshowtheyare
connectedtoeachother.WhenyouareusingBash,atanygiventimeyouareatoneofthe
locationsonthetree,andapathisawayofexpressingthatlocation.Therearetwowaysof
expressingalocationonUnix-likeoperatingsystems:absolutepathsandrelativepaths.An
absolutepathisawayofexpressingalocationstartingfromtherootdirectory.Anabsolute
pathismadeupofthenameoffoldersinthetree,inorder,separatedbybackslashes.The
absolutepathtotheberniedirectory(fortheoperatingsystemillustratedabove)is
/home/bernie.Thefirstslashrepresentstherootdirectory;followedbythehomedirectory;
followedbyanotherslashandtheberniedirectory.
Theotherwayofspecifyingalocationonyourcomputeriscalledusingarelativepath.
Insteadofstartingattherootdirectory,arelativepathisrelativetothedirectoryyouarein.
Youroperatingsystemknowsapathisrelativewhenitdoesn’tbeginwithabackslash.Ifwe
werelocatedinthehomedirectoryintheexampleabove,therelativepathtotheprojects
directorywouldbebernie/projects.Ifwewereinthehomedirectory,therelativepathto
bernieissimplybernie.
Navigating
Youcanusethecdcommand,whichstandsforchangedirectory,tonavigatetoa
directoryusinganabsoluteorrelativepath.Enterthecdcommandfollowedbytheabsolute
path/tonavigatetotherootdirectory:
$cd/
Youarenowinyourrootdirectory,whichyoucanverifywiththecommandpwd:
$pwd
>>/
Thecommandls,whichstandsforlistdirectories,printsthefoldersinthedirectoryyouare
in:
$ls
>>bindevinitrd.imglost+found
...
Youcancreateanewdirectorywiththecommandmkdir.Directorynamescannothave
spacesinthem.Usethemkdircommandtomakeanewdirectorycalledtstp:
$mkdirtstp
>>
Verifythenewdirectoryexistswiththecommandls.Nowusethecdcommandtoenterthe
tstpdirectorybypassingintherelativepathtotstpasaparameter:
$cdtstp
Nowusethecdcommandfollowedbytwoperiodstoreturntothefolderyouwereinbefore
youenteredthetstpdirectory:
$cd..
Deletethenewdirectoryusingthecommandrmdir,whichstandsforremovedirectory:
$rmdirtstp
Usethecommandlstoverifythedirectorywasdeleted.
Flags
Commandshaveaconceptcalledflagsthatallowtheissuerofthecommandtochange
thecommandsbehavior.Ifyouuseacommandwithoutanyflags,allofthecommandsflags
aresettofalse.Butifyouaddaflagtoacommand,theflagissettotrueandthebehaviorof
thecommandchanges.The-and--symbolsareusedtoattachflagstoacommand.--author
isanexampleofaflagyoucanaddtothelscommandtoprinttheauthorofallofthe
directoriesandfilesthatgetlisted.ThisexampleisforLinux,onOSXyoucanusethesame
flagbutyouneedtouseonedashinsteadoftwo.Hereisanexampleofusingthe--authorflag
withthelscommand:
$ls--author
>>drwx------+13coryalthoff442BSep1617:25Pictures
...
Whenyouaddthe--authorflagtoyourlscommand,thenameofeachdirectoryandfolder
inyourcurrentdirectorywillprint—aswellassomeadditionalinformation,includingthe
nameofthepersonthatcreatedthem.
vim
Vimisacommandlinetexteditor.It'slikeMicrosoftWord,exceptyouuseitfromthe
commandline.IfyouareusingUbuntu,firstinstallvimwiththecommandapt-getinstall
vim.MakesuretoenterYwhenprompted.IfyouareusingaMac,itshouldcomewithvim
Ifyouareusingtheonlinebashshell,italreadyhasviminstalled.
Youcancreateanewfilewithvimbygoingtothecommandlineandusingthecommand
vim[nameofthefiletocreate].Usethecommandvimself_taught.txttocreateanewtextfile
calledself_taught.txt.Presstheiorinsertkeyandtypeaparagraphoftextintothefile.Any
paragraphoftextyoufindontheinternetwilldo.Thereasonyouhavetohittheiorinsert
keywhenyoufirstenterafileisbecausevimhasdifferentmodesoptimizedfordifferent
activities.vimstartsin,NormalMode,whichisnotmeantforaddingtexttothefile(itis
meantforeasynavigation)—youcandeletetextinnormalmode,butyoucannotinsertnew
text.Onceyoupresstheiorinsertkeyandenternormalmode,youcanusevimlikeaword
processor—trytypingintovim.
Sinceyoucan’tuseyourmousetomovethecursoraround,it’simportanttolearnafew
shortcutstojumptodifferentlocationsinyourdocument,sothatyoudon’tendupusingthe
arrowkeysonyourkeyboard(becausethat’sslow).Topracticemovingaround,firstmake
sureyouareinNormalMode(control-c).Youcanmovetothebeginningofawordby
pressingbandtheendofawordwithe.0willmoveyoutothebeginningofthelineyouare
on,whilethedollarsign$willmoveyoutotheendoftheline.Hwillmoveyoutothefirst
lineofthepageandLwillmoveyoutothelast.Youcandeleteentirelinesoftextinnormal
modebypressingthedkeytwice.Spendsometimeusingthesekeystogetfamiliarwithwith
navigatingthroughafileusingvim.
ToexitvimyouneedtofirstswitchtoNormalModebypressingcontrol-c.Nextpress
theshiftkeyandthenhitthecolonkey(whilestillholdingtheshiftkey).Fromhereyoucan
typeq!ifyouwanttoquitwithoutsavingyourchangestothefile,ortypexifyouwantsave
yourchangesandquit.Onceyou’vetypedoneoftheseoptions,presstheenterkeytoexit.
vimisusefulinafewsituations:serversareusuallyonlyaccessedwithacommandlineshell,
soifyouwanttomakechangestoafileonaserver,youneedtouseacommandlinetext
editor,andonceyougetgoodatusingvim,itisoftenfastertouseittomakechangesthan
usingaconventionalwordprocessor.TrytypingvimtutorintheBashcommandlineshell
andseewhathappens.
Touch
Youcanusethecommandtouchfollowedbyafilepathtoquicklycreateanewfile:
$touchpurple_carrots.txt
>>
ViewAFileWithless
Thelesscommandenablesyoutoviewfilesfromthecommandline.Pickafileon
yourcomputerandusethecommandless[filepath]toviewthefilefromthecommandlinein
thelessprogram.Makesureyouareinthesamefolderyoucreatedthefileselftaught.txtin
withvimandpassthefilenameasaparameter:
$lessself_taught.txt
>>whatevertextyouputinyourfile
Pressqtoexit.
Users
Operatingsystemshavedifferentusers.Auserrepresentsapersonintheworld—
someoneusingtheoperatingsystem—andeachuserhasdifferentactivitiestheyareallowed
todo—calledtheirpermissions—aswellastheirowndirectoriesandfilesthatotherusers
can’taccess.Userscanalsobeplacedintogroups,withanentiregroupgiven.Youcansee
thenameoftheuseryouareonyouroperatingsystemwiththecommandwhoami:
$whoami
>>cory
Normallyyoustartastheuseryoucreatedwhenyouinstalledyouroperatingsystem.This
userisnotthemostpowerfulusertheoperatingsystemhas.Therootuseristhehighestlevel
user,whichmeansithasthehighestlevelofpermissions.Everysystemhasarootuser.The
rootusercandoananything:forexampletherootusercancreateordeleteotherusers.
Forsecurityreasons,younormallydonotloginastherootuser.Insteadyouusea
commandcalledsudoinfrontofanothercommandtotemporarilyusethepoweroftheroot
usertoissuethecommand.sudoallowsyoutodomostthingsastherootuser,butnot
everything.Makesuretobecarefulusingsudo,becauseusingcommandswithsudocanharm
youroperatingsystemifyoudon’tknowwhatyouaredoing.Wearenotgoingtocover
usingsudointhisbook,butI’vesetupatutorialonusingsudoat
theselftaughtprogrammer.io/sudoyoucanreadthroughonceyoufeelcomfortableusingthe
Bashcommandlineshell.
Permissions
Everydirectoryandfileonyourcomputerhasasetofpermissions.Thesepermissions
definewhatactionsausercanperformonthem.Therearethreetypesofpermissions:r,rw
andx,whichstandforread,writeandexecute.Readingafilemeansviewingit,writingafile
meanschangingit,andexecutingafilemeansrunningitasaprogram.Youcanviewafileor
directory’spermissionswiththecommandls-lah[name_of_file].Usethecommandtouchto
createanewfilecalledtstp:
$touchtstp
>>
Nowwecanviewthefilespermissions:
$ls-lahtstp
-rw-r--r--1coryalthoffstaff5BFeb2111:55test.py
Ourfilehasthreepermissionsthatapplytothreedifferentgroupsofusers—representedby-
rw-r--r--.Thefirstsetappliestotheownerofthefile,thesecondsetappliestothegroup
assignedthefileandthethirdsetappliestoeveryone.Sointhisexample,theownerofthefile
haspermissiontoreadandwritethefile,whereaseveryoneelsecanonlyreadthefile.
Youcanaddandsubtractpermissionswiththecommandchmod(shortforchange
mode).Youcanaddpermissiontoreadwiththecommandchmod+r,permissiontoread
andwritewiththecommandchmod+rw,andpermissiontoexecutewithchmod+x.Youcan
subtractthesamepermissionswithchmod-r,chmod-wandchmod-x,respectively.
BashPrograms
Ifwewanttorunabashscriptfromthecommandline,weneedtogivethefileowner
permissiontoexecute.Createanewfilecalledhello_world.shandtypeechoHello,World!in
it.Thereasonweusethe.shextensionistoletanyonewhoseesthisfileknowitisaBash
script.Thesyntax./[file_name]isusedtoexecuteafileinthedirectoryyouarein.You
executeafilewith/[file_name]andtheperiodmeanslookforthefileinthecurrentdirectory.
Youcouldalsoreplacetheperiodwiththepathtothefile.Tryexecutingthefile(makesure
youareinthesamedirectoryasthefile):
$./hello_world.sh
>>Permissiondenied
Thereasontheconsoleprintedpermissiondeniedisbecausewedonothavepermissionto
executethefile.Wecanchangethatwiththechmodcommand.
$chmodu+xhello_world.sh
$./hello_world.sh.
>>Hello,World!
Intheexampleabove,weaddeduto+xtoformu+xbecauseustandsforuser,andweonly
wantedtogiveourselvestheabilitytoexecutethefile.Ifyouarerepeatinganargumentfrom
thepreviouscommandline,inthiscasethefilename,youcanuse!$torepresenttherepeated
argument.
Processes
HiddenFiles
Youroperatingsystem,anddifferentprogramsonyourcomputer,usehiddenfilesto
storedata.HiddenfilesinUnix-likesystemsarefilesthatbydefaultarenotshowntousers
becausechangingthemcanaffecttheprogram(s)thatdependsonthem.Hiddenfilesstartwith
aperiod—forexample—.hidden.Youcanseehiddenfilesusingthecommandlswiththe
flag-a,whichstandsforall.Createahiddenfilenamed.self_taughtwithtouch.hiddenand
testifyoucanseeitwiththecommandslsandls-a.
EnvironmentalVariables
Youroperatingsystemcanstoreandusevariablescalledenvironmentalvariables.You
cancreateanewenvironmentalvariablefromthecommandlinewiththesyntaxexport
variable_name=[variable_value].Inordertouseanenvironmentalvariable,youmustputa
dollarsigninfrontoftheenvironmentalvariable’sname.Forexample:
$exportx=100
$echo$x
>>100
Creatinganenvironmentalvariablefromthecommandlineisnotpermanent.Ifyouquitthe
commandline,reopenit,andtypeecho$x,theterminalwillnolongerprint100.
Wecanmaketheenvironmentalvariablexpersistentbyaddingittoahiddenfileusedby
Unix-likeoperatingsystemscalled.profilelocatedinyourhomedirectory.Gotoyourhome
directorywiththecommandcd~(~isashortcutforrepresentingyourhomedirectoryon
Unix-likeoperatingsystems)andopenyour.profilefileusingvimwiththecommandvim
.profile.MakesuretoenterNormalMode,typeexportx=100intothefirstlineofthefile,
andexitwith:x.Closeandreopenyourcommandline,andyoushouldstillbeabletoprint
theenvironmentalvariableyoudefined:
$echo$x
>>100
Theenvironmentalvariablexgets100aslongasit’sdefinedinyour.profilefile.
$PATH
WhenyoutypeacommandintotheBashcommandshell,itlooksforthecommandinall
ofthedirectoriesstoredinanenvironmentalvariablenamed$PATH.$PATHisastringof
directorypathsseparatedbycolons.TheBashcommandshelllooksineachofthese
directoriesforafilematchingthenameofthecommandyoutyped.Ifthefileisfound,the
commandisexecuted,ifnotthecommandshellprints“commandnotfound”.Useechoto
printyour$PATHenvironmentalvariable(your$PATHmaylookdifferentthanmine):
echo$PATH
>>/usr/local/sbin:/user/local/bin:/usr/bin:
...
Inthisexampletherearethreedirectorypathsin$PATH:/usr/local/sbin,/user/local/binand
/usr/bin.TheBashcommandlineshellwillbeabletoexecuteanycommandfoundinthese
threedirectories.Youcanseewhereacommandlikegrepisinstalledbypassingitasa
parametertothecommandwhich:
$whichgrep
>>/usr/bin/grep
Thegrepcommandislocatedin/usr/bin,oneofthelocationsinmyoperatingsystem’s
$PATHenvironmentalvariable.
pipes
InUnix-likesystems,thecharacter|isknownasapipe.Youcanuseapipetousethe
outputofonecommandastheinputforanothercommand.Forexample,wecanpassthe
outputofthelscommandastheinputofthelesscommandtoopenlesswithallofthefilesin
thecurrentdirectory:
$ls|less
>>Applications
...
Theresultisatextfilewiththeoutputoflsopenedupintheprogramless(pressqtoquit
less).Youarenotlimitedtopipingtwocommands—youcanchainmultiplecommands
togetherusingpipes.
cat
Youcanusetheversatilecatcommandtodisplaythecontentsofafileandtocatenate,
whichmeans“toconnectinaseries.”11.Createanewfilecalledhello.txtandaddHello,
World!asthefirstline.Nowusethecatcommandtoprintthecontentsofthefile:
$cathello.txt
>>echoHello,World!
Tousethecatcommandtocatenatetwofiles,firstcreateafilecalledc1.txtandaddthetext
Boy.Thencreateanotherfilecalledc2.txtandaddthetextMeetsWorld.Nowwecan
catenatethembypassingbothfilesasparameterstothecatcommand,followedbythe
greaterthansymbol(>),andthenameofthenewfiletocreate:
$catc1.txtc2.txt>combined.txt
>>
Openupthenewlycreatedcombined.txtwhichshouldsayBoyMeetsWorld.
RecentCommands
Youcanscrollthroughyourrecentcommandsbypressingtheupanddownarrowsin
thecommandlineshell.Toseealistofallofyourrecentcommandsusethecommandhistory
:
$history
>>1.echoHello,World!
>>2.pwd
>>3.ls
…
JumpAround
Whenyouaretypingacommandintheterminal,therewillbetimeswhereyouwantto
editthecommandyou’vealreadytyped.Yourfirstinstinctwillbetousethearrowrightor
leftkeytomovethecursortothepartyouwanttochange.Butthisisslow.Insteadyoushould
useshortcutsthatwillgetyoutherefaster.
Typeechohello,world!(withoutpressingEnter)intheterminalanduseescbtomove
thecursorbackoneword,andescftomovethecursorforwardoneword.Youcanalsomove
thecursortothebeginningofthelinewithcontrolaortheendofthelinewithcontrole.
TabComplete
Tabcompleteisafeaturethatwillhelpimprovethespeedyougetthingsdonefromthe
commandlineshell.Ifyouareinthemiddleoftypingacommandyouandpressthetab
buttononyourkeyboard,thecommandlineshellwilltrytoautocompletethecommandfor
you.
Tryitforyourselfbytypingechinthecommandlinefollowedbytab;echwill
automaticallygetturnedintoecho.Youcanalsousetabtocompletefileordirectorypaths.
Starttypingthepathofthedirectoryyouareinandfinishitoffbypressingtab.Ifyoupress
tabandnothinghappens,itisbecausetwocommandsorpathsarenamedsimilarly,andthe
shelldoesn’tknowwhichtochoose.Forexampleifyouhaveadirectorynamedcar,and
anotherdirectorynamedcandy,andyoutypecaandtrytotabcomplete,nothingwillhappen
becausetheshellwon’tknowwhethertochoosecarorcandy.Ifyouaddannsoyou’vetype
can,andpresstabcomplete,theshellwillautocompletecandybecausetheshellknowscaris
notcorrect.
Wildcard
Awildcardisacharacterusedtomatchapattern.Twoexamplesofwildcardsarean
asteriskandaquestionmark.Theasteriskwildcardmatcheseverythingeitherstartingor
endingwithapattern,dependingifyouputitbeforeorafterthepattern.Asteriskwildcards
arecommonlyusedwiththecommandls.Thecommandls*.txtwillshowanyfilesinthe
currentdirectorythatendwith.txt,whereasls.txt*willshowanyfilesthatwith.txt.
Aquestionmarkwillmatchanysinglecharacter.Solst?owillshowanyfilesor
directoriesthatmatchtfollowedbyanycharacterfollowedbyo.Ifyouhaddirectories
namedtwoandtoo,theybothwouldgetprinted.
OtherTools
Ifyourterminalgetscluttered,youcanclearitwiththecommandclear.Ifaprocessis
takingtoolong,youcankillitwithcontrol+c.Anotherpowerfulcommandisgrep,usedto
searchfilesforpatternsandcoveredinthenextchapter.
TheOneWeekChallenge
Ichallengeyoutoonlyusethecommandlineforoneweek.Thatmeansnographical
userinterfaceforanythingotherthanusingtheinternet!Youshouldcompletethischallenge
duringaweekyouareactivelyworkingonaprogrammingproject.
Chapter18.RegularExpressions
quote
Aregularexpressionisasequenceofcharactersusedtolookforapatterninastring.Inthis
chapterwearegoingtopracticeusingregularexpressionsfromthecommandlineusing
grep,acommandforusingregularexpressionstosearchfiles.Wewillalsolearntouse
regularexpressionsinPython.
Setup
Togetstartedpracticingusingregularexpressions,createafilecalledzen.txt.Fromthe
commandline(makesureyouareinsidethedirectorywhereyoucreatedzen.txt)enterthe
commandpython3-c“importthis”.ThiswillprintoutTheZenofPython,apoemwrittenby
TimPeters:
TheZenofPython
Beautifulisbetterthanugly.
Explicitisbetterthanimplicit.
Simpleisbetterthancomplex.
Complexisbetterthancomplicated.
Flatisbetterthannested.
Sparseisbetterthandense.
Readabilitycounts.
Specialcasesaren'tspecialenoughtobreaktherules.
Althoughpracticalitybeatspurity.
Errorsshouldneverpasssilently.
Unlessexplicitlysilenced.
Inthefaceofambiguity,refusethetemptationtoguess.
Thereshouldbeone--andpreferablyonlyone--obviouswaytodoit.
Althoughthatwaymaynotbeobviousatfirstunlessyou'reDutch.
Nowisbetterthannever.
Althoughneverisoftenbetterthan*right*now.
Iftheimplementationishardtoexplain,it'sabadidea.
Iftheimplementationiseasytoexplain,itmaybeagoodidea.
Namespacesareonehonkinggreatidea--let'sdomoreofthose!
The-cflaginPythonexecutesPythonpassedinasastring,andimportthisprintstheZenof
Pythonwhenexecuted.CopyandpastetheZenofPythonintothefilezen.txt.Ialso
recommendreadingitasitcontainssomegreatwisdom.Usethefunctionexit()toexit
Python.Ifyouareusingtheonlinebashshell.
IfyouareusingaMac,setthefollowingenvironmentalvariablesintheterminal:
$exportGREP_OPTIONS='--color=always'
$exportGREP_COLOR='1;35;40'
Thiswillmakegrephighlightthewordsmatchedintheterminal,whichhappensbydefault
onUbuntubutnotonOSX.Remember,settinganenvironmentalvariablefromtheterminalis
notpermanent,soifyouexittheterminalandcomebackyouwillhavetosetthe
environmentalvariablesagain.Addtheenvironmentalvariablestoyour.profilefileifyou
wanttomakethechangepermanent.
SimpleMatch
Thecommandgrepacceptsaregularexpressionandthepathtoafiletolookforthe
regularexpressioninasparameters.Fromthecommandline,inthedirectorywhereyou
createdthefilezen.txt,enterthefollowingcommand:
$grepBeautifulzen.txt
>>Beautifulisbetterthanugly.
Beautifulistheregularexpressionandzen.txtisthepathtothefiletolookfortheregular
expressionin.BeautifulisasequenceofcharactersthatmatchthewordBeautiful.Thisisthe
simplestkindofregularexpression.YourconsoleprintedthelineBeautifulisbetterthan
ugly.withBeautifulhighlightedbecauseitisthewordtheregularexpressionmatched.
IgnoreCase
IfwechangeourregularexpressionfromBeautifultobeautiful,itwillnolonger
matchanythingintheZenofPython.Entergrepbeautifulzen.pytoseeforyourself.Ifwe
wantourregularexpressiontomatchthewordbeautifulregardlessofcase(whetherornot
charactersarecapitalized),wecanusetheflag-i:
$grep-ibeautifulzen.txt
>>Beautifulisbetterthanugly.
Becauseweaddedtheflag-itoourcommand,grepignorescaseandhighlightsBeautiful
again.
OnlyReturnMatched
grepreturnstheentirelineofthefileamatchwasfoundon.Wecanreturntheexact
wordmatchedbyusingtheflag-o:
$grep-oBeautifulzen.txt
>>Beautiful
Normally,grepwouldhavereturnedBeautifulisbetterthanugly.butbecauseweaddedthe
flag-o,onlytheexactmatch,Beautiful,getsreturned.
MatchBeginningandEnd
Regularexpressionshavecertaincharactersthatdon’tmatchthemselves,butinsteaddo
somethingspecial.Forexamplethe^characterisusedtolookformatchesonlyiftheyoccur
atthebeginningofaline:
$grep^Ifzen.txt
>>Iftheimplementationishardtoexplain,it'sabadidea.
>>Iftheimplementationiseasytoexplain,itmaybeagood
idea.
Similarly,wecanusethedollarsigntoonlymatchthelinesthatendwithapattern:
$grepidea.$zen.txt
>>Iftheimplementationishardtoexplain,it'sabadidea.
>>Iftheimplementationiseasytoexplain,itmaybeagoodidea.
Thelinesmatchedbothendwithidea.ThelineNamespacesareonehonkinggreatidea--let's
domoreofthose!wasignoredbecausealthoughitincludesthewordidea,itdoesnotend
withthewordidea.Youcanalsocombinethetwoanchormatcheswe’vecoveredintothe
regularexpression^$tosearchforemptylinesinafile.
MatchMultipleCharacters
Youcanusebracketsinaregularexpressiontomatchanyofthecharactersinsidethe
brackets.Inthisexample,insteadofmatchingtextfromourzen.txtfile,wearegoingtousea
pipetopassinastringtogrep:
$echoTwoisanumberandtooisnot.|grep-it[ow]o
>>Twoisanumberandtooisnot.
Remember,thepipesymbolpassestheoutputofonecommandastheinputofthenext.Inthis
casetheoutputofechoispassedastheinputofgrep.ThecommandechoTwoisanumber
andtooisnot.|grep-it[ow]owillmatchbothtwoandtoobecausetheregexislookingfor
atfollowedbyeitheranoorawfollowedbyano.
Repetition
Wecanaddrepetitioninourpatternswiththeasterisksymbol.Theasterisksymbol
means“Theprecedingitemwillbematchedzeroormoretimes.”56Wemightwanttosay,
matchtwfollowedbyanyamountofo’s.Theregularexpressiongreptwo*accomplishes
this:
$echotwotwoootwoonottoo.|grep-otwo*
>>two
>>twooo
>>twoo
Addinga*aftertwomeanstheregularexpressionshouldmatchanythingwithtwofollowed
byanynumberofo’s.
Range
Youcanmatcharangeoflettersornumbersbyputtingtherangeinsidebrackets.For
example,wemightwanttoonlymatchthenumbersin122233hello334443goodbye939333.
Theregex[[:digit:]]*willmatchallthenumbersinthestringbecauseitincludesarangeof
numbers(0-9),followedby*whichtellstheregextomatchasmanynumbersinarowasit
can.
$echo122233hello334443goodbye939333|grep-o[0-9]*
>>122233
>>33443
>>939333
Similarly,youcanmatcharangeofcharacters(allcharactersinthiscase)withtheregex
[[:alpha:]]:
$echo122233hello334443goodbye939333|grep[[:alpha:]]*
Escaping
Whatifwewanttomatchoneofthespecialcharacterswe’vebeendiscussing,likethe
dollarsign?Wecandothisbyescapingthecharacter.Wecoveredescapinginthechapter
ManipulatingStrings:escapingmeansprefixingacharacterwithaspecialcharactertoletthe
programevaluatingthesyntaxknowyouwanttousetheactualcharacterandnotthespecial
meaningthecharacternormallyhas.Escapinginregularexpressionsisdonewithabackward
slash:
$echoIlove$|grep\\$
>>Ilove$
Normally,thedollarsignhasthespecialmeaningofonlymatchingsomethingattheendofa
line,howeverbecauseweescapedit,ourregexlooksforthedollarsign.
RegularExpressionsinPython
PythonhasalibrarycalledrethatletsyouuseregularexpressionsinPythonprograms.
ThesymbolsinPythonregularexpressionsarenotalwaysthesameasgrep,buttheconcept
ofregularexpressionsisthesame.
#
https://github.com/calthoff/tstp/blob/master/part_III/regular_expressions/reg_expr_ex1.py
importre
line="Matchthis."
matchObj=re.search('this',line)
ifmatchObj:
print(matchObj.group())
else:
print("Nomatch!")
>>this
recomeswithdifferentmethodslikesearch,whichreturnsthefirstoccurrenceofthe
patternyouarelookingfor.Ifamatchisfound,anSRE_Matchobjectisreturned,andwecan
getthematchbycallingthefunctiongroup().Ifnomatchisfound,re.search()returnsNone.
Wecanusethefunctionfindall()intheremoduletofindeveryoccurrenceofapattern,
insteadofjustthefirstmatch.
#
https://github.com/calthoff/tstp/blob/master/part_III/regular_expressions/reg_expr_ex2.py
importre
line="""Thenumbers172canbefoundonthebackoftheU.S.$5
dollarbillinthebushesatthebase
oftheLincolnMemorial."""
matchObj=re.findall('\d+',line)
ifmatchObj:
printmatchObj
else:
print("Nomatch!")
>>[‘172’,‘5’]
re.findall()returnsalistofallthestringsmatchingthepattern,andanemptylistifthereisno
match.The+symbolmatchestheprecedingcharacteroneormoretimes.
Ifwewanttomatcheverythinginbetweentwounderscores,wecandoitwith:
#
https://github.com/calthoff/tstp/blob/master/part_III/regular_expressions/reg_expr_ex3.py
importre
line="""__yellow____red__and__blue__arecolors"""
matchObj=re.findall('__.*?__',line)
ifmatchObj:
printmatchObj
else:
print("Nomatch!")
Thetwounderscoresmatchtwounderscores,theperiodmatchesanycharacter,andthe
questionmarkfollowedbyanasteriskmeanskeepmatchinganycharacteruntilthereis
anotherdoubleunderscore.
HereisafunexampleofusingregularexpressionsinPythontocreatethegameMad
Libs:
“““https://github.com/calthoff/tstp/blob/master/part_III/lets_read_some_code/lets_read_some_code.py”””
importre
text="""
Giraffeshavearousedthecuriosityof__PLURAL_NOUN__since
earliesttimes.Thegiraffeisthe
tallestofallliving__PLURAL_NOUN__,butscientistsareunable
toexplainhowitgotitslong
__PART_OF_THE_BODY__.Thegiraffe'stremendousheight,which
mightreach__NUMBER____PLURAL_NOUN__,comesfromitslegsand_
_BODYPART__.
"""
defmad_libs(mls):
"""
:parammls:Stringwithpartstheusershouldfilloutsurrounded
bydoubleunderscores.Underscorescannot
beinsidehinte.g.,no__hint_hint__only__hint__.
"""
hints=re.findall("__.*?__",mls)
ifhints:
forwordinhints:
new_word=input("entera{}".format(word))
mls=mls.replace(word,new_word,1)
print('\n')
print(mls)
else:
print("invalidmls")
mad_libs(text)
ZenChallenge
WritearegularexpressiontofindeverywordintheZenofPythonendingwiththelettery.
Chapter19.PackageManagers
“Everyprogrammerisanauthor.”
―SercanLeylek
Packagemanagersareprogramsthatinstallandmanageotherprogramsonyour
operatingsystem.Anexampleofmanagingaprogramiskeepingtheprogramuptodate
whenanewversioncomesout.Inthischapterwearegoingtolearntouseseveralpackage
managers.Packagemanagersareusefulbecauseprogrammersuseprogramstocreatenew
programs.Forexample,mostprogramsuseoneormoredatabases(welearntousea
databaseinthelastchapterofthissection),andwhichareprogramsthemselves.
Programmersusepackagemanagerstodownloadandkeeptheirdatabasesuptodate,aswell
astoinstallandmanagethewidevarietyofotherprogramstheyuseintheircraft.
Inthischapterwewilllearntousethreepackagemanagers:apt-get,Homebrew,andpip.
apt-getisapackagemanagerforUbuntu,HomebrewisapackagemanagerforOSX,OneGet
isapackagemanagerforWindowsandpipisapackagemanagerthatcomeswithPythonand
isusedtodownloadandmanagePythonprograms.
Packages
Apackageissoftware“packaged”fordistribution—itincludesthefilesthatmakeupthe
actualprogram,aswellasfileswithmetadata(dataaboutdata);suchasthesoftware’sname,
versionnumber,anddependencies(programsthatneedtobedownloadedinorderforitto
runproperly).Packagemanagersdownloadpackages,installthem—whichmeans
downloadinganydependenciesthepackagehas.
Apt-get
Apt-getisapackagemanagerthatcomeswithUbuntu.Youcannotuseapt-getfromthe
onlineBashcommandlineshellemulator.YouneedtohaveUbuntuinstalledsoyoucanuse
thecommandsudo.IfyoudonothaveUbuntuinstalledonyourcomputer,youcanskipthis
section.
Youinstallapackagewithsudoapt-getinstall[package_name].Hereisanexampleof
installingapackagenamedaptitude:
$sudoapt-getinstallaptitude
>>Doyouwanttocontinue?[Y/N]Y
…
>>Processingtriggersforlibc-bin(2.19-0ubuntu6.7)…
MakesuretotypeYwhenpromptedwith“DoyouwanttoContinue[Y/N]”.Theaptitude
packageshouldnowbeinstalled.YoucanuseitbytypingaptitudeintotheBashcommand
lineshell:
$aptitude
>>
ThiswillopenupAptitude,ahelpfulprogramthatshowsinformationaboutpackages.You
shouldseealistofallthepackagesinstalledonyouroperatingsystemundertheoption
InstalledPackages,aswellasalistofpackagesyou’veyettoinstall,underNotInstalled
Packages.
Youcanalsolistthepackagesthathavebeeninstalledwithapt-getwiththecommandapt
list--installed:
$aptlist--installed
>>python3-requests...
Youcanremovepackagesusingaptitudewiththesyntaxapt-getuninstall
[package_name].Ifyouwanttoremoveaptitudefromyourcomputer,youcanuninstallit
withsudoapt-getuninstallaptitude.That’sallthereistoit—installingandremoving
programsisassimpleasusingtwocommandswithapt-get.
Homebrew
HomebrewisapopularpackagemanagerforOSX.IfyouhaveaMac,youcaninstall
Homebrewfromthecommandline.Gotohttp://brew.sh/andcopyandpastetheprovided
scriptintheterminal.Onceinstalled,youshouldseealistofcommandswhenyoutypebrew
intoyourterminal:
$brew
>>Exampleusage:
>>brew[info|home|options][FORMULA...]
...
WecaninstallpackageswithHomebrewusingthesyntaxinstall[package_name].Usethe
followingcommandtoinstallapackagecalledcalc:
$brewinstallcalc
...
>>==>Pouringcalc-2.12.5.0.el_capitan.bottle.tar.gz
>>/usr/local/Cellar/calc/2.12.5.0:518files,4.4M
Acalculatorprogramcalledcalcisnowinstalled.Youcanuseitbytypingcalcfromthe
commandline(typequittoexit);
$calc
>>2+2
>>4
Thecommandbrewlistprintsthesoftwareyou’veinstalledusingHomebrew:
$brewlist
>>calc
YoucanremovepackagesusingHomebrewwiththesyntaxbrewuninstall[package_name].If
youwanttoremovecalcfromyourcomputer,youcanuninstallitwiththecommandbrew
uninstallcalc.Thatisallyouneedtoknow—youarereadytouseHomebrewtomanageyour
packages.
OneGet
OneGetisthefirstpackagemanagertocomewithWindows:itwasfirstreleasedwith
WindowsTen.IfyoudonothaveWindowsTen,youcandownloadOneGetbyfollowingthe
instructionsonitsGitHubpage:https://github.com/OneGet/oneget.
WecaninstallpackagesonOneGetusingthecommand
pip
PipisusedtodownloadPythonpackages.PipwillinstallaPythonpackageforyou,and
onceinstalled,youcanimporttheprogramyoudownloadedasamoduleinyourPython
programs.First,checkPipisinstalledbygoingtoeithertheBashcommandlineshell,orthe
CommandPromptifyouareusingWindows(whichyoucanfindbysearchingCommand
PromptfromtheRunWindow)andtypingpip.
$pip
>>Usage:
pip<command>[options]
Commands:
installInstallpackages.
downloadDownloadpackages.
....
Alistofcommandsyoucanusewithpipshouldprint.PipcomeswithPythonwhenyou
downloadit,butinearlierversionsitdidn’tsoifnothinghappenswhenyouenterpip,google
“installingpipwitheasy_install”.
Wecanusepipinstall[package_name]toinstallanewpackage.Youcanfindallofthe
Pythonpackagesavailablefordownloadathttps://pypi.python.org/pypi.Therearetwoways
tospecifyapackage:youcanjustusethepackagename,oryoucangivethepackagename
followedby==andtheversionnumber.Ifyouusethepackagename,themostrecentversion
willgetdownloaded.Sometimesyoudon’twantthemostrecentversion,youwantaspecific
version,whichiswhythesecondoptionexists.Hereisanexampleofhowwecaninstalla
packagecalledFlask,apopularPythonpackagethatletsyoueasilycreatewebsites,usinga
versionnumber:
pipinstallFlask==0.10.1
>>
OnUnix-likesystemsyouneedtousesudo:
sudopipinstallFlask==0.10.1
>>
Whentheinstallationisfinished,theflaskmodulewillbeinstalledinaspecialfolderonyour
computercalledsite-packages.Site-packagesisautomaticallyincludedinyourPythonpath,
sowhenPythonimportsamodule,itlooksinsite-packagestoseeifit'sthere.
Now,ifyouwriteaprogram,youwillbeabletoimportandusetheFlaskmodule.
CreateanewPythonfileandaddthefollowingcode:
fromflaskimportFlask
app=Flask(__name__)
@app.route('/')
defindex():
return"Hello,World!"
app.run(port='8000')
Ifyougotohttp://127.0.0.1:8000/inyourwebbrowser,youwillseeawebsitethatsays
“Hello,World!”.ThisexamplewastakenfromFlask’stutorial,whichisavailableat
theselftaughtprogrammer.io/flaskifyouareinterestedinlearningabouthowFlaskworks.
Youcanviewthepackagesyou’veinstalledwithpipwiththecommandpipfreeze:
pipfreeze
>>Flask==0.10.1
…
Youcanusethesyntaxpipfreeze>[filename]tosavenamesofallthepackagesyou’ve
installedwithpiptoafile.Createarequirementsfilewith:
pipfreeze>requirements.txt
>>
Openrequirements.txtwithatexteditortoseethenewfile.Thisfileisusefulbecauseyoucan
usethesyntaxpipinstall[requirements_file]toinstallallthepackageslistedina
requirementsfile.Thiscommandisusefulforquicklydownloadingthedependenciesofa
Pythonprogramthathasnotbeenlistedonpypi,andthusisnotavailableonpip.
Finally,youcanuninstallprogramsyou’vedownloadedwithpipuninstall
[package_name].TouninstallFlask,usethefollowingcommand:
pipuninstallflask.
…
>>Proceed(y/n)?y
...
Challenge
FindthreeprogramsthatinterestyouonUbuntuusingaptitudeandinstallthemusingapt-get
.
Chapter20.VersionControl
“Iobjecttodoingthingsthatcomputerscando.”
—OlinShivers
Writingsoftwareisateamsport.Whenyouareworkingonaprojectwithmorethanone
person,youwillbothbemakingchangestothecodebase—thefoldersandfilesthatmakeup
yoursoftware—andyouneedtokeepthosechangesinsync.Youcouldbothperiodically
emaileachotherwithyourchanges,andcombinethetwodifferentversionsyourself,butthat
wouldquicklybecometedious.Alsowhatwouldhappenifyoubothmadechangestothesame
partoftheproject?Whosechangesshouldbeused?Thesearethekindsofproblemsa
versioncontrolsystemsolves.Aversioncontrolsystemissoftwaredesignedtoletyou
easilycollaborateonprojectswithotherprogrammers.
Therearemanydifferentversioncontrolsystems.GitandSVNarebothpopular
choices.Versioncontrolsystemsareprograms,usuallyusedinconjunctionwithaservicethat
storesyoursoftwareonthecloud,likeGitHub.InthischapterwearegoingtouseGit,a
versioncontrolsystem,toputsoftwareonGithub,awebsitethatstoresyourcodeonthe
cloud.
Repositories
ArepositoryisadatastructurecreatedbytheprogramGittomanageasoftware
project.Adatastructureisawayoforganizingandstoringinformation:listsanddictionaries
areexamplesofdatastructures(youwilllearnmoreaboutdatastructuresinPartIV).
Repositoriesareusedtokeeptrackofallthechangesinasoftwareproject.
WhenyouareworkingonaprojectmanagedbyGit,therearemultiplerepositories
(usuallyoneforeachpersonworkingontheproject).Atypicalsituationlookslikethis:
everybodyworkingontheprojectalsohastheirownrepositoryontheircomputercalleda
localrepositorywhichkeepstrackofallthechangestheymakeontheirowncomputer;
thereisalsoacentralrepositoryhostedonawebsitelikeGitHubwhichallofthelocal
repositoriescommunicatewithtostayinsyncwitheachother.Aprogrammerworkingonthe
projectcanupdatethecentralrepositorywiththechangesthey’vemadeintheirlocal
repositoryortheycanupdatetheirlocalrepositorywiththenewestchangesother
programmershavemadetothecentralrepository.
imageofrepositoriescommunicating
ThisisdonefromthecommandlineusingtheprogramGit.
YoucancreateanewrepositoryusingtheGitprogramfromthecommandlineoron
GitHub’swebsite.Onceyoucreatearepository,youcanusetheGitprogramtomanageitand
communicatewithacentralrepository.
GettingStarted
Togetstarted,youneedtocreateaGithubaccount:gotoGithub.com/jointocreateone.
CreateanewrepositoryonGithub.LogintoyourGitHubaccountatgithub.comand
clickonthe+buttonatthetoprightcornerofthescreen.ClickCreaterepositoryfromthe
dropdownmenu.Givetherepositorythenamehangman.Makeitpublic,andcheckinitialize
therepositorywithareadme.NowclickCreatearepository.Ifatanytimeyouruninto
troubleanddosomethingwronginthissection,gotothesettingspageofyourrepository,
deleteit,andstartover.YoualsoneedtoinstallGit.YoucaninstallGitusingyourpackage
managerofchoice.
OnGitHub,hitthebuttoninthetoprightcornerandselectYourProfile.
image
Youwillseethenameofyourrepository:hangman.Clickonit.Youwillseeabuttonthat
saysCloneOrDownload.Whenyouclickonit,youwillseeHTTPS:followedbyalink.You
canusethislinktodownloadyourrepositorytoyourcomputerusingthecommandgitclone
[repository_url].Therepositorywilldownloadinwhateverdirectoryouissuethecommand
from.Copythelink,orpressthecopylinktoclipboardbuttonanduseitwiththegitclone
command:
$gitclone
>>
Uselstoverifytherepositorydownloaded:
$ls
>>my_project
PushingandPulling
TherearetwomainthingsyouwillbedoingwithGit.Thefirstisupdatingthecentral
repositorywithchangesfromyourlocalrepository.Thisiscalledpushingbecauseyouare
pushingnewdatatoyourcentralrepository.
Thesecondthingyouwillbedoingiscalledpulling.Pullingdatameansupdatingyour
localrepositorywithallofthenewchangesfromthecentralrepository.
Thecommandgitremote-vshowsyouwhatcentralrepositoryyourlocalrepositoryis
pushingandpullingfrom.The-vflagstandsforverbose,whichmeansthecommandwill
usuallyprintoutextrainformation.Usethiscommandinsidetoseethecentralrepository
yourlocalrepositoryispushingandpullingfrom:
$gitremote-v
>>originhttps://github.com/[username]/my_git_project.git
(fetch)
>>originhttps://github.com/[username]/my_git_project.git(push)
Thefirstlineshowsthecentralrepositoryyourprojectwillpulldatafromandthesecondline
showsthecentralrepositoryyourprojectwillpushdatato.Generally,youwillpushandpull
fromthesamecentralrepository.
PushingExample
Inthissection,youwillmakeachangetothelocalrepositoryforyourhangmanproject
andpushthatchangetoyourcentralrepository.
MovethePythonfilewiththecodeweusedtocreateHangmanintothehangman
directorywecreatedinGettingStarted.Ourlocalrepositorynowhasafilethatdoesnotexist
inourcentralrepository—ourlocalrepositoryisoutofsyncwithourcentralrepository.We
canfixthisbypushingthechangeswemadeinourlocalrepositorytoourcentralrepository.
Pushingchangesfromyourlocalrepositorytoyourcentralrepositoryhappensinthree
steps.FirstyoustageyourfileswhichiswhereyoutellGitwhichfileshavechangesthatyou
wanttopushtoyourcentralrepository.Onceyou’vereviewedeverything,youcommitthe
files.
,thenyoucommitthemandfinallyyoupushthem.InthefirststepyoutellGitwhatfiles
youwanttopushtoourcentralrepository.Thisiscalledstagingafile.Whenafileisstaged,
wehavethechancetochangeourmindandunstageit.Thesyntaxgitadd[file_path]isused
tostageafile.Usethecommandgitaddhangman.pytostageourfile:
$gitaddhangman.py
>>
Thecommandgitstatusshowsyouthecurrentstateofyourprojectinrelationtoyour
repository.Newfilesthatexistinyourlocalproject,butdonotexistinyourrepositoryare
showningreen,asaredeletedfiles.Filesthathavebeenmodifiedareinred.Afileis
modifiedifitisinyourrepository,butlocalversionandtheoneinyourrepositorydiffer.
Enterthecommandgitstatus:
$gitstatus
>>Onbranchmaster
Changestobecommitted:
(use"gitresetHEAD<file>..."tounstage)
newfile:hangman.py
Youshouldseethefilehangman.pyinagreenfont.Youhavetostageeachfileyouwantto
pushtoyourrepositorywiththecommandgitadd[file].Ifyoustageafileandchangeyour
mind,youcanunstageitwithoutmakingchangestoyourcentralrepository.Youcanunstage
afilewiththesyntaxgitreset[file_path].Unstagehangman.pywiththecommandgitreset
hangman.py.Usegitstatustoseetheresult,andadditagainwithgitaddhangman.py:
$gitresethangman.py
$gitstatus
>>
$gitaddhangman.py
>>
Oncewe’vestagedourfiles,andeverythinglooksthewaywewant,wearereadytomoveto
thenextstep—committingourfilestoourcentralrepository.Whenyoucommitfiles,you
wanttousethe-mflagsoyoucanpassalongamessage.Themessagewillbesavedalong
withyourcommitinyourrepositorytohelpyourememberwhatchangesyoumadeinthat
commit.
$gitcommit-m“myfirstcommit”
>>1filechanged,1insertion(+)
createmode100644hello_world.py
ThefinalstepistoactuallypushyourchangestoGitHub.Thisisdonewiththecommandgit
pushoriginmaster:
$gitpushoriginmaster
>>Countingobjects:3,done.
Deltacompressionusingupto4threads.
Compressingobjects:100%(2/2),done.
Writingobjects:100%(3/3),309bytes|0bytes/s,done.
Total3(delta0),reused0(delta0)
Tohttps://github.com/[your_username]/my_project.git
0eb3a47..48acc38master->master
OnceyouenteryourGitHubusernameandpasswordfromthecommandline,yourchanges
willbepushedtoGitHub.IfyoulookatyourrepositoryonGitHub’swebsite,youwillsee
hangman.pyisnowinyourproject.
PullingExample
Inthissection,welearnhowtoupdateyourrepositorywithchangesfromthecentral
repository.First,wehavetomakeachangeinourcentralrepository.Usethecommandgit
pulloriginmastertoupdateyourlocalrepositorywiththechangewemade:
$gitpulloriginmaster
>>Fromhttps://github.com/calthoff/my_project
>>*branchmaster->FETCH_HEAD
Gitappliedthechangesfromourcentralrepositorytoourlocalrepository,andtheyarenow
insync.YoucanviewtheREADME.mdfileonyourcomputertoseethechange.
RevertingVersions
Whenyouuseversioncontrol,theentirehistoryofyourprojectissavedandavailable
foryoutouse.Ifyoudecideyouwanttoreverttoaversionofyourprojectfrom10daysago,
youareabletodoso.Youcanviewyourproject’shistoryofcommitswithgitlog,which
shouldoutputsomethinglikethis:
$gitlog
commitaeb4ef3cf3aabdb9205ea9e96e8cab5c0f5ca7ea
Author:CoryAlthoff<coryalthoff@Corys-MacBook-Pro.local>
Date:ThuJan2113:52:022016-0800
Thestringofnumbersandlettersaftercommitisthecommitnumber.Wecanusethis
numbertorevertourprojecttoexactlyhowitwasatthattime.Wecantravelbackintime
withthecommandgitcheckout[oldcommit].Inthiscasethecommandwouldbegitcheckout
aeb4ef3cf3aabdb9205ea9e96e8cab5c0f5ca7ea.
diff
Wecanusethecommandgitdifftoseethedifferencebetweentheversionofafileinour
localproject,andtheversioninourrepository.Addx=100tothesecondlineofour
hello_world.pyfile.Enterthecommand:
$gitdiffhello_world.py
>>diff--gita/hello_world.pyb/hello_world.py
indexb376c99..83f9007100644
---a/hello_world.py
+++b/hello_world.py
@@-1+1,2@@
print('hello')
+x=100
Githighlights+x=100ingreenbecausethelinechanged,the+istosignifythelinex=100
wasadded.
TheOtherPullRequest
Confusingly,therearetwoconceptsnamedpullinversioncontrol.Wepreviouslytalked
aboutpullingdatafromyourrepository.Thereisalsoanunrelatedconceptcalledapull
request.ApullrequesttakesplaceonGitHub.Ifyouareworkingonabranchofaproject,and
youwanttomergeitwiththemasterrepository,youwouldissueapullrequestonGitHubto
mergethetwo.Thisgivesyourteammatesthechancetoreviewandcommentonyour
changes.Ifeverythinglooksgood,someoneonyourteamcanapprovethepullrequestand
mergethetwobranches.
LearningMore
Youtypicallycreateanewbranchwhenyouwanttodevelopanewfeature
foryourprogramorfixabugorproblem.Whenyouarefinishedwithwhatever
youaredoingonyourbranch,younormallymergeyourchangeswiththe
masterbranch,whichistheprocessofcombiningthetwobranches.Covering
mergingisoutsidethescopeofthisbook,butgit-scmhasagreattutorialonit
youcancheckoutat:https://git-scm.com/book/en/v2/Git-Branching-Basic-
Branching-and-Merging.
Challenge
Chapter21.SQLite
“Data!Data!Data!Ican’tmakebrickswithoutclay!”
—SirArthurConanDoyle
Databasesareprogramsusedtopersistdata.Datapersistsifitoutlivestheprocessthat
createdit46.Mostoftheprogramswe’vebuiltsofarworkfinewithoutpersistinganydata,
withonenotableexception—ourwebscrapingprogramwebuiltinpartIItocollectheadlines
fromGoogleNews.Alloftheheadlineswecollectarelostaftertheprogramstopsrunning.
Butwhatifwewanttoanalyzetheheadlinesfromthelastyear?Thisiswherepersistence
comesin,andwhydatabasesareimportant.Databasesperformtwomainoperations—read
andwrite.Whenyouwritetoadatabase,youaregivingitinformationtostore.Whenyou
readfromadatabase,youareretrievinginformationfromit.Inanincreasinglydata-centric
world,databasesarebecomingexceedinglyimportant.Inthischapter,wegooverthebasics
ofdatabases.
NoSQLvs.SQL
RelationaldatabaseswerefirstproposedbyEdgarF.Coddin1970.Relationaldatabases
storedatalikeanExcelspreadsheet—dataisstoredinrowsandcolumns.Thedataisstored
andretrievedusingthequerylanguageSQL,whichstandsforStructuredQueryLanguage.
PostGreSQLandMySQLareexamplesofpopularrelationaldatabases.
Recentlyanewbreedofdatabases,calledNoSQLhavegainedpopularity.NoSQL
literallymeans“noSQL.”Inotherwords,thethingthesenewbreedofdatabaseshavein
commonistheyarenotrelationaldatabasesusingSQL.Redisisanexampleofapopular
NoSQLdatabases.Redisisa“keyvaluestore”whichmeansyoucanstoredatalikea
dictionaryinPython,i.e.,storeakeyandavalue.Soforexampleyoucouldstore“Monty”as
akeyand“Python”asavalue;queryRedisfor“Monty”anditwillreturn“Python”.While
NoSQLdatabasesareuseful,,inthischapterwearelearningaboutrelationaldatabases,
becausetheyareveryimportantinthesoftwareindustry.Onceyouarefamiliarwith
relationaldatabases,IencourageyoutolearnmoreaboutNoSQL.
GettingStarted
WewillgetstartedwithSQLbyusingSQLite.SQLiteisalightweightdatabasethat
comesbuiltintoyouroperatingsystem,sowedon’thavetoinstallanything.Gotothe
commandlineandtypethecommand“sqliteself_taught.db”tocreateanewdatabase.Ifyou
arecomingbacktoYoucanopenSQLiteanytimewiththecommand“sqlite3”.Ifyoualready
createdthedatabase,openitwith“.openself_taught.db”.ExitSQLitewiththecommand
“.exit”.
#explainrelationaldatabaseswhatisacolumn,whatisarow
DataTypes
CreateaTable
CREATETABLEcustomers(
idint
first_nametext
last_nametext,
date_createddate,
PRIMARYKEY(column1)
);
Everytableinarelationaldatabasehastohavesomethingcalledaprimarykey
—auniqueidentifierforeachrow.Inthiscaseourprimarykeyisourfirst
column—calledid.Wecansetourprimarykeytoautoincrement,whichmeans
theprimarykeywillbeanumberthatthedatabaseautomaticallyincrementsfor
youwheneveryouaddnewdata.Ourcustomerstablealsohasarowforthe
customer’sfirstnameandlastname.
RelationaldatabasesaremadeupoftablesthatstoredatalikeanExcelspreadsheet,in
columnsandrows.Let’screateourfirsttable.Typethefollowinginsidesqlite.Makesure
youareinsideSQLitewiththecommand“sqlite3”andyourcommandlinesayssqlite.
CREATETABLEself_taught.bdfl(
namestring
projectstring
ageint
birthdaydate
);
Ournewinourdatabaseself_taughtiscalled“bdfl”,whichstandsforbenevolentdictatorfor
life.BDFLisatitlegiventothecreatorsofopensourceprogrammingprojectslikeLinus
Torvalds,creatorofLinux,GuidovanRossumcreatorofPython,DavidHeinemeierHansson
creatorofRubyonRailsorMattMullenwegcreatorofWordpress.Wewilluseourtableto
storedataaboutthesebdfls,suchastheirname,project,andbirthdays.
Constraints
#findsomewheretoput#CreatingarelationshipwithanothertableisdonewithSQL,
whichmakessurethedatabaseisawareoftherelationship.Youcan’tjustenteranyinteger
underthecustomercolumn,yourdatabasewillonlyletyouenteravalidprimarykeyfora
customerinthecustomertable.
Aconstraintisaruleyoucanapplytoacolumnthatisenforcedbythedatabase.Examplesof
constraintsare:notnull,unique,primarykeyandforeignkey.
Ifyouputtheconstraint“notnull”onacolumn,thatcolumncannotbe“null”whichislike
“None”inPython.Thismeansthecolumnmusthavedatainit.Sayyouarecollectingthefirst,
middleandlastnameofsubscribersforanewsletteronyourwebsite.Thetableinyour
databasetocollectthisinformationmightlooklikethis:
subscribers
first|middle|last
Youprobablywouldwanttoput“notnull”constraintsonthefirstcolumn,whileallowing
“null”inthemiddleandlastcolumns.Thereasonbeingthateveryonehasafirstname,butnot
everyonehasamiddleorlastname.WhatifPrincesignedupforyournewsletter?Ifyouputa
notnullconstraintonlast,Princewouldn’tbeabletosignup.
Constraintsareimportantbecausetheymakeguaranteesaboutdata.Inthiscase,ifwe
mightwanttocreateaprogramthatanalyzesthefirstnamesofallofoursubscribers,our
programwouldprobablyperformsomesortofoperationoneachstring.Ifthe“first”column
gaveourprogramanullvalueinsteadofastring,andourprogramtreatedthenullvaluelike
astring,itwouldcauseanerrorinourprogram.Byaddinganotnullconstrainttoourfirst
column,ourprogramcannowbeassuredthateveryfirstnameitgetsfromthedatabaseis
goingtobeastringandourprogramcantreatitassuch.
Ifwearecollectingdataforanewsletter,wealsoneedtocollectourusers’email
addresses.Wecandothisbyaddinganemailcolumntoourtable:
ALTERTABLEcustomer
ADDemailstringUNIQUE
Weaddeda“unique”constrainttothisnewcolumn,sothattheemailcolumnmustbeunique.
Thismeansifanemailisusedinonerowofourtable,itcannotbeusedinanotherrow.This
makessensebecauseeveryemailaddressintheworldisunique,andiftwosubscribershave
thesameemail,thereisaproblem.
Aforeignkeyletsyouenforcethatthedataenteredinthetableisaprimarykeyin
anothertable.WealreadysawanexampleofusingaforeignkeyinourAmazonexample.
HereishowwecouldchangeatabletoaddaforeignkeyP_Idtoamadeuptablecalled
Persons:
ALTERTABLEcustomer
ADDFOREIGNKEY(P_Id)
REFERENCESPersons(P_Id)
Acheckconstraintisusedtomakesuredataenteredinatablemeetscertain
specifications.Hereisanexampleofacheckconstraintwecouldaddtoouremailcolumn:
ALTERTABLEcustomer
ADDCHECKemailvarchar(255)
Thisenforcesthatallemailsenteredintoourdatabasehavetobelessthan255
characters.
InsertData
TimetoinsertdataaboutourfirstBDFLintoourtable.EnterthefollowingSQL
statementsoneatatime:
INSERTINTObdfl(name,project,age)
VALUES(GuidovanRossum,Python,1-31-1956);
INSERTINTObdfl(name,project,age)
VALUES(DavidHeinemeierHansson,RubyonRails,10-15-1979);
INSERTINTObdfl(name,project,age)
VALUES(LinusTorvalds,Linux,12-28-1969);
INSERTINTObdfl(name,project,age)
VALUES(MattMullenweg,WordPress,1-11-1984);
INSERTINTObdfl(name,project,age)
VALUES(DriesBuytaert,Drupal,11-19-1978);
INSERTINTObdfl(name,project,age)
VALUES(LarryWall,Perl,9-27-1954);
QueryData
Queryingdatameanslookingupdatainyourdatabasegivenspecificparameters.
Exampleoftheparametersyoumightgivearewhattablethedataisin,anddifferentattributes
youwantthedatatocontain.Let’sstartbyqueryingforeverythinginourbdfltableby
enteringSELECT*frombdflintosqlite.InSQL,SELECT*meansselecteverything.
Ifwedonotwanttoselecteverything,andjustwanttoselectthenameofthebdflof
Linux,wecandothiswithSELECTnameFROMbfdflWHEREproject=“Linux”
orQuery
Youcanadd“or”toyourquerytoselectfromarowifeithertheconditionbeforethe
or,ortheconditionaftertheoristrue.Selecteverythingfromourtablewheretheprojectis
RubyonRailsorWordpresswiththestatementSELECT*FROMbfdlWHEREproject=
“Linux”OR
project=”RubyonRails”.
andQuery
Adding“and”toyourquerywillonlyselectarowifbothconditionsaretrue.Firsttry
SELECT*FROMbfdlWHEREbfdl.nameLIKED%.Thisreturnsanyrowwherethename
startswithaD.ItshouldreturnthedatastoredforDavidHeinemeierHanssonandDries
Buytaert.
Nowtry“”SELECT*FROMbfdlWHEREnameLIKE“D%”ANDwhereA.Date>=
Convert(datetime,'2010-04-01').ThiswillonlyreturnDavidHeinemeierHansson.While
DriesBuytaertstartswith“D”,hisbirthdayisnotgreaterthanthedateweprovided.
Count
Countthenumberofrowsinatablewith:
SELECTCOUNT(*)FROMbfdl
>>
CommunicatingwithDatabases
Sofar,we’velearnedtocommunicatewithadatabasebyusingSQLfromtheterminal.
Butwhatifwewanttostoredatacollectedinadatabasefromaprogramwe’vewrittenina
database?ThereistoolwecanusetoabstractSQLawayentirelycalledanORM,orobject
relationalmap.Anobjectrelationalmapissoftwarethatletsyouinteractwithyourdatabase
usingclassesinyourprogramminglanguage,insteadofusingSQL.
UsinganORMlibrary,suchastheORMthatcomeswiththePythonwebframework
Django,wecanrepresentadatabasetableasaclassinPython.InsteadofusingSQL
statementstoreadandwritefromourtable,wecanusemethodsfromthetableclasswe
defined.HereisanexampleofhowwecouldrepresentandqueryourAmazoncustomertable
usingDjango’sORM:
fromdjango.dbimportmodels
classCustomer(models.Model):
username=models.CharField()
Thisexampledefinesanewdatabasetablecalled“customer.”Inthiscasewehavetwo
columns,idwhichisourprimarykeyautomaticallycreatedbyourORMandnotreflectedin
ourcode,andusernamewhichisavariablesettoaCharFieldobjectwhichDjangousesto
createadatabasecolumnthatstoresstrings.
Now,whenwewanttointeractwithourdatabase,wecandoitusingourCustomerclass.
Forexample,wecancreateanewuser,whichwillbestoredinMySQL,withthefollowing
code:
user1=Customer.objects.create(username=”eddie”)
Toqueryourdatabasewesimplyneedtousetheclasswecreated:
eddie=Customer.objects.get(username=”eddie”)
printeddie.username
>>“eddie”
Withthiscode,wewereabletosuccessfullyreadandwritefromourdatabasewithoutany
SQL.Django’sORMtranslatesourcodetoSQLforus,sowedon’thavetoworryaboutit.
Challenge
Chapter22.BringingItAllTogether
“Themagicofmythandlegendhascometrueinourtime.Onetypesthecorrectincantation
onakeyboard,andadisplayscreencomestolife,showingthingsthatneverwerenorcould
be”....
—FrederickBrooks
Inthischapter,wewillseehowpowerfulprogrammingcanbebybuildingaweb
scraper:aprogramthatextractsdatafromawebsite.Learningtoscrapedatafromwebsitesis
powerful.Itgivesyoutheabilitytoextractanydatayouwantfromthelargestcollectionof
informationthathaseverexisted.Seeingthepowerofwebscrapers,andhoweasytheyareto
build,isoneofthereasonsIgothookeduponprogramming,andIhopeithasthesameeffect
onyou.
HTML
Beforewebuildourwebscraper,weneedaquickprimeronHTML,orhypertext
markuplanguage.HTMLisoneofthefundamentaltechnologiesusedtobuildwebsites,along
withCSSandJavaScript.YoucanthinkofHTMLasitsownlittlelanguage,usedtogivea
websitestructure.HTMLismadeupoftagsthatawebbrowserusestolayoutawebpage.In
fact,youcanbuildanentirewebsiteusingonlyHTML.Itwon’tbeinteractiveorlookvery
good,becauseJavaScriptiswhatmakeswebsitesinteractive,andCSSiswhatgivesthem
style,butitwillbeawebsite.HereisanexampleofawebsitethatwilldisplaythetextHello,
World!:
<!--ThisisacommentinHTML.Savethisfileasindex.html-->
<htmllang="en">
<head>
<metacharset="UTF-8">
<title>MyWebsite</title>
</head>
<body>
Hello,World!
<ahref="https://news.ycombinator.com/">here</a>
</body>
</html>
Takeaminutetostudythiscode.NowsavethisHTMLintoafileandopenthefilewithyour
webbrowserbyclickingonthefile(youmayhavetorightclickandchangethedefault
programtoopenthefilewithtoawebbrowserlikeChrome).Onceyouopenthefilewith
yourwebbrowser,youwillseeawebsitethatsaysHelloWorld!withalinktotheY
Combinatorwebsite.
YourwebbrowserusesthedifferenttagsinourHTMLtofigureouthowtodisplaythis
website.Tagshaveabeginningtagandclosingtag,oftenwithsomethingliketextinbetween.
Forexample,yourbrowserdisplaysthetextinbetweenthe<title></title>tagsinthetabof
yourbrowser.Anythinginbetweenthe<body></body>tags,makesuptheactualwebsite.
Anythinginside<a></a>tagsisalink.ThereisalotmoretoHTML,butthisisallyouneed
toknowinordertobuildyourfirstwebscraper.
ScrapeGoogleNews
NowwecanbuildascraperthatfetchesalloftheheadlinesfromGoogleNews.Wewill
dothisbyextractingallofthe<a></a>tagsinGoogleNews’sHTML.Aswesawinour
HTMLexample,each<a></a>taghasavariableinitcalledhref,e.g.,<a
href=”theselftaughtprogrammer.io”></a>.Wearegoingtoextractallofthehrefvariables
fromallofthe<a></a>tagsonGoogleNews’swebsite.Inotherwords,wearegoingto
collectalloftheURLsGoogleNewsislinkingtoatthetimewerunourprogram.Wewill
usetheBeautifulSouplibraryforparsingourHTML(convertingHTMLtoPythonobjects),
sofirstinstallitwith:
pipinstallbeautifulsoup4==4.4.1
OnceBeautifulSoupisinstalled,wecangetGoogleNews’sHTMLusingPython’sbuilt-in
urllib2libraryforworkingwithURLs.Startbyimportingurllib2andBeautifulSoup:
“““https://github.com/calthoff/tstp/blob/master/part_III/lets_read_some_code/lets_read_some_code.py”””
importurllib2
frombs4importBeautifulSoup
Nextwecreateascraperclass
classScraper:
def__init__(self,site):
self.site=site
defscrape(self):
pass
Ourmethodtakesawebsitetoscrapefrom,andhasamethodcalledscrapewhichweare
goingtocallwheneverwewanttoscrapedatafromthewebsitewepassedin.
Nowwecanstartdefiningourscrapemethod.
defscrape(self):
response=urllib2.urlopen(self.site)
html=response.read()
Theurlopen()functionmakesarequesttoGoogleNewsandreturnsaresponseobject,which
includesGoogleNews’sHTMLinitasavariable.Wesavetheresponseinourresponse
variableandassignthevariablehtmltoresponse.read()whichreturnstheHTMLfrom
GoogleNews.AlloftheHTMLfromGoogleNewsisnowsavedinthevariablehtml.This
isallweneedinordertoextractdatafromGoogleNews.However,westillneedtoparsethe
HTML.ParsingHTMLmeansreadingitintoourprogramandgivingitstructurewithour
code,suchasturningeachHTMLtagintoaPythonobject,whichwecandousingthe
BeautifulSouplibrary.First,wecreateaBeautifulSoupobjectandpassinourhtmlvariable
andthestring‘html.parser ’asaparametertoletBeautifulSoupknowweareparsingHTML:
defscrape(self):
response=urllib2.urlopen(self.site)
html=response.read()
soup=BeautifulSoup(html,'html.parser')
Movingforward,wecannowprintoutthelinksfromGoogleNewswith:
defscrape(self):
response=urllib2.urlopen(self.site)
html=response.read()
soup=BeautifulSoup(html,'html.parser')
fortaginsoup.find_all('a'):
url=tag.get('href')
ifurland'html'inurl:
print("\n"+url)
find_all()isamethodwecancallonBeautifulSoupobjects.Ittakesastringrepresentingan
HTMLtagasaparameter(‘a’representing<a></a>),andreturnsaResultSetobject
containingalltheTagobjectsfoundbyfind_all().TheResultSetissimilartoalist—you
caniteratethroughit(weneversaveResultSetinavariable,itissimplythevaluereturnedby
soup.find_all('a')),andeachtimethroughtheloopthereisanewvariable:tag,representinga
tagobject.Wecallthemethodget()onthetagobject,passinginthestring‘href’asa
parameter(hrefisthepartofanHTML<ahref=“url”></a>tagwhichholdstheURL),andit
returnsastringURLwhichwestoreinthevariableurl.
ThelastthingwedoistochecktomakesureURLisnotNonewithifurl,becausewe
don’twanttoprinttheurlifitisempty.Wealsomakesure‘html’isintheurl,becausewe
don’twanttoprintGoogle’sinternallinks.Iftheurlpassesbothofthesetests,weuse‘\n’to
printanewlineandthenprinttheurl.Hereisourfullprogram:
importurllib2
frombs4importBeautifulSoup
classScraper:
def__init__(self,site):
self.site=site
defscrape(self):
response=urllib2.urlopen(self.site)
html=response.read()
soup=BeautifulSoup(html,'html.parser')
fortaginsoup.find_all('a'):
url=tag.get('href')
ifurland'html'inurl:
print("\n"+url)
Scraper().scrape('https://news.google.com/')
Runthescraper,andyoushouldseearesultsimilartothis:
https://www.washingtonpost.com/world/national-security/in-foreign-bribery-cases-leniency-
offered-to-companies-that-turn-over-employees/2016/04/05/d7a24d94-fb43-11e5-9140-
e61d062438bb_story.html
http://www.appeal-democrat.com/news/unit-apartment-complex-proposed-in-
marysville/article_bd6ea9f2-fac3-11e5-bfaf-4fbe11089e5a.html
http://www.appeal-democrat.com/news/injuries-from-yuba-city-bar-violence-hospitalize-
groom-to-be/article_03e46648-f54b-11e5-96b3-5bf32bfbf2b5.html
...
NowthatyouhaveallofGoogleNews’sheadlinesavailableinyourprogram,the
possibilitiesarelimitless.Youcouldwriteaprogramtoanalyzethemostusedwordsinthe
headlines,andbuildawordcloudtovisualizeit.Youcouldbuildaprogramtoanalyzethe
sentimentoftheheadlines,andseeifithasanycorrelationwiththestockmarket.Asyouget
betteratwebscraping,alloftheinformationintheworldwillbeopentoyou,andIhopethat
excitesyouasmuchasitexcitesme.
Challenge
ModifytheGoogleScrapertosavetheheadlinesinafile.
Chapter23.Practice
Exercises
0.Buildascraperforanotherwebsite.
0.WriteaprogramandthenreverttoanearlierversionofitusingPyCharm.
0.Downloadpylintusingpip,readthedocumentationforitandtryitout.
Read
0.http://www.tutorialspoint.com/python/python_reg_expressions.htm
PartIVIntroductiontoComputerScience
DataStructures&Algorithms
“Iwill,infact,claimthatthedifferencebetweenabadprogrammerandagoodoneiswhether
heconsidershiscodeorhisdatastructuresmoreimportant.Badprogrammersworryabout
thecode.Goodprogrammersworryaboutdatastructuresandtheirrelationships.”
—LinusTorvalds
Thischapterisalightintroductiontoalgorithmsanddatastructures,perhapsthemost
importantsubjectinComputerScience.ThetitleoftheinfluentialbookAlgorithms+Data
Structures=Programssummarizestheirimportance.Mygoalinthischapteristointroduce
youtothesubject,andclarifysomethingsIfoundconfusingwhenIwaslearningaboutthem
(whichIstillam).Inadditiontoreadingthischapter,youdefinitelyneedtoreadmoreabout
algorithmsanddatastructuresoutsideofthisbook,andalsospendalotoftimepracticingthe
conceptsintroducedhere.Manyoftheexamplesinthischaptercomefromtheamazingbook
PythonAlgorithmsandDataStructuresbyBradMillerandDavidRanum.Itisoneof
myfavoritebooks,availableonlineforfreeat:http://interactivepython.org.
WhatAreAlgorithms&DataStructures?
Analgorithmisaseriesofstepsthatcanbefollowedtosolveaproblem.Theproblemcould
beanything,likesortingorsearchingalist,ortraversingeatree.
Adatastructureisawaytostoreandorganizeinformation.Datastructuresare
fundamentaltoprogramming,andwhateverprogramminglanguageyouusewillcomewith
built-indatastructures.Commondatastructuresincludehashtables,stacks,andlists.Data
structurescomewithdifferenttradeoffs,withcertaindatastructuresbeingbettersuitedfor
specifictasksthanothers.
BigONotation
InComputerScience,wesolveproblemsusingalgorithms.Butwhatifyoucomeupwith
twodifferentalgorithmstosolvethesameproblem?Howdoyoudecidewhichisbest?BigO
Notationgivesyouaframeworkfordecidingifonealgorithmisbetterthananotherby
lookingatthenumberofstepseachalgorithmtakes,andchoosingtheonethattakestheleast
amountofsteps.
WecanuseanequationlikeT(n)=ntodescribeanalgorithm.
Thefollowingsectionsintroducesomealgorithmsyoushouldlearn.
Modulo
Themodulooperator“%”returnstheremainderoftwonumberswhenyoudividethem.
Forexample“2%2”returns0,whereas“3%2”returns1.Moduloishelpfulinsolvinga
varietyofproblemslikeFizzbuzz,apopularfirstinterviewquestiondesignedtoweedout
peoplethatcannotprogram.ThequestionwasintroducedbyImranGhoryinhisblogpost
UsingFizzBuzztoFindDeveloperswhoGrokCoding.Ifyouknowthecorrectapproachtothe
problem,itiseasytosolve,whereasifyoudon’t,itcanappearcomplicated.Theproblemis
usuallygivenas:
Writeaprogramthatprintsthenumbersfrom1to100.Butformultiplesofthreeprint“Fizz”
insteadofthenumberandforthemultiplesoffiveprint“Buzz”.Fornumberswhichare
multiplesofboththreeandfiveprint“FizzBuzz”.
Thekeytosolvingthisproblemisusingmodulo.Tosolvethisproblemyousimply
needtoiteratefrom1to100andcheckifeachnumberisdivisibleby3,5orboth.Hereisthe
solution:
deffizz_buzz():
foriinrange(0,101):
ifi%3==0andi%5==0:
print'FizzBuzz'
elifi%3==0:
print'Fizz'
elifi%5==0:
print'Buzz'
else:
printi
Westartbyiteratingthroughthenumbers1to100withaforloop.Thenwesimplycheck
eachoftheconditions.Weneedtocheckifthenumberisdivisibleby3or5first,becauseifit
is,wecanmovetothenextnumber.Thisisnottruewithbeingdivisiblebyeither5or3,
becauseifeitheraretrue,westillhavetocheckifthenumberisdivisiblebyboth.Wethen
cancheckifthenumberisdivisibleby3or5(inanyorder).Finally,ifnoneoftheconditions
aretrue,wesimplyprintthenumber.
Hereisanotherproblemwhereusingmoduloisthekeytofiguringouttheanswer:
Rotateanarrayofnelementstotherightbyksteps.
Forexample,withn=7andk=3,thearray[1,2,3,4,5,6,7]isrotatedto[5,6,7,1,2,3,4],i.e.,
youaremovingeveryelementinthelistkpositions.
Whenyousolvethisproblem,yourfirstinstinctmightbetosimplyaddntotheindex
ofeachnumberandmovethenumbertothenewposition.Theproblemisthisdoesnotwork.
Thekeytosolvingthisproblemisonceagainmodulo.Saywehadalistofnumbersfrom1-
12.Wecanthinkofthislistasaclock.Ifwestartat12o’clock,andgoaroundtheclock
twelvehours,wearebacktowherewestarted.
Wecanachievethisbytakingthecurrentindex,addingthekeynewindexandgettingthe
remainder.Ifyoutakeourlistof12numbersandyouwanttocalculatewherethenumber12
wouldbe(startingatindex11)ifweadded12totheindex,youcancalculatethatwith11+
BubbleSort
Asortingalgorithmisanalgorithmthattakesagroupofnumbersandputsthemina
certainorder.Therearemanydifferentalgorithmssuchasselectionsort,insertionsort,shell
sort,mergesort,andquicksort.Inthissectionwewillimplementbubblesort,asorting
algorithmthatisnotveryefficient—buteasytounderstandandusefulforunderstanding
sortingalgorithms.Hereisanimplementationofbubblesort:
defbubble_sort(num_list):
"""
:paramnum_list:Listofnumbers
:return:Sortedlistofnumbers
"""
foriinrange(len(num_list)-1,0,-1):
forjinrange(i):
ifnum_list[j]>num_list[j+1]:
temp=num_list[j]
num_list[j]=num_list[j+1]
num_list[j+1]=temp
my_list=[4,266,9,24,44,54,41,89,20]
bubble_sort(my_list)
print(my_list)
>>[4,9,20,24,41,44,54,89,266]
Inthisalgorithm
SequentialSearch
Searchalgorithmsareusedtofindanumberinalistofnumbers.Sequentialsearchisa
simplesearchalgorithmthatcheckseachnumberinthelistonebyonetoseeifitmatchesthe
numberitislookingfor.Asequentialsearchisthewayyousearchadeckofcardsfora
specificcard.Yougoonebyonethrougheachcard,ifthecardisthecardyouarelooking
for,youstop.Ifyoumakeitthroughtheentiredeckwithoutfindingthecard,youknowthe
cardisn’tthere.Hereisanexampleofsequentialsearch:
defsequential_search(number_list,number):
"""
:paramnumber_list:Listofintegers.
:paramnumber:Integertolookfor.
:return:Trueifthenumberisfoundotherwisefalse.
"""
found=False
foriinnumber_list:
ifi==number:
found=True
break
returnfound
printsequential_search(range(0,100),2)
>>True
printsequential_search(range(0,100),202)
>>False
First“found”issettofalse.Thenweloopthrougheverynumberinthelistandcheckifitis
equaltothenumberwearelookingfor.Ifitequaltothenumberwearelookingfor,weset
foundtoTrue,exitourloopandreturnTrue.Otherwise,wecontinuetothenextnumberinthe
list.IfwegetthroughtheentirelistandfoundhasneverbeensettoTrue,w.Wereturnfound
whichwillstillbesettoFalse.
SequentialsearchisanO(n)algorithm.Inthebestcasescenario,thenumberweare
lookingforcouldbethefirstnumberinthelist,inwhichcaseouralgorithmwouldtakeonly
onestep.However,intheworstcasescenario,thenumberisnotinthelistandwehaveto
checkeverysinglenumberinthelist,ornsteps.Ifwehaveamillionitemsinourlist,worst
casewehavetosearchthroughamillionitems.Ifwehaveabillionitems,worstcasewehave
tosearchthroughabillionitems.Asourlistgrows,theworstcasescenarioforouralgorithm
growsbythesizeofourlist,makingthisalgorithmO(n).
BinarySearch
Binarysearchisalogarithmicalgorithmusedtosearchfornumbersinalist,butthe
numbershavetobeordered.Rememberthis:,inorderforanalgorithmtobelogarithmic,it
needstoeitherbedividingormultiplyingtoasolution.Anyguesseshowbinarysearch
works?Binarysearchworksbycontinuallycuttingthelistinhalf.Thealgorithmpicksa
numberinthemiddleofalist,andlooksatwhetherit’stherightnumber.Ifitistheright
number,thesearchiscomplete.Ifit’snottherightnumber,thealgorithmthrowsawayhalf
thelist.Ifthenumberwastoobig,itthrowsawayeverythingabovethenumberitselected.If
thenumberwastoosmall,itthrowsawayeverythingbelowthenumberitselected.
Imagewehaveanorderedlistfrom1to10,andwewanttosearchforthenumber3.Our
listlookslikethis:
[1,2,3,4,5,6,7,8,9,10]
Ouralgorithmwouldfirstpickthenumber5becauseit’sinthemiddleofthelist.Since5is
notthenumberwearelookingfor,and3issmallerthanfive,ouralgorithmwouldthrowout
everythingabove5.Nowourlistlookslikethis:
[1,2,3,4,5]
Ourlistwouldnowpickthenumberthree,sinceit’sinthemiddleofthelist.Since3isthe
numberwearelookingfor,ouralgorithmwouldstopandreturnthatthenumber3wasfound
inourlist.Noticeouralgorithmonlytooktwostepstofigureoutthreewasinourlist.Ifwe
searchedthroughthelistlinearly,onebyone,lookingforthenumberthree,itwouldtakesus
threesteps.Hereisanexampleofabinarysearchalgorithmsearchingforthenumber3:
defbinary_search(number_list,number):
"""Logarithmicbinarysearchalgorithm.
:paramnumber_list:Listoforderedintegers.
:paramnumber:Integertosearchforinthepassedinlist.
:return:Trueifthenumberisfound,otherwiseFalse.
"""
first=0
last=len(number_list)-1
number_found=False
whilefirst<=lastandnotnumber_found:
middle=(first+last)/2
ifnumber_list[middle]==number:
number_found=True
else:
ifnumber<number_list[middle]:
last=middle-1
else:
first=middle+1
returnnumber_found
binary_search([1,2,3,4,5,6,7,8,9,10],3)
Weuseawhileloopthatcontinuesaslongasthevariablefirstisnotgreaterthanorequalto
thevariablelast,andthevariablenot_trueisFalse.
Wecalculatethemiddleindexofthelistbyaddingthefirstindexofthelistwiththelast
indexofthelistanddividingthembytwo.Thereasonwesubtractonefromlastisbecausethe
lengthofalistiscalculatedstartingfromone,whereasindexesstartatzero.
Wechecktoseeifthemiddlenumberisthenumberwearelookingforbylookingup
themiddlenumberinourlistwithnumber_list[middle].Ifthemiddlenumberisthenumber
wearelookingfor,not_trueissettoTrue,andweexittheloop.Otherwise,wechecktoseeif
thenumberwearelookingforisbiggerorsmallerthanthemiddlenumber.Ifourmiddle
numberissmallerthanthenumberwearelookingfor,wechangelasttothemiddleindex-1,
whichonthenextloopwillsplitthelistinhalfwitheverythingsmallerthanourcurrent
middlenumber,whereasifourmiddlenumberisbiggerthanthenumberwearelookingfor,
wechangefirsttomiddle+1,dividingthelistinhalfsoitcontainseverythingbiggerthan
ourmiddlenumber.
Recursion
Recursionisnotoriousasoneofthetoughestconceptsfornewprogrammerstograsp.If
itisconfusingtoyouatfirst,don’tworry,it’sconfusingtoeveryone.Recursionisamethod
ofsolvingproblemsbybreakingtheproblemupintosmallerandsmallerpiecesuntilitcan
beeasilysolved.Thisisachievedwithafunctionthatcallsitself.Anyproblemthatcanbe
solvedrecursivelycanalsobesolvediteratively,howeverincertaincases,recursionoffersa
moreelegantsolution.
Arecursivealgorithmmustfollowthethreelawsofrecursion:
“1.Arecursivealgorithmmusthaveabasecase.
2.Arecursivealgorithmmustchangeitsstateandmovetowardthebasecase.
3.Arecursivealgorithmmustcallitself,recursively.”
Let’sgooveranexampleofarecursivefunctionthathasallthree,afunctiontoprint
outthelyricstothepopularchildren’ssong“99BottlesofBeerontheWall”19:
defbottles_of_beer(bob):
"""Userecursiontoprintthebottlesofbeersong.
:parambob:Integernumberofbeersthatarestartonthewall.
"""
ifbob<1:
print"Nomorebottlesofbeeronthewall.Nomorebottlesofbeer."
return
tmp=bob
bob-=1
print"{}bottlesofbeeronthewall.{}bottlesofbeer.Takeonedown,passitaround,
{}bottlesofbeeronthewall.".format(tmp,tmp,bob)
bottles_of_beer(bob)
bottles_of_beer(99)
>>99bottlesofbeeronthewall.99bottlesofbeer.Takeonedown,passitaround,
98bottlesofbeeronthewall.
>>98bottlesofbeeronthewall.98bottlesofbeer.Takeonedown,passitaround,
97bottlesofbeeronthewall.
…
>>Nomorebottlesofbeeronthewall.Nomorebottlesofbeer.
Inthisexample,thebasecaseis:
ifbob<1:
print"Nomorebottlesofbeeronthewall.Nomorebottlesofbeer."
return
Thebasecaseofarecursivealgorithmiswhatfinallystopsthealgorithmfromrunning.
Ifyouhavearecursivealgorithmwithoutabasecase,itwillcontinuetocallitselfforever,
andyouwillgetaruntimeerrorsayingthemaximumrecursiondepthhasbeenexceeded.The
line:
bob-=1
satisfiesoursecondrulethatwemustmovetowardourbasecase.Inourexample,wepassed
inthenumber“99”toourfunctionastheparameterbob.Sinceourbasecaseisbobbeingless
than1,andbecausebobstartsat99,withoutthislinewewillneverreachourbasecase,which
willalsogiveusamaximumrecursiondepthruntimeerror.
Ourfinalruleissatisfiedwith:
bottles_of_beer(bob)
Withthisline,wearecallingourfunction.Thefunctionwillgetcalledagain,butwithone
importantdifference.Insteadofpassingin99likethefirsttimethefunctionwascalled,this
time98willbepassedin,becauseofrulenumbertwo.Thethirdthefunctioniscalled97will
becalled.Thiswillcontinuetohappenuntileventuallybobisequalto0,andwehitourbase
case.Atthatpointweprint“Nomorebottlesofbeeronthewall.Nomorebottlesofbeer.”
andreturn,signalingthealgorithmtostop.
Let’sgooveronemorerecursivealgorithm.Sayyouaregiventhefollowingproblem:
Givenanon-negativeintegernum,repeatedlyaddallitsdigitsuntiltheresulthasonly
onedigit.
Forexample:
Givennum=38,theprocessislike:3+8=11,1+1=2.Since2hasonlyonedigit,
returnit.
Onewaywecansolvethisproblemisusingrecursion:
defadd_digits(number):
"""
:paramnumber:Int
:return:Singledigitint
"""
number=str(number)
iflen(number)==1:
returnint(number)
the_sum=0
forcinnumber:
the_sum+=int(c)
returnadd_digits(the_sum)
printadd_digits(99)
>>9
Inthisexample,ourfunctionacceptsanumberasaparameter,calculatesitssumandcalls
itselfuntilithitsthebasecasewhichisanumberwithonlyonedigit.Ifwepassinthenumber
ninetynine,ithastwodigitssoitfailsthebasecase.Wethencalculatethatthe_sumequals18,
sowecalladd_digitsagainandpassin18.Onceagain18doesnotpassourbasecasesowe
calculatethesumofthedigitswhichthistimeis9,andcalladd_digitswith9asaparameter.
Thethirdtimearound,ournumberis9andsinceit’sonlyonedigit,itdoessatisfyourbase
caseandwereturntheanswer.
AbstractDataTypes
Whenlearningaboutdatastructures,youwillcomeacrossthetermabstractdatatype.I
rememberwhatanabstractdatatypeisbythinkingoftherelationshipbetweenanabstractdata
typeandadatastructureassimilar(althoughnotexactlythesame)astherelationshipbetween
aclassandanobject.
Ifyouwanttomodelanorangeinobject-orientedprogramming,aclassrepresentsthe
ideaofanorange,whereastheobjectrepresentstheactualorange.Similarly,anabstractdata
typerepresentstheideaofacertaintypeofdatastructure.Anexampleofanabstractdatatype
isalist.Alistisanabstractdatatypethatcanbeimplementedseveralwayswithdata
structuressuchasanarray,oralinkedlist.
Nodes
NodeisatermusedfrequentlyinComputerScience.Youcanthinkofanodeasapointona
graph.Theinternetismadeupofroutersthatcommunicatewitheachother.Eachrouterisa
nodeinthenetwork.Nodesareusedinseveraldatastructures,includinglinkedlists,treesand
graphs.
Stacks
Astackisalast-in-first-outdatastructure.Thisisbestenvisionedasastackofdishes.
Sayyoustackfivedishesontopofeachother.Inordertogettothelastdishinthestack,you
havetoremovealloftheotherdishes.Thisishowastackdatastructureworks.Youputdata
intoastack.Everypieceofdataislikeadish,andyoucanonlyaccessthedatabypullingout
thedataatthetopofthestack.HereisanexampleofastackimplementedinPython:
classStack:
def__init__(self):
self.items=[]
defisEmpty(self):
returnself.items==[]
defpush(self,item):
self.items.append(item)
defpop(self):
returnself.items.pop()
defpeek(self):
returnself.items[len(self.items)-1]
defsize(self):
returnlen(self.items)
Thetwomostimportantmethodsinthedefinitionarepushandpop.Pushletsyouput
dataontopofthestack,andpopletsyoutakeitoffthestack.
Sowhatarestacksusedfor?Wellfirstofall,stacksareimportantforunderstanding
recursion.Recursionisafundamentalpartofprogrammingwegooverinalatersection.
Mostnewprogrammersstrugglewithrecursion,butthekeytounderstandingrecursionisto
deeplyunderstandhowastackworks.
Furthermore,stacksareusedtoreversethings.Whateveryouputonastackcomesout
inreverseorderwhenyoutakeitoff.Forexample,let’ssayyouwanttoreverseastring.We
couldreversethestringbyfirstputtinginonastack,andthentakingitoff,likethis:
fromcollectionsimportstackass
my_string=“Hello”
stack=s()
forcinmy_string:
stack.push(c)
new_string=“”
forcinstack:
new_string+=c
printnew_string
>>>olleH
Inthisexamplewewentthrougheachcharacterintheword“Hello”,andputitonourstack.
Thenweweiteratedthroughourstack,andtookeverythingwejustputonthestack,offofit,
andsavedtheorderinthevariablenew_string.Bythetimewegettothelastline,ourwordis
reversedandourprogramprints“olleH”.
I’mgoingtoshareanotherexampleofusingastack.It’sarareexampleofaquestion
I’vebeenaskedtosolveinaninterview,andactuallyhaveusedonthejob.Theproblemis,
writeaprogramthattestsastringforbalancedparentheses.So,forexample,“(hello)”would
pass,but“(hello”wouldfail.“()()()”wouldpass,but“()(”wouldfail.Thislookseasyatfirst,
untilwegetsomethinglikethis“((()((()((()((((()))))”.
Howareyousupposedtokeeptrackofalltheparenthesis?Thekeytosolvingthis
problemistouseastack.Everytimewecomeacrossanopenparen,weputitonastack.Ifwe
comeacrossaclosedparen,wepullanopenparenoffthestack.
defbalanced_paren(expression):
stack=[]
forcinexpression:
ifc=='(':
stack.append(c)
elifc==')':
iflen(stack)<1:
returnFalse
stack.pop()
iflen(stack)==0:
returnTrue
returnFalse
Iftheparenthesisarebalanced,ourstackwillbeemptyafterourloop,andwecanreturn
“True.”Onethingweneedtowatchoutfor,iftheparenthesisareunbalanced,wewilltryto
popfromanemptystackwhichwillcauseanerror.Thatiswhywhenwecomeacrossa
closedparen,wehavetofirstmakesurethestackisnotemptybeforewepopitoffthestack.
Ifwecomeacrossaclosedparen,andthestackisempty,weknowtheparenthesisarenot
balancedandwecanreturn“False.”Ifattheendofthelooptherearestillopenparenthesison
thestack,wecanalsoreturn“False.”
LinkedLists
Alinkedlistismadeupofaseriesofnodes,witheachnodepointingtothenextnodein
thelist.MyfriendStevegaveagreatmetaphorforthinkingaboutlinkedlists.Imagineyou
areintheMafiaandwanttogiveordersinsuchawaythatnooneknowswhoyouare.You
couldsetupastructurewhereyougiveanorderanonymouslytothenextpersondownthe
chainofcommand.Youknowwhotheyarebuttheydon’tknowyou,theyonlyknowthenext
persontheyshouldgivetheorderto.Thepersontheygivethenextordertodoesn’tknow
them,butonlyknowsthenextpersontoreceivetheorder.Thischainofinformationiswhata
singlylinkedlistis.Inasinglylinkedlist,allofthenodesinthelistonlyknowaboutthenext
node.Theydon’tkeeptrackofthenodebehindthem.Youcangettoeverypieceofdataina
linkedlistbystartingattheheadofthelist,andmovingonebyonetoeachnextnode.A
doublylinkedlististhesamethingexcepteachnodekeepstrackofthenodebehindit,in
additiontokeepingtrackofthenextnode.Alinkedlistcanalsobeorderedorunordered.In
thissection,wewillimplementanunorderedsinglyanddoublylinkedlist:
classNode:
"""Classrepresentingonenodeinalinkedlist."""
def__init__(self,data):
self.data=Node(data)
self.next=None
classLinkedList:
"""Classrepresentingalinkedlistdatastructure"""
def__init__(self,head):
self.head=head
defadd(self,data):
"""Addanewnodetothelinkedlist."""
previous_head=self.head
self.head=Node(data)
self.head.next=previous_head
Ourlinkedlistclassissimple.Itstorestheheadofthelinkedlistandhasamethodcalled
“add”toaddanewnodetothelist.Sincethelistisunordered,wedon’tcarewhereweputthe
nextnode,soit’seasiesttoputitattheheadofthelist.Themethodaddstoresthecurrenthead
intheprevious_headvariable,sowedon’tlosetrackofit,createsanewnodewiththepassed
indata,andsetsthenewnodeastheheadofthelist.Thenodethatusedtobetheheadisthen
setasthenextnodeafterthenewhead.Nowwecancreatealinkedlistandadddatatoit:
linked_list=LinkedList()
linked_list.add(1)
linked_list.add(2)
linked_list.add(3)
Togetthedatafromourlinkedlist,westartwiththehead,andvisiteverynodeuntilwehita
nextnodethatequalsnone.
node=linked_list.head
whilenode:
printnode.data
node=node.next
>>3
>>2
>>1
Wecanchangeoursinglylinkedlisttoadoublylinkedlistbykeepingchangingournode
classtokeeptrackofthenodebehindit.
classNode:
"""Classrepresentingonenodeinalinkedlist."""
def__init__(self,data):
self.data=data
self.next=None
self.previous=None
classLinkedList:
"""Classrepresentingalinkedlistdatastructure"""
def__init__(self,data):
self.head=Node(data)
defadd(self,data):
"""Addanewnodetothelinkedlist."""
previous_head=self.head
self.head=Node(data)
previous_head.previous=self.head
self.head.next=previous_head
Ournewlinkedlististhesameasourpreviousone,exceptnoweverynode
knowsthenodeinfrontandbackofit.
Arrays
Anarrayisanimplementationofthelistabstractdatatype.Everypieceofdatainan
arrayhastobethesametype.Soforexample,youcanhaveanarraymadeupofstringsor
intsbutyoucannothaveanarraymadeupofboth.Python’sbuiltinlistdatastructureis
implementedinternallyinCasanarrayofpointers.
BinaryTrees
Atreeisanotherdatastructurethatgetsitsnamefromlookinglike,youguessedit,a
tree.Agreatexampleofatreedatastructureisthefilesystemonyourcomputer,whichis
implementedusingatree.Therearemanydifferentkindsoftreessuchasredandblacktrees,
AVLtreesandbinarytrees.Inthissection,wewillbuildabinarytree.
Abinarytreeismadeupofnodescontainingdata.Eachnodecanhavealeftchildanda
rightchild.Inthesamewayalinkedlistpointstothenextelementinthelist,anodecanpoint
totwoothernodescalledtheleftandrightchild.Likeadoublylinkedlist,theleftandright
childkeeptrackoftheirparent.HereisanexampleofabinarytreeimplementedinPython:
classNode:
"""Binarytreenode."""
def__init__(self,value):
self.value=value
self.left_child=None
self.right_child=None
classBinaryTree:
"""Thisclassrepresentsabinarytreedatastructure."""
def__init__(self,root):
"""
:paramroot:Binarytreenode.
"""
self.root=root
Ourbinarytreeissimplymadeupofaroot.Seenextsectionforatreewithchildren.
BreadthFirst&DepthFirstSearch
Ifwewanttovisiteverynodeinabinarytree,therearetwosearchalgorithmswecan
use:breadthfirstsearch,anddepthfirstsearch.Ifyouthinkofatreeofbeingmadeupof
rowsandcolumns,inabreadthfirstsearchwevisiteachrowonebyone,whereasindepth
firstsearchwevisiteachcolumnonebyone.Wewillusethebinarytreefromtheprevious
exampletocreateourtreetotraverse:
tree=BinaryTree(“a”)
tree.left=TreeNode(“b”)
tree.right=TreeNode(“c”)
tree.left.left=TreeNode(“d”)
tree.right.right=TreeNode(“e”)
Nowwecanwriteafunctionthattakesatreeasaparameteranddoesabreadthfirstsearchof
thetree,printingoutthevalueofeachnodeitvisits:
defbreadth_first(tree):
"""Breadthfirstsearchofbinarytreeprintouteachnode.
:paramroot:BinaryTree
"""
current_level=[root]
next_level=[]
whilecurrent_level:
fornodeincurrent_level:
printnode.val
ifnode.left_child:
next_level.append(node.left_child)
ifnode.right_child:
next_level.append(node.right_child)
current_level=next_level
next_level=[]
Weusethelistcurrent_leveltokeeptrackofallthenodesinthelevelofthetreeweare
currentlyin.Whenthereisnomorecurrentlevel,ouralgorithmstops.Ourtreelookslike
this:
a
/\
bc
/\
de
“current_level”willstartas[a],become[b,c],become[d,e]andfinallybecomeanemptylist
[]atwhichpointouralgorithmisfinished.Weareabletodothisbykeepingtrackofthenext
levelofnodesinourlistnext_level.
Beforeourwhileloop,weaddtherootofourtreetocurrent_level,thentellourwhile
looptocontinue,aslongascurrent_levelisn’tempty.Inourwhileloop,weiteratethrough
everynodeincurrent_level,printingouteachnode.Thenwecheckifthenodehasany
children.Ifitdoes,weaddthosechildrentoournext_levellist.Attheendofourforloop,we
switchthetwolists,sothecurrent_levellistissettothenext_levellist,andthenext_levellist
becomesempty.Thisiswhatallowsustomovefromoneleveltothenext.Eventually,when
wereachthelastlevel,thenodeswillnothaveanychildren,next_levelwillbeempty,
current_levelwillbesettonext_level,andbecauseit’semptythealgorithmwillstop.
Adepthfirstsearchsearchesthetreeverticallyinsteadoftraversingacrossit
horizontally.Ourtreefromthepreviousexamplewouldbesearched“a”,“b”,“d”,“e”,“c”.
Wecanimplementdepthfirstsearchusingrecursion:
HashTables
InthechapterContainers,wecoveredPython’sbuiltindictionarydatatype.Dictionaries
arehelpfulbecausetheycanstorekeysandvaluesandareincrediblyfastatgettingandsetting
data.Torecap,dictionariesmapkeystovalues.Forinstanceyoucouldaddthekey
“super_computer”toadictionarywiththevalue“Watson”withthefollowingcode:
my_dictionary={}
my_dictionary[“super_computer”]=“Watson”
Nowwecanretrievethekey“super_computer”inwith:
printmy_dictionary[“super_computer”]
>>>Watson
Theamazingthingaboutdictionariesistheycansetandgetdatainconstanttime.Itdoesn’t
matterhowmanyrowsofdatawehaveinourdictionary.Wecouldhaveonebillionrows,and
stilladdandretrievethevaluefor“super_computer”toourdictionaryinO(1)time.
Internally,Pythonusesahashtabletoimplementitsdictionary.Ahashtableisadata
structurethatusesalistandahashfunctiontostoredatainO(1)time.Whenyouaddavalue
toahashtable,itusesahashfunctiontocomeupwithanindexinthelisttostorethedata.
Whenyouretrievedatafromahashtable,itusesthesamehashfunctiontofindtheindexsoit
canretrieveitfromthelist.Inthisexample,ourhashtableisonlygoingtostorenumbers.
Thehashfunctionwillreturntheresultofthenumbermoduloeleven.Soforexample,our
hashfunctionforonewouldreturnone,sowestorethenumberoneatindexoneinourlist.
Ourhashfunctionforthenumberfivewouldreturnfive,andsowewouldstorethenumber
fiveatindexfiveinourlist.Hereisanexampleofahashtable:
classHashTable:
"""Hashtabledatastructure"""
def__init__(self):
self.list=[None]*11
@staticmethod
defhash(n):
"""
:paramn:int
:return:returnindexinlisttostorenumber.
"""
returnn%11
defset(self,n,v):
"""
:paramn:int
:paramv:canbeanytype.
"""
self.list[self.hash(n)]=v
defget(self,n):
"""
:paramn:int
:return:intvaluefromlist
"""
returnself.list[hash(n)]
hash_table=HashTable()
hash_table.set(1,'Disrupted')
hash_table.set(5,'HubSpot')
print(hash_table.get(1))
print(hash_table.get(5))
Thisisanoversimplifiedexamplethatclearlyhasproblems.However,thegoalistoillustrate
howahashtableworks.bc
Challenge
ChapterX.RelationalDatabaseDesign
Whenyoucreateusearelationaldatabase,youhavetodesignthedifferenttablesyour
databasewillhave,howthetablesrelatetoeachother,whatcolumnstheywillhave,andwhat
constraintsareputonthosecolumns.Together,thismakesupyourdatabaseschema.Inthis
section,wearegoingtotodesignaschematostoredataforawebsitelikeAmazon.com.
First,weneedtothinkaboutthedataAmazonneedstostore.Thefirstthingthatcomes
tomindisproducts;Amazonclearlymusthaveadatabasewheretheystorealloftheir
products.Amazonalsohastokeeptrackofcustomers—youdon’thavetoregisteranew
accounteverytimeyouordersomethingonAmazon,sotheymuststoretheircustomers
informationaswell.Acustomermightordermorethanoneproduct,soAmazonmustalso
haveawaytostoreorders.Let’sstartbydesigningatabletoholddataaboutproducts:
product
_______________________
id|name|price
1|ThePragmaticProgrammer|14.99
Ourproducttablehasacolumncalledidthatservesastheprimarykey,andcolumnsforthe
nameoftheproductandtheprice.Ourprimaryidisanintegerthatautoincrements,thename
columnacceptsstrings,andthepricecolumnacceptsintegers.Thedatashowninthetable
suchas1,“ThePragmaticProgrammer,”and14.99arenotpartofthedatabasedesign,butare
anexampleofhowdatawouldlookinourtable.Thisconventionisusedthroughoutthis
chapter.
Notethatwhenyoudesignadatabaseschema,youwanttopickanamingconvention
andsticktoit.Inthefollowingthiscasewewilluselowercaselettersandanunderscorein
betweenwords.Nowweneedatabletokeeptrackofourcustomers.Ourcustomertableis
verysimilartoourproducttable:
Customer
id|first_name|last_name
______________
1|Steve|Smith
Ourcustomertablehasaprimarykey,andtwocolumnsthatacceptstrings.Thisisallweneed
tokeeptrackofourcustomers.
Wearegoingtokeeptrackofourordersusingtwodifferenttables.Thefirsttablewill
mapanorderidtoaspecificcustomer,andthesecondtablewillkeeptrackoftheproductsin
eachorder.Hereisourfirsttableshownalongwiththecustomertable:
customer
id|first_name|last_name
______________
1|Steve|Smith
order
_______
id|customer
1|1
Ourordertablehasaprimarykeycalledidandacolumncalledcustomer.Ourcustomer
columnisdifferentthantherestofthecolumnswe’veseensofarbecauseitusesaconstraint
calledaforeignkey(coveredinthechapterSQL).Thecustomercolumnofourordertable
acceptsanintegerthatrepresentstheprimarykeyofacustomerinourcustomertable.Inour
example,thefirstentryintheordertablecustomercolumnis1.Ifwelookuptherowwith1
asitsprimarykeyininourcustomertablewewouldgettherow“1|Steve|Smith”.By
usingaforeignkey,we’vesuccessfullylinkedourordertabletoourcustomertable.Thisis
calledcreatingarelationship.
Imagineifwedecidedtoputtheinformationfromour“customer”tableinthe“order”
tableinstead:
order
id|username|order
_____________
1|Steve|NoSQLDistilled
2|Cory|ThinkPython
3|Steve|TheTalentCode
Theproblemwiththisdesignisthatdataisduplicatedinourtable.TheusernameSteveis
repeatedtwice.IfweneededtochangeSteve’susernameto“Steven,”wemightaccidentally
onlychangethenameinthefirstrow,andforgettochangeitinthethird.Thiswouldcorrupt
ourdata:
order
id|username|order
_____________
1|Steven|NoSQLDistilled
2|Cory|ThinkPython
3|Steve|TheTalentCode
Inouroriginaldesignthisisnotpossible.Takeanotherlookatourpreviousdesign:
customer
id|username
_________
8|Cory
9|Steve
orders
id|username|order
_________________
1|9|NoSQLDistilled
2|8|ThinkPython
3|9|TheTalentCode
Whenweneedtochangeausername,weonlyhavetochangeitinoneplace—thecustomer
table.Oncewechangethenameinourcustomertable,anyonelookingupusernamewitha
foreignkeyof9willseethecustomer ’susernameisSteve.Thereisnochanceofaccidentally
corruptingthedatabecauseitonlyexistsinonelocation.
Tablescanhavethreetypesofrelationships:onetoone,onetomanyandmanytomany.
Thisisanexampleofaonetoonerelationship.Youcreatebothaonetoonerelationshipand
amanytoonerelationshipusingaforeignkey.Thedifferenceis,inaonetoonerelationship,
bothtablescanhaveforeignkeystoeachother,althoughlikeinthiscase,theydon’thaveto.
Inaonetomanyrelationship,onlythemanysidehasaforeignkeylinkingittotheone.This
isnotsomethingyourdatabaseknowsabout,butratheraconstructinventedtohelpyou
designdatabases.
Inthisexample,acustomercanhavemanyorders,butanordercannothavemany
customers.Anotherexampleofamanytoonerelationshipisaclassroom.Ateachcanhave
manyclasses,butaclasscannothavemanyteachers.Inaonetoonerelationshiphowever,the
relationshipcangobothways.Onepersonhasonepassport,andonepassporthasoneperson.
Thefinalrelationshiptablescanhaveiscalledmanytomany.Inordertodothatweneedto
createajunctiontable,whichweneedtodoinordertocompleteourAmazondesign.Our
finaltableorder_itemwillkeeptrackofproducts:
order_item
__________
id|order_id|product_id
1|1|1
Thistablehasidasaprimarykey,andtwoforeignkeys—order_idandproduct_idlinking
thetabletoourorderandproducttable.Ourdesigniscomplete,wecanstoreandlookupall
oftheinformationweneedtofulfillanorder.Hereareallofourtablestogether:
product
_______________________
id|name|price
1|ThePragmaticProgrammer|14.99
customer
id|first_name|last_name
______________
1|Steve|Smith
order
_______
id|customer
1|1
order_item
__________
id|order_id|product_id
1|1|1
2|1|1
Ifwearereadytoshipanorder,wecangetalloftheinformationweneedbylookingatthe
order_itemtableandusingthatinformationtoqueryothertables.Firstwewouldselectallthe
rowsfromourorder_idwithanorder_idof1.Thenwewouldlookupalltheproductsusing
theproduct_idineachrow.Finally,wewouldusetheorder_idkeytolookupthenameofthe
foreigncustomerkeyinourordertable,andlookupthecustomer'snameinthecustomer
tableusingthatinformation.
Normalization
Oneofthechallengesyoufacewhenworkingwithadatabaseismaintainingdata
integrity,whichmeans“assuringtheaccuracyandconsistencyofdataoveritsentirelife-
cycle”49.Normalizationandreferentialintegrityaresomeoftheconceptsthathelpensure
dataintegrity.
Datanormalizationistheprocessofdesigningarelationaldatabaseinordertoreduce
dataredundancy,whichcanleadtoinaccuratedata.Whiletherearemanyrulesfordata
normalization,therearethreespecificrulesthateverydatabaseshouldfollow.Eachofthese
rulesiscalleda“normal“form.”Ifthefirstruleisfollowed,thedatabaseisin"firstnormal
form"or1nf.Ifallthreerulesarefollowed,thedatabaseisin"thirdnormalform"or3nf.52
Inordertoreacheachsuccessivelevelofnormalization,allofthepreviousrulesmustbe
followed.Inotherwords,iftherulefor2nfissatisfied,but1nfisnot,thedatabaseisnot
considered2nf.
Toreachthefirstnormalform,youneedtoavoidduplicatingdatainmultiplerow,
avoidstoringmorethanonepieceofinformationinarowandthetablemusthaveaprimary
key.Hereisanexampleofstoringduplicatedata:
t-shirt
____________
color|color
blue|blue
Andanexampleofstoringmorethanonepieceofdatainonerow:
t-shirt
______
color
blue,large
Inthisexampleweareusingacommatostoretwopiecesofdatainonecolumn—“blue”and
“large.”Thisissomethingyoushouldneverdo.Furthermore,neitheroftheseexamplesare
1nfbecausetheydonothaveaprimarykey.
Hereisanexampleofatablethatis1nf:
t-shirt
primary_key=id
_____________
id|color
1blue
Inorderforatabletobe2nf,allnonprimarykeycolumnsmustrelatetotheprimarykey.
Let’slookatanexamplethatviolates2nf:
t-shirt
primary_key=item
primary_key=color
__________
item|color|price|tax
t-shirtred19.99.90
t-shirtblue18.00.78
poloyellow321.4
pologreen401.8
poloorange432
Thistableisnot2nfbecausethetwocolumnsthatarenotprimarykeys,priceandtaxrelateto
item,butdonotrelatetocolor.
dealership
________
id|location|available
1Portland|Yes
53
Normalizationisanimportantpartofdatabasedesign.Whilethereevenmore
normalizationruleswedidnotcover,itisimportanttoalwaysnormalizeyourdatabaseto
3nf.Tohelpyouremembertherules,programmersoftenusethephrase“Thedatadepends
onthekey[1NF],thewholekey[2NF]andnothingbutthekey[3NF]sohelpmeCodd”
(Codd,mentionedearlier,isthecreatorofrelationaldatabases)tohelpthemrememberthe
rulesofofnormalization.
ReferentialIntegrity
Referentialintegrityisanotherwayofensuringdataintegrity.Itisameasureof
consistencyinadatabase.Ifforexamplewehavethefollowingtables:
customer
id|username
_________
8|Cory
9|Steve
order
id|username|order
_________________
1|9|NoSQLDistilled
andwedeletethesecondrowfromthecustomertable:
customer
id|username
_________
8|Cory
order
id|username|order
_________________
1|9|NoSQLDistilled
Ourusernamecolumninourordertablereferencesaforeignidthatnolongerexists.Thisis
aviolationofreferentialintegrity.Fortunately,yourdatabasemanagesreferentialintegrity
foryou.Ifyoutrytodothisinarelationaldatabase,itwon’tletyou,youwillgetanerror.
Indexing
Youcanindexacolumninatableinordertomakereadsfaster.Indexesworklikea
telephonebooksortedbylastname.Ifyouwerelookingthroughsuchatelephonebook,you
wouldn’tlookthrougheverysingleentry,youwouldimmediatelyskiptothesectionofthe
phonebookthatmatchesthelastnameofthepersonyouwerelookingfor.Thisiswhatan
indexdoes.Whenyouindexacolumninadatabase,internallythedatabaseduplicatesthedata
inthecolumn,butarrangesitinaspecificorder—alphabeticallyforexample,thatallowsit
tolookupdatafaster.HereisanexampleofcreatinganindexwithSQL:
CREATEINDEXmy_index
ONtable_name(customers)
Thiswillcauseourdatabasetointernallyduplicateallofthedatainourcustomertableand
arrangeitalphabetically.Nowwecanlookupcustomersmuchfaster.Thedrawbackto
creatinganindexisthatduplicatingdataandorganizingithasacost—itincreasesthetimeit
takestowritetoyourdatabase.
Challenge
ChapterX.ComputerArchitecture
“Thereare10kindsofpeopleintheworld—thosewhounderstandbinaryandthosewho
don't.”
—Anonymous
Nowthatwe’vecoveredthefundamentalsofprogramming—andsometoolsto
programmoreeffectively—wearegoingtogooversomeofthebasicsofComputerScience.
ComputerScienceisusuallytaughtasabstruselyaspossible—soI’veattemptedtomakeitas
friendlyaspossible—whilefocusingonthemostpracticalparts.Inthischapterwetakealook
underthehoodofwhatwe’velearnedsofar—weexplorehowPython,youroperatingsystem
andcomputerwork.
HowYourComputerWorks
Computerscanonlyunderstandbinary,soinordertounderstandhowacomputerworks,
youshouldhaveabasicunderstandingofbinary.Whenyounormallycount,youcountin
baseten,whichmeanswerepresenteverynumberintheworldusingonlytendigits.The
“base”ofacountingsystemisthenumberofnumbersusedtorepresentallthenumberinthe
world.Inthebase10countingsystem,oncewegetoverthenumbernine,werecombine
numbersfromonetotentocreatenewnumbers:
0123456789
11121314151617181920
Basetwoisasystemofcountingjustlikebaseten.However,insteadofcombining
existingnumbersaftertennumbers,basetwostartsdoingitaftertwonumbers
0->0
1->1
10->2
11->3
0and1arethesameasbaseten.However,oncewegetto2,we’vegonepasttwonumbers,
andweneedtocombineourfirsttwonumberstocreateanewnumber.Henceoneandzero
are“10”iscombinedtorepresent2.Eachnumberstartingfromtheleftrepresentswhetheror
notthereisapowerof2.Soforexample,“10”meanstherearezero2**0’sand12**1:
10
2+0=2
Yourcomputerismadeupofhardware—aCPU,memoryandinput/outputdevices.All
computingisdonewiththesephysicalpiecesofhardware.Hardwareonlyunderstandsbinary.
Thatmeansnomatterwhatprogramminglanguageyouuse,nomatterwhatoperatingsystem
youuse,everyinstructionacomputereverexecutesisinbinary.
[explainrepresentingthingsinbinary]
Thischartshowshoweachcharacterismappedtoanumber.Forexample,97ismappedto
theletter“a”.AprogramusingASCIIwouldstorean“a”as1100001,or97inbinary.When
thesameprogramneedstoretrievetheletter“a”,itlooksup97intheASCIItableandseesit
representstheletter“a”.
YoucanusePythontoeasilygetthenumberacharactermapstoinASCII.Thefunction
“ord”takesacharacterandreturnsthenumberitmapsto:
ord(“a”)
>>97
ord(“z”)
>>122
AtahighlevelacomputerismadeupofaCPUandmemory.TheCPUisthepartofa
computerthatexecutestheinstructionsprovidedbyaprogram.Acomputercanhaveoneor
moreCPUS:acomputerwithmultipleCPUsissaidtohaveamulti-coreprocessor.ACPU
hasaclockgeneratorthatproduces“clockcycles.”InaCISC,orComplexInstructionSet
processor,eachinstructiontakesoneormore“clockcycles,”whereasRISC,orReduced
InstructionSetComputingprocessors,arefaster,abletoexecutemultipleinstructionsper
clockcycle.Clockspeedreferstothespeedamicroprocessorexecutesinstructions.
pythonversionofjavaCPUprograminStructuredComputerOrganization
Computershavetwotypesofmemory,RAMandROM.RAMstandsforrandomaccess
memoryandisvolatile,meaningitiserasedwhenacomputerturnsoff.Allarithmeticand
logicoperationstakeplaceinram.ROMstandsforreadonlymemory.Itisnotvolatile;it
persistsafteracomputeristurnedoffandisusedforthemostfundamentalpartsofan
operatingsystemneededwhenacomputerstartsup.
Everythinginmemoryisstoredinbinary,sequencesofzerosandones.Thezerosand
onesarestoredinchipsmadeupofmillionsorbillionsoftransistors.Eachtransistorcanbe
turnedonoroffusingaflowofelectricity.Whenatransistoristurnedoffitrepresentsazero
inbinary,andwhenatransistoristurnedonitrepresentsaone.
Memoryisnotwhereeverythingonacomputerisstored,despitethenamesounding
likeitis.MemoryisusedincombinationwiththeCPUtoexecutetasks,likearithmetic
operations.Whenyoustoreinformationinadatabaseforexample,itiswrittentodisk
storage,notmemory.
I/O
TheCPUcombinedwithmemorymakeupisthebrainofacomputer.I/Oisthetransfer
ofdatatoorfromthebrainbyotherdevices.Readingandwritingdatafromdiskstorageis
anexampleofI/O.
Aswelearnedearlier,computersonlyunderstandbinary.Sohowdocomputerswork
withcharacters?TheanswerischaracterencodingschemeslikeASCIIandUnicode.Toa
computer,charactersdonotexist.Charactersarerepresentedusingencodingschemeslike
ASCIItomapcharacterstobinarynumbers.
HowProgrammingLanguagesWork
Machinecodeiscodemadeupentirelyofbinary(orhexwhichisbase16butignore
thatfornow)thatcanbeexecuteddirectlybyyourcomputer ’shardware.Youcanprogram
everyPythonprogramwe’vewrittensofarinbinary.
ComputerScientistsuseaconceptcalledabstractiontomanagecomplexity.When
somethingisanabstraction,itmeansyoucanuseitwithoutneedingtounderstandhowit
works.Whenyouprogram,youareprogrammingbeneathseverallayersofabstraction:you
canprograminPythonwithoutknowinghowPythoncommunicateswithyouroperating
system,howyouroperatingsystemcommunicateswithyourcomputer ’shardware,orhow
yourcomputer ’shardwareexecutesbinary.
Programminglanguagesarebuiltonabstractions.Machinecodeisthelowestlevel
programminglanguage—thatmeansithasnoabstractionsbeneathit.Aprogramming
languagebecomeshigher-levelthefurtherawayitgetsfrommachinecode(thedirect
instructionsexecutedbytheCPU).Thehigher-levelaprogramminglanguageis—themore
abstractionsithasbeneathit.Themoreabstractionsaprogramminglanguagehasbeneathit
thesloweritruns.IfyoucanwriteanyPythonprograminmachinecode—andtheprograms
willrunfaster—youmightbewondering,“Whydon’tprogrammerswritealloftheir
programsinmachinecode?”Thereasonprogrammersdon’twriteprogramsinmachine
codeisbecauseitisinsanelytedious.Peopledon’tthinklikecomputers—theydon’tthink
purelyinnumbers—whichledtotheideaofwritingaprograminmachinecodethatcould
translateamorehumanreadablelanguageintomachinecode,sohumanscouldwriteina
languagethatwasmorenaturaltothem,andthisnewprogram(calledanassembler)would
automaticallytranslateassemblycodeittomachinecode(calledanassembler)—this
programminglanguageiscalledAssembly,anditisthefirstabstractionabovemachinecode
andtheprogramthattranslatesassemblycodetomachinecode.Hereisanexampleofa
programaddingtwonumbersandstoringtheminavariableinmachinecode:
addmachinecodeexample
HereisthesameprogramwritteninAssembly:
addassemblycodeexample
Asyoucansee,theAssemblylanguagecoderepresentsthewayhumanthinksmuchthanthe
machinecode;however,whileassemblycodeiseasiertoread,theinstructionsstillareoneto
onewithmachinecode;thatis,everylineofassemblycodetranslatestoexactlyonelineof
machinecode—soprogramminginassemblyisstilltediousbecauseitrequiresthe
programmertowritesomanylinesofcodetogetanythingdone.Thisledprogrammersto
developevenhigherlevelprogramminglanguages—likeC—withinstructionsthatarenot
onetoonewithmachinelanguageandthuslettheprogrammerwriteprogramswithfewer
linesofcode.Ccodeisfirsttranslatedintoassemblycode,andthenitistranslatedinto
machinecode.Herethesameprogramaddingtwonumbersandstoringtheminavariable
fromthepreviousexamplesinC:
intx
x=2+2
ThisissimilartohowwewouldwritethisprograminPython,althoughtherearetwosteps
insteadofone.InC,unlikePython,UnlikePython—inC—youhavetodeclarewhattypea
variableisbeforeyouuseit.Thisisaconceptcalledstatictyping,whichwecoverlater.
Althoughthiscodelooksfamiliar,CisalowerlevellanguagethanPython,whichisone
abstractionlevelaboveit.Ccodegetsconvertedtoassembly,andthentomachine,codebya
programcalledacompiler.Thedifferencebetweenanassemblerandacompileristhata
compilerincludesextrafunctionalitytooptimizeperformancewhenittranslatesahigher
levellanguagetomachinecode(compilersareusedineveryprogramminglanguagenotjust
C).Buttheconceptisstillthesame—ahighlevellanguageistranslatedtomachinecode.
WhenyouprograminC,youhavetomanagethememoryyourprogramusesyourself;
whenyouprograminPython—yourprogram’smemoryismanagedforyou,andyouhave
noaccesstoit.WhenyouprograminPython,youcancreatealist,andappendasmanyitems
tothatlistasyou’dlike.WhenyouprograminC,thisisnotpossible.Whenyoudefinean
arrayinC,youhavetoallocateacertainamountofmemorytoit.Youcan’tjustdefinean
array—youhavetoincludehowmanyitemswillbeinthatarray.Ifyoudefineanarraythat
canholdtenintegers—anddecideyouactuallyneedeleven—youhavetocreateabrandnew
array.InPython,aswe’veseen,yousimplycreatealistandappendasmanyitemstoitas
needed.
Pythonachievesthisbyaddingonemorelayerofabstraction.Pythoncodedoesnotget
compiledtomachinecodethewayCcodedoes.Instead,whenyourunaPythonprogram,itis
executedintwosteps.First,thePythoncompiler(writteninC)translatesPythoncodeto
bytecode—aspecialkindofcodeconsistingonlyofnumbers—butmeanttobeexecutedbya
virtualmachine(softwarethatemulateshardware).Python’svirtualmachineprogramisthen
executedbythehardware,anditgoeslinebylinethroughthebytecodeandexecuteseach
instruction.Thisdesignofferstwoimportantadvantages.Becauseofthis,youcanwritea
programinPythonwithoutworryingaboutmanagingmemory,andwhyyouca
ThereareseveraldifferentimplementationsofPython.TheversionofPythonweare
usingiscalledCPython.
WhenyouareprogramminginC,youhavetocompileyourprogrambeforeitwillrun.
TheCcompilertakesyourcodeandtranslatesitintomachinecodeyourcomputercan
understand.Oncecompiled,younolongerneedanysortofprogramtorunyourcode,you
canexecuteitdirectly.
Pythoncode,however,mustalwaysberunusingthe“python”program.Thisisbecause
Pythoniswhatisoftencalledan“interpreted”language,atermusedtodifferentiateitfroma
languagelikeC,whichiscalleda“compiled”language.Thisisconfusing,becausewhile
Pythonhasaninterpreter,italsohasacompiler.WhenyourunaPythonprogram,its
compilertranslatesyourcodetosomethingcalledbytecode,aspecialkindofcodethatislike
binarybutmeanttobeconsumedbyavirtualmachine.Atruntime,Python’svirtualmachine
translatesthebytecodeintomachinecodeandexecutesitlinebyline.
Thesetwoapproachesbothhaveadvantagesanddisadvantages.OneadvantageofC’s
approachisspeed—compilingdirectlytomachinecodemakesC’sprogramsrunfasterthan
Pythonprograms.C’sapproachalsoallowsforvariablestobestaticallytyped.This
eliminatesaclassoferrors,butalsohasdrawbackssuchasgivingprogrammersless
flexibility.Python’sapproachisadvantageousbecauseitallowsittobeplatform-independent.
Python’suseofaninterpreteralsoallowsitsvariablestobedynamicallytyped,whichmakes
programminginPythonmuchmoreflexiblethanC.
Programminglanguagescanbeeitherdynamicallyorstaticallytyped.PythonandRuby
areexamplesofdynamicallytypedlanguages.Inadynamicallytypedlanguage,youcan
declareavariableasonetype,andlaterchangethatsamevariabletoavalueofadifferent
type.Forexample,thisisallowedinPython:
x=5
x=“HelloWorld”
Inastaticallytypedlanguage,tryingtochangeavariablefromonetypetoanotherwill
causeanerrorbecauseinastaticallytypedlanguage;onceyoudeclareavariable’stype,you
cannotchangeit.Understandingthedifferencebetweenstaticallyanddynamicallytyped
languageswillpayoffwithhoursoffunarguingwithyourfriendswithComputerScience
degreesaboutwhetherornottheformerisbetterthanthelatter.
Yourcomputerismadeupofphysicalhardware.ThishardwareiswhatrunsourPython
programsanditonlyunderstandsonething—binary.Binaryisacountingsystem.Countingin
binaryisnodifferentthanwhenyounormallycount—whichiscalledcountinginbaseten—
exceptthereareonlytwodigitsinsteadoftendigits.Whenwecountinbaseten,westartat
zero,andwhenwegettoninewesayohno!Weranoutofdigits.Wesolvethisbytakinga
digitwealreadyhave—zero—andputtingitafterthefirstdigit(one)tocreatethenumberten.
Binaryworksthesameway.Zeroinbinaryiszero.Oneinbinaryisone.Inbinary,azeroor
oneiscalledabit—shortforbinarydigit.Inbinary,afterthenumberone,werunoutof
digits,andlikebaseten,wereusedigitswealreadyhave.Weaddazerototheendofthenext
number—two—whichbecomes10.Wedothesamethingforthree—whichbecomes11.
Countingthiswayiscalledcountinginbasetwo.
(binary)
base2base10
00
11
102
113
1004
1015
1106
1117
10008
10019
101010
101111
Icoverhowtoconvertanumberinbase10toitsbinaryequivalentlaterinthe
book.Fornow,justunderstandingwhatbinaryiswillsuffice,andallowsustoexplorea
fundamentalprogrammingconcept—datatypes,ortypesforshort.
Atthebeginningofthissectionwelearnedcomputersonlyunderstandbinary.Python
needstostorebothintegersandstringsinyourcomputersmemory(thepartofyour
computerthatsavesdataandcanonlystorebinary)—sohowcanPythonstoreintegersand
stringsinmemory?Wealreadywentoverhowthenumbertwoisrepresentedinbinary—soit
iseasytounderstandhowPythonstoresintegerslike2inyourcomputer’smemory—itjust
convertsthemtobinary.ButhowdoesyourPythonrepresentastringlike“z”whenittalksto
yourcomputer?Pythonrepresents“z”justlikethenumbertwo—inbinary(everythingisin
binary!).Toyourcomputer“z”is:
01111010
01111010isalsothenumber122inbinary.Sinceeverythingmustberepresentedinbinary,
Pythonrepresentsstrings(like“z”or“a”)withbinarynumbers,andithasatablethatmaps
eachbinarynumbertoacharacterinthealphabet.Thistableiscalledaunicodetable.Youcan
checkoutacoolexampleofaunicodetablehere:http://unicode-table.com/en/#0046(click
onaletterandthenclickonthelinkinthepopuptoseethebinary).01111010ismappedto
“z”.So01111010canrepresenteitherthenumber122or“z”.Whenyouputquotesaround
“z”,thequotesletPythonknowyouarerepresentingastringandnotanothertypelikean
integer.Ittakestheletterinquotesandlooksupintheunicodetablewhichbinarynumber
represents“z”andthenusesthatwhenittalkstoyourcomputer.Thatiswhytypeisso
important.Acomputercanonlyunderstandbinaryandsoprogramminglanguageslike
Pythonhavetodifferentiatebetweendifferenttypesofdata—likestringsandintegers—soit
canknowhowtorepresentthemtoyourcomputer.
HowYourOperatingSystemWorks
Anoperatingsystemisthesoftwarethatmanagesacomputer’shardware,allowingyou
touseyourcomputer.Yourcomputerhasalimitedamountofresourcessuchasmemoryand
CPU,andyouroperatingsystemdeterminestheresourceseachprogramreceives,alongwith
creatingastructureformanagingfiles,managingdifferentusersandmanagingother
commonoperationsneededbyprogrammers.
Thekernelisthemostfundamentalpartofanoperatingsystem,responsiblefor
allocatingresourceslikeCPUandmemorytodifferentprocesses.Processesareprograms
thatareexecuting.Thekernelassignsmemoryandastacktoeachnewprogramwhenitstarts
running.Thestateofthecurrentprocessissavedinadatastructurecalledaprocesscontrol
block.
Thekernelcannotbeaccesseddirectly,sothereisanotherlayerofsoftwarebuiltontop
ofthekernelinordertoaccessitcalledtheshell(becauseitisashellaroundthekernel).We
learnedhowtousetheshellinthechapterTheCommandLineinPartIII.Operatingsystems
haveotherresponsibilitiesotherthanresourcesharing,butinordertolimitthesizeofthe
kernel’scode,otheroperatingsystemjobs(calleddaemons)arerunalongsideuser
programs.
Whenthekernelswitchesfromoneprocesstoanother,itiscalledacontextswitch.
InterruptsInterupts
ThefollowingexplanationonStackOverflowhelpedmeunderstandthedifference
betweenconcurrencyandparallelism:
“Concurrencyiswhentwoormoretaskscanstart,run,andcompleteinoverlappingtime
periods.Itdoesn’'tnecessarilymeanthey'lleverbothberunningatthesameinstant.E.g.,
multitaskingonasingle-coremachine.
Parallelismiswhentasksliterallyrunatthesametime,e.g.,onamulticoreprocessor.”
RichieHindle4.
Howcanasinglecoreprocessorrunmultipletasksatonce?
Challenge
ChapterX.NetworkProgramming
Inthischapter,welookintohowcomputerscommunicatewitheachotherover
networks.Anetworkisagroupofcomputersconnectedthroughsoftwareandhardwarethat
allowsthemtoexchangemessages.54.TheInternetisanexampleofanetwork.Inthis
chapterwewillexplorethefoundationoftheinternet—theclientservermodelandthe
TCP/IPprotocol.Thenwewilldivedeeperintothesesubjectsbybuildingbothaclientanda
server.
Client-ServerModel
TheInternetcommunicatesusingtheclient-servermodel.Intheclient-servermodel,
thereisaserveractivelylisteningforrequests(likeGoogle),sentbyaclient(yourweb
browser).Clientssendrequeststoserversaskingfortheresourcestheyneedtorendera
webpage,andifeverythinggoeswell,theserverrespondsbysendingtheresourcestothe
browser.RequestsaremadeusingHTTP,orhypertexttransferprotocol,whichwecoverin
thenextsection.WhenIsayresources,ImeantheHTML,JavaScriptandCSSfilesthe
browserneedstodisplayawebsitealongwithanyimages.WhenyouvisitGoogle’swebsite,
youareseeingtheclient-servermodelinaction.Googlewaitsforyoutosendarequest,and
respondswiththeresourcesyourwebbrowserneedstodisplayGoogle’swebsitetoyou.
TrygoingtoGoogleinyourbrowserandcopyandpastetheURLintoaword
processor.Youwillseethereisaslashaddedtotheendoftheurl.Thatisbecausewhenyou
gotoawebsitelikeGoogle,youarereallygoingto“http://www.google.com/”.The“/”is
referencingtherootpageofthewebsite,whichyouwillrecallfromtheCommandLine
chapterishowishowyoureferencetherootofanoperatingsystem.Sowhenyougoto“
http://www.google.com/”,youarerequestingGooge’srootpage,whereasifyougoto“
http://www.google.com/news”youarerequesting“/news”andGooglewillrespondwith
differentresources.
However,beforeanyofthiscanhappen,yourwebbrowserneedstotranslate“
http://www.google.com”intoanIPaddress.ThisiswheresomethingcalledtheDNS,or
domainnamesystemcomesin.TheDNSisagianttablethatmapsallofthedomainsinthe
worldtotheirIPaddresses,maintainedbydifferentinternetauthoritiessuchastheInternet
AssignedNumbersAuthority.Anipaddressisauniquenumberthatrepresentseach
computerontheinternet.TocommunicatewithGoogle,yourbrowserneedstogetitsIP
address,whichitdoesbylookingitupintheDNS.
Atalowlevel,allthiscommunicationhappensthroughsockets.Socketsarethefunctions
thatgiveprogramsaccesstoacomputer ’snetworkhardware.Socketsarecreatedbyyour
operatingsystemasadatastructure,allowingcomputerstoestablishconnectionswitheach
other.Aserveropensapassivesocket,andaclientopensanactivesocket.Apassivesocket
staysopenindefinitely,listeningforconnections,whereasanactivesocketrequestsdatafrom
apassivesocketandthencloses.
Torecap,theclient-servermodelworksasfollows—auserentersadomainnameinto
thebrowser,andthebrowserlooksupthedomain’sIPaddressintheDNS.Thebrowsersends
anhttprequesttotheIPaddressitlookedup,andtheserverrespondswithanhttprequest
lettingthebrowserknowitreceiveditsrequestandthensendstheresourcesthewebbrowser
needstodisplaytherequestedwebpagetoyou.
TCP/IP
Thecommunicationintheclient-servermodelfollowstheTCP/IPprotocol.Aprotocol
isanagreeduponwayofdoingthings,usedtostandardizeaprocess.Protocolsarenot
limitedtoComputerScience.IfyouweretomeettheQueenofEngland,therewouldbea
protocolinplace—asetofruleseverypersonhastofollowwhenmeetingher.Youwouldn’t
justwalkuptoherandsay“Heybro!”Youwouldaddressheracertainway,speakpolitely,
sticktocertainsubjectsetc.Thatisaprotocol.
ComputerscommunicatingovertheInternetuseaprotocolcalledTCP/IP.Imaginean
internetwithoutanagreeduponprotocol.Withnostandardforcommunicating,everytime
twocomputersneededtopassdatatooneanother,theywouldhavetonegotiatethetermsof
theircommunication.Nothingwouldevergetdone.LuckilywehaveprotocolslikeTCP/IP
thatensurecommunicationhappensseamlessly.
TCP/IPiswhatiscalledaprotocolstack.Itisismadeupoffourlayers,witheachlayer
usingitsownprotocol.Eachlayerisaprogramresponsibleforaccomplishingatask,and
communicatingwiththelayersaboveandbelowit.WhiletheInternetcoulduseoneprotocol
(insteadofastack),thebenefitofusingaprotocolstackseparatedintolayersisthatyoucan
makechangestoonelayerwithoutneedingtochangetheothers.Thinkaboutthepostoffice.
Someoneatthepostofficeacceptspackages,thensomeoneelsesortsthemandpassesthem
offtosomeonewhodeliversthem.Eachpersonhastheirownprotocolforaccomplishing
theirtask(andtheyallcommunicatewitheachother).Ifthedeliveryguydecidestodeliver
packagesusingdronesinsteadofatruck,thechangeinprotocoldoesn’taffectthepersonwho
acceptspackagesorthepersonwhosortsthem.ThisisthesamereasonwhyTCP/IPusesa
protocolstack,sochangestooneprotocolwon’taffecttheothers.
ThefourlayersofTCP/IParetheApplicationLayer,theTransportLayer,theInternet
LayerandtheNetworkLayer.
[tcp/ippicture]
Let’stakealookatanexampleofdatamovingthroughTCP/IPbyonceagainthinkingabout
mail.ThinkoftheApplicationLayerasaprogramcontainingaletter.Whenyoutypeaurl
intoyourwebbrowser,theApplicationLayerwritesamessageontheletterthatlooks
somethinglikethis:
[pictureofletterwithhttponit]
TheinformationontheletterisanHTTPrequest.Again,HTTPisaprotocolthatserversand
clientsusetosendmessagestoeachother.TheHTTPrequestItcontainsinformationsuchas
therequestedresource,thebrowsertheclientisusingandafewmorepiecesofinformation.
Theletteristhenpassedtothenextlayer,theTransportLayer.Youcanthinkofthe
TransportLayerputtingtheletterinanenvelope.Outsidetheenvelope,theTransportLayer
putsmoreinformation:
[pictureofenvelopewithwritingonit]
Theinformationincludesthedomainnametosendtherequestto,thedomaintherequestis
comingfrom,theportnumbertheserverison,andsomethingcalledachecksum.Dataisnot
sentacrossthenetworkallatonce,itisbrokenupintopacketswhicharesentoneatatime.
TheTransportLayerkeepsuseschecksumtomakesureallthepacketsgetdeliveredproperly.
NowtheTransportLayerpassestheenveloptotheInternetLayerwhichtakesthe
envelopeandputsitinanevenbiggerenvelop,withmoreinformationwrittenonit:
[pictureofenvelopwithwritingonit]
TheinformationwrittenontheInternetLayerenveloponlycontainstheinformationthe
routerneedstodeliverthedatatotheserveritissendingthedatato.ItcontainstheIPaddress
oftheserverandtheIPaddressofthecomputermakingtherequest.ItalsocontainstheTTL,
whichstandsfortimetoliveand(explainttl)Atthispoint,theenvelopisconsideredapacket
.
Thisfinalenvelopeissenttothebottomlayer,theNetworkLayer,whichuseshardware
andsoftwaretophysicallysendthedata.ThedataisreceivedbytheNetworkLayeronthe
serverscomputersandtheenvelopispassedinreverseorderuptheprotocolstackwitheach
layerremovinganenvelopuntiltheletterisrevealedattheApplicationLeveloftheserver.
TheserverthengoesthroughthesameprocessthroughtheTCP/IPstack,sendinganHTTP
requestbacksignallingthattherequestwaseithervalidorinvalid.Iftherequestwasvalid,it
startssendingtheresourcestheclientneeds.
Itisimportanttorememberthatdatadoesnotgetsentallatonce,itgetsbrokendown
intopackets.Thebottomlayerofthestack,theNetworkLayermaysendthousandsofpackets
toaclient,
Challenge
Chapter#TK.BringingItAllTogether
CreateaServer
InthissectionwearegoingtousethePythonsocketlibrarytocreateasimpleweb
serverandclient,usingPython’sbuilt-inlibraryforcreatingandmanagingsockets.Weare
goingtocreateaserverthatlistensforrequestsandrespondstothemwiththedata,eanda
clientwecanusetomakethoserequests.
Awebservercreatesasocket,bindsittoaport,andthenrunsaninfiniteloop,
respondingtorequestsastheycomethroughthesocket;whereasaclientsimplyopensupa
socketand,connectstoaservertogettheinformationitneeds.Wewillstartbybuildinga
serverinPython.Thefirststepistoimportthesocketanddatelibraries:
importsocket
importdatetime
Firstwegettoday’sdate:
today=str(datetime.datetime.today())
Nowwecancreateasocket:
s=socket.socket(socket.AF_INET,socket.SOCK_STREAM)
socket.AF_INETisanaddressfamilyspecifyingwhataddressesinyoursocketcan
communicatewith.AF_INETisusedforcommunicatingovertheinternet.Thereareother
addressfamilieslikeAF_BLUETOOTHthatcanbepassedinforcommunicatingover
Bluetooth.“socket.SOCK_STREAM”meanswewanttouseTCPtoensuredelivery.#fact
check
NextwebindoursockettoTCPport8888:
s.bind(“”,8888)
Andsetthelengthofthequeue(aqueueisusedbecausemultiplerequestscancomeinatthe
sametimeandadatastructureisneededtoprocessthem):
s.listen(10)
Nowwecancreatetheserver ’sinfiniteloopwhichwaitsforaconnectionandsendsthedate
backasaresponse:
whileTrue:
connect,address=s.accept()
resp=(connect.recv(1024)).strip()#limitrequestto1024bytes
connect.send(“receivedhttprequest”)
#Closetheconnectionwhenwearefinished:
connect.close()
Hereisourfullserver:
importsocket
importdatetime
today=datetime.datetime.today()
s=socket.socket(socket.AF_INET,socket.SOCK_STREAM)
s.bind(("",8888))
s.listen(5)
whileTrue:
connect,address=s.accept()
resp=(connect.recv(1024)).strip()
connect.send(today)
connect.close()
Youcantestthisserverbyrunningtheprogramandgoingtolocalhost:8888inyourbrowser.
Youshouldseethedatewhenyoudo.
CreateaClient
Nowletscreateaclienttomakerequeststoourserver.Justlikecreatingawebserver,
westartoutbycreatingasocket:
importsocket
s=socket.socket(socket.AF_INET,socket.SOCK_STREAM)
Toconnecttoourserver,wegetthenameofourlocalmachineandsetthevariableporttothe
portourserverusessowecanuseitlater:
#getnameofourlocalmachine
host=socket.gethostname()
#setport
port=8888
Nowwecanconnecttoourhostnameatport8888bypassinginatuplewithourhostnameand
port:
s.connect((hostname,port))
Savetheresponseandclosethesocket:
msg=s.recv()
s.close()
Printthemessagewereceived:
print(“{}”.format(msg))
That’sallthereistoit.We’vebuiltafunctioningclient.Whenyourunourclient,youwillget
thedatefromourserver.
Challenge
PartVProgrammingforProduction
Chapter#TK.Testing
“Ifdebuggingistheprocessofremovingbugs,thenprogrammingmustbetheprocessof
puttingthemin.”
-EdsgerDijkstra
Whenyoustartbuildingsoftwareotherpeoplearegoingtouse,thecodeintheproduct
peopleendupusingiscalledproductioncode.Whenyouputsoftwareintoproduction,it
meansyouputitlivewherepeopleareusingit.PartVofthisbookisabouthowyoushould
programwhenyourgoalistoputsomethingintoproduction;inthissectionofthebookwe
learnaboutthesoftwaredevelopmentprocess,withafocusontesting.Wealsolearnsome
bestprogrammingpractices.
TheWaterfallSoftwareDevelopmentProcess
Asoftwaredevelopmentprocessisawayof“splittingofsoftwaredevelopmentwork
intodistinctphases(orstages)containingactivitieswiththeintentofbetterplanningand
management,”63TheWaterfallmodelisa“sequential(non-iterative)designprocess,used
insoftwaredevelopmentprocesses,inwhichprogressisseenasflowingsteadilydownwards
(likeawaterfall)throughthephasesofconception,initiation,analysis,design,construction,
testing,production/implementationandmaintenance.”64Itismadeupoffivephases:
PlanningandRequirementsAnalysis,DefiningRequirements,SystemDesign,Implementation
andDeployment,SystemTestingandSystemMaintenance.
InthefirstphaseoftheSystemsDevelopmentCycleyoudeterminewhatproblemyou
wanttosolve.Youlookathowthenewsystemaffectsyourcurrentpriorities,youanalyzethe
resourcesyouwouldneedtobuildthenewsystem,andyouthinkaboutthesystem’s
requirements.Youcandoafeasibilitystudyatthisstage,lookingatwhetherthesystemis
operationally,economically,andtechnicallyfeasible,amongotherconsiderations.
IntheDefiningRequirementsphaseyoudefineanddocumentthesystemrequirements.
Thisisdonebyworkingwiththeprojectstakeholders—“anindividual,group,or
organization,whomayaffect,beaffectedby,orperceiveitselftobeaffectedbyadecision,
activity,oroutcomeofaproject”.61
IntheSystemDesignphase,differentdesignapproachesarediscussed,andafinal
approachisagreeduponandoutlinedinadocumentcalledaDesignDocumentSpecification
whichoutlinesthe“designapproachfortheproductarchitecture.”62Adesignapproach
“clearlydefinesallthearchitecturalmodulesoftheproductalongwithitscommunicationand
dataflowrepresentationwiththeexternalandthirdpartymodules(ifany).Theinternaldesign
ofallthemodulesoftheproposedarchitectureshouldbeclearlydefinedwiththeminutestof
thedetailsinDDS.”62Inotherwords,youdecidehowyouaregoingtoarchitectthesystem
youarebuilding,andcommitittotheDDS.
Thisiswherethemagichappens.Afterthreestagesofplanning,codingbeginsinthe
ImplementationandDeploymentphase.Theproductisbuiltinthisphase,followingthe
DesignDocumentSpecification.
TheSystemTestingPhaseiswheretheproductistestedforbugsandtestedtomake
sureitmeetstherequirementsoutlinedintheDefiningRequirementsstage.Thenextchapter
ofthisbookcoverstestingindetail.
Oncetheproducthasbeentested,itisputintoproductionandreleasedtothepublic.
Duringthisphasetheproductisliveandthesoftwareteamperformsanynecessary
maintenance.
OtherSoftwareDevelopmentProcesses
Thewaterfallmethodisoneofmanysoftwaredevelopmentprocesses—likethe
incrementalmodel:“amethodofsoftwaredevelopmentwheretheproductisdesigned,
implementedandtestedincrementally(alittlemoreisaddedeachtime)untiltheproductis
finished.”65TherearealsovariousimplementationsofthepopularAgilemethodology
whichyoucanfindoutmoreaboutbyreadingtheManifestoforAgileSoftware
Development:http://agilemanifesto.org.
Whilenotasoftwaredevelopmentprocess,inthenextchapter,weutilizeTestDriven
Development,adevelopmenttechniquewhereyoumustfirstwriteatestthatfailsbefore
youwritenewfunctionalcode.”66TestDrivenDevelopmenthelpsyouwritebetter
programsbyforcingyoutothinkclearlyaboutwhatyouaredesigning.
Testing
Inthischapterwewillfocusontesting,partofthesoftwaredevelopmentprocess.
Testingaprogrammeanscheckingthattheprogram“meetstherequirementsthatguidedits
designanddevelopment,respondscorrectlytoallkindsofinputs,performsitsfunctions
withinanacceptabletime,issufficientlyusable,canbeinstalledandruninitsintended
environments,andachievesthegeneralresultitsstakeholdersdesire.”57Inordertoachieve
this,wewriteprogramstotestourprograms.Inmostcases,testingisnotoptional;you
shouldconsidereveryprogramyouintendtoputintoproductionincompleteuntilyouhave
writtentestsforit,unlessyouhaveaverygoodreasonnotto.Ifyouwriteaquickprogramto
dosomethinglikemanipulateafile,andyouneveraregoingtouseitagain,testingmightbe
awasteoftime.Butifyouarewritingaprogramthatotherpeoplearegoingtobeusing,you
shouldwritetests.Assomeonesmartoncesaid,“Untestedcodeisbrokencode.”Inthis
chapter,wearegoingtogooversomeofthefundamentalsoftesting.Funfact—theword
computer“bug”originatedfromanincidentin1942,whereGraceMurrayHopperfoundand
removedamothstuckinaMarkAikenRelayCalculator.
Assertions
Assertionsarethefoundationoftests.Anassertionisastatementthataprogrammer
expectstobeTrue(inwhichcaseitdoesn’tdoanything)andraisesanexceptionifitisFalse.
Pythonhasabuilt-inassertkeywordforcreatingassertions.Hereisanexample:
x=1
assertx==1
>>
x=1
assertx==2
>>Traceback(mostrecentcalllast):
File"/Users/coryalthoff/PycharmProjects/self_taught/st.py",line
2,in<module>
assertx==1
AssertionError
Inthefirstexample,theconditionfollowingassertisTrue,sonoerrorisraised.Inthe
secondexample,theconditionfollowingassertisFalse,andanAssertionErrorisraised.
Assertionsareusedtocheckwhetherornotatest(whichwewilllearnhowtowriteshortly)
passed.IfanAssertionErrorisraised,thetestfailed,otherwiseitpassed.
TypesofTests
Testingisusuallydoneinfourdifferentphases:unittesting,integrationtesting,system
testing,andacceptancetesting.Inthissection,wewillbrieflyexploreeachofthesephases.
Theunittestingphaseinvolveswritingunitteststhattestindividualpiecesofcodesuch
asfunctions,methodsandclasses.Eachunittesttestsoneaspectofapieceofcodewithan
assertion.Sayforexampleyouhaveafunctionthatprintswhateverstringyoupassit.One
unittestmightpassthefunctionthestring“Hello”andtestthat“Hello”wasprinted;whereas
anotherunittestmightpassthefunctionanintegertomakesurethefunctionisableto
properlyhandleaninputthatisnotastring.Unittestsshouldtestthegeneral-usecaseforall
ofyourfunctionsclassesandmethods;checkingwhathappenswhentheyreceiveinputvalues
youweren’texpecting,andtestboundaryconditions—whenthingslikealistgetbigorfull.
Youcanwriteyourunittestswithinaunittestingframework—aprogramforcreatingand
structuringunittests,likePython’sbuilt-inunittestingframeworkcalledunittest,which
comeswithdifferentassertionmethodsthatletyoueasilytestdifferentconditions.
Integrationtestingisperformedafterunittesting.Whileunittestingtestseachindividual
pieceofcodeinamodule,integrationtestingteststomakesurethedifferentmodulesina
projectworkwitheachother.Forexample,sayyouarebuildingabankingapplicationand
youhavetwomodules:transfer—fortransferringmoney—andbalance—forshowingthe
customer'sbalance.Anintegrationtestmightcheckthatwhenthetransfermodulemoves
1,000dollarsfromacustomer'saccount,andthebalancemodulecorrectlyreducesthe
customer ’sbalanceby1000dollars.59
Systemtestingteststheentiresystem.Therearemanydifferenttypesoftestsusedin
systemstesting:graphicaluserinterfacetesting—whichteststhepartofproductyoucansee;
usabilitytesting—whichteststhatpeoplecanfigureouthowtousetheproduct;andsoftware
performancetesting—whichteststoseehowtheproductperformsunderaheavyworkload
suchasalargenumberofusers;amongothers.
Finallythelastphaseofthetestingcycleisacceptancetesting—whichcheckstomake
surethesoftwaremeetstherequirementsagreeduponbytheprojectstakeholders.Acceptance
testingisnotdoneprogrammatically—itisdonebypeoplewhomakesurethetheproduct
requirementslaidoutintherequirementsdocumentaremet.Therestofthischapterwill
focusonunittesting,becauseasanewsoftwareengineeronateam,youwillberesponsible
forwritingunittests—butyoumostlikelywillnotberesponsibleforintegration,systemsand
acceptancetesting.
TDD
Aswelearnedinthepreviouschapter,TDDstandsfortestdrivendevelopment,andisa
softwaredevelopmenttechniquethathelpsyoudesignbetterprograms.Whenyoufollowtest
drivendevelopment,youwriteyourunittestsbeforeyouwriteyourprogram.Following
TDDforcesyoutobreakoutofthepatternofputtingwritingunittestsoffuntiltheendof
yourdevelopmentcycle,andthendecidingnottowritethem.Italsoguaranteesyouwillhave
unitteststhroughoutyourdevelopmentcycle.Lastly,TDDhelpsyoutodesignbettersoftware
byforcingyoutothinkaboutthedesignrequirementsofyourprogrambywritingyourtests
first.
InthissectionwearegoingtolearntowriteunittestsbycreatingastackusingTDD
andPython’sunittesttestingframework.Wewillstartourdevelopmentprocessbywriting
unitteststhatwillfail(butwouldworkifourstackwasproperlydesigned),andthenwriting
codetomakethetestspass.
Tobettervisualizethetestsweneedtocreate,wWewillstartbydefiningastackthat
doesn’tdoanythingtobettervisualizethetestsweneedtocreate.InthissectionIassumeyou
rememberhowastackworks(Ifyouforgethowastackworks,pleaserevisitthechapterData
Structures&Algorithms).
importunittest
classStack:
def__init__(self):
self.stack=[]
defpush(self,item):
pass
defpop(self):
pass
defpeak(self):
pass
defis_empty(self):
pass
>>>
We’vedefinedaStackclass,however,noneofthemethodsinourclassactuallydoanything.
BeforewedefinethemethodsforourStackclass,wearegoingtowriteallofourtests.
classStackTests(unittest.TestCase):
defsetUp(self):
self.stack=Stack()
deftearDown(self):
delself.stack
deftest_is_empty(self):
self.assertTrue(self.stack.is_empty())
deftest_push(self):
self.stack.push(100)
self.assertFalse(self.stack.is_empty())
deftest_peak(self):
self.stack.push('test')
self.assertEqual(self.stack.peak(),'test')
deftest_pop(self):
self.stack.push(10.1)
self.stack.pop()
self.assertTrue(self.stack.is_empty())
deftest_pop_value(self):
self.stack.push('test_value')
value=self.stack.pop()
self.assertEqual(value,'test_value')
>>>
Whenyouwriteunittestswiththeunittestframework,youstartbydefiningaclassthat
inheritsfromunittest.TestCase—inthisourclassiscalledStackTests.Theunittest
frameworkusestheclassthatinheritsfromunittest.TestCasetoruntests;witheachmethod
youdefineinyourclass—aslongasitstartswithtest—runsasanisolatedtest.Atestthat
passeswillnotraiseanAssertionError,andatestthatfailswill.Ifyourunourprogram,you
willseetheresultsofeachtest(theyallwillfailwithAssertionErrors).
Hereisamoredetailedexplanationofthetestingcodeyouwrote.Thefirsttwo
methodssetUpandtearDowndonotstartwithtestbecausetheyarenottests:theyaremethods
inheritedfromunittest.TestCaseusedtohelpsetupourtests.
defsetUp(self):
self.stack=Stack()
deftearDown(self):
delself.stack
setUprunsbeforeeachtest,andtearDownrunsaftereachtest.InthiscaseweusesetUp
tocreateanewStackobjectbeforeeachtest,andusetearDowntodeleteitaftereachtest.We
dothisinordertomakesurewehaveabrandnewStackobjectineachtest,sothetestsdon’t
interferewitheachother.
Inthefirsttestwedefine,test_is_empty(),usingtheassertionmethodassertTrue()
whichtakesaparameterandraisesanAssertionErroriftheparameterevaluatestoFalse
becauseifwedon’tputanythinginourstack,itshouldbeempty.
deftest_is_empty(self):
self.assertTrue(self.stack.is_empty())
Ournexttestistest_push().Itcallspush()onourstackandpassesin100.Weusethe
assertionmethodassertFalseandpassinself.stack.is_empty()becauseafterpushing
somethingtoourstack,thestackshouldnolongerbeempty:self.stack.is_empty()shouldbe
False.
deftest_push(self):
self.stack.push(100)
self.assertFalse(self.stack.is_empty())
Inournexttest,test_peak(),wepush‘test’ontotheStackanduseassertEqual()to
checkthatself.stack.peek()returnsthevaluewepushedtoourstack,inthiscase
self.stack.peek()and‘test’shouldbeequal.
deftest_peak(self):
self.stack.push('test')
self.assertEqual(self.stack.peak(),'test')
Ournexttestistest_pop().Wepush10.1toourstack,callself.stack.pop()andusethe
assertionmethodassertTrue()tocheckthatourpop()methodsuccessfullyremovedanitem
andourstackisnowempty.
deftest_pop(self):
self.stack.push(10.1)
self.stack.pop()
self.assertTrue(self.stack.is_empty())
Ourfinaltest,test_pop_value(),pushesthestring‘test_value’toourstackandusesthe
assertionmethodassertEqual()tocheckthatpop()returns‘test_value’.
deftest_pop_value(self):
self.stack.push('test_value')
value=self.stack.pop()
self.assertEqual(value,'test_value')
Runourtests.YouwillbenotifiedallfivetestsfailedbyraisingfiveAssertionErrors
.ThisisbecausewehavenotdefinedanyofthemethodsinourStackclass.Ourmethods
don’tdoanythingandsoallofourtestsfail.OneoftheadvantagesofTDDisthatwriting
yourtestsfirsthelpsclarifyyourthinking.Wenowknowexactlyhowweneedtodefineeach
methodinourStackclassinordertopasseachtest.Hereiswhatweneedtodo:
classStack:
def__init__(self):
self.stack=[]
defpush(self,item):
self.stack.append(item)
defpop(self):
returnself.stack.pop()
defpeek(self):
returnself.stack[-1]
defis_empty(self):
iflen(self.stack)>0:
returnFalse
returnTrue
>>>
Nowthatwe’vedefinedthemethodsinourStack,whenyourunourtestsagain,thetestswill
allpass.We’vetestedourclass,ourmethodsandgeneralusecases;andnowweneedtotest
unexpectedusecases,badinputvalues,andboundaryconditions.Anexampleofan
unexpectedusecaseiscallingpoponanemptystack,whichwillcauseanerror.Tofixthis,
firstweshouldwriteanewtest.
classStackTests(unittest.TestCase):
defsetUp(self):
self.stack=Stack()
deftearDown(self):
delself.stack
deftest_empty_pop(self):
withself.assertRaises(IndexError):
self.stack.pop()
Ournewtesttest_empty_popusestheassertionmethodself.assertRaises()withthewith
statementtotestifanexceptionisraised.Thecodeinsidethewithstatementisexpectedto
raisetheexceptionpassedtoassertRaises.Iftheexceptionisraisedthetestpasses;ifthe
exceptionisnotraisedanAssertionErrorisraisedandthetestfails.Nowwecanfixourstack
tohandleemptypops:
classStack:
def__init__(self):
self.stack=[]
defpush(self,item):
self.stack.append(item)
defpop(self):
ifnotself.stack:
raiseIndexError("Cannotpopfromemptystack")
returnself.stack.pop()
defpeek(self):
returnself.stack[-1]
defis_empty(self):
iflen(self.stack)>0:
returnFalse
returnTrue
>>>
Weneedtothinkaboutbadinputvalues.Inthisexample,therearenobadinputvalues
becauseanyobjectinPythoncanbeaddedtoalist,andthereforecanbeaddedtoourstack.
However,youalwayswanttotakethetimetoatleastthinkaboutanyinputvaluesthatcould
breakyourprogram.Finally,weshouldthinkaboutboundarycases:the“behaviorofa
systemwhenoneofitsinputsisatorjustbeyonditsmaximumorminimumlimits.”58In
Python,thereisnolimittohowmanyobjectswecanputinalist,thereforethesizeofour
stackisonlylimitedbytheamountofmemorythecomputerthatcreatedithas.However,
someprogramminglanguageslimitthenumberofobjectsyoucanputinalist,andifthat
werethecaseinPython,wewouldhavetowriteatestforthatcondition.
WritingGoodTests
Goodtestsarerepeatable.Thatmeanswhenyourunyourtests,theyshouldworkin
anyenvironment—ifyouwriteatestonOSX,itshouldalsoworkonWindowswithout
havingtomakeanychangestotothetest.Anexampleofviolatingthiswouldbeincluding
hardcodeddirectorypathsinyourtest.WindowsandOSXusedifferentslashesfordirectory
paths—soyourtestinaWindowsenvironmentwouldnotworkinanOSXenvironment.This
meansthetestisnotrepeatable,andneedstoberewritten.Testsshouldalsorunquickly.Tests
needtorunoften;trynottowriteteststhattakealongtimetorun.Finallyyourtestsshouldbe
orthogonal—onetestshouldnotaffecttheother.
CodeCoverage
Codecoverageisthethetotalnumberoflinesofcodeinyourprojectcalledduring
yourtestsdividedbythetotalnumberoflinesofcodeinyourproject.Codecoveragedoes
notmeasuretheefficiencyofyourtests,butisusefulforfindinguntestedpartsofyourcode.
Yougenerallywanttohavecodecoverageabove80%.Ifyourcodecoverageislow,itmeans
youhavenottestedenoughofyourcode.TheprofessionalversionofPyCharm(theonethat
costsmoney)isintegratedwithatoolforanalyzingcodecoverage.IfyouusePyCharm’sfree
CommunityEdition,youareinluckbecausethetooltheprofessionalversionisintegrated
withiscalledcoverage.pyandisfreetouse.Thedocumentationforcoverage.pyisavailable
at:https://coverage.readthedocs.io.
TestingSavesTime
It’seasytogetlazyandskiptesting,justifyingyourlazinessbysayingyoudon’thave
timetowritetests.Counterintuitively,takingthetimetowritetestswillsaveyouasubstantial
amountoftimeinthelongrun.Thereasonisbecauseifyoudon’twritetests,youwillendup
testingyoursoftwaremanually—runningyourprogramyourselfwithvariousdifferent
inputsandunderdifferentconditionstoseeifanythingbreaksasopposedtotestingyour
programprogrammatically.Whileyoushouldtestyourentireprogrambytestingitmanually
andlookingforbugs,youdon’twanttosolelyrelyonthis.Spendingyourtimemanually
testinginputsandconditionswhenyoucouldeasilyautomatetheprocessisahugewasteof
time.Finally,ifyoucomebacktotheprojectinamonth,youwon’tbeabletorememberthe
differenttestsyouweremanuallyrunning.
Testingisanimportantpartofprogramming.Mostsoftwareteamshaveatleastone
persondedicatedtotesting.GettingintothehabitoffollowingTDDwillimproveyourcode
bymakingsureyoualwayswritetestsandthereforedecreasethenumberoferrorsinyour
code,andbyhelpingyoutothinkcarefullyabouthowyoudesignyourprograms.
Challenge
WriteunittestsfortheHangmanprogramwebuiltinPartI.
Wheneveryouarefollowingprogramminginstructions,andsee$,itmeanswhatever
followsisacommandthatyoushouldtypeintothecommandline(noneedtotypethedollar
sign).
Programminglanguageshaveconventionsaswell.Conventionsarerulesgenerally
followedbythecommunityusingthelanguage.
Chapter#TK.BestProgrammingPractices
“Alwayscodeasiftheguywhoendsupmaintainingyourcodewillbeaviolentpsychopath
whoknowswhereyoulive.”
-MartinGolding
Inthischapterwewillcoverafewgeneralprogrammingprinciplesthatwillhelpyouwrite
production-readycode.Manyoftheseprinciplesoriginatedintheexcellentbook“The
PragmaticProgrammer”byAndyHuntandDaveThomas,abookthatdramaticallyimproved
thequalityofmycode.
WriteCodeAsALastResort
Yourjobasasoftwareengineeristowriteaslittlecodeaspossible.Whenyouhavea
problemyouneedtosolve,yourfirstthoughtshouldnotbe“HowcanIsolvethis?”Itshould
be,“HassomeoneelsesolvedthisproblemalreadyandcanIusetheirsolution?”Ifyouare
tryingtosolveacommonproblem,chancesaresomeoneelsehasalreadysolvedit.Startby
lookingonlineforasolution.Onlyafteryou’vedeterminednooneelsehasalreadysolved
theproblemshouldyoustartsolvingityourself.
DRY
DRYisaprogrammingprinciplethatstandsforDon’tRepeatYourself.FollowingDRY
iseasy:ifyouarewritingcodeandfindyourselfrepeatingthesamecode—stop.Donot
repeatyourself.Ifyoufindthatyouarecopyingpiecesofcode,pastingthemsomewhereelse
inyourprogram,andmakingsmallchangestoittocreatenewcode—stop.Donotrepeat
yourself.Wewillillustratethiswithaprogramthatmakeschangestoalistofwords.
defcapitalize_item(word,word_list):
forindex,iteminenumerate(word_list):
ifitem==word:
word_list[index]=word_list[index].capitalize()
defchange_letter(word,word_list,old_letter,new_letter):
forindex,iteminenumerate(word_list):
ifitem==word:
word_list[index]=word_list[index].replace(old_letter,new_letter)
words=['Programming','is','fun']
upper('Programming',words)
print(words)
change_letter('fun',words,'u','$')
print(words)
>>['PROGRAMMING','is','fun']
>>['PROGRAMMING','is','f$n']
BothfunctionsusePython’sbuilt-inenumeratefunction,whichtakesalistasaparameter,and
allowsyoutouseaforlooptoeasilycapturethecurrentindexofthelist,aswellasthe
currentiteminthelist.“Index”referstoanitem’spositionwithinalist.Ourprogramworks
—butifyoulookcloselyatthecode—youwillseebothofourfunctionslookforawordina
listtoreplaceitwithsomethingnew.Insteadofhavingcodetosearchforawordinboth
functions,weshouldcreateonefunctionthatreturnstheindexofthewordwearelookingfor
(Pythonhasabuilt-infunctionindex()thatfindstheindexofastringinalist,butforthesake
ofthisexamplewearenotusingit).Hereishowweshouldrefactorourcodetoavoid
repeatingourselves:
deffind_index(word,word_list):
forindex,iteminenumerate(word_list):
ifitem==word:
returnindex
defupper(word,word_list):
index=find_index(word,word_list)
word_list[index]=word_list[index].upper()
defchange_letter(word,word_list,old_letter,new_letter):
index=find_index(word,word_list)
word_list[index]=word_list[index].replace(old_letter,new_letter)
words=['Programming','is','fun']
upper('Programming',words)
print(words)
change_letter('fun',words,'u','$')
print(words)
>>>
['PROGRAMMING','is','fun']
['PROGRAMMING','is','f$n']
>>>
Bycreatinganewfunctionthatreturnstheindexofaword,wearenolongerrepeatingcode.
Ifwedecidetochangethewaywesearchforanindex,weonlyneedtochangeourfind_index
function,insteadofchangingthecodetofindanindexinmultiplefunctions.
Orthogonality
OrthogonalityisanotherimportantprinciplepopularizedbythebookthePragmatic
Programmer.TheauthorsAndyHuntandDaveThomasexplain,“Incomputing,thetermhas
cometosignifyakindofindependenceordecoupling.Twoormorethingsareorthogonalif
changesinonedonotaffectanyoftheothers.Inawell-designedsystem,thedatabasecode
willbeorthogonaltotheuserinterface:youcanchangetheinterfacewithoutaffectingthe
database,andswapdatabaseswithoutchangingtheinterface.”16Putthisinpracticeby
rememberingthatasmuchaspossible,“ashouldnotaffectb”.Ifyouhavetwomodules—
moduleaandmoduleb—moduleashouldnotmakechangestothingsinmoduleb,andvice
versa.Ifyoudesignasystemwhereaaffectsb;whichaffectsc;whichaffectsd;things
quicklyspiraloutofcontrolandthesystembecomesunmanageable.
EveryPieceOfDataShouldHaveOneRepresentation
Thisisbestexplainedwithanexample.Sayyouarebuildingaproject,andinthat
projectyouareusingTwitter ’sAPI(applicationprogramminginterface)—aprogramthat
givesyouaccesstodatafromawebsitelikeTwitter.Twitterprovidesamoduleyoucan
downloadonpiptoquerytheirAPIfordata.InordertouseTwitter ’sAPI,youhaveto
registerforanAPIkey—astringyousendtoTwitterwhenyouusetheirAPIsotheycan
verifyitisyou.
Onceyou’veobtainedyourAPIkeyfromTwitter,youstartusingtheAPIintwo
functions.Thefirstfunctiongetsdatafromcelebrities,andthesecondfunctiongetsdatafrom
noncelebrities.BothfunctionsneedtheAPIkeyinordertouseTwitter ’sAPI.Hereisan
exampleofwhatthiscouldlooklike:
#WARNINGthiscodedoesnotactuallywork
importtwitter_api
importcelebrity_list
importregular_list
defcelebrity_data():
api_key='11330000aazzz22'
returntwitter_api.get_data(api_key,celebrity_list)
defpeople_data():
api_key='11330000aazzz22'
returntwitter_api.get_data(api_key,regular_list)
Thisisamadeupexamplethatwon’tactuallywork,butthegistisweareusingamadeup
modulecalledtwitter_apitocallthefunctionget_datawithourapikeyandalistofpeople.A
fewmonthsgobyandyouendupgettinganewAPIkeyfromTwitter.Yougotothecelebrity
functionandchangethevariableapi_keytothenewAPIkey.It’sbeenalongtimesinceyou
wrotethiscode,andyoucompletelyforgettheAPIkeyisusedinthesecondfunction.Youput
yourcodeintoproductionandaccidentlyleavingtheoldAPIkeyinthesecondfunction;
everythingbreaks,andyougetfired.Youcould'veavoidedyourtragicfatebyfollowingthe
rulethateverypieceofdatashouldhaveonerepresentation.Thecorrectwaytohandlethis
situationistomakeaPythonfilewithavariablecalledapi_key.Thisiscalledaconfiguration
file.Yourprogramshouldimportapi_keyfromthisfile,andbothfunctionsshoulduseit.
Thisway,thepieceofdata(theAPIkey)isonlyrepresentedonce.Thatway,nomatterhow
manyplacestheAPIkeyisused,ifyouhavetochangetheAPIkey,youonlyneedtochangeit
inoneplace,theconfigurationfile,andthepreviouslydiscusseddisasterwillbeaverted.
FunctionsShouldDoOneThing
Everyfunctionyouwriteshoulddoonething,andonethingonly.Ifyoufindyour
functionsgettingtoolong,askyourselfifthefunctionyouarewritingisaccomplishingmore
thanonetask.Limitingfunctionstoaccomplishingonetaskoffersseveraladvantages:your
codewillbeeasiertoreadbecausethenameofyourfunctionwilldescribeexactlywhatit
does;andifyourcodeisn’tworkingitwillbeeasiertodebugbecauseeveryfunctionis
responsibleforaspecifictask,soyoucanquicklyisolateanddiagnosethefunctionthatisn’t
working.AsRyanSingersays,“Somuchcomplexityinsoftwarecomesfromtryingtomake
onethingdotwothings.”
UseDummyData
WhileIwasateBayIwasgivenanassignmenttofixanerrorinourcode.TheprogramI
wasdebuggingprocessedalargetextfileandtookfiveminutestorun.Iwouldmakea
changetotheprogramtotrytogetsomeinformationaboutwhatwaswrong,runthe
program,andwaitfiveminutesfortheresults.Iwasnotmakinganyprogress,becauseIhad
towaitfiveminuteseverytimeImadeachange,andthatquicklyaddedup.Ifinallytookthe
timetosubstitutethelargetextfilewithdummydata—fakedatamyprogramcouldusebut
wouldonlytakeafewsecondstoprocess.ThiswayIcouldstilllookforthebuginthe
program,butmuchfaster.Takingthetimetosetupdummydata—evenifittakesyoutwenty
minutes—willquicklypayoffbyshorteningyourdebugcycle.
IfIt’sTakingTooLongYouAreProbablyMakingaMistake
Ifyouarenotworkingonsomethingobviouslycomplexlikeworkingwithalarge
amountsofdata,andyourprogramistakingaverylongtimetoload,assumeyouaredoing
somethingwrong.
Logging
Loggingisthepracticeofrecordingdatawhenyoursoftwareruns.Youcanuse
loggingtohelpdebugyourprogram,andtogainadditionalinsightintowhatwhathappened
whenyourprogramran.Pythoncomeswithagreatloggingmodulethatletsyoulogeitherto
theconsoleorafile.
Whensomethinggoeswronginyourprogram,youdon’twantittogounnoticed—you
shouldloginformationaboutwhathappenedsoyoucanreviewitlater.Loggingisalso
usefulforcollectingandanalyzingdata.Forexample,youmightsetupawebserversettolog
data—includingthedateandtime—everytimeitreceivesarequest.Youcouldstoreallof
yourlogsinadatabase,andcreateanotherprogramtoanalyzethatdataandcreateagraph
displayingthetimesofdayyourwebsiteisvisitedthemost.
Goodprogrammersuselogging,summarizednicelybyHenrikeWarnewhenhesaid
“Oneofthedifferencesbetweenagreatprogrammerandabadprogrammeristhatagreat
programmeraddsloggingandtoolsthatmakeiteasytodebugtheprogramwhenthingsfail.”
YoucanlearnhowtousePython’sloggingmoduleat
https://docs.python.org/3/howto/logging.html.
DoThingsTheBestWayTheFirstTime
Ifyouareinasituationwhereyouareprogrammingandyouthink,“Iknowthereisa
betterwayofdoingthis,butI’minthemiddleofcodinganddon’twanttostopandfigureout
howtodoitbetter.”Stop.Doitbetter.
FollowConventions
Takingthetimetolearntheconventionsofthenewprogramminglanguageyouare
tryingtolearnwillhelpyoureadcodewritteninthenewlanguagefaster.Pep8isasetof
guidelinesforwritingPythoncode,andyoushouldreadit.It’savailableat:
https://www.python.org/dev/peps/pep-0008.
UseaPowerfulIDE
AnIDE,orInteractiveDevelopmentEnvironment,isaprogramyouusetowriteyour
code.Thusfar,we’vebeenusingIDLE,theIDEthatcomeswithPython.However,IDLEis
justoneoptionofmanydifferentIDEsavailable,andIdonotrecommendusingitlongterm,
becauseitisnotverypowerfulcomparedtootherIDEs.Forexample,ifyouopenupaPython
projectinabetterIDE,therewillbeadifferenttabsforeachPythonfile.InIDLE,youhaveto
openanewwindowforeachfile,whichinbigprojectsthisquicklygetstediousandit’s
difficulttonavigatebackandforthbetweenfiles.
IuseanIDEcalledPyCharmcreatedbyJetBrains.Theyofferafreeversionaswellasa
professionalversion.Eitheronewillwork.SublimeisanotherpopularIDE.InthischapterI
willbegoingoversomeofthefeaturesIuseinJetBrainsIDEthatincreasesmyproductivity.
BecauseanyIDEisliabletochangeitscommandsatanytime,therearenoexamplesinthis
chapter.InsteadIdescribesomeofthefeaturesPyCharmhas,togiveyouanideaofwhatan
IDEiscapableof,soyouwon’twastetimedoingthingsmanuallythatyoucanquicklydo
withanIDE.Iputatutorialattheselftaughtprogrammer.io/idesoIcankeepituptodate.
Ifyouseeavariable,functionorobjectbeingusedandyouwouldliketoseeits
definition,PyCharmhasashortcuttojumptothethecodethatdefinedit(evenifitisina
differentfile).Thereisalsoashortcuttojumpbacktothepageyoustartedfrom.
PyCharmhasafeaturethatsaveslocalhistorywhichhasdramaticallyimprovedmy
productivity.PyCharmwillautomaticallysaveanewversionofyourprojecteverytimeit
changes.ThismeansyoucanusePyCharmaslocalversioncontrolsystembutwithouthaving
topushtoarepository.Youdon’thavetodoanything,ithappensautomatically.BeforeIknew
aboutthisfeature,Iwouldoftensolveaproblem,changethesolution,andthendecideI
wantedtogobacktotheoriginalsolution.IfIdidn’tpushtheoriginalsolutiontoGitHub,the
originalsolutionwaslonggone,andIwouldhavetorewriteitagain.Withthisfeature,you
cansimplyjumpbackintime10minutes,andreloadyourprojectexactlyhowitwas.Ifyou
changeyourmindagain,youcanjumpbackandforthbetweenthedifferentsolutionsasmany
timesasyouwant.
Inyourworkflowyouareprobablycopyingandpastingcodealot,movingitfromone
locationonapagetoanother.InPyCharm,insteadofcopyingandpasting,youcanmove
codeupanddownonthepageyouareon.
PyCharmisintegratedwithpopularversioncontrolsystemslikeGitandSVN.Instead
ofhavingtogotothecommandline,youcanuseGitfromPycharm.Thefewertripsyou
havetomakebackandforthbetweenyourIDEandthecommandline,themoreproductive
youwillbe.
PyCharmalsohasabuiltincommandlineandPythonShell.
InPycharmsProversion,thereisabuiltintoolforconnectingtorelationaldatabases,
suchasMySQL.Ifyouareusingarelationaldatabase,thisisahugetimesaver.Remembering
SQLsyntaxisdifficult,havingautocompleteforyourSQLisagamechanger.
PyCharmhasapowerfuldebuggerthatletsyoupausetheexecutionofaprogramat
certainpoints,checkthestatusofdifferentvariables,andwalkthroughaprogramstepby
step.
CodeReviews
Acodereviewiswhensomeonereadsyourcodeandgivesyoufeedback.Youshoulddo
asmanycodereviewsasyoucan—especiallyasaself-taughtprogrammergettingstarted.
Evenifyoufollowallofthebestpracticeslaidoutinthischapter,youarestilldoingthings
wrong—youneedsomeonemoreexperiencedthanyoutoreadoveryourcodeandtellyou
themistakesyouaremakingsoyoucanfixthem.
CodeReviewonStackExchangeisawebsitewhereyoucangetacodereviewfroma
communityofprogrammers.AnyonecangoonCodeReviewandposttheircode.Other
membersoftheStackExchangecommunityreviewyourcodeandgiveyoufeedbackabout
whatyoudidwellandofferhelpfulsuggestionsonhowyoucanimprove.
Security
Securityisaneasysubjectfortheself-taughtprogrammertoignore.Youprobablywon’tbe
askedaboutsecurityinyourinterview,andsecurityisnotimportantfortheprogramsyou
writewhileyouarelearningtoprogram.However,onceyougetyourfirstjobprogramming,
youaredirectlyresponsibleforthesecurityofthecodeyouwrite.Thelastthingyouwantis
foryourcompany'sdatabasetogethackedbecauseyoumadeamistake.Bettertolearnsome
practicaltipstokeepyourcodesafenow.
InthechapterTheCommandLinewelearnedtousesudotoissueacommandastheroot
user.Imentionedyoushouldneverrunaprogramassudo.Youshouldalsodisableroot
loginsifyouaremanagingthesystem.Thereareseveralreasonsyoushoulddothis:every
hackerisawarethereisarootaccountandsoitisaneasytargetwhenattackingasystem
(henceitshouldbedisabled),ifyouareloggedinasrootyoucanirreversiblydamagethe
systemyouarerunning,andifyourunaprogramasrootanditgetstakenoverbyahacker,
thecompromisedprogramnowhasrootaccess.
Thereareseveralkindsofmaliciousattacksthatrelyonexploitingprogramsthat
acceptuserinput,soyoushouldalsoassumealluserinputismaliciousandprogram
accordingly.AnSQLinjectionisatypeofattackthatoccurswhenausersubmitsinputtoa
programandaddsSQLtoit,allowingthemtoexecuteSQLinyourdatabase.Forinstance,you
mighthaveacarwebsite,andonthatwebsiteyouwanttoreturndetailsaboutaspecificcara
userenters.Youexpecttheusertoprovideyouwiththenameofacar,suchas“NissanLeaf”.
Onceyoureceivethenameofthecarfromtheuser,youwritemorecodetoqueryaMySQL
tablecalledcarstodisplaydetailsaboutthecar.Youusethefollowingquery:
SELECT*FROMcarWHEREname=query_data;
Thevariablequery_datarepresentsthedatayoucollectedfromtheuser.Iftheuserenters
somethinglike“NissanLeaf”likeyouexpected,thisworksfine.Theproblemistheusercan
enteranythingtheywant.Amalicioususercouldentersomethinglikethisintothewebsite:
“Ferrari”;DROPTABLEcars;
ThiswouldcauseyourprogramtoqueryMYSQLwiththefollowing,validSQL:
SELECT*FROMcarWHEREname=“Ferrari”;DROPTABLEcars
ThequerywouldgetallofthedataforacarnamedFerrari,thenitwoulddeleteyourdatabase
table—wipingoutallofyourdata.Tocombatthis,youassumealluserinputismalicious,and
insteadofwritingrawSQL—youuseanORMlikeSQLAlchemyoranotherlibrarythat
protectsagainstSQLinjections.Youdon’twanttowriteasolutiontothisproblemyourself
becauseyoucouldtooeasilymakeamistake,whichsomeonecouldexploit.
Afinalstrategyforkeepingyoursoftwaresecureistominimizeyourattacksurface
—thedifferentareasofyourprogramwhereattackerscouldextractdataorattackyour
systeminsomeway.Bymakingyourattackareaassmallaspossible,youcanreducethe
likelihoodofvulnerabilitiesinyourprogram.Herearesomestrategiesforminimizingyour
attacksurface:avoidstoringconfidentialdataifyoudon’thaveto,giveusersthelowestlevel
ofaccessyoucan,useasfewthirdpartylibrariesaspossible(thelesscodethelessamountof
possibleexploits),andgetridoffeaturesthatarenolongerbeingused(lesscodeless
exploits).
Avoidinglogginginastherootuseronyoursystem,nottrustinguserinputand
minimizingyourattacksurfaceareimportantstepstomakingsureyourprogramsaresecure.
However,thesearejuststartingpoints.Youshouldalwaystaketimetotrytothinklikea
hacker.Howwouldahackertrytoexploityourcode?Thiscanhelpyoufindvulnerabilities
youotherwisewould’veoverlooked.Thereisalotmoretolearnaboutsecuritythanwecan
coverinthisbook,somakeanefforttoalwaysbethinkingandlearningaboutsecurity.Bruce
Schneiersaiditbest—“Securityisastateofmind”.
Challenge
Reviewaprogramyou’vewrittenandseeifyoufollowedtheprogrammingpractices
outlinedinthischapter.
Chapter#TK.BringingItAllTogether
“Idon'tcareifitworksonyourmachine!Wearenotshippingyourmachine!”
—VidiuPlaton
Congratulationsonmakingittothefinalcodingexerciseofthisbook.Inthischapterweare
goingtocreateaprogramthatscrapesalyricspageonlyrics.wikia.comandcreatesaword
cloudwiththelyrics.Ifyou’veneverseenawordcloud,itisanimagederivedfromapiece
oftextinwhichthesizeofeachwordcorrespondstohowmanytimesitappearedinthetext.
[pictureofawordcloud]
WewilldevelopourprogramusingtheWaterfallmodelandtestdrivendevelopment.
WordCloudLyrics
WebeginoursoftwaredevelopmentprocessinthePlanningandRequirementsAnalysis
phase.First,wedefinetheproblemwewanttosolve:ourinabilitytoscrapeasonglyrics
websiteandturnthelyricsintoawordcloud.Weneverwanttowritecodeunlesswehaveto,
sothefirstthingwedoischeckifsomeonehasalreadysolvedthisproblem.SomeGoogle
searchingrevealsthisproblemhasbeensolved,andthecodeisavailableonGitHub.
Normallywewouldusetheavailablesolutionifitmeetsourrequirements,andthesoftware
developmentprocesswouldend,butforthesakeoflearningwecanforgetwesawthe
solution,andconcludenoonehassolvedthisproblem.
NextweentertheDefiningRequirementsphase,wherewegooverourproject’s
requirements,andtheresourceswewillneedtocompleteit.Ourprojectrequirestwomain
piecesoffunctionality:theabilitytoscrapedatafromawebsite,andtheabilitytocreatea
wordcloud.We’vealreadyscrapeddatafromGoogleNews,soweknowwehavethetools
andexpertisetoscrapesonglyricsfromawebsite.Whatwedon’tknowishowtobuilda
wordcloud.Buildinganicelookingwordcloudisnoteasy,soifsomeonehasnotalready
solvedthisproblemforus,itisgoingtosignificantlyincreasethetimeneededtofinishthis
project.Luckily,aGooglesearchrevealsthereisaPythonmodulecalledword_cloudthat
letsyoueasilycreatebeautifulwordclouds,sowecanconcludeweshouldbeableto
completethisprojectinareasonableamountoftime.Wealsohavetodefinetherequirements
forourproject—weneedtobeabletopassourprogramaurlthathassonglyricsfrom
lyrics.wikia.com,andourprogramneedstogenerateawordcloud.Wefinishthisphaseby
writingourrequirementsdowninarequirements.txtdocument.
WearenowinthethirdphaseoftheWaterfallModel—SystemDesign—wherewe
decidehowoursystemwillbearchitected.Ourprogramwillconsistoftwofunctions—oneto
getthelyricsfromthewebsite,andtheothertocreatethewordcloud,whichwillalsobethe
mainfunctionweusetorunourprogram.
Afterdesigningoursystem,wearereadytoentertheImplementationandDeployment
phaseandstarttocode.SincewearefollowingTestDrivenDevelopment,wewillstartby
writingourtests.Weknowourprogramneedstodotwothings:getlyricsfrom
lyrics.wikia.com,andcreateawordcloud;sowewillstartbywritingatestforeachofthese
piecesoffunctionality.
importunittest
importos
defget_lyrics():
pass
defcreate_wordcloud(wiki_url):
pass
classTestBIAT(unittest.TestCase):
deftest_lyrics(self):
"""Testthatastringgetsreturnfromget_lyrics()"""
self.assertIsInstance(get_lyrics(),str)
deftest_wordcloud_creation(self):
"""Testthatanewfileiscreatedwhencreate_wordcloud()is
called"""
filecount_before=len(os.listdir())
create_wordcloud('http://lyrics.wikia.com/wiki/The_Beatles:Lucy_In_The_S
ky_With_Diamonds')
self.assertEqual(filecount_before+1,len(os.listdir()))
Ourfirsttestchecksthatget_lyrics()functionreturnsastringbecausewewanttoscrapethe
lyricsfromthesiteandreturnthemasastringforourotherfunctiontouse.Oursecondtest
checksthattest_wordcloud_creation()createsanewfile.Whenwecreateawordcloudweare
goingtosaveitasafileinthefolderwhereweranourprogram,whichiswhywetestto
makesurecreate_wordcloud()createsanewfileinthecurrentdirectorywhenit’scalledwith
theurlofalyrics.wikia.comlyricspage.Wecheckthatanewfileiscreatedusingtheos
module—abuilt-inPythonlibrarythathasdifferentfunctionalityforinteractingwithyour
operatingsystem.os.list.dir()returnsalistofallthefilesinthecurrentdirectory.Inour
assertionintest_wordcloud_creation(),wecheckthenumberoffilesinthecurrentdirectory
beforewecallcreate_wordcloud()+1isthesameasthenumberoffilesafter
create_wordcloud()wascalled.Thisisbecauseafterwecallcreate_wordcloud(),there
shouldbethesamenumberoffilesastherewere,onlyonemore(thenewfilethatwas
created).Whenwerunthesetests,theybothshouldfailwithanAssertionError.Nowallwe
havetodoiswritethecodetomakethempass.First,weneedtoinstallthelibrariesweare
goingtousewithpip.Thefirsttwoarerequirementsforthewordcloudlibrary.
$pip3installnumpy
$pip3installpillow
$pip3installwordcloud
YoushouldalreadyhaveBeautifulSoupinstalledfromwhenwebuiltourfirstscraper,butif
not:
$pip3installBeautifulSoup4
Westartourprogrambyimportingthelibrariesweinstalled:
fromwordcloudimportWordCloud
frombs4importBeautifulSoup
importrequests
fromosimportpath
Nowwecanwriteafunctiontoscrapesonglyrics:
defget_lyrics(wiki_url,tag,tag_name):
"""Takesaurlforalyricspageoflyrics.wikia.com,atagandatagnamesearchesforthattag
intheurlsHTMLthathasthetag_namepassedin.Returnsthesonglyricsfound.
:paramwiki_url:stringlyrics.wikialyricsurle.g.http://lyrics.wikia.com/wiki/The_Beatles:Girl.
:paramtag:stringHTMLtagtolookforlyricsine.g."div"
:paramtag_name:stringnameofHTMLtagtolookforlyricsine.g.lyricbox
:return:stringsonglyrics
"""
response=requests.get(wiki_url)
soup=BeautifulSoup(response.text,'html.parser')
ifsoup.find_all(tag,tag_name):
returnsoup[0].text
Ourget_lyrics()functionshouldlookfamiliar.ItissimilartoourGoogleNews
scraper:wegettheHTMLfromthelyricswebsiteandpassitintoaBeautifulSoupobject;
thenwepasstheparametertagandtag_nameintosoup.find_all()whichlooksforanHTML
tagwiththetagandtag_namewepassedin.Wereturntheresult—thesonglyrics(ifthey
werefound).Wepasstagandtag_nameintoourfunctioninsteadofpassingitinto
soup.find_all()directlybecausewearegettingdatafromalivewebsite,andthetagweare
lookingforislikelytochange,sowedon’twanttohardcodethatdataintoourfunction,
isnteadwepassitinsothecallerofthefunctioncandecidethecorrecttagandtag_nameto
passin.Atthetimeofthiswriting,thesonglyricsareheldintheHTMLtag<div
class=“lyricbox”></div>.IftheHTMLtageverchanges,IwillupdateitonGitHub,sowatch
outforthatifthisprogramstopsworking.Hereishowwewouldcallourcurrentfunction:
get_lyrics('http://lyrics.wikia.com/wiki/The_Beatles:Lucy_In_The_Sky_With_Diamonds','div',
'lyricbox')
Nowwecancodeafunctiontocreateourwordcloud:
defcreate_wordcloud(wiki_url,file_name,tag,tag_name):
"""Takesaurlforalyricspageoflyrics.wikia.comandcreatesa
wordcloudfromthelyrics.
:paramwiki_url:stringlyrics.wikialyricsurle.g.
http://lyrics.wikia.com/wiki/The_Beatles:Girl.
"""
lyrics=get_lyrics(wiki_url,tag,tag_name)
wordcloud=WordCloud().generate(lyrics)
image=wordcloud.to_image()
image.show()
image.save(path.dirname(__file__)+'/wordcloud.jpg')
create_wordcloud('http://lyrics.wikia.com/wiki/The_Beatles:Lucy_In_The_Sky_With_Diamonds',‘wordcloud.jpg’,
‘div’,‘lyricbox’)
Westartbygettingthelyricsfromthewiki_urlwepassedinbycallingourget_lyrics()
functionandpassingitthewiki_url.Fromhereonout,theTheWordCloudmoduledoesall
thehardworkforus.WecreateaWordCloudobjectandpassitthelyrics.Nowwecanuse
themethodstheWordCloudobjecthastocreateourwordcloud.Thetricktoknowingwhat
methodsexistandfiguringouthowtousethemistoreadtheWordcloudmodule’s
documentation,availableat:https://github.com/amueller/word_cloud.
Firstwecallto_image()onourWordCloudobject,andthencallshow()ontheresult
tocreateourwordcloud.Thiscreatesawordcloudthatwillpopuponourdesktop,butwe
alsowanttosavethewordcloudasafile,sowecallsave()—whichtakesapathtowherewe
shouldsavethefileasaparameter,andsavesthefilethere;soifwewanttonameourfile
“my_file.txt”andsaveittoourdesktop,wewouldpassinsomethinglike
“/users/desktop/my_file.txt”.Inthiscase,wewanttosaveourfileinthefolderwhereour
programisrunning,sowepassinpath.dirname(__file__)(pathisafunctionfromtheos
modulewhichreturnsthepathwhereourprogramisrunning),andconcatenatethatwiththe
file_namepassedtotocreate_wordcloud()asaparameter—resultinginsomethinglike
“users/program/word_cloud.jpg”(whichiswhereourwordcloudwillbesaved).That’sall
thereistooit,ourfunctionnowcreatesabeautifulwordcloudbasedonthesonglyricsfrom
theurlwepassin.
Whenyouruntheprogram,itshouldcreateawordcloudthatpopsuponyourdesktop
(ifnotchecktheversionpostedonGitHubbecausethetagsmayhavechanged),savethefile
tothefolderwheretheprogramisrunning,andwhenyourunourtests,theyshouldallpass.
Hereisourfinishedcode:
fromwordcloudimportWordCloud
frombs4importBeautifulSoup
importrequests
fromosimportpath
defget_lyrics(wiki_url,tag,tag_name):
"""Takesaurlforalyricspageoflyrics.wikia.com,ataganda
tagnamesearchesforthattag
intheurlsHTMLthathasthetag_namepassedin.Returnsthesong
lyricsfound.
:paramwiki_url:stringlyrics.wikialyricsurle.g.
http://lyrics.wikia.com/wiki/The_Beatles:Girl.
:paramtag:stringHTMLtagtolookforlyricsine.g."div"
:paramtag_name:stringnameofHTMLtagtolookforlyricsine.g.
lyricbox
:return:stringsonglyrics
"""
response=requests.get(wiki_url)
soup=BeautifulSoup(response.text,'html.parser')
ifsoup.find_all(tag,tag_name):
returnsoup[0].text
defcreate_wordcloud(wiki_url,file_name,tag,tag_name):
"""Takesaurlforalyricspageoflyrics.wikia.comandcreatesa
wordcloudfromthelyrics.
:paramwiki_url:stringlyrics.wikialyricsurle.g.
http://lyrics.wikia.com/wiki/The_Beatles:Girl.
"""
lyrics=get_lyrics(wiki_url,tag,tag_name)
wordcloud=WordCloud().generate(lyrics)
image=wordcloud.to_image()
image.show()
image.save(path.dirname(__file__)+'/wordcloud.jpg')
create_wordcloud('http://lyrics.wikia.com/wiki/The_Beatles:Lucy_In_The_Sky_With_Diamonds','word_cloud.jpg',
'div',
'lyricbox')
Nowthatwe’vewrittenourprogram,thenextphaseisSystemTesting.We’vealready
writtenunittestsusingTestDrivenDevelopment,butwestillneedtowriteintegrationtests
andsystemtests,andhavepeopletesttheprogramasusers.Inthiscase,wearecreatinga
productforotherprogrammerstouse,soweshouldgetsomeprogrammerstotryoutour
program,andseeiftheyareabletoeasilyuseit.Wewon’twriteanymoretestsinthis
example,buttrytothinkabouthowyouwouldapproachthis.
ThefinalstepinourdevelopmentprocessisImplementationandDeployment.Sincewe
arebuildingaproductforotherprogrammers,puttingourprogramintoproductionmeans
uploadingittoGitHubandPyPi.Oncewedothat,itisavailableforotherprogrammersto
use,anditisuptoustocontinuouslycheckinonthecodeandprovidemaintenanceforthe
codeifanythingbreaks.IfyouareinterestedinlearninghowtouploadyourcodetoPyPiso
itisavailableonpip,followtheinstructionshere:http://peterdowns.com/posts/first-time-
with-pypi.html.
PartVILandaJob
ChapterX.YourFirstProgrammingJob
“Bewareof‘therealworld.’Aspeaker ’sappealtoitisalwaysaninvitationnottochallenge
histacitassumptions.”
—EdsgerW.Dijkstra
WelcometoPartVI,thefinalpartofthisbook,dedicatedtohelpingyouwithyour
careerasasoftwareengineer.Gettingyourfirstprogrammingjobrequiresextraeffort,butif
youfollowtheadviceinthischapter,youshouldhavenoproblem.Luckily,onceyouland
yourfirstprogrammingjobandgetsomeexperience,whenitcomestimetolookforyour
nextjob,recruiterswillstartreachingouttoyou.
ChooseaPath
Onethingtokeepinmindisthatprogrammingjobsarelumpedintospecificdomains:
eachwiththeirownsetoftechnologiesandskillsets.Ifyoulookatprogrammingjobads,the
headlinewillbesomethinglike“PythonBackendProgrammerWanted.”Thismeanstheyare
lookingforsomeonethatprogramsthebackendofawebsite,andisalreadyfamiliarwith
Python.Ifyougotothejobdescription,therewillbealistoftechnologiestheidealcandidate
willbefamiliarwith,alongwithskillstheyshouldhave.Whileit’sfinetobeageneralist(a
programmerthatdabblesineverything)whileyouarelearningtoprogram,anditispossible
togetajobasageneralistprogrammer,youprobablyshouldfindaspecificareaof
programmingyouenjoyandstartbecominganexpertinit.Thiswillmakegettingajob
significantlyeasier.
Webandmobiledevelopmentaretwoofthemostpopularprogrammingpaths.
Applicationdevelopmentisoftensplitintotwoparts—thefrontendandthebackend.The
frontendofanapplicationisthepartthatyoucansee—liketheinterfaceofawebapp.The
backendiswhatyoucan’tsee—thepartthatprovidesthefrontendwithdata.Some
companieswillhaveateamdedicatedtothefrontend,andateamdedicatedtothebackend.
Othercompaniesonlyhirefullstackdevelopers—programmersthatcanworkonboththe
frontandbackend.However,thisonlyappliestoapplicationdevelopment(buildingwebsites
orapps).Thereareallkindsofotherprogrammingareasyoucanworkinlikesecurity,
platformengineering,anddatascience.Tolearnmoreaboutthedifferentareasof
programming,gotositeslistingprogrammingjobs,andreadthroughthedescriptions.The
PythonJobBoardisagoodplacetostart:https://www.python.org/jobs.Lookatthe
requirementsforthedifferentjobs,aswellasthetechnologiestheyuse—thiswillgiveyouan
ideawhatyouneedtolearntobecompetitiveforthetypeofjobyouwant.
GettingInitialExperience
Inordertogetyourfirstprogrammingjobyouneedexperience.Buthowdoyouget
programmingexperienceifnoonewillhireyouwithoutit?Thereareafewwaystosolve
thisproblem.Onesolutionistofocusonopensource.Youcaneitherstartyourownopen
sourceproject,orcontributetothethousandsofopensourceprojectsonGitHub.
Anotheroptionistodofreelancework;youcancreateaprofileonsiteslikeUpwork,
andstartgettingsmallprogrammingjobsrightaway.Togetstarted,Irecommendfinding
someoneyouknowthatneedssomeprogrammingworkdone.Havethemsignupforan
Upworkaccountandofficiallyhireyoutheresotheycanleaveyouagreatreviewforyour
work.UntilyouhaveatleastonegoodreviewonasitelikeUpwork,itisdifficulttoland
jobs.Oncepeopleseethatyou’vesuccessfullycompletedatleastonejob,gettinghiredfor
jobsbypeopleyou’venevermetbecomeseasier,becauseyou’veestablishedsomec
redibility.
GettinganInterview
Onceyou’vegainedprogrammingexperiencethrougheitheropensourceorfreelance
workit’stimetostartinterviewing.I’vefoundthemosteffectivewaytogetaninterviewisto
focusonLinkedIn.Ifyoudon’thaveaLinkedInaccount,createonetogetstartednetworking
withpotentialemployers.Addasummaryaboutyourselfatthetopofofyourprofile,and
makesuretohighlightyourprogrammingskills.Forexample,alotofpeopleaddsomething
like“Languages:Python,JavaScript”atthetopoftheirprofile,whichhelpsrecruiters
searchingforthosekeywordsfindthem.Makesuretoaddyouropensourceorfreelancing
experienceasyourmostrecentjob.
Onceyourprofileiscomplete,startconnectingwithtechnicalrecruiters—therearetons
oftechnicalrecruitersonLinkedIn,sofindthemandsendthemarequesttoconnect.Theyare
alwayslookingfornewtalent,sotheywillbeeagertoconnectwithyou.Oncetheyaccept
yourinvitation,reachoutandaskiftheyhaveanyopenpositionsavailable.
TheInterview
Ifarecruiterthinksyoulooklikeagoodfitfortheroletheyarehiringfor,theywill
sendyouamessageonLinkedInaskingtosetupaphonescreen.Thephonescreenwillbe
withtherecruiter,soitisusuallynon-technical(althoughI’vehadrecruitersaskmesome
technicalquestionsthey’vememorizedtheanswertoduringthefirstinterview).The
conversationisaboutthetechnologiesyouknow,yourpreviousexperienceandseeingifyou
wouldfitintothecompany’sculture.
Ifyoudowell,youwilladvancetothesecondround—atechnicalphonescreen—where
youspeakwithoneormoremembersoftheengineeringteamandtheyaskyouthesame
questionsaboutyourexperienceandskillsasthefirstinterview.Thistimehowever,the
questionsarefollowedbyatechnicaltestoverthephone.Theengineerswillgiveyouthe
addressofawebsitewhereyoucaneditcode;theywillpostaprogrammingquestion;andask
youtosolveit.
Ifyoumakeitpastthesecondround—andthisnerve-rackingprocesshasn’tcausedyou
toabandonthisbookandquitprogramming—youwillhaveathirdinterview.Thethird
interviewisonsite,inpersonatthecompany’soffice.Thethirdroundisalotlikethefirst
two.Youmeetwithdifferentengineersontheteam,theyaskyouaboutyourskillsand
experience,andtherearemoretechnicaltests.Sometimesyouevenstayforlunchtoseehow
youinteractwiththeteam.Thethirdroundiswherethefamouswhiteboardcodingtests
happen.Ifthecompanyyouareinterviewingfordoeswhiteboarding,youwillbeasked
severalprogrammingproblemsandaskedtosolvethemonawhiteboard.Irecommend
buyingawhiteboardandpracticingthisbeforehandbecausesolvingaprogramming
problemonawhiteboardismuchharderthansolvingitonacomputer.Ofcoursenotall
companiesfollowthisexactformula,youmayrunintodifferentvariationsofit,butin
general,thisiswhatyoushouldexpect.
HackingTheInterview
Themajorityofprogramminginterviewsfocusontwosubjects—datastructuresand
algorithms.Thatmeanstopassyourprogramminginterview,youknowexactlywhatyou
mustdo—getverygoodattwospecificareasofComputerScience.Fortunately,thiswill
alsohelpyoutobecomeabetterprogrammer.
Youcannarrowdownthequestionsyoushouldfocusonevenfurtherbythinkingabout
theinterviewfromtheinterviewer's’perspective.Thinkaboutthesituationyourinterviewers
areisin;theysaysoftwareisneverfinished,andit’strue.Yourinterviewersmostlikelyhasa
lotofworktheyneedtogetdone,anddon’twanttodedicatealotofhistimetointerviewing
candidates.Comingupwithgoodprogrammingquestionsishard.Aretheygoingtospend
theirvaluabletimecomingupwithoriginalprogrammingquestions?Probablynot.Theyare
probablygoingtogoogle“programminginterviewquestions”andaskoneofthefirstones
theyfind.Thisleadstothesameinterviewquestionscomingupoverandoveragain—and
therearesomegreatresourcesouttheretopracticethem.LeetcodeisoneIrecommendyou
checkout.AlmosteveryquestionI’veeverbeenaskedinaprogramminginterviewison
Leetcode.
ChapterX.WorkingonaTeam
“Youcan’thavegreatsoftwarewithoutagreatteam,andmostsoftwareteamsbehavelike
dysfunctionalfamilies.”
-JimMcCarthy
Comingfromaself-taughtbackground,youareprobablyusedtoprogrammingalone.
Onceyoujoinacompany,youwillneedtolearnhowtoworkonateam.Evenifyoustart
yourowncompany,eventuallyyouwillneedtohireadditionalprogrammers,atwhichpoint
youwillalsoneedtolearntoworkasateam.Programmingisateamsportandlikeanyteam
sport,youneedtogetalongwithyourteammates.Thischapterprovidessometipsfor
successfullyworkinginateamenvironment.
MastertheBasics
Whenyouarehiredbyacompany,youareexpectedtobecompetentintheskills
coveredinthisbook.Itisnotenoughtosimplyreadthisbook—youneedtomastertheskills
init.Startasideprojectwithafriendanduseversioncontrol.Writenewtestseveryday.
Scheduleprogramstorun.Ifyourteammateshavetoconstantlyhelpyouwiththebasics,they
willbecomefrustrated.
Don’tAskWhatYouCanGoogle
Asanew,self-taughtmemberofaprogrammingteam,youwillhaveplentytolearn,
andyouwillneedtoaskalotofquestions.Thisisagreatwaytolearn,butyouwanttomake
sureyouareonlyaskinggoodquestions.Ageneralruleistoonlyaskaquestionifyou’ve
spentatleastfiveminutestryingtoGoogletheansweryourself.Whileaskingquestionsisa
positivething,ifyouasktoomanyquestionsyoucouldhaveeasilyfiguredoutyourself,you
willannoyyourteammates.Makesureyouareonlyaskinggoodquestionsbytryingtofind
theanswerforatleastfiveminutesbeforeyouaskit.
ChangingCode
Byreadingthisbook,you’vedemonstratedyouarethetypeofpersonwhoisconstantly
lookingtoimprove.Unfortunately,noteveryoneonyourteamwillshareyourenthusiasmfor
becomingabetterprogrammer.Manyprogrammersdon’thavethedesiretokeeplearning,
andtheyarefinedoingthingssuboptimally.Thiscanbeespeciallyprevalentinstartups,
whereshippingcodefastisoftenmoreimportantthanshippinghighqualitycode.Ifyoufind
yourselfinthissituation,youshouldlistentoWalterWhiteand“treadlightly.”Programmers
cangettheiregoshurtveryeasilywhenyouchangetheircode.Evenworse,ifyouspendalot
oftimefixingotherpeople’scode,youwillnothaveenoughtimetocontributetonew
projects,anditmaylooklikeyouarenotworkinghardenough.Thebestdefenseforthisisto
carefullyquestionanycompanyyoujoinabouttheirengineeringculture.Ifyoustillfind
yourselfinthissituation,itisbesttolistentoEdwardYourdon,“Ifyouthinkyour
managementdoesn’tknowwhatit’sdoingorthatyourorganisationturnsoutlow-quality
softwarecrapthatembarrassesyou,thenleave.”
Evenifyouworkatagreatcompany,therearestilltimeswhenyouwillneedtofix
otherpeople’scode,whichyoushouldapproachdelicately.MikeCoutermarshwrotean
articleonMediumcalled“Jr.Developers#5:HowtoImproveCodeWithoutAnyoneHating
You”whichIrecommendyoureadforadviceonthesubject.Asanewprogrammerentering
yourfirstprogrammingjob,it’simportantforyoutogetalongwithyourteam—alienating
yourteammemberscanhappenmucheasierthanitmayseem—somakesuretoalwaysstay
cognizantofhowyourteammateswillreactifyouwanttomakechangestotheircode.
ImposterSyndrome
Everyonethatprogramsfeelsoverwhelmedattimesand,nomatterhowhardyouwork
therearegoingtobethingsyoudon’tknow.Asaself-taughtprogrammer,itisespecially
easytofeelinadequatebecausesomeoneaskedyoutodosomethingyou’veneverheardof,
orbecauseyoufeelliketherearesomanyconceptsinComputerScienceyoustilldonot
understand.Remember—thishappenstoeveryone—notjustyou.
IwassurprisedwhenmyfriendwithaMastersdegreeinComputerSciencefrom
Stanfordtoldmehefeelsthiswayaswell.Hesaideveryoneinhisprogramdealtwith
impostersyndrome,andhenoticedtheyreactedinoneoftwoways:theyeitherstayedhumble
andwerealwayswillingtoadmitwhentheydidn’tknowsomething—andtriedlearnit—or
theypretendedtheykneweverything(whentheydidn’t)andstifledtheirlearning.Makesure
yourememberyougottowhereyouarebyworkinghard,andit’sokifyoudon’tknow
everything,nobodydoes.Juststayhumble,relentlesslystudyanythingyoudon’tunderstand,
andyouwillbeunstoppable.
FurtherLearning
“Thebestprogrammersarenotmarginallybetterthanmerelygoodones.Theyareanorder-
of-magnitudebetter,measuredbywhateverstandard:conceptualcreativity,speed,ingenuity
ofdesign,orproblem-solvingability.”
—RandallE.Stross
ThearticleABC:AlwaysBeCodingbyDavidByttowgivesgreatadviceonhowtogeta
jobasasoftwareengineer.Thearticleissummarizedinthetitle—alwaysbecoding.Ifyou
combineABCwithanewacronymImadeup—ABL—alwaysbelearning—youaresureto
haveanexceptionalcareer.Inthischapter,Iamgoingtogooversomeprogramming
resourcesI’vefoundhelpful.
TheClassics
Thereareafewprogrammingbooksthatareconsideredmustreads:ThePragmatic
ProgrammerbyAndyHuntandDaveThomas;DesignPatternsbyErichGamma,John
Vlissides,RalphJohnson,andRichardHelm(designpatternsareanimportantsubjectwe
didn’tgetachancetocover);CodeCompletebySteveMcConnell;Compilers:Principles,
Techniques,andTools,byAlfredAho,JeffreyUllman,MonicaS.LamandRaviSethi;and
IntroductiontoAlgorithmsbyTheMITPress.IalsohighlyreccomendProblemSolvingwith
DataStructuresandAlgorithms,afree,interactive,excellentintroductiontoalgorithms,
mucheasiertounderstandthanMIT’sIntroductiontoAlgorithms.
OnlineClasses
Onlinecodingclassesareapopularwaytolearntoprogram.Courseraisoneofthe
mostpopular,butIalsolovethelesserknownCodeschool.Ihighlyrecommendyoutake
someoftheirclasses.Notonlydotheyhavegreatclassesondifferentprogramming
languages,buttheyalsohaveclassesonGit,SQLandothersubjectsdiscussedinthisbook.
HackerNews
HackerNewsisaplatformforuser-submittednewsonthetechnologyincubatorY
Combinatorswebsitefoundathttps://news.ycombinator.com.Itispopularwithprogrammers
andwillhelpyoukeepuptodatewiththenewesttrendsandtechnologies.
Other
Thisarticleisgreatforfindingnewthingstolearn:http://matt.might.net/articles/what-cs-
majors-should-know.
NextSteps
Firstofall—thankyouforpurchasingthisbook.Ihopeit’shelpedyoubecomeabetter
programmer.Theprogrammingcommunityhasgivenmesomuchsupport,andbywriting
thisbook,IhopeI’vemanagedtohelpyouinyourjourneythewaysomanypeoplehelped
me.
Nowthatyou’vefinished,it’stimeforyoutogetdowntobusiness.Wheredoyougo
fromhere?Datastructuresandalgorithms,algorithmsalgorithms.GetonLeetCodeand
practicethosealgorithms.Thenpracticethemsomemore!InthischapterIgivesomefinal
thoughtsonhowyoucancontinuetoimproveasaprogrammer(onceyoufinishedpracticing
youralgorithms).
FindaMentor
Findingamentorwillhelpyoutakeyourprogrammingskillstothenextlevel.Oneof
thehardthingsaboutlearningtoprogramisthattherearesomanythingsyoucando
suboptimally,withoutknowingit.Imentionedearlieryoucanhelpcombatthisbydoingcode
reviews:amentorcandocodereviewswithyouandhelpyouimproveyourcodingprocess,
recommendbooks,andhelpyouwithprogrammingconceptsyouarehavingtrouble
understanding.
StrivetoGoDeep
Thereisaconceptinprogrammingcalleda“blackbox”thatreferstosomethingyou
use,butdonotunderstandhowitworks.Whenyoufirststartprogramming,everythingisa
blackbox.Oneofthebestwaystogetbetteratprogrammingistotrytoopenupeveryblack
boxyoucanfindandtrytounderstandhowitworks.Oneofmyfriendstoldmeitwasa
major“aha”momentforhimwhenherealizedthecommandlineitselfisaprogram.Opening
upablackboxiswhatIcallgoingdeep.
Writingthisbookhelpedmegodeep.TherewerecertainconceptsIthoughtI
understood,onlytofindoutIcouldn’texplainthem.Ihadtogodeep.Don’tjuststopatone
explanation,readalltheexplanationsonthetopicyoucanfind.Askquestionsandread
differingopinionsonline.
Ifindoneofthemosthelpfulquestionstobe“Whatproblemdoesthissolve?”For
example,welearnedaboutObject-orientedprogramminginthisbook.Butwhywasobject-
Orientedprogramminginvented?Whatproblemdoesitsolve?Arethereothersolutions?
Pursuingtheanswerstothesetypesofquestionswillhelpyoubecomeabetterprogrammer.
Anotherwaytogodeepistobuildthingsyouwanttounderstandbetter.Havingtrouble
understandingdatabases?Buildasimpledatabaseinyourfreetime.WhenIwashaving
troubleunderstandingcompilers—Ibuiltone.Takingthetimetodoaprojectlikethisiswell
worththeinvestment,becauseitwillimproveyourunderstandingofwhateveryouare
strugglingwith.
OtherAdvice
Ioncecameacrossaforumdiscussingwaystobecomeabetterprogrammer.Thetop
votedanswerwasasomewhatsurprising:“Dothingsotherthanprogramming.”I’vefound
thistobetrue—readingbookslikeTheTalentCodebyDanielCoylehasmademeabetter
programmer;becausehelaysoutexactlywhatyouneedtodotomasteranyskill.Another
goodbookaboutlearningaskillisMasterybyGeorgeLeonard.Keepyoureyeoutforthings
outsideofprogrammingyoucanbringtoyourprogramminggame.
ThelastpieceofadviceIwillleaveyouwithistospendasmuchtimeasyoucanreading
otherpeople’scode.Readingotherpeople’scodeisoneofthebestwaystoimproveasa
programmer.Whenyouarelearning,youneedtomakesuretostrikeabalancebetween
writingcode,andreadingcode.Readingotherpeople’scodeisgoingtobedifficultatfirst,
butitisimportanttodoitbecauseyoucanlearnsomuchfromotherprogrammers.
IhopeyouenjoyedthisbookasmuchasIenjoyedwritingit.Pleasefeelfreetoemail
meatcory@theselftaughtprogrammer.ioforanyreason.Ialsohaveaprogramming
newsletteryoucansignupforattheselftaughprogrammer.ioandaforumwhereyoucanget
intouchwithmeandacommunityofpeoplelearningtoprogram.Ifyoulikedthisbook,
pleasealsoconsiderleavingareviewonAmazon,ithelpsgetthisbookinthehandsofmore
people,andIreallyappreciateeveryreviewIreceive.Bestofluckontherestofyourjourney.
Andremember—ABL!
Acknowledgements
Parents,Pam,Randee,Anzar,CoverDesigner,Lauren,Antoine,Torrey,JinChun.
Glossary
Biography
Citations
0.http://stackoverflow.com/questions/466790/assembly-code-vs-machine-code-vs-
object-code
0.
0.http://man7.org/linux/man-pages/man1/ls.1.html
0.
0.http://www.webopedia.com/TERM/C/clock_speed.html
0.
0.http://stackoverflow.com/questions/1050222/concurrency-vs-parallelism-what-is-the-
difference
0.
0.http://berb.github.io/diploma-thesis/original/022_webapps.html
0.
0.http://stackoverflow.com/questions/11828270/how-to-exit-the-vim-editor
0.
0.http://superuser.com/questions/666942/why-it-is-not-recommend-to-use-root-login-
in-linux
0.
0.http://programmers.stackexchange.com/questions/37294/logging-why-and-what
0.
0.http://stackoverflow.com/questions/10925478/how-to-read-api-documentation-for-
newbs
0.
0.http://www.infoworld.com/article/2908474/application-development/stack-overflow-
survey-finds-nearly-half-have-no-degree-in-computer-science.html
0.
0.http://www.merriam-webster.com/dictionary/catenate
0.
0.DesignPatternskindlelocation546
0.
0.DesignPatternskindlelocation650
0.
0.DesignPatternskindlelocation824
0.
0.https://hungred.com/wp-content/uploads/2009/05/ascii-table-cheat-sheet1.png
0.
0.ThePragmaticProgrammerkindlelocation830
0.
0.https://automatetheboringstuff.com/chapter0/
0.
0.http://kevinlondon.com/2015/07/26/dangerous-python-functions.html
0.
0.
http://interactivepython.org/runestone/static/pythonds/Recursion/TheThreeLawsofRecursion.html
0.
0.https://automatetheboringstuff.com/chapter1/
0.https://www.quora.com/What-are-some-interesting-facts-about-computer-
programming
0.http://www.dkfindout.com/us/explore/eight-cool-facts-about-computer-coding/
0.http://thenextweb.com/insider/2016/02/26/8-facts-every-computer-programmer-
should-know/#gref
0.http://cs.lmu.edu/~ray/notes/x86assembly/
0.http://www.tutorialspoint.com/python/python_files_io.htm
0.http://www.wsj.com/articles/computer-programming-is-a-trade-lets-act-like-it-
1407109947?mod=e2fb
0.https://en.wikipedia.org/wiki/Persistence_(computer_science)
0.https://en.wikipedia.org/wiki/Column_family
0.http://stackoverflow.com/questions/2570756/what-are-database-constraints
0.https://en.wikipedia.org/wiki/Data_integrity
0.https://www.sitepoint.com/understanding-the-observer-pattern/
0.http://codedx.com/how-to-minimize-your-softwares-attack-surface/
0.https://support.microsoft.com/en-us/kb/283878
0.http://www.slideshare.net/jagaarj/database-design-normalization
0.http://www.tcpipguide.com/free/t_WhatIsNetworking.htm
0.Readwrite.com
0.http://www.gnu.org/software/grep/manual/grep.html
0.https://en.wikipedia.org/wiki/Software_testing
0.https://en.wikipedia.org/wiki/Boundary_case
0.http://www.guru99.com/integration-testing.html
0.https://en.wikipedia.org/wiki/Systems_development_life_cycle#cite_note-1
0.ProjectManagementInstitute,2013
0.http://www.tutorialspoint.com/sdlc/sdlc_overview.htm
0.https://en.wikipedia.org/wiki/Software_development_process
0.https://en.wikipedia.org/wiki/Waterfall_model
0.https://en.wikipedia.org/wiki/Incremental_build_model
0.http://agiledata.org/essays/tdd.html
0.http://www.agilenutshell.com/
0.https://maryrosecook.com/blog/post/a-practical-introduction-to-functional-
programming
0.http://stackoverflow.com/questions/1031273/what-is-polymorphism-what-is-it-for-
and-how-is-it-used
0.https://en.wikipedia.org/wiki/Syntax