V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
ksco
V2EX  ›  程序员

数据库的事务隔离问题,求指点

  •  
  •   ksco · 2016-08-19 14:01:57 +08:00 · 2211 次点击
    这是一个创建于 3017 天前的主题,其中的信息可能已经有所发展或是发生改变。

    数据库是 PostgreSQL ,程序一共有三个线程在操作数据库,两个线程不断执行 Insert 和 Update 操作,一个线程不断执行 Select 操作。

    如果操作比较频繁,就会出现 Insert 和 Select 都失败的情况。

    搜索了一番发现是因为没有事务隔离导致的冲突(对,我对数据库基本一无所知 (*´Д`*)

    参考了 事务隔离,对涉及 Insert 和 Update 的操作开头和结尾处分别加入了 BEGIN 和 COMMIT ,问题就没有了。

    现在想问的是,我这种解决方法是正确的吗?

    16 条回复    2016-08-21 01:34:12 +08:00
    xenme
        1
    xenme  
       2016-08-19 14:13:30 +08:00   ❤️ 1
    治标了:以前抢着来,出错了。
    现在有人锁了,就等着,排队来,后续可能会有性能问题。
    ksco
        2
    ksco  
    OP
       2016-08-19 14:20:29 +08:00
    @xenme 不会出现性能问题,因为每个表的最大数据量也不过几千条而已。

    对于不改变数据的 SELECT 操作,没有必要加上 BEGIN 和 COMMIT 吧?
    xenme
        3
    xenme  
       2016-08-19 14:23:56 +08:00   ❤️ 1
    @ksco 一般只有更新数据的时候才需要事务,所以 SELECT 一般 没必要加上
    ksco
        4
    ksco  
    OP
       2016-08-19 14:26:12 +08:00
    @xenme 好的,谢谢你
    rahuahua
        5
    rahuahua  
       2016-08-19 23:48:29 +08:00
    你确定加了 Begin , commit 就没问题了? insert/update/delete 是自带事务的,如果事务里只有一条 sql 语句,加和不加 begin/commit 是一样的。
    还有事务隔离是不会导致 select 失败的(如果你没有修改隔离级别到 repeated read 以上并且 select 是包在一个事务里面并且这个事务有修改其他的数据导致一致性检查失败)。
    ksco
        6
    ksco  
    OP
       2016-08-20 01:13:04 +08:00
    @rahuahua Insert 和 Update 操作都有多条语句的,不是单条。

    加上事务隔离之后 Select 确实就正常了,其它的操作也正常了。
    rahuahua
        7
    rahuahua  
       2016-08-20 08:34:27 +08:00
    @ksco
    加上事务隔离之后 Select 确实就正常了,其它的操作也正常了
    rahuahua
        8
    rahuahua  
       2016-08-20 08:34:44 +08:00
    rahuahua
        9
    rahuahua  
       2016-08-20 08:36:10 +08:00   ❤️ 1
    手滑了,囧

    加上事务隔离之后 Select 确实就正常了,其它的操作也正常了

    很好奇你是怎么加事务隔离的
    ksco
        10
    ksco  
    OP
       2016-08-20 10:17:40 +08:00 via iPhone
    @rahuahua 对于有数据更新(Update Insert Delete)的两个线程,我把每个完整的操作流程都封装在某个函数中,这些流程都涉及到多条 SQL 语句,所以我就在这些函数的开头和结尾部分分别加上了 Begin 和 Commit 操作,把它包装成一个事务。

    所以我想问这样能真正解决我遇到的问题吗?
    虽然目前来看是解决了。
    rahuahua
        11
    rahuahua  
       2016-08-20 12:07:52 +08:00   ❤️ 1
    @ksco 如果你没有修改事务隔离级别,只是将多个 SQL 放一个事务是能保证原子性,和事务隔离没有直接的关系。如果你修改隔离级别,那么事务一致性要求就会更严格( PG 默认的 read committed 是一致性保证最低的隔离级别),失败的概率应该更大才对
    ksco
        12
    ksco  
    OP
       2016-08-20 12:21:54 +08:00 via iPhone
    @rahuahua 原来如此,我没动过隔离级别的,应该就是默认的,谢谢!
    pathbox
        13
    pathbox  
       2016-08-20 20:44:45 +08:00
    mysql innodb 有事物 不会有你的这个问题吧
    ksco
        14
    ksco  
    OP
       2016-08-20 20:51:46 +08:00
    @pathbox 只要是 SQL 都会有这种问题吧
    alcarl
        15
    alcarl  
       2016-08-20 23:13:17 +08:00
    没用过 pg ,简单 google 了一下, pg 默认是隐式提交也就是自动提交。默认隔离级别是 readcommit 。按一般来说这种默认设置不应该报错才对。。。。。我很好奇为毛 select 也会报错,我很怀疑是有人改了默认配置。。。。。
    ksco
        16
    ksco  
    OP
       2016-08-21 01:34:12 +08:00
    @alcarl 嗯,我也觉得很奇怪,下周再看看。
    关于   ·   帮助文档   ·   博客   ·   API   ·   FAQ   ·   实用小工具   ·   2189 人在线   最高记录 6679   ·     Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 · 23ms · UTC 00:03 · PVG 08:03 · LAX 16:03 · JFK 19:03
    Developed with CodeLauncher
    ♥ Do have faith in what you're doing.