mirror of
https://github.com/python/cpython.git
synced 2025-08-03 08:34:29 +00:00
Initial revision
This commit is contained in:
parent
04691fc1c1
commit
e876949f2b
7 changed files with 488 additions and 0 deletions
73
Demo/classes/Complex.py
Executable file
73
Demo/classes/Complex.py
Executable file
|
@ -0,0 +1,73 @@
|
|||
# Complex numbers
|
||||
|
||||
|
||||
from math import sqrt
|
||||
|
||||
|
||||
def complex(re, im):
|
||||
return Complex().init(re, im)
|
||||
|
||||
|
||||
class Complex:
|
||||
|
||||
def init(self, re, im):
|
||||
self.re = float(re)
|
||||
self.im = float(im)
|
||||
return self
|
||||
|
||||
def __repr__(self):
|
||||
return 'complex' + `self.re, self.im`
|
||||
|
||||
def __cmp__(a, b):
|
||||
a = a.__abs__()
|
||||
b = b.__abs__()
|
||||
return (a > b) - (a < b)
|
||||
|
||||
def __float__(self):
|
||||
if self.im:
|
||||
raise ValueError, 'cannot convert complex to float'
|
||||
return float(self.re)
|
||||
|
||||
def __long__(self):
|
||||
return long(float(self))
|
||||
|
||||
def __int__(self):
|
||||
return int(float(self))
|
||||
|
||||
def __abs__(self):
|
||||
# XXX overflow?
|
||||
return sqrt(self.re*self.re + self.im*self.im)
|
||||
|
||||
def __add__(a, b):
|
||||
return complex(a.re + b.re, a.im + b.im)
|
||||
|
||||
def __sub__(a, b):
|
||||
return complex(a.re - b.re, a.im - b.im)
|
||||
|
||||
def __mul__(a, b):
|
||||
return complex(a.re*b.re - a.im*b.im, a.re*b.im + a.im*b.re)
|
||||
|
||||
def __div__(a, b):
|
||||
q = (b.re*b.re + b.im*b.im)
|
||||
re = (a.re*b.re + a.im*b.im) / q
|
||||
im = (a.im*b.re - b.im*a.re) / q
|
||||
return complex(re, im)
|
||||
|
||||
def __neg__(self):
|
||||
return complex(-self.re, -self.im)
|
||||
|
||||
|
||||
def test():
|
||||
a = complex(2, 0)
|
||||
b = complex(3, 4)
|
||||
print a, b
|
||||
print a+b, a-b, a*b, a/b
|
||||
print b+a, b-a, b*a, b/a
|
||||
i = complex(0, 1)
|
||||
print i, i*i, i*i*i, i*i*i*i
|
||||
j = complex(1, 1)
|
||||
print j, j*j, j*j*j, j*j*j*j
|
||||
print abs(j), abs(j*j), abs(j*j*j), abs(j*j*j*j)
|
||||
print i/-i
|
||||
|
||||
test()
|
71
Demo/classes/Dbm.py
Executable file
71
Demo/classes/Dbm.py
Executable file
|
@ -0,0 +1,71 @@
|
|||
# A wrapper around the (optional) built-in class dbm, supporting keys
|
||||
# and values of almost any type instead of just string.
|
||||
# (Actually, this works only for keys and values that can be read back
|
||||
# correctly after being converted to a string.)
|
||||
|
||||
|
||||
def opendbm(filename, mode, perm):
|
||||
return Dbm().init(filename, mode, perm)
|
||||
|
||||
|
||||
class Dbm:
|
||||
|
||||
def init(self, filename, mode, perm):
|
||||
import dbm
|
||||
self.db = dbm.open(filename, mode, perm)
|
||||
return self
|
||||
|
||||
def __repr__(self):
|
||||
s = ''
|
||||
for key in self.keys():
|
||||
t = `key` + ': ' + `self[key]`
|
||||
if s: t = t + ', '
|
||||
s = s + t
|
||||
return '{' + s + '}'
|
||||
|
||||
def __len__(self):
|
||||
return len(self.db)
|
||||
|
||||
def __getitem__(self, key):
|
||||
return eval(self.db[`key`])
|
||||
|
||||
def __setitem__(self, key, value):
|
||||
self.db[`key`] = `value`
|
||||
|
||||
def __delitem__(self, key):
|
||||
del self.db[`key`]
|
||||
|
||||
def keys(self):
|
||||
res = []
|
||||
for key in self.db.keys():
|
||||
res.append(eval(key))
|
||||
return res
|
||||
|
||||
def has_key(self, key):
|
||||
return self.db.has_key(`key`)
|
||||
|
||||
|
||||
def test():
|
||||
d = opendbm('@dbm', 'rw', 0666)
|
||||
print d
|
||||
while 1:
|
||||
try:
|
||||
key = eval(raw_input('key: '))
|
||||
if d.has_key(key):
|
||||
value = d[key]
|
||||
print 'currently:', value
|
||||
value = eval(raw_input('value: '))
|
||||
if value == None:
|
||||
del d[key]
|
||||
else:
|
||||
d[key] = value
|
||||
except KeyboardInterrupt:
|
||||
print ''
|
||||
print d
|
||||
except EOFError:
|
||||
print '[eof]'
|
||||
break
|
||||
print d
|
||||
|
||||
|
||||
test()
|
11
Demo/classes/README
Normal file
11
Demo/classes/README
Normal file
|
@ -0,0 +1,11 @@
|
|||
Examples of classes that implement special operators (see class.doc):
|
||||
|
||||
Complex.py Complex numbers
|
||||
Dbm.py Wrapper around built-in dbm, supporting arbitrary values
|
||||
Range.py Example of a generator: re-implement built-in range()
|
||||
Rat.py Rational numbers
|
||||
Vec.py A simple vector class
|
||||
|
||||
(For straightforward examples of basic class features, such as use of
|
||||
methods and inheritance, see the library code -- especially the window
|
||||
modules are full of them.)
|
72
Demo/classes/Range.py
Executable file
72
Demo/classes/Range.py
Executable file
|
@ -0,0 +1,72 @@
|
|||
# Example of a generator: re-implement the built-in range function
|
||||
# without actually constructing the list of values. (It turns out
|
||||
# that the built-in function is about 20 times faster -- that's why
|
||||
# it's built-in. :-)
|
||||
|
||||
|
||||
# Wrapper function to emulate the complicated range() arguments
|
||||
|
||||
def range(*a):
|
||||
if len(a) == 1:
|
||||
start, stop, step = 0, a[0], 1
|
||||
elif len(a) == 2:
|
||||
start, stop = a
|
||||
step = 1
|
||||
elif len(a) == 3:
|
||||
start, stop, step = a
|
||||
else:
|
||||
raise TypeError, 'range() needs 1-3 arguments'
|
||||
return Range().init(start, stop, step)
|
||||
|
||||
|
||||
# Class implementing a range object.
|
||||
# To the user the instances feel like immutable sequences
|
||||
# (and you can't concatenate or slice them)
|
||||
|
||||
class Range:
|
||||
|
||||
# initialization -- should be called only by range() above
|
||||
def init(self, start, stop, step):
|
||||
if step == 0:
|
||||
raise ValueError, 'range() called with zero step'
|
||||
self.start = start
|
||||
self.stop = stop
|
||||
self.step = step
|
||||
self.len = max(0, int((self.stop - self.start) / self.step))
|
||||
return self
|
||||
|
||||
# implement `x` and is also used by print x
|
||||
def __repr__(self):
|
||||
return 'range' + `self.start, self.stop, self.step`
|
||||
|
||||
# implement len(x)
|
||||
def __len__(self):
|
||||
return self.len
|
||||
|
||||
# implement x[i]
|
||||
def __getitem__(self, i):
|
||||
if 0 <= i < self.len:
|
||||
return self.start + self.step * i
|
||||
else:
|
||||
raise IndexError, 'range[i] index out of range'
|
||||
|
||||
|
||||
# Small test program
|
||||
|
||||
def test():
|
||||
import time, builtin
|
||||
print range(10), range(-10, 10), range(0, 10, 2)
|
||||
for i in range(100, -100, -10): print i,
|
||||
print
|
||||
t1 = time.millitimer()
|
||||
for i in range(1000):
|
||||
pass
|
||||
t2 = time.millitimer()
|
||||
for i in builtin.range(1000):
|
||||
pass
|
||||
t3 = time.millitimer()
|
||||
print t2-t1, 'msec (class)'
|
||||
print t3-t2, 'msec (built-in)'
|
||||
|
||||
|
||||
test()
|
86
Demo/classes/Rat.py
Executable file
86
Demo/classes/Rat.py
Executable file
|
@ -0,0 +1,86 @@
|
|||
# Rational numbers
|
||||
|
||||
|
||||
def rat(num, den):
|
||||
return Rat().init(num, den)
|
||||
|
||||
|
||||
def gcd(a, b):
|
||||
while b:
|
||||
a, b = b, a%b
|
||||
return a
|
||||
|
||||
|
||||
class Rat:
|
||||
|
||||
def init(self, num, den):
|
||||
if den == 0:
|
||||
raise ZeroDivisionError, 'rat(x, 0)'
|
||||
g = gcd(num, den)
|
||||
self.num = num/g
|
||||
self.den = den/g
|
||||
return self
|
||||
|
||||
def __repr__(self):
|
||||
return 'rat' + `self.num, self.den`
|
||||
|
||||
def __cmp__(a, b):
|
||||
c = a-b
|
||||
if c.num < 0:
|
||||
return -1
|
||||
if c.num > 0:
|
||||
return 1
|
||||
return 0
|
||||
|
||||
def __float__(self):
|
||||
return float(self.num) / float(self.den)
|
||||
|
||||
def __long__(self):
|
||||
return long(self.num) / long(self.den)
|
||||
|
||||
def __int__(self):
|
||||
return int(self.num / self.den)
|
||||
|
||||
def __coerce__(a, b):
|
||||
t = type(b)
|
||||
if t == type(0):
|
||||
return a, rat(b, 1)
|
||||
if t == type(0L):
|
||||
return a, rat(b, 1L)
|
||||
if t == type(0.0):
|
||||
return a.__float__(), b
|
||||
raise TypeError, 'Rat.__coerce__: bad other arg'
|
||||
|
||||
def __add__(a, b):
|
||||
return rat(a.num*b.den + b.num*a.den, a.den*b.den)
|
||||
|
||||
def __sub__(a, b):
|
||||
return rat(a.num*b.den - b.num*a.den, a.den*b.den)
|
||||
|
||||
def __mul__(a, b):
|
||||
return rat(a.num*b.num, a.den*b.den)
|
||||
|
||||
def __div__(a, b):
|
||||
return rat(a.num*b.den, a.den*b.num)
|
||||
|
||||
def __neg__(self):
|
||||
return rat(-self.num, self.den)
|
||||
|
||||
|
||||
def test():
|
||||
print rat(-1L, 1)
|
||||
print rat(1, -1)
|
||||
a = rat(1, 10)
|
||||
print int(a), long(a), float(a)
|
||||
b = rat(2, 5)
|
||||
l = [a+b, a-b, a*b, a/b]
|
||||
print l
|
||||
l.sort()
|
||||
print l
|
||||
print rat(0, 1)
|
||||
print rat(1, 0)
|
||||
print a+1
|
||||
print a+1L
|
||||
print a+1.0
|
||||
|
||||
test()
|
65
Demo/classes/Vec.py
Executable file
65
Demo/classes/Vec.py
Executable file
|
@ -0,0 +1,65 @@
|
|||
# A simple vector class
|
||||
|
||||
|
||||
def vec(*v):
|
||||
return apply(Vec().init, v)
|
||||
|
||||
|
||||
class Vec:
|
||||
|
||||
def init(self, *v):
|
||||
self.v = []
|
||||
for x in v:
|
||||
self.v.append(x)
|
||||
return self
|
||||
|
||||
|
||||
def fromlist(self, v):
|
||||
self.v = []
|
||||
if type(v) <> type([]):
|
||||
raise TypeError
|
||||
self.v = v[:]
|
||||
return self
|
||||
|
||||
|
||||
def __repr__(self):
|
||||
return 'vec(' + `self.v`[1:-1] + ')'
|
||||
|
||||
def __len__(self):
|
||||
return len(self.v)
|
||||
|
||||
def __getitem__(self, i):
|
||||
return self.v[i]
|
||||
|
||||
def __add__(a, b):
|
||||
# Element-wise addition
|
||||
v = []
|
||||
for i in range(len(a)):
|
||||
v.append(a[i] + b[i])
|
||||
return Vec().fromlist(v)
|
||||
|
||||
def __sub__(a, b):
|
||||
# Element-wise subtraction
|
||||
v = []
|
||||
for i in range(len(a)):
|
||||
v.append(a[i] - b[i])
|
||||
return Vec().fromlist(v)
|
||||
|
||||
def __mul__(self, scalar):
|
||||
# Multiply by scalar
|
||||
v = []
|
||||
for i in range(len(self.v)):
|
||||
v.append(self.v[i]*scalar)
|
||||
return Vec().fromlist(v)
|
||||
|
||||
|
||||
|
||||
def test():
|
||||
a = vec(1, 2, 3)
|
||||
b = vec(3, 2, 1)
|
||||
print a
|
||||
print b
|
||||
print a+b
|
||||
print a*3.0
|
||||
|
||||
test()
|
110
Demo/classes/class.doc
Executable file
110
Demo/classes/class.doc
Executable file
|
@ -0,0 +1,110 @@
|
|||
New features of classes
|
||||
=======================
|
||||
|
||||
A class can implement certain operations that are invoked by special
|
||||
syntax (such as subscription or arithmetic operations) by defining
|
||||
methods with special names.
|
||||
|
||||
|
||||
Special methods for any type
|
||||
----------------------------
|
||||
|
||||
__repr__(self) --> string
|
||||
|
||||
Used by the print statement and conversions (reverse quotes) to
|
||||
compute the string representation of an object.
|
||||
|
||||
__cmp__(self, other) --> int
|
||||
|
||||
Used by all comparison operations. Should return -1 if self<other, 0
|
||||
if self==other, +1 if self>other. Due to limitations in the
|
||||
interpreter, exceptions raised by comparisons are ignored, and the
|
||||
objects will be considered equal in this case.
|
||||
|
||||
|
||||
Special methods for sequence and mapping types
|
||||
----------------------------------------------
|
||||
|
||||
__len__(self) --> int
|
||||
|
||||
Used by the built-in function len(). Should return the length of the
|
||||
object, which should be >= 0. Also, an object whose __len__() method
|
||||
returns 0
|
||||
|
||||
__getitem__(self, key) --> value
|
||||
|
||||
Used to implement value = self[key]. Note that the special
|
||||
interpretation of negative keys (if the class wishes to emulate a
|
||||
sequence type) is up to the __getitem__ method.
|
||||
|
||||
__setitem__(self, key, value)
|
||||
|
||||
Used to implement self[key] = value. Same note as for __getitem__.
|
||||
|
||||
__delitem__(self, key)
|
||||
|
||||
Used to implement del self[key]. Same note as for __getitem__.
|
||||
|
||||
|
||||
Special methods for sequence types
|
||||
----------------------------------
|
||||
|
||||
__getslice__(self, i, j) --> sequence
|
||||
|
||||
Used to implement self[i:j]. Note that missing i or j are replaced by
|
||||
0 or len(self), respectively, and len(self) has been added to negative
|
||||
i or j.
|
||||
|
||||
__setslice__(self, i, j, sequence)
|
||||
|
||||
Used to implement self[i:j] = value. Same note as for __getslice__.
|
||||
|
||||
__delslice__(self, i, j)
|
||||
|
||||
Used to implement del self[i:j]. Same note as for __getslice__.
|
||||
|
||||
|
||||
Special methods for numeric types
|
||||
---------------------------------
|
||||
|
||||
__add__, __sub__, __mul__, __div__, __mod__, __divmod__, __pow__,
|
||||
__lshift__, __rshift__, __and__, __xor__, __or__
|
||||
|
||||
Used to implement the binary arithmetic operations (divmod and pow are
|
||||
called by built-in functions). All have the call pattern
|
||||
func(self, other) --> number.
|
||||
|
||||
__neg__, __pos__, __abs__, __invert__
|
||||
|
||||
Used to implement the unary arithmetic operations (-, +, abs and ~).
|
||||
All have the call pattern func(self) --> number.
|
||||
|
||||
__nonzero__(self) --> int
|
||||
|
||||
Used to implement boolean testing. An alternative name for this
|
||||
method is __len__.
|
||||
|
||||
__coerce__(self, other) --> (self1, other1) or None
|
||||
|
||||
Used to implement "mixed-mode" numeric arithmetic. Either return a
|
||||
tuple containing self and other converted to some common type, or None
|
||||
if no way of conversion is known. When the common type would be the
|
||||
type of other, it is sufficient to return None, since the interpreter
|
||||
will also ask the other object to attempt a coercion (but sometimes,
|
||||
if the implementation of the other type cannot be changed, it is
|
||||
useful to do the conversion to the other type here).
|
||||
|
||||
__int__(self) --> int
|
||||
__long__(self) --> long
|
||||
__float__(self) --> float
|
||||
|
||||
Used to implement the built-in functions int(), long() and float().
|
||||
|
||||
|
||||
Notes
|
||||
-----
|
||||
|
||||
Except for __repr__ and __cmp__, when no appropriate method is
|
||||
defined, attempts to execute the operation raise an exception. For
|
||||
__repr__ and __cmp__, the traditional interpretations are used
|
||||
in this case.
|
Loading…
Add table
Add a link
Reference in a new issue