MySQL 触发器 - 5.7.23-0ubuntu0.16.04.1 - 奇怪的行为并非每次都有效

标签 mysql triggers

我使用tornado框架在python 3上开发了一个应用程序,该框架连接到一个巨大的MySQL数据库,里面有数百万行。

这个数据库是实时同步的,所以数据会插入、删除和更新,所以创建了3个触发器插入、更新和删除。

但是触发器的行为非常奇怪,有时它们起作用,有时它们不起作用,就像当它们决定拾取数据并做它们必须做的事情时,其他时候它们只是忽略它。

MySQL版本是这个5.7.23-0ubuntu0.16.04.1

我还插入了数据库中的 3 个触发器,也许是我的触发器中的逻辑错误:

插入

BEGIN
  IF (SELECT count(*) FROM USERS_STATS WHERE LOGIN = NEW.LOGIN) = 0 THEN
    INSERT INTO USERS_STATS (LOGIN) values (NEW.LOGIN);
  END IF;
  UPDATE USERS_STATS SET
    TOTAL_DEPOSIT = TOTAL_DEPOSIT + IF(NEW.CMD = 6 AND NEW.COMMENT NOT LIKE '%internal%' AND NEW.COMMENT NOT LIKE '%Tr %' AND NEW.COMMENT NOT LIKE '%agent%' AND NEW.COMMENT NOT LIKE '%rebate%' AND NEW.COMMENT NOT LIKE '%MAM:CASH%' AND NEW.PROFIT > 0, NEW.PROFIT, 0),
    TOTAL_WITHDRAW = TOTAL_WITHDRAW + IF(NEW.CMD = 6 AND NEW.COMMENT NOT LIKE '%internal%' AND NEW.COMMENT NOT LIKE '%Tr %' AND NEW.COMMENT NOT LIKE '%agent%' AND NEW.COMMENT NOT LIKE '%rebate%' AND NEW.COMMENT NOT LIKE '%MAM:CASH%' AND NEW.PROFIT < 0, NEW.PROFIT, 0),
    TOTAL_VOLUME = TOTAL_VOLUME + IF(NEW.CMD IN(0, 1) AND NEW.CLOSE_TIME > '1970-01-01', NEW.VOLUME, 0),
    TOTAL_REBATE = TOTAL_REBATE + IF((NEW.COMMENT LIKE '%agent%' OR NEW.COMMENT LIKE '%rebate%' OR NEW.COMMENT LIKE '%MAM:CASH%'), ROUND(NEW.PROFIT, 2), 0)
  WHERE LOGIN = NEW.LOGIN;
END

更新

BEGIN
  DECLARE old_is_deposit integer;
  DECLARE new_is_deposit integer;
  DECLARE old_is_withdraw integer;
  DECLARE new_is_withdraw integer;
  DECLARE old_is_rebate integer;
  DECLARE new_is_rebate integer;
  SET old_is_deposit := OLD.CMD = 6 AND OLD.COMMENT NOT LIKE '%internal%' AND OLD.COMMENT NOT LIKE '%Tr %' AND OLD.COMMENT NOT LIKE '%agent%' AND OLD.COMMENT NOT LIKE '%rebate%' AND OLD.COMMENT NOT LIKE '%MAM:CASH%' AND OLD.PROFIT > 0;
  SET new_is_deposit := NEW.CMD = 6 AND NEW.COMMENT NOT LIKE '%internal%' AND NEW.COMMENT NOT LIKE '%Tr %' AND NEW.COMMENT NOT LIKE '%agent%' AND NEW.COMMENT NOT LIKE '%rebate%' AND NEW.COMMENT NOT LIKE '%MAM:CASH%' AND NEW.PROFIT > 0;
  SET old_is_withdraw := OLD.CMD = 6 AND OLD.COMMENT NOT LIKE '%internal%' AND OLD.COMMENT NOT LIKE '%Tr %' AND OLD.COMMENT NOT LIKE '%agent%' AND OLD.COMMENT NOT LIKE '%rebate%' AND OLD.COMMENT NOT LIKE '%MAM:CASH%' AND OLD.PROFIT < 0;
  SET new_is_withdraw := NEW.CMD = 6 AND NEW.COMMENT NOT LIKE '%internal%' AND NEW.COMMENT NOT LIKE '%Tr %' AND NEW.COMMENT NOT LIKE '%agent%' AND NEW.COMMENT NOT LIKE '%rebate%' AND NEW.COMMENT NOT LIKE '%MAM:CASH%' AND NEW.PROFIT < 0;
  SET old_is_rebate := OLD.COMMENT LIKE '%agent%' OR OLD.COMMENT LIKE '%rebate%' OR OLD.COMMENT LIKE '%MAM:CASH%';
  SET new_is_rebate := NEW.COMMENT LIKE '%agent%' OR NEW.COMMENT LIKE '%rebate%' OR NEW.COMMENT LIKE '%MAM:CASH%';
  IF (SELECT count(*) FROM USERS_STATS WHERE LOGIN = NEW.LOGIN) = 0 THEN
    INSERT INTO USERS_STATS (LOGIN) values (NEW.LOGIN);
  END IF;
  IF (old_is_deposit = 1 AND new_is_deposit = 1 AND OLD.PROFIT != NEW.PROFIT) THEN
    UPDATE USERS_STATS SET
      TOTAL_DEPOSIT = TOTAL_DEPOSIT + (NEW.PROFIT - OLD.PROFIT)
    WHERE LOGIN = OLD.LOGIN;
  ELSEIF (old_is_withdraw = 1 AND new_is_withdraw = 1 AND OLD.PROFIT != NEW.PROFIT) THEN
    UPDATE USERS_STATS SET
      TOTAL_WITHDRAW = TOTAL_WITHDRAW + (NEW.PROFIT - OLD.PROFIT)
    WHERE LOGIN = OLD.LOGIN;
  ELSEIF (old_is_deposit = 1 AND new_is_withdraw = 1) THEN
    UPDATE USERS_STATS SET
      TOTAL_DEPOSIT = TOTAL_DEPOSIT - OLD.PROFIT,
      TOTAL_WITHDRAW = TOTAL_WITHDRAW + NEW.PROFIT
    WHERE LOGIN = OLD.LOGIN;
  ELSEIF (old_is_withdraw = 1 AND new_is_deposit = 1) THEN
    UPDATE USERS_STATS SET
      TOTAL_DEPOSIT = TOTAL_DEPOSIT + NEW.PROFIT,
      TOTAL_WITHDRAW = TOTAL_WITHDRAW - OLD.PROFIT
    WHERE LOGIN = OLD.LOGIN;
  ELSEIF (old_is_deposit = 0 AND new_is_deposit = 1) THEN
    UPDATE USERS_STATS SET
      TOTAL_DEPOSIT = TOTAL_DEPOSIT + NEW.PROFIT
    WHERE LOGIN = OLD.LOGIN;
  ELSEIF (old_is_deposit = 1 AND new_is_deposit = 0) THEN
    UPDATE USERS_STATS SET
      TOTAL_DEPOSIT = TOTAL_DEPOSIT - OLD.PROFIT
    WHERE LOGIN = OLD.LOGIN;
  ELSEIF (old_is_withdraw = 0 AND new_is_withdraw = 1) THEN
    UPDATE USERS_STATS SET
      TOTAL_WITHDRAW = TOTAL_WITHDRAW + NEW.PROFIT
    WHERE LOGIN = OLD.LOGIN;
  ELSEIF (old_is_withdraw = 1 AND new_is_withdraw = 0) THEN
    UPDATE USERS_STATS SET
      TOTAL_WITHDRAW = TOTAL_WITHDRAW - OLD.PROFIT
    WHERE LOGIN = OLD.LOGIN;
  END IF;
  IF (OLD.VOLUME != NEW.VOLUME) THEN
    UPDATE USERS_STATS SET
      TOTAL_VOLUME = TOTAL_VOLUME + (NEW.VOLUME - OLD.VOLUME)
    WHERE LOGIN = OLD.LOGIN;
  END IF;
  IF (old_is_rebate = 1 AND new_is_rebate = 1 AND OLD.PROFIT != NEW.PROFIT) THEN
    UPDATE USERS_STATS SET
      TOTAL_REBATE = TOTAL_REBATE + (ROUND(NEW.PROFIT, 2) - ROUND(OLD.PROFIT, 2))
    WHERE LOGIN = OLD.LOGIN;
  ELSEIF (old_is_rebate = 0 AND new_is_rebate = 1) THEN
    UPDATE USERS_STATS SET
      TOTAL_REBATE = TOTAL_REBATE + ROUND(NEW.PROFIT, 2)
    WHERE LOGIN = OLD.LOGIN;
  ELSEIF (old_is_rebate = 1 AND new_is_rebate = 0) THEN
    UPDATE USERS_STATS SET
      TOTAL_REBATE = TOTAL_REBATE - ROUND(OLD.PROFIT, 2)
    WHERE LOGIN = OLD.LOGIN;
  END IF;
END

删除

BEGIN
  IF (SELECT count(*) FROM USERS_STATS WHERE LOGIN = OLD.LOGIN) = 0 THEN
    INSERT INTO USERS_STATS (LOGIN) values (OLD.LOGIN);
  END IF;
  UPDATE USERS_STATS SET
    TOTAL_DEPOSIT = TOTAL_DEPOSIT - IF(OLD.CMD = 6 AND OLD.COMMENT NOT LIKE '%internal%' AND OLD.COMMENT NOT LIKE '%Tr %' AND OLD.COMMENT NOT LIKE '%agent%' AND OLD.COMMENT NOT LIKE '%rebate%' AND OLD.COMMENT NOT LIKE '%MAM:CASH%' AND OLD.PROFIT > 0, OLD.PROFIT, 0),
    TOTAL_WITHDRAW = TOTAL_WITHDRAW - IF(OLD.CMD = 6 AND OLD.COMMENT NOT LIKE '%internal%' AND OLD.COMMENT NOT LIKE '%Tr %' AND OLD.COMMENT NOT LIKE '%agent%' AND OLD.COMMENT NOT LIKE '%rebate%' AND OLD.COMMENT NOT LIKE '%MAM:CASH%' AND OLD.PROFIT < 0, OLD.PROFIT, 0),
    TOTAL_VOLUME = TOTAL_VOLUME - IF(OLD.CMD IN(0, 1) AND OLD.CLOSE_TIME > '1970-01-01', OLD.VOLUME, 0),
    TOTAL_REBATE = TOTAL_REBATE - IF((OLD.COMMENT LIKE '%agent%' OR OLD.COMMENT LIKE '%rebate%' OR OLD.COMMENT LIKE '%MAM:CASH%'), ROUND(OLD.PROFIT, 2), 0)
  WHERE LOGIN = OLD.LOGIN;
END

我还附上了触发器如何放置的屏幕截图: enter image description here

需要帮助为什么我的触发器无法正常工作。 预先感谢您。

最佳答案

所以最后我发现了一种不同的逻辑,在我的情况下无法摆脱死锁,因此我可以在触发器执行的一个时间间隔内更新所有必要的数据,而不是使用 cronjob 脚本来执行上述触发器。

关于MySQL 触发器 - 5.7.23-0ubuntu0.16.04.1 - 奇怪的行为并非每次都有效,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52741674/

相关文章:

php - 通过 ajax 将变量传递给外部 php 文件,然后将其插入到 mysql 数据库中

Oracle:搜索所有存储的过程/触发器/其他数据库代码?

linux - 在 PostgreSQL 函数中使用参数运行系统命令

jquery - 使用 jquery 事件 keydown 模拟在文本区域上输入不起作用

php - Codeigniter 删除连接子句中的 "("

mysql - mysql查询两列中的匹配值

sql - 我可以按多个列排序并以某种方式保持 MySQL 中列之间的排序相关吗?

java - 尝试连接 docker mysql 时,ResourcePool 无法从其主要工厂或源错误获取资源

MYSQL SELECT INSERT 触发器内

sql-server - 如何获取数据库上的事件触发器列表?