• [牛客数据库SQL实战] 41~50题及个人解答


    41. 构造一个触发器audit_log,在向employees_test表中插入一条数据的时候,触发插入相关的数据到audit中

    • 这是题目所给的前置条件
    CREATE TABLE employees_test(
    	ID INT PRIMARY KEY NOT NULL,
    	NAME TEXT NOT NULL,
    	AGE INT NOT NULL,
    	ADDRESS CHAR(50),
    	SALARY REAL
    );
    CREATE TABLE audit(
    	EMP_no INT NOT NULL,
    	NAME TEXT NOT NULL
    );
    
    -- 在SQLite环境下,建立触发器
    CREATE TRIGGER audit_log AFTER INSERT ON employees_test
    BEGIN
        INSERT INTO audit VALUES(NEW.ID, NEW.NAME);
    END;
    

    运行时间:19ms

    占用内存:3304k

    • 注意:BEGIN ... END之间的语句是要加';'的,MySQL因为这方面还很复杂。。。

    • 在MySQL环境下

    -- 建立触发器
    CREATE TRIGGER audit_log
    AFTER INSERT
    ON employees_test FOR EACH ROW
    INSERT INTO audit VALUES(NEW.ID, NEW.NAME);
    
    -- 使用BEGIN ... END
    -- 无法在dbeaver客户端执行 不支持delimiter语法。。。
    -- 以下程序在命令行是可以执行的
    DELIMITER $$
    CREATE TRIGGER audit_log 
    AFTER INSERT ON employees_test FOR EACH ROW
    BEGIN
        INSERT INTO audit VALUES(NEW.ID, NEW.NAME);
    END$$
    DELIMITER ;
    

    MySQL官方文档:触发器实例

    42. 删除emp_no重复的记录,只保留最小的id对应的记录。

    • 这是题目所给的前置条件
    CREATE TABLE IF NOT EXISTS titles_test (
    	id int(11) not null primary key,
    	emp_no int(11) NOT NULL,
    	title varchar(50) NOT NULL,
    	from_date date NOT NULL,
    	to_date date DEFAULT NULL);
    
    insert into titles_test 
    values 
    ('1', '10001', 'Senior Engineer', '1986-06-26', '9999-01-01'),
    ('2', '10002', 'Staff', '1996-08-03', '9999-01-01'),
    ('3', '10003', 'Senior Engineer', '1995-12-03', '9999-01-01'),
    ('4', '10004', 'Senior Engineer', '1995-12-03', '9999-01-01'),
    ('5', '10001', 'Senior Engineer', '1986-06-26', '9999-01-01'),
    ('6', '10002', 'Staff', '1996-08-03', '9999-01-01'),
    ('7', '10003', 'Senior Engineer', '1995-12-03', '9999-01-01');
    
    -- 带条件的删除操作 删除重复emp_no 保留最小id
    DELETE FROM titles_test 
    WHERE id NOT IN (
    	SELECT id FROM (
    		SELECT MIN(id) AS id FROM titles_test GROUP BY emp_no
    	) AS tt_x
    );
    

    运行时间:15ms

    占用内存:3404k

    • 注意:以下语句在MySQL中是不执行的
    DELETE FROM titles_test 
    WHERE id NOT IN (
    	SELECT MIN(id) FROM titles_test GROUP BY emp_no
    );
    -- You can't specify target table 'titles_test' for update in FROM clause
    -- 不允许在删除条件内 使用要处理的表
    

    MySQL DELETE FROM with subquery as condition

    43. 将所有to_date为9999-01-01的全部更新为NULL,且 from_date更新为2001-01-01。

    • 这是题目所给的前置条件
    CREATE TABLE IF NOT EXISTS titles_test (
    	id int(11) not null primary key,
    	emp_no int(11) NOT NULL,
    	title varchar(50) NOT NULL,
    	from_date date NOT NULL,
    	to_date date DEFAULT NULL);
    
    insert into titles_test 
    values 
    ('1', '10001', 'Senior Engineer', '1986-06-26', '9999-01-01'),
    ('2', '10002', 'Staff', '1996-08-03', '9999-01-01'),
    ('3', '10003', 'Senior Engineer', '1995-12-03', '9999-01-01'),
    ('4', '10004', 'Senior Engineer', '1995-12-03', '9999-01-01'),
    ('5', '10001', 'Senior Engineer', '1986-06-26', '9999-01-01'),
    ('6', '10002', 'Staff', '1996-08-03', '9999-01-01'),
    ('7', '10003', 'Senior Engineer', '1995-12-03', '9999-01-01');
    
    -- 更新数据 UPDATE
    UPDATE titles_test 
    SET to_date=NULL, from_date='2001-01-01'
    WHERE to_date='9999-01-01';;
    

    运行时间:19ms

    占用内存:3432k

    • 注意:题目这里有个误区
      即to_date和from_date是否绑定在一起判断的
      1. 当to_date='9999-01-01'时,from_date才更新
      2. 还是只要是from_date都更新,不管to_date的值
        答案是不清楚的,因为所给的数据都很特殊,以上解题方法是按情况1,即两者绑定
    • 以下解法是符合情况2 即to_date有条件更新,from_date全更新
    UPDATE titles_test
    SET to_date=(
    	CASE to_date
    		WHEN '9999-01-01' THEN NULL
    		ELSE to_date
    	END
    ), from_date='2001-01-01';
    

    44. 将id=5以及emp_no=10001的行数据替换成id=5以及emp_no=10005,其他数据保持不变,使用replace实现。

    • 这是题目所给的前置条件
    CREATE TABLE IF NOT EXISTS titles_test (
    	id int(11) not null primary key,
    	emp_no int(11) NOT NULL,
    	title varchar(50) NOT NULL,
    	from_date date NOT NULL,
    	to_date date DEFAULT NULL);
    
    insert into titles_test 
    values 
    ('1', '10001', 'Senior Engineer', '1986-06-26', '9999-01-01'),
    ('2', '10002', 'Staff', '1996-08-03', '9999-01-01'),
    ('3', '10003', 'Senior Engineer', '1995-12-03', '9999-01-01'),
    ('4', '10004', 'Senior Engineer', '1995-12-03', '9999-01-01'),
    ('5', '10001', 'Senior Engineer', '1986-06-26', '9999-01-01'),
    ('6', '10002', 'Staff', '1996-08-03', '9999-01-01'),
    ('7', '10003', 'Senior Engineer', '1995-12-03', '9999-01-01');
    
    -- 使用replace更新数据 直接全字段更新
    REPLACE INTO titles_test
    VALUES('5', '10005', 'Senior Engineer', '1986-06-26', '9999-01-01');
    

    运行时间:18ms

    占用内存:3408k

    -- 使用replace函数更新数据 选择字段更新
    UPDATE titles_test
    SET emp_no=REPLACE(emp_no, 10001, 10005)
    WHERE id=5;
    

    运行时间:18ms

    占用内存:3420k

    • 此外,使用UPDATE语句也很方便
    -- 使用update更新
    UPDATE titles_test SET emp_no=10001 WHERE id=5;
    

    45. 将titles_test表名修改为titles_2017。

    • 这是题目所给的前置条件
    CREATE TABLE IF NOT EXISTS titles_test (
    	id int(11) not null primary key,
    	emp_no int(11) NOT NULL,
    	title varchar(50) NOT NULL,
    	from_date date NOT NULL,
    	to_date date DEFAULT NULL);
    
    insert into titles_test 
    values 
    ('1', '10001', 'Senior Engineer', '1986-06-26', '9999-01-01'),
    ('2', '10002', 'Staff', '1996-08-03', '9999-01-01'),
    ('3', '10003', 'Senior Engineer', '1995-12-03', '9999-01-01'),
    ('4', '10004', 'Senior Engineer', '1995-12-03', '9999-01-01'),
    ('5', '10001', 'Senior Engineer', '1986-06-26', '9999-01-01'),
    ('6', '10002', 'Staff', '1996-08-03', '9999-01-01'),
    ('7', '10003', 'Senior Engineer', '1995-12-03', '9999-01-01');
    
    -- SQLite3.7.9 修改表名的函数为RENAME TO
    ALTER TABLE titles_test RENAME TO titles_2017;
    

    运行时间:16ms

    占用内存:3556k

    • 注意:MySQL RENAME TORENAME 都可以

    46. 在audit表上创建外键约束,其emp_no对应employees_test表的主键id。

    • 这是题目所给的前置条件
    CREATE TABLE employees_test(
        ID INT PRIMARY KEY NOT NULL,
        NAME TEXT NOT NULL,
        AGE INT NOT NULL,
        ADDRESS CHAR(50),
        SALARY REAL
    );
    
    CREATE TABLE audit(
        EMP_no INT NOT NULL,
        create_date datetime NOT NULL
    );
    
    -- 这是看了讨论区 才知道的。。。
    -- 它是先删表 再建表 在建表的同时创建外键约束
    DROP TABLE audit;
    CREATE TABLE audit(
        EMP_no INT NOT NULL,
        create_date datetime NOT NULL,
        FOREIGN KEY(EMP_no) REFERENCES employees_test(ID));
    

    运行时间:17ms

    占用内存:3424k

    • 注意:OJ系统的坑,');'不要在单独一行。。。
    • 实际环境MySQL,只需修改添加即可
    -- 添加外键约束
    ALTER TABLE audit ADD FOREIGN KEY (emp_no) REFERENCES employees_test(id);
    

    47. 如何获取emp_v和employees有相同的数据?

    • 这是题目所给的前置条件
      存在如下的视图:
    create view emp_v as select * from employees where emp_no >10005;
    CREATE TABLE `employees` (
        `emp_no` int(11) NOT NULL,
        `birth_date` date NOT NULL,
        `first_name` varchar(14) NOT NULL,
        `last_name` varchar(16) NOT NULL,
        `gender` char(1) NOT NULL,
        `hire_date` date NOT NULL,
        PRIMARY KEY (`emp_no`));
    
    -- 最直观的 emp_v只是在employees的基础上生成的
    SELECT * FROM emp_v;
    

    运行时间:15ms

    占用内存:3416k

    -- 使用where
    SELECT em.*
    FROM employees AS em, emp_v AS ev 
    WHERE em.emp_no=ev.emp_no;
    

    运行时间:22ms

    占用内存:4564k

    -- 使用连接 找交集
    SELECT em.*
    FROM employees AS em 
    INNER JOIN emp_v AS ev 
    ON em.emp_no=ev.emp_no;
    

    运行时间:23ms

    占用内存:4588k

    48. 将所有获取奖金的员工当前的薪水增加10%。

    • 这是题目所给的前置条件
    create table emp_bonus(
    	emp_no int not null,
    	recevied datetime not null,
    	btype smallint not null);
    	CREATE TABLE `salaries` (
    	`emp_no` int(11) NOT NULL,
    	`salary` int(11) NOT NULL,
    	`from_date` date NOT NULL,
    	`to_date` date NOT NULL, PRIMARY KEY (`emp_no`,`from_date`));
    
    -- 借助其他表,更新薪水表数据 使得获得奖金的员工 薪水增长10%
    UPDATE salaries
    SET salary = salary*1.1
    WHERE emp_no IN (
    	SELECT emp_no FROM emp_bonus	
    ) AND to_date='9999-01-01';
    

    运行时间:21ms

    占用内存:3556k

    -- 上一更新方法是有瑕疵的 不够严谨
    UPDATE salaries
    SET salary = salary*1.1
    WHERE emp_no IN (
            -- 关键在这里 emp_bonus内的员工是有可能不存在薪水表内
    	SELECT s.emp_no 
    	FROM emp_bonus AS eb 
    	INNER JOIN salaries AS s 
            -- 注意加上当前条件 不然emp_no会出现重复的
    	ON eb.emp_no=s.emp_no AND s.to_date='9999-01-01'
    );
    

    运行时间:25ms

    占用内存:3408k

    49. 针对库中的所有表生成select count(*)对应的SQL语句

    -- SQLite获得所有表的索引 通过sqlite_master表的`name`列
    -- 本题需要注意小写。。。
    SELECT 'select count(*) from ' || name || ';' AS cnts
    FROM sqlite_master
    WHERE type='table';
    

    运行时间:19ms

    占用内存:4808k

    • 实际环境下
    -- 通过information_schema.TABLES获得所有数据库的表名,指定获取哪个数据库的表名
    SELECT table_name 
    FROM information_schema.TABLES 
    WHERE TABLE_SCHEMA='employees';
    -- 字符串拼接
    SELECT CONCAT('SELECT COUNT(*) FROM ', table_name, ';') AS cnts 
    FROM information_schema.TABLES 
    WHERE TABLE_SCHEMA='employees';
    

    MySQL 中的 information_schema 数据库

    50. 将employees表中的所有员工的last_name和first_name通过"'"连接起来

    -- 简单的字符串拼接 就是需要注意'的表达
    SELECT last_name || "'" || first_name FROM employees;
    

    运行时间:25ms

    占用内存:3404k

    • MySQL环境下
    -- 简单的字符串拼接
    SELECT CONCAT(last_name, "'", first_name) FROM employees;
    

    完整的个人练习代码

    我的练习SQL代码已经上传至Github:https://github.com/slowbirdoflsh/newcode-sql-practice
    仅供参考~~~

  • 相关阅读:
    WPF Image Binding Uri Source 失败解决办法
    redis哈希表数据类型键的设置
    redis字符串类型键的二进制操作
    redis字符串类型的基本命令
    redis字符串数据类型基本概念和应用场景
    redis键的迁移操作
    redis键的排序操作
    Redis键的序列化和反序列化
    Redis过期命令
    Redis键的基本操作
  • 原文地址:https://www.cnblogs.com/slowbirdoflsh/p/11223413.html
Copyright © 2020-2023  润新知