- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
考虑使用 Python 3.6.5 和 SQLAlchemy 1.2.7 的以下示例代码
import threading
from concurrent.futures import ThreadPoolExecutor
from sqlalchemy import create_engine, Column, Integer, Boolean
from sqlalchemy.exc import OperationalError
from sqlalchemy.ext.declarative import declarative_base
from sqlalchemy.orm import sessionmaker, scoped_session, Session
engine = create_engine("sqlite:///threading_sqlalchemy.db")
base = declarative_base(engine)
smaker = sessionmaker(engine)
scopedmaker: scoped_session = scoped_session(smaker)
dblock = threading.Lock()
class Key(base):
__tablename__ = "Key"
id = Column(Integer, primary_key=True)
value = Column(Integer, nullable=False, unique=True, index=True)
taken = Column(Boolean, nullable=False, default=False)
def __repr__(self):
return f"<Key id={self.id}, value={self.value}, taken={self.taken}>"
try:
Key.__table__.drop()
# this is also quite funny, if the table doesn't exist it throws:
# sqlite3.OperationalError: no such table: Key
# when there is literally a sqlalchemy.exc.NoSuchTableError
except OperationalError:
pass
base.metadata.create_all()
def gen_keys(n):
print(f"made in {threading.current_thread()}")
with dblock:
session: Session = scopedmaker()
session.bulk_save_objects([Key(value=i * 100) for i in range(0, n)])
session.commit()
def take_keys(n):
print(f"used in {threading.current_thread()}")
with dblock:
session: Session = scopedmaker()
keys = session.query(Key).filter(Key.taken == False).limit(n).all()
for key in keys:
key.taken = True
print(keys)
session.commit()
def take_keys_2(n):
print(f"used in {threading.current_thread()}")
with dblock:
session: Session = scopedmaker()
keys = session.query(Key).filter(Key.taken == False).limit(n).all()
for key in keys:
key.taken = True
session.commit()
print(keys)
gen_keys(100)
# take_keys works just as expected
with ThreadPoolExecutor() as executor:
for _ in range(0, 5):
executor.submit(take_keys, 10)
# take_keys_2 breaks, raises following error
# >>> sqlite3.ProgrammingError: SQLite objects created in a thread can only be used in that same thread.
# >>> The object was created in thread id 12340 and this is thread id 4312
# according to the console log, 12340 is one of the ThreadPoolExecutor threads, and 4312 is the main thread
with ThreadPoolExecutor() as executor:
for _ in range(0, 5):
executor.submit(take_keys_2, 10)
我只有一个非常简单的类(class) Key
它有一个值,可以标记为 taken
,想象一下类似赠品的东西,您不想将同一个赠品分发给不同的潜在客户。我用它来测试确实存在的竞争条件,并迫使我在数据库访问上使用锁,没什么大不了的,我可以忍受。
我真正不明白的是为什么take_keys
有效,但是 take_keys_2
当它们之间唯一的区别是 print(keys)
的位置时,就会中断。陈述。特别是因为在非功能性示例中,错误消息似乎是我在错误的线程中使用创建的对象(我不是,我只是在创建它的同一线程中的 session.commit()
之后使用它。
如果有人能解释为什么会发生这种情况,我会很高兴。
最佳答案
现在,我还没有掌握所有详细信息,但足以让您了解您的情况。 Threading support in SQLite不太好。因此,如果使用内存数据库,SQLAlchemy 的池行为默认为 SingletonThreadPool
;如果使用文件,则默认为 NullPool
。后者意味着根本没有池化,或者换句话说,连接始终根据请求打开和关闭。
print()
的位置很重要,因为上面对 session.commit()
的调用会使 session 中对象的所有数据库加载状态过期。因此,为了打印键列表(最终调用它们的__repr__
),SQLAlchemy 必须重新获取每个对象的状态。如果您将 echo=True
添加到对 create_engine()
的调用中,这一点就会变得很明显。
毕竟,您在 take_keys_2
中的 session
正在与开放事务保持连接。这就是它变得有点困惑的地方:当函数退出时, session 超出范围,这意味着它所持有的连接最终会返回到池中。但该池是一个 NullPool,因此它最终确定并关闭连接并丢弃它。最终确定意味着回滚任何打开的事务,这就是失败的原因:
Traceback (most recent call last):
File "~/Work/sqlalchemy/lib/sqlalchemy/pool.py", line 705, in _finalize_fairy
fairy._reset(pool)
File "~/Work/sqlalchemy/lib/sqlalchemy/pool.py", line 876, in _reset
pool._dialect.do_rollback(self)
File "~/Work/sqlalchemy/lib/sqlalchemy/engine/default.py", line 457, in do_rollback
dbapi_connection.rollback()
sqlite3.ProgrammingError: SQLite objects created in a thread can only be used in that same thread.The object was created in thread id 140683561543424 and this is thread id 140683635291968
最终确定是在解释器关闭期间在“虚拟”线程中执行的,而不是在工作线程中执行的,因为连接一直处于延迟状态。
例如,如果您在 print(keys)
之后添加对 session.rollback()
的调用:
def take_keys_2(n):
...
with dblock:
...
session.commit()
print(keys)
session.rollback()
the connection is returned to the pool explicitly ,并且 take_keys_2
也可以工作。另一种选择是使用 expire_on_commit=False
,这样提交后就不需要额外的查询来打印 Key
对象的表示:
def take_keys_2(n):
with dblock:
session: Session = scopedmaker(expire_on_commit=False)
...
关于python - SQLAlchemy 奇怪的线程行为,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/50578890/
我正在处理一组标记为 160 个组的 173k 点。我想通过合并最接近的(到 9 或 10 个组)来减少组/集群的数量。我搜索过 sklearn 或类似的库,但没有成功。 我猜它只是通过 knn 聚类
我有一个扁平数字列表,这些数字逻辑上以 3 为一组,其中每个三元组是 (number, __ignored, flag[0 or 1]),例如: [7,56,1, 8,0,0, 2,0,0, 6,1,
我正在使用 pipenv 来管理我的包。我想编写一个 python 脚本来调用另一个使用不同虚拟环境(VE)的 python 脚本。 如何运行使用 VE1 的 python 脚本 1 并调用另一个 p
假设我有一个文件 script.py 位于 path = "foo/bar/script.py"。我正在寻找一种在 Python 中通过函数 execute_script() 从我的主要 Python
这听起来像是谜语或笑话,但实际上我还没有找到这个问题的答案。 问题到底是什么? 我想运行 2 个脚本。在第一个脚本中,我调用另一个脚本,但我希望它们继续并行,而不是在两个单独的线程中。主要是我不希望第
我有一个带有 python 2.5.5 的软件。我想发送一个命令,该命令将在 python 2.7.5 中启动一个脚本,然后继续执行该脚本。 我试过用 #!python2.7.5 和http://re
我在 python 命令行(使用 python 2.7)中,并尝试运行 Python 脚本。我的操作系统是 Windows 7。我已将我的目录设置为包含我所有脚本的文件夹,使用: os.chdir("
剧透:部分解决(见最后)。 以下是使用 Python 嵌入的代码示例: #include int main(int argc, char** argv) { Py_SetPythonHome
假设我有以下列表,对应于及时的股票价格: prices = [1, 3, 7, 10, 9, 8, 5, 3, 6, 8, 12, 9, 6, 10, 13, 8, 4, 11] 我想确定以下总体上最
所以我试图在选择某个单选按钮时更改此框架的背景。 我的框架位于一个类中,并且单选按钮的功能位于该类之外。 (这样我就可以在所有其他框架上调用它们。) 问题是每当我选择单选按钮时都会出现以下错误: co
我正在尝试将字符串与 python 中的正则表达式进行比较,如下所示, #!/usr/bin/env python3 import re str1 = "Expecting property name
考虑以下原型(prototype) Boost.Python 模块,该模块从单独的 C++ 头文件中引入类“D”。 /* file: a/b.cpp */ BOOST_PYTHON_MODULE(c)
如何编写一个程序来“识别函数调用的行号?” python 检查模块提供了定位行号的选项,但是, def di(): return inspect.currentframe().f_back.f_l
我已经使用 macports 安装了 Python 2.7,并且由于我的 $PATH 变量,这就是我输入 $ python 时得到的变量。然而,virtualenv 默认使用 Python 2.6,除
我只想问如何加快 python 上的 re.search 速度。 我有一个很长的字符串行,长度为 176861(即带有一些符号的字母数字字符),我使用此函数测试了该行以进行研究: def getExe
list1= [u'%app%%General%%Council%', u'%people%', u'%people%%Regional%%Council%%Mandate%', u'%ppp%%Ge
这个问题在这里已经有了答案: Is it Pythonic to use list comprehensions for just side effects? (7 个答案) 关闭 4 个月前。 告
我想用 Python 将两个列表组合成一个列表,方法如下: a = [1,1,1,2,2,2,3,3,3,3] b= ["Sun", "is", "bright", "June","and" ,"Ju
我正在运行带有最新 Boost 发行版 (1.55.0) 的 Mac OS X 10.8.4 (Darwin 12.4.0)。我正在按照说明 here构建包含在我的发行版中的教程 Boost-Pyth
学习 Python,我正在尝试制作一个没有任何第 3 方库的网络抓取工具,这样过程对我来说并没有简化,而且我知道我在做什么。我浏览了一些在线资源,但所有这些都让我对某些事情感到困惑。 html 看起来
我是一名优秀的程序员,十分优秀!