设为首页 收藏本站
查看: 695|回复: 0

[经验分享] PostgreSQL环境下,触发器的创建

[复制链接]

尚未签到

发表于 2016-11-20 07:49:25 | 显示全部楼层 |阅读模式
  前言:
  有时候我们向数据库表中插入数据时,需要在插入前保证数据的约束。有两种方法可以保证约束不被破坏,1、插入前手动检查数据,2、使用触发器。今天我们就用触发器来保证约束完整性。
  假设有如下两个关系

课程编号(course_name)上课教室(room)课程时间(time)
EnglishBO-101AM
HistoryBO-102AM
PhysicsBO-103PM
BiologyBO-104PM

                                   course关系

 


教师编号(id)教授课程(course_name)
10001 History
10002 English
10003 Physics
10001 Biology
  teaches关系
  现在要求实现约束“同一个教师不可能在同一时间在两个不同的教室上课”
  这就要求我们每次向teaches关系中插入的数据,必须保证教师不会在同一时间,出现在两个不同的教室。转换成SQL语句就是

NOT EXISTS (
SELECT id
FROM teaches NATURAL JOIN course
GROUP BY id, time
HAVING COUNT(id) > 1);
  创建触发器的SQL语句

CREATE TRIGGER example_trigger BEFORE INSERT ON teaches
FOR EACH ROW
EXECUTE PROCEDURE example_function();
  example_function定义如下

CREATE FUNCTION example_function()
RETURNS TRIGGER
LANGUAGE plpgsql
AS $fun_trigger$
BEGIN
IF EXISTS (
SELECT id
FROM (SELECT id, course_name FROM teaches UNION (SELECT NEW.id, NEW.course_name)) AS T NATURAL JOIN course
GROUP BY id, time
HAVING COUNT(id) > 1)
THEN
RAISE EXCEPTION 'CONSTRAINT ERROR: A teacher cannot be in two classrooms at the same time';
END IF;
RETURN NEW;
END;
$fun_trigger$;
  返回为TRIGGER(RETURNS TRIGGER)的存储过程中会自动创建一些变量
  NEW:INSERT/UPDATE 操作新记录
  OLD:DELETE/INSERT/UPDATE操作旧记录。更多的参数请参阅postgreSQL文档
  course关系的数据:

SELECT course_name, room, time FROM course;
DSC0000.jpg
 

  teaches关系的数据:

SELECT id, course_name FROM teaches;
DSC0001.jpg
 

  插入一条有冲突的语句:

INSERT INTO teaches(id, course_name) VALUES('10003', 'Finance');
DSC0002.jpg
 

  大功告成。
  附注:
  1.如果你需要在一个表(TABLE)更新(UPDATE)和插入(INSERT)时都触发触发器,SQL为

CREATE TRIGGER example_trigger BEFORE INSERT OR UPDATE ON teaches
FOR EACH ROW
EXECUTE PROCEDURE example_function();
  2.如果你需要在一个表(TABLE)的指定列(COLUMN)更新时触发触发器,SQL为

CREATE TRIGGER example_trigger BEFORE UPDATE OF course_name ON teaches
FOR EACH ROW
EXECUTE PROCEDURE example_function();
  3.如果你想用一个触发器(TRIGGER)作用在多个表上(TABLE),该怎么办呢?

CREATE TRIGGER example_trigger BEFORE INSERT ON teaches, course
FOR EACH ROW
EXECUTE PROCEDURE example_function();
  不幸的是,这样的语法在postgreSQL中是不行的,你只能这样

CREATE TRIGGER example_trigger1 BEFORE INSERT ON teaches
FOR EACH ROW
EXECUTE PROCEDURE example_function();
CREATE TRIGGER example_trigger2 BEFORE INSERT ON course
FOR EACH ROW
EXECUTE PROCEDURE example_function();
  参见点击查看详情
  结束语:以上的SQL语句全部是在PostgreSQL(9.4.4)中运行通过,由于各种数据库的语法的差别,在其他数据库中并不能保证不会报错。由于本人水平所限,如有错误,欢迎批评指正。
  参考文献:
  1. 《数据库系统概念》第六版 5.3节 触发器
  2. http://www.postgresql.org/docs/current/static/plpgsql-trigger.html
  3. http://www.postgresql.org/docs/9.4/static/sql-createtrigger.html
  4. http://www.tutorialspoint.com/postgresql/postgresql_triggers.htm
  5. http://stackoverflow.com/questions/16102188/postgresql-insert-trigger-to-set-value
  6. http://www.postgresql.org/message-id/17673.1129178350@sss.pgh.pa.us
  版权声明:本文为博主原创文章,未经博主允许不得转载。

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-302644-1-1.html 上篇帖子: 【Postgresql】字符串操作函数 下篇帖子: PostgreSQL 的昨天今天和明天 转载
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表