我们知道在python的类里面可以定义三种类型的方法:
class A(object):
# 实例方法
def foo(self,x):
print "executing foo(%s,%s)"%(self,x)
# 类方法
@classmethod
def class_foo(cls,x):
print "executing class_foo(%s,%s)"%(cls,x)
# 静态方法
@staticmethod
def static_foo(x):
print "executing static_foo(%s)"%x
a = A()
这三个方法究竟有什么不同呢?
\ | 实例方法 | 类方法 | 静态方法 |
---|---|---|---|
a=A() | a.foo(x) | a.class_foo(x) | a.static_foo(x) |
A | 不可用 | A.class_foo(x) | A.static_foo(x) |
可以看到,实例是三种方法都可以调用的,而类只可以调用两种。所以这些方法的使用就是看你是否要让类不需要通过实例来调用方法,而类方法和静态方法的区别就是你是否要在该方法引导类的属性或者其他类的类方法。
看过flask web development 这本书的同学应该记得这一段代码:
from config import config
# ...
config[config_name].init_app(app)
其中config是一个字典,在config.py中定义好的,其键所对的是在config.py中定义好的类,代码如下:
class Config:
SECRET_KEY = os.environ.get('SECRET_KEY') or 'you will never know'
SQLALCHEMY_COMMIT_ON_TEARDOWN = True
FLASKY_MAIL_SUBJECT_PREFIX = '[lulalula]'
FLASKY_MAIL_SENDER = 'flasky Admin <dengbulieduo@163.com>'
FLASKY_ADMIN = os.environ.get('FLASKY_ADMIN')
FLASKY_POSTS_PER_PAGE = 10
FLASKY_FOLLOWERS_PER_PAGE = 10
FLASKY_COMMENTS_PER_PAGE = 10
SQLALCHEMY_RECORD_QUERIES = True
FLASKY_SLOW_DB_QUERY_TIME = 0.5
# init_app函数的函数体为空只是预留一个方法,方便调用
@staticmethod
def init_app(app):
pass
# ...
class ProductionConfig(Config):
SQLALCHEMY_DATABASE_URI = os.environ.get('DEV_DATABASE_URL') or \
'sqlite:///' + os.path.join(basedir, 'data.sqlite')
@classmethod
def init_app(cls, app):
Config.init_app(app)
import logging
from logging.handlers import SMTPHandler
credentials = None
secure = None
if getattr(cls, 'MAIL_USERNAME', None) is not None:
credentials = (cls.MAIL_USERNAME, cls.MAIL_PASSWORD)
if getattr(cls, 'MAIL_USE_TLS', None):
secure = ()
mail_handler = SMTPHandler(
mailhost=(cls.MAIL_USERNAME, cls.MAIL_PASSWORD),
fromaddr=cls.FLASKY_MAIL_SENDER,
toaddrs=[cls.FLASKY_ADMIN],
subject=cls.FLASKY_MAIL_SUBJECT_PREFIX + 'Application Error',
credentials=credentials,
secure=secure)
mail_handler.setLevel(logging.Error)
app.logger.addHandler(mail_handler)(Config):
SQLALCHEMY_DATABASE_URI = os.environ.get('DEV_DATABASE_URL') or \
'sqlite:///' + os.path.join(basedir, 'data.sqlite')
config = {
'development': DevelopmentConfig,
'testing': TestingConfig,
'production': ProductionConfig,
'default': DevelopmentConfig
}
可以知道如果config_name = 'production'的时候就会调用ProductionConfig这个类,而ProductionConfig继承与Config,所以有Config的所有属性。如果只是为了让类调用init_app方法,大可以像Config用静态方法,但是可以看到init_app里面调用了很多属性,所以只能用类方法来实现。这样我们就可以理清这三者之间的关系,和恰当的使用者这三种方法了。