Python-luento

Table of Contents

1 It's…

Käynnistys ja sammutus komentoriviltä tapahtuu seuraavasti.

$ python
Python 2.5 (r25:51908, Jan 23 2007, 09:20:42) 
[GCC 4.1.1 20070105 (Red Hat 4.1.1-51)] on linux2
Type "help", "copyright", "credits" or "license" for more information.
>>> 'hello world'
'hello world'
>>> print 'hello world'
hello world
>>> quit
Use quit() or Ctrl-D (i.e. EOF) to exit
>>> quit()

Yllä $ on komentorivikehote (Unix-tyylinen, Windowsissa erilainen). Jokainen Python-tulkin kehote alkaa merkeillä >>>, joiden jälkeen on näytetty syöte. Voit kirjoittaa syötteen ilman kehoitemerkkejä omaan Python-tulkkiisi ja rivinvaihdon jälkeen sen pitäisi viimeisen kehoterivin jälkeen tulostaa esimerkissä näkyvä teksti.

Graafisempi versio Python-tulkista löytyy yleensä nimellä idle. Ohjelmia voi ajaa joko tuplaklikkaamalla niiden ikoneita tai antamalla ohjelmatiedosto parametrina python-komennolle.

Komentorivillä auttaa funktio help. Sen kutsuminen ilman argumenttia käynnistää avustuskehotteen. Argumentin kanssa kutsuttuna funktio tulostaa argumenttiin liittyvän avustuksen. (Kokeile antaa syöte help() Pythonin kehotteelle.)

Kutsumalla funktiota quit() voit poistua tulkista. Graafiset versiot toimivat tässä suhteessa kuten muutkin graafiset ohjelmat.

Nykyisin käytössäsi on todennäköisesti Pythonin versio 2.7 tai 3.0. Nämä esimerkit toimivat molemmissa versiossa joitain mainittuja poikkeuksia lukuunottamatta.

1.1 Laskentaa

>>> 1+1
2
>>> 1+2*3
7
>>> 5/2
2
>>> 5/2.
2.5
>>> 2**256
115792089237316195423570985008687907853269984665640564039457584007913129639936L
>>> 2.**256
1.157920892373162e+77
>>> (0+1j) * (0+1j)
(-1+0j)
>>>

Liukuluvut ovat liukulukuja Pythonissakin:

>>> 1.0 + 2.0
3.0
>>> 10.0 + 20.0
30.0
>>> 0.1 + 0.2
0.30000000000000004
>>> from decimal import Decimal
>>> Decimal('0.1')+Decimal('0.2')
Decimal('0.3')
>>> 

Johtuen liukulukujen esitystavasta tietokoneessa, ne eivät koskaan ole tarkkoja lukuja. Pyöristysvirheitä syntyy helposti, kuten yllä nähdään. Tämän takia monista ohjelmointikielistä löytyy reaalilukujen aritmetiikkaa paremmin mallintavia tyyppejä, kuten Pythonin Decimal.

Tärkeintä on muistaa, että jo laskettaessa rahasummia voi ohjelmointikielen perusliukulukutyyppien käyttö aiheuttaa ikäviä yllätyksiä, joten rahamäärille kannattaa käyttää omaa luokkaansa.

1.2 Muuttujista

Muuttujat Pythonissa ovat käytännössä viitteitä. Ne antavat uuden nimen osoittamallensa oliolle, eivät varaa muistista tilaa, johon arvo sijoitetaan.

>>> a
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
NameError: name 'a' is not defined
>>> a = 2
>>> a
2
>>> a = 3.0
>>> a
3.0
>>> a = 'kissa istuu'
>>> a
'kissa istuu'
>>> a
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
NameError: name 'a' is not defined
>>> a = 2
>>> a
2
>>> a = 3.0
>>> a
3.0
>>> 

Jos kaksi muuttujaa osoittaa samaan muuttumattomaan olioon, ja toiseen sijoitetaan toinen olio, ei ensimmäinen muutu.

>>> a = 'kissa istuu'
>>> a
'kissa istuu'
>>> b = a
>>> b
'kissa istuu'
>>> b = 'istuuko kissa?'
>>> b
'istuuko kissa?'
>>> a
'kissa istuu'
>>> 

Jos kaksi muuttujaa osoittaa samaan muuttuvaan olioon, ja toisen muuttujan kautta muutetaan oliota, näkyy muutos myös toisen muuttujan kautta.

>>> a=[1,2,3]
>>> b=a
>>> b[0]=2
>>> b
[2, 2, 3]
>>> a
[2, 2, 3]
>>> 

Muuttujan sisältämän tai viittaaman arvon tyypin voi aina tarkistaa funktiolla type(). Tätä ei tule käyttää liikaa, se vain osoittaisi hutiloitua oliosuunnittelua.

>>> a=2
>>> type(a)
<type 'int'>
>>> a='kissa'
>>> type(a)
<type 'str'>
>>> type(2)
<type 'int'>
>>> type(type)
<type 'type'>
>>> type(Decimal)
<type 'type'>
>>> type(Decimal('1.0'))
<class 'decimal.Decimal'>
>>> 

1.3 Muita tietotyyppejä

Listat muuttuvien ja järjestettyjen tietojoukkojen käsittelyyn:

>>> lista=[1, 2, 'kolme', 4]
>>> lista[0]
1
>>> lista[2]
'kolme'
>>> lista[-1]
4
>>> lista[1:3]
[2, 'kolme']
>>> lista[3:]
[4]
>>> lista[:2]
[1, 2]
>>> lista[-2]=3
>>> lista
[1, 2, 3, 4]
>>> 

Monikot ovat kuten listat, mutta ne eivät voi muuttua: niiden koko pysyy samana, eikä edes alkioita voi muuttaa.

>>> monikko=(1, 2, 'kolme', 4)
>>> monikko
(1, 2, 'kolme', 4)
>>> monikko[0]
1
>>> monikko[-1]
4
>>> monikko[1:3]
(2, 'kolme')
>>> monikko[-2]=3
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: 'tuple' object does not support item assignment
>>> len(lista)
4
>>> len(monikko)
4
>>> lista.append(5)
>>> lista
[1, 2, 3, 4, 5]
>>> len(lista)
5
>>> monikko.append(5)
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AttributeError: 'tuple' object has no attribute 'append'
>>> monikko
(1, 2, 'kolme', 4)
>>> 

Joitain listaolioiden metodeja:

>>> lista.reverse()
>>> lista
[5, 4, 3, 2, 1]
>>> lista.sort()
>>> lista
[1, 2, 3, 4, 5]
>>> lista[-3]='kolme'
>>> lista
[1, 2, 'kolme', 4, 5]
>>> lista.sort()
>>> lista
[1, 2, 4, 5, 'kolme']
>>> lista[-1]=3
>>> lista.sort()
>>> lista.insert(0,0)
>>> lista
[0, 1, 2, 3, 4, 5]
>>> lista.pop()
5
>>> lista.append(5)
>>> lista
[0, 1, 2, 3, 4, 5]
>>> lista.extend([6,7])
>>> lista
[0, 1, 2, 3, 4, 5, 6, 7]
>>> lista.pop()
7
>>> lista
[0, 1, 2, 3, 4, 5, 6]
>>> del lista[-1]
>>> lista
[0, 1, 2, 3, 4, 5]
>>> lista.append(0)
>>> lista
[0, 1, 2, 3, 4, 5, 0]
>>> lista.count(0)
2
>>> lista.remove(0)
>>> lista
[1, 2, 3, 4, 5, 0]
>>> lista.reverse()
>>> lista
[0, 5, 4, 3, 2, 1]
>>> lista.sort()
>>> lista
[0, 1, 2, 3, 4, 5]
>>> lista[-3]='kolme'
>>> lista
[0, 1, 2, 'kolme', 4, 5]
>>> lista.sort()
>>> lista
[0, 1, 2, 4, 5, 'kolme']
>>> 

Hajautustaulut (sanasto, dictionary, hash table, map) ovat paljon käytetty tietorakenne. Avain, jonka perusteella arvo haetaan, voi olla lähes mitä tahansa tyyppiä (tyypille täytyy löytyä funktioiden hash ja cmp toteutus, tai vastaavien metodien toteutus).

>>> htaulu={}
>>> htaulu={'sata':100, 'tuhat':1000, 10:'kymmenen'}
>>> htaulu
{'sata': 100, 10: 'kymmenen', 'tuhat': 1000}
>>> htaulu[10]
'kymmenen'
>>> htaulu[100]
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
KeyError: 100
>>> htaulu['sata']
100
>>> htaulu.keys()
['sata', 10, 'tuhat']
>>> htaulu.values()
[100, 'kymmenen', 1000]
>>> htaulu.items()
[('sata', 100), (10, 'kymmenen'), ('tuhat', 1000)]
>>> htaulu.has_key(1)    
False
>>> htaulu[1]='yksi'
>>> htaulu
{'sata': 100, 10: 'kymmenen', 1: 'yksi', 'tuhat': 1000}
>>> 

Joukot sisältävät alkiot ilman järjestystä ja moninkertoja.

>>> joukko = set([1,1,2,3,5,8])
>>> joukko
set([8, 1, 2, 3, 5])
>>> joukko2 = set([3,4,5])
>>> joukko.union(joukko2)
set([1, 2, 3, 4, 5, 8])
>>> joukko.intersection(joukko2)
set([3, 5])
>>> joukko.difference(joukko2)
set([8, 1, 2])
>>> joukko2.difference(joukko)
set([4])
>>> joukko.symmetric_difference(joukko2)
set([1, 2, 4, 8])
>>> 

2 Rakenteiden määrittely

2.1 Funktion määrittely ja dokumentointi

>>> def summa(a,b):
...     "Palauttaa kahden argumenttinsa summan."
...     return a+b
... 
>>> summa(2,3)
5
>>> summa(0.1,0.2)
0.30000000000000004
>>> summa(Decimal('0.1'),Decimal('0.2'))
Decimal('0.3')
>>> summa('kissa ','istuu')
'kissa istuu'
>>> help(summa)

Help on function summa in module __main__:

summa(a, b)
    Palauttaa kahden argumenttinsa summan.
(END)
[Paina Q-näppäintä palataksesi kehoitteeseen.] 
>>> type(summa)
<type 'function'>
>>> 

Funktioita voi myös antaa argumentteina toisille funktioille ja palauttaa funktioiden arvoina. Funktio-ohjelmoijat ovat siis melkein kuin kotonaan Pythonin parissa.

2.2 Ehto ja toistolauseet: if, while, for, break, continue

>>> if 2>3:
...     print 'oletko aivan varma?'
... else:
...     print 'näin on!'
... 
näin on!
>>> 
>>> a=1
>>> while a<5:
...     print '%d. Spam' % a
...     a = a + 1
... 
1. Spam
2. Spam
3. Spam
4. Spam
>>> 
>>> menu = ('Spam','Spam','Spam', 'Spam')
>>> for i in menu:
...     print i
... 
Spam
Spam
Spam
Spam
>>> 
>>> a=1
>>> while True:
...     print '%d. Spam' % a
...     if a>4: break
...     elif a==2: 
...         print '...with Spam, please!'
...         a = a + 1
...     else:
...         a = a + 1
... 
1. Spam
2. Spam
...with Spam, please!
3. Spam
4. Spam
5. Spam
>>> 

2.3 Luokan määrittely ja dokumentointi, olion luonti

Vaihdetaan tässä vaiheessa hieman esitystapaa. Voit kirjoittaa allaolevat määrittelyt tulkin kehoitteella, mutta järkevämpää on käyttää erillistä editoria. Aiemmin mainittu idle tulee useimpien Python-asennusten mukana, mutta mikä tahansa Python-kielen tunnistava editori, siis useimmat editorit, on hyvä vaihtoehto. Itse käytän Emacsia ja sen Python-moodia. Tätä tehdessä olen käyttänyt Emacsin Org-moodia ja sen Babel-laajennosta. Vim, varsinkin Python-skriptauslaajennoksella, on myös oiva työväline. Laajennoksia löytyy myös muille, kuten Eclipselle, Netbeansille ja VisualStudiolle.

Rivinumeroita ei kirjoiteta, ne ovat vain olemassa, jotta voin viitata tekstissä jollekin tietylle riville esimerkissä.

Asiaan… Seuraavassa määritellään yksinkertainen henkilöluokka vanhalla ja uudella tavalla. Vanha tapa on vielä käytössä Pythonin versiossa 2, mutta Python 3 käyttää vain uutta tapaa. Version näet helpoiten Pythonin käynnistyksen yhteydessä, katso vaikka ihan ensimmäistä esimerkkiä tässä paperissa.

 1:  # -*- coding: iso-8859-15 -*-
 2:  # Tuo ylempi rivi kertoo, mitä merkistökoodausta lähdekoodissa käytetään.
 3:  # Kätevämpää kuin osaa ensin arvatakaan! Ääkkösilläkin on mahdollisuus
 4:  # tulostua oikein. Kannattaa käyttää samaa merkistökoodausta kuin
 5:  # tiedostojärjestelmässä, ja sen kannattaisi jo nykyään olla ainakin UTF-8.
 6:  # Python-tulkki kyllä murmuttaa, jos tuo ei ole kunnossa, ja neuvoo,
 7:  # mistä löytyy apu.
 8:  #
 9:  # Jos tuntui liian nörtiltä, kopioi vain ensimmäinen rivi tai älä käytä
10:  # ääkkösiä :)
11:  
12:  class Person:
13:      """Person is a class for objects holding a persons name and phone number."""
14:  
15:      def __init__(self, name, phone):
16:          self._name = name
17:          self._phone = phone
18:  
19:      def name(self):
20:          return self._name
21:      def set_name(self, name):
22:          self._name = name
23:          return self._name
24:  
25:      def phone(self, phone=None):
26:          """This style of attribute accessor combines
27:          the setter and getter in one method. Call it
28:          without arguments to get the value, and with
29:          single argument to set the value.
30:  
31:          This is also an example of multi-line documentation string :)
32:          Try help(Person) or help(Person.phone) to see the documentation.
33:          """
34:          if phone:
35:              self._phone = phone
36:          return self._phone
37:  
38:      def __str__(self):
39:          return 'Person name: %s phone: %s' % (self.name(), self.phone())
40:  
41:  class NewStylePerson(object): # New style class definition
42:      def __init__(self, name):
43:          self._name = name
44:  
45:      def get_name(self):
46:          return self._name
47:      def set_name(self, name):
48:          self._name = name
49:          return self._name
50:      name = property(get_name, set_name)
51:  
52:      # Property phone done likewise, left out for clarity.
53:  
54:      def __str__(self):
55:          return 'Person name: %s' % self.name
56:  
57:  
58:  donald = Person('Donald','555-313')
59:  daisy = NewStylePerson('Daisy')
60:  print donald
61:  print daisy
62:  print Person.__doc__
63:  # Remember to try help(Person)
64:  
Person name: Donald phone: 555-313
Person name: Daisy
Person is a class for objects holding a persons name and phone number.

Vanhan ja uuden ero on, että uuden mallisen luokkamäärittelyn tulee mainita yliluokkana object perintälistassa, eli suluissa uuden luokan nimen perässä, kts. NewStylePerson rivillä 41. Metodien ensimmäinen parametri tulee olla self, eli viite olioon itseensä. Lohkot merkitään sisennyksellä kuten aiemminkin. Metodi __init__() toimii rakentimena, ja kaikki olion attribuutit onkin määriteltävä siellä. Jos määrittelet attribuutin metodin ulkopuolella, siitä tulee luokan attribuutti.

Metodi __str__ vastaa Javan toString()-metodia, eli sen pitää palauttaa tulostamiskelpoinen ja ihmiselle lukukelpoinen merkkijono, joka kertoo tarpeelliset asiat oliosta.

Olio luodaan yksinkertaisesti antamalla luokan nimen perässä suluilla ympäröitynä olion rakentimen argumentit. Sulun ja luokan nimen välissä ei ole välilyöntiä. Palautuva olioviite tallennetaan muuttujaan normaalilla sijoituslauseella.

2.4 Poikkeuksellisesti: try … except

Tulossa myöhemmin.

3 Testilähtöinen ohjelmistokehitys

 1:  # -*- coding: iso-8859-15 -*-
 2:  import unittest
 3:  from money import Money
 4:  
 5:  class MoneyTest(unittest.TestCase):
 6:      def setUp(self):
 7:          self.m12CHF = Money(12, 'CHF')
 8:          self.m14CHF = Money(14, 'CHF')
 9:          self.m28USD = Money(28, 'USD')
10:      def testSimpleAdd(self):
11:          expected = Money(26, 'CHF')
12:          result = self.m12CHF.add(self.m14CHF)
13:          self.assertEquals(expected, result, 'not expected value')
14:  
15:  if __name__=='__main__':
16:      unittest.main()

Yllä on testilähtöisen kehityksen perinne-esimerkistä pätkä kirjoitettuna Pythonilla. Koetapa toteuttaa Money-luokka tuon avulla! Seuraava tehtävä: Tee keilailu-kata Pythonilla: http://butunclebob.com/files/downloads/Bowling%20Game%20Kata.ppt .

4 Moduulit ja paketit

 1:  # -*- coding: iso-8859-15 -*-
 2:  
 3:  # tiedosto: kala.py
 4:  
 5:  a=175
 6:  
 7:  def foo():
 8:      print 'foo!'
 9:  
10:  def bar():
11:      print 'bar!'
12:  
13:  # 
14:  bar()
15:  
16:  if __name__=='__main__':  # aja seuraavaa import-lauseessa
17:      foo()  
bar!
foo!

Jos halutaan rajoittaa osa tiedoston suorituksesta vain siihen, kun tiedosto ajetaan Pythonilla, se tehdään rivillä 16 alkavalla if-lohkolla.

Jos tämä puuttuu, ja tilalla on pelkkä unittest.main(), ajetaan tuo unittest.main() joka kerta kun tiedosto luetaan, eli esimerkiksi joka kerta kun tiedosto otetaan mukaan toiseen import komennolla.

Funktio dir listaa argumenttinsa attribuutit. Muista myös funktio help.

>>> import kala
>>> dir(kala)
['__builtins__', '__doc__', '__file__', '__name__', 'a', 'foo']
>>> dir()
['__builtins__', '__doc__', '__name__', 'kala']
>>> 

Moduulin attribuuttien tuominen osaksi käytössä olevaa nimiavaruutta ei yleensä ole järkevää, vaan kuormittaa nimiavaruuden turhilla määrityksillä:

>>> from kala import *
>>> dir(kala)
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
NameError: name 'kala' is not defined
>>> dir()
['__builtins__', '__doc__', '__name__', 'a', 'foo', 'os']
>>> 

Tuo vain tarpeellinen, tai sitten koko moduuli.

>>> from kala import foo
>>> dir(kala)
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
NameError: name 'kala' is not defined
>>> dir()
['__builtins__', '__doc__', '__name__', 'foo']
>>> 

Hätätapauksessa voit uudelleennimetä tuomasi kohteen, mutta tätä on syytä käyttää harkiten.

>>> from kala import foo as bar
>>> dir(kala)
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
NameError: name 'kala' is not defined
>>> dir()
['__builtins__', '__doc__', '__name__', 'bar']
>>> bar()
foo!
>>> 

4.1 Paketit

Paketit toteutetaan hakemistohierarkiana. Esimerkiksi paketti a.b.c.foo löytyy hakemistopolusta a/b/c/foo.py. Jokaiseen alihakemistoon tulee laittaa tiedosto __init__.py, joka voi olla tyhjäkin. Tiedostoon voi kuitenkin kirjoittaa paketin alustuskoodin, joka ajetaan, kun paketin sisältö otetaan käyttöön (import). Tiedostossa voi myös määritellä attribuutin __ALL__, joka kertoo, mitä paketista otetaan käyttöön, jos käytetään muotoa import * from paketti.

4.2 Esimerkkejä kirjastoista

Käydään läpi esimerkkejä Pythonin varsin kattavasta oheiskirjastosta. Leikkisästi voidaan sanoa, että jos jotain ei Pythonille löydy, sitä ei tarvita. (Eli kannattaa etsiä ennen kuin itse alkaa koodaamaan…)

5 Tieto talteen

>>> f=open('/tmp/kala.txt','wt')
>>> f.write('kissa istuu puussa\n')
>>> print >>f, 'jos kuu on juustoa, niin minä...'
>>> f.close()
$ cat /tmp/kala.txt 
kissa istuu puussa
jos kuu on juustoa, niin minä...
$
>>> l=('kissa istuu puussa\n','jos kuu on...\n')
>>> open('/tmp/kala-b.txt','wt').writelines(l)
$ cat /tmp/b
kissa istuu puussa
jos kuu on...
$
>>> f=open('/tmp/kala.txt','rt')
>>> f.read()
'kissa istuu puussa\njos kuu on juustoa, niin minä...\n'
>>> f.seek(0)
>>> f.readline()
'kissa istuu puussa\n'
>>> f.readline()
'jos kuu on juustoa, niin minä...\n'
>>> f.readline()
''
>>> f.read()
''
>>> f.seek(0)
>>> for rivi in f:
...     print rivi
... 
kissa istuu puussa

jos kuu on juustoa, niin minä...

>>> f.close()
>>> for rivi in open('/tmp/kala.txt','rt'):
...     print rivi
... 
kissa istuu puussa

jos kuu on juustoa, niin minä...

>>> for rivi in open('/tmp/kala.txt'):
...     print rivi
... 
kissa istuu puussa

jos kuu on juustoa, niin minä...

>>>
>>> from __future__ import with_statement
>>> with open('/tmp/kala.txt') as f:
...     for rivi in f:
...             print rivi
... 
kissa istuu puussa

jos kuu on juustoa, niin minä...

>>> 

5.1 Tiedon sarjallistaminen

>>> class foo:
...     def __init__(self, x):
...             self.__x = x
...     def __repr__(self):
...             return '<foo@%x arvolla %d.>' % (id(self), self.__x)
... 
>>> data=(1, 2.0, 'kissa', [4, 5, 6], foo(16))
>>> data
(1, 2.0, 'kissa', [4, 5, 6], <foo@2aaaae4c7e60 arvolla 16.>)
>>> import pickle
>>> pickle.dump(data, open('/tmp/dumppi','w'))
>>> pickle.load(open('/tmp/dumppi')) 
(1, 2.0, 'kissa', [4, 5, 6], <foo@2aaaaeb51a70 arvolla 16.>)
>>> 

5.2 Liitynnät tietokantoihin

Seuraavat esimerkit ovat (lähes) suoraan Pythonin sqlite3-moduulin dokumentaatiosta.

 1:  import sqlite3
 2:  
 3:  conn = sqlite3.connect('/tmp/example')
 4:  c = conn.cursor()
 5:  
 6:  # Create table
 7:  c.execute('''create table stocks
 8:  (date text, trans text, symbol text,
 9:   qty real, price real)''')
10:  
11:  # Insert a row of data
12:  c.execute("""insert into stocks
13:            values ('2006-01-05','BUY','RHAT',100,35.14)""")
14:  
15:  
16:  ##  ***  NEVER do this -- INSECURE!!!!  ***
17:  ## symbol = 'IBM'
18:  ## c.execute("... where symbol = '%s'" % symbol)
19:  
20:  # Do this instead
21:  t = (symbol,)
22:  c.execute('select * from stocks where symbol=?', t)
23:  
24:  # Larger example
25:  for t in (('2006-03-28', 'BUY', 'IBM', 1000, 45.00),
26:            ('2006-04-05', 'BUY', 'MSOFT', 1000, 72.00),
27:            ('2006-04-06', 'SELL', 'IBM', 500, 53.00),
28:           ):
29:      c.execute('insert into stocks values (?,?,?,?,?)', t)
>>> import sqlite3
>>> conn = sqlite3.connect('/tmp/example')
>>> c = conn.cursor()
>>> c.execute('select * from stocks order by price')
>>> for row in c:
...    print row
...
(u'2006-01-05', u'BUY', u'RHAT', 100, 35.140000000000001)
(u'2006-03-28', u'BUY', u'IBM', 1000, 45.0)
(u'2006-04-06', u'SELL', u'IBM', 500, 53.0)
(u'2006-04-05', u'BUY', u'MSOFT', 1000, 72.0)
>>>

Lähde: http://docs.python.org/lib/module-sqlite3.html

 1:  # A minimal SQLite shell for experiments
 2:  
 3:  import sqlite3
 4:  
 5:  con = sqlite3.connect(":memory:")
 6:  con.isolation_level = None
 7:  cur = con.cursor()
 8:  
 9:  buffer = ""
10:  
11:  print "Enter your SQL commands to execute in sqlite3."
12:  print "Enter a blank line to exit."
13:  
14:  while True:
15:      line = raw_input()
16:      if line == "":
17:          break
18:      buffer += line
19:      if sqlite3.complete_statement(buffer):
20:          try:
21:              buffer = buffer.strip()
22:              cur.execute(buffer)
23:  
24:              if buffer.lstrip().upper().startswith("SELECT"):
25:                  print cur.fetchall()
26:          except sqlite3.Error, e:
27:              print "An error occurred:", e.args[0]
28:          buffer = ""
29:  
30:  con.close()

Lähde: http://docs.python.org/lib/sqlite3-Module-Contents.html

Yleisesti Pythonille tehtyjen tietokantaliityntöjen tulisi toteuttaa API, joka on esitelty osoitteessa http://www.python.org/dev/peps/pep-0249/. Yllä esitelty sqlite3-moduuli on yksi esimerkki Python-DB-rajapinnan toteutuksesta.

Pythonille tietokantaa etsiessä kannattaa myös vilkaista ZOPE-sovelluspalvelimen1 tietokantaa ZODB2. ZODB on oliotietokanta, joten ainakin puhtaat oliosovellukset hyötyvät sen käytöstä. Seuraava esimerkki on ZODB:n ohjeesta osoitteesta http://wiki.zope.org/ZODB/guide/node3.html.

 1:  from ZODB import FileStorage, DB
 2:  
 3:  storage = FileStorage.FileStorage('/tmp/test-filestorage.fs')
 4:  db = DB(storage)
 5:  conn = db.open()
 6:  
 7:  from persistent import Persistent
 8:  
 9:  class User(Persistent):
10:      pass
11:  
12:  dbroot = conn.root()
13:  
14:  # Ensure that a 'userdb' key is present 
15:  # in the root
16:  if not dbroot.has_key('userdb'):
17:      from BTrees.OOBTree import OOBTree
18:      dbroot['userdb'] = OOBTree()
19:  
20:  userdb = dbroot['userdb']
21:  
22:  # Create new User instance
23:  import transaction
24:  
25:  newuser = User() 
26:  
27:  # Add whatever attributes you want to track
28:  newuser.id = 'amk' 
29:  newuser.first_name = 'Andrew' ; newuser.last_name = 'Kuchling'
30:  ...
31:  
32:  # Add object to the BTree, keyed on the ID
33:  userdb[newuser.id] = newuser
34:  
35:  # Commit the change
36:  transaction.commit()
>>> newuser
<User instance at 81b1f40>
>>> newuser.first_name           # Print initial value
'Andrew'         
>>> newuser.first_name = 'Bob'   # Change first name
>>> newuser.first_name           # Verify the change
'Bob'
>>> transaction.abort()          # Abort transaction
>>> newuser.first_name           # The value has changed back
'Andrew'
>>>

Footnotes:

1 http://www.zope.org/ Pythonilla tehty WWW-sovelluspalvelin. [fn:zodb]

2 http://wiki.zope.org/ZODB/FrontPage Tietokantatoteutus ZOPE:n alla.

Date: 2012-04-11 09:53:45 EEST

Author: Jonne Itkonen

Org version 7.8.03 with Emacs version 23

Validate XHTML 1.0