掌握Python设计模式,SQL Alchemy打破ORM与模型类的束缚

发布时间:2024年01月12日

大家好,反转软件组件之间的依赖关系之所以重要,是因为它有助于降低耦合度和提高模块化程度,进而可以提高软件的可维护性、可扩展性和可测试性。

当组件之间紧密耦合时,对一个组件的更改可能会对其他组件产生意想不到的影响,从而导致整个系统中的连锁更改效应。通过反转依赖关系,可以确保组件仅依赖于抽象而不是具体实现来减轻这个问题。

就ORM和模型类而言,反转依赖关系允许ORM和模型类相互独立地发展,有助于使代码更加模块化、更具可维护性。例如,如果数据库模式发生更改,可以更新ORM以反映新的模式,而无需修改模型类。同样,如果想切换到不同的ORM,我们也可以这样做,而无需修改模型类。

此外,反转依赖关系还允许对每个组件进行单独测试,从而使代码更具可测试性。例如,可以为模型类编写单元测试,而无需设置数据库连接,因为模型类并不依赖于ORM。同样可以为ORM编写集成测试,而无需实例化模型类,因为ORM仅依赖于模型类的抽象。

1.代码示例

以下是一个使用Python中的SQLAlchemy反转ORM和模型类之间依赖关系的示例。首先使用元数据定义模式,然后定义模型类,最后使用映射器和关系设置ORM。

from?sqlalchemy?import?create_engine,?Column,?Integer,?String,?ForeignKey
from?sqlalchemy.orm?import?sessionmaker,?relationship,?mapper
from?sqlalchemy.ext.declarative?import?declarative_base,?declared_attr
from?sqlalchemy.schema?import?MetaData

#?使用元数据定义模式
metadata?=?MetaData()

class?UserTable:
????@declared_attr
????def?__tablename__(cls):
????????return?cls.__name__.lower()?+?'s'

????id?=?Column(Integer,?primary_key=True)
????name?=?Column(String)
????email?=?Column(String,?unique=True)

class?PostTable:
????@declared_attr
????def?__tablename__(cls):
????????return?cls.__name__.lower()?+?'s'

????id?=?Column(Integer,?primary_key=True)
????title?=?Column(String)
????content?=?Column(String)
????user_id?=?Column(Integer,?ForeignKey('users.id'))

#?定义模型类
class?User:
????def?__init__(self,?name,?email):
????????self.name?=?name
????????self.email?=?email

class?Post:
????def?__init__(self,?title,?content,?author):
????????self.title?=?title
????????self.content?=?content
????????self.author?=?author

#?使用映射器和关系设置ORM
engine?=?create_engine('sqlite:///example.db')
Session?=?sessionmaker(bind=engine)
Base?=?declarative_base()

class?UserORM(Base,?UserTable):
????@property
????def?model(self):
????????return?User(self.name,?self.email)

class?PostORM(Base,?PostTable):
????@property
????def?model(self):
????????return?Post(self.title,?self.content,?self.author.model)

????author?=?relationship(UserORM,?backref='posts')

Base.metadata.create_all(engine)

#?在应用程序中使用ORM和模型类
session?=?Session()


#?创建一些用户和帖子
user1?=?UserORM(name='Alice',?email='alice@example.com')
user2?=?UserORM(name='Bob',?email='bob@example.com')

post1?=?PostORM(title='My?first?post',?content='Hello,?world!',?author=user1)
post2?=?PostORM(title='Another?post',?content='This?is?a?test',?author=user2)

#?将用户和帖子保存到数据库
session.add(user1)
session.add(user2)

session.add(post1)
session.add(post2)

session.commit()

#?通过电子邮件地址检索用户
user?=?session.query(UserORM).filter_by(email='alice@example.com').one()
print(user.model.name)??#?打印出?"Alice"

#?检索用户的帖子
for?post?in?user.posts:
????print(post.model.title)??#?打印出?"My?first?post"

在这段代码中,首先使用元数据定义了模式,并为数据库中的每个表定义了单独的类。然后定义模型类UserPost,它们与相应的数据库表具有相同的属性。

接下来,使用映射器将模型类映射到数据库表,还定义了User模型和Post模型之间的关系,其中User模型具有一个posts属性,该属性是用户的帖子列表。

最后,在应用程序中使用ORM和模型类,创建一些用户和帖子,并将它们保存到数据库中。然后,通过电子邮件地址检索用户并打印他们的姓名,以及检索用户的帖子并打印它们的标题。

通过以这种方式反转ORM和模型类之间的依赖关系,使代码更加解耦且易于维护。我们可以更改数据库模式或ORM实现而不必修改模型类,反之亦然,这使得轻松测试模型类,而无需设置数据库连接。

此外,通过在ORM类中使用model属性来返回相应模型类的实例,进一步反转了依赖关系,因为现在ORM类依赖于模型类,而不是相反。

通过使用元数据定义模式、定义模型类并使用映射器和关系来设置ORM,可以实现ORM和模型类之间的依赖关系反转。通过这种方法,可以使代码更加解耦且易于维护。

2.反转依赖关系

在上面的代码中,通过以下方式实现了ORM和模型类之间的依赖关系反转:

  1. 使用元数据定义数据库模式:创建了一个metadata对象,并使用它来使用SQLAlchemy的表结构定义数据库模式。

  2. 定义模型类:定义了UserPost模型类,它们代表了系统中的实体,并定义这些实体的属性和行为。

  3. 使用映射器和关系设置ORM:使用SQLAlchemy的mapper函数将ORM类映射到第1步中定义的数据库模式。还使用SQLAlchemy的relationship函数定义ORM类之间的关系,例如UserPost之间的一对多关系。

通过这样做,将ORM类与数据库模式和模型类解耦,并使其依赖于SQLAlchemy提供的抽象,例如元数据和映射器函数。这使得修改数据库模式或切换到不同的ORM实现变得更容易,而无需修改模型类。它还使模型类更容易进行测试,因为可以在不需要实例化ORM类或连接到数据库的情况下对其进行隔离测试。

文章来源:https://blog.csdn.net/csdn1561168266/article/details/135541479
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。