物化视图(Materialized View)和普通视图(View)都是数据库中的虚拟表,但它们在实现和使用上有显著的不同。下面详细介绍它们的区别,并通过代码示例来解释它们的用法。
普通视图(View)
普通视图是基于SQL查询定义的虚拟表,它不存储实际数据,而是每次查询时动态生成结果集。普通视图适用于简化复杂查询、提高代码可读性和安全性。
创建普通视图
假设我们有一个包含销售记录的表sales
和一个包含客户信息的表customers
。
CREATE TABLE sales (
sale_id NUMBER,
sale_date DATE,
amount NUMBER,
customer_id NUMBER
);
CREATE TABLE customers (
customer_id NUMBER,
customer_name VARCHAR2(100)
);
我们可以创建一个普通视图来显示销售记录和对应的客户名称。
CREATE VIEW sales_view AS
SELECT s.sale_id, s.sale_date, s.amount, c.customer_name
FROM sales s
JOIN customers c ON s.customer_id = c.customer_id;
查询普通视图
每次查询普通视图时,数据库都会动态执行视图定义中的查询。
SELECT * FROM sales_view WHERE amount > 100;
物化视图(Materialized View)
物化视图是基于SQL查询定义的虚拟表,但它会将查询结果存储在数据库中。物化视图适用于需要频繁访问的大型查询结果集,可以显著提高查询性能。与普通视图不同,物化视图的数据是物理存储的,需要定期刷新以保持数据的最新状态。
创建物化视图
继续使用前面的sales
和customers
表,我们可以创建一个物化视图来存储销售记录和对应的客户名称。
CREATE MATERIALIZED VIEW sales_mv
AS
SELECT s.sale_id, s.sale_date, s.amount, c.customer_name
FROM sales s
JOIN customers c ON s.customer_id = c.customer_id;
查询物化视图
查询物化视图时,数据库直接访问预计算并存储的结果集,从而提高查询性能。
SELECT * FROM sales_mv WHERE amount > 100;
刷新物化视图
物化视图的数据需要定期刷新以保持最新状态。刷新方式可以是完全刷新(重新计算整个结果集)或增量刷新(只更新变化的数据)。
完全刷新
EXEC DBMS_MVIEW.REFRESH('sales_mv', 'C');
增量刷新
EXEC DBMS_MVIEW.REFRESH('sales_mv', 'F');
区别总结
特性 | 普通视图 (View) | 物化视图 (Materialized View) |
---|---|---|
数据存储 | 不存储实际数据,每次查询时动态生成结果集 | 存储实际数据,预计算并存储查询结果 |
查询性能 | 每次查询时执行视图定义的SQL,性能取决于原始表 | 查询性能较高,直接访问预计算并存储的结果集 |
数据一致性 | 总是最新的,因为每次查询时重新计算 | 可能不是最新的,需要定期刷新以保持数据一致性 |
维护开销 | 无需维护,动态生成结果 | 需要定期刷新以保持数据最新状态 |
使用场景 | 简化复杂查询、提高代码可读性和安全性 | 需要频繁访问的大型查询结果集,提高查询性能 |
代码示例总结
创建和使用普通视图
-- 创建普通视图
CREATE VIEW sales_view AS
SELECT s.sale_id, s.sale_date, s.amount, c.customer_name
FROM sales s
JOIN customers c ON s.customer_id = c.customer_id;
-- 查询普通视图
SELECT * FROM sales_view WHERE amount > 100;
创建和使用物化视图
-- 创建物化视图
CREATE MATERIALIZED VIEW sales_mv
AS
SELECT s.sale_id, s.sale_date, s.amount, c.customer_name
FROM sales s
JOIN customers c ON s.customer_id = c.customer_id;
-- 查询物化视图
SELECT * FROM sales_mv WHERE amount > 100;
-- 刷新物化视图
EXEC DBMS_MVIEW.REFRESH('sales_mv', 'C'); -- 完全刷新
EXEC DBMS_MVIEW.REFRESH('sales_mv', 'F'); -- 增量刷新
通过理解普通视图和物化视图的区别以及如何创建和使用它们,可以根据具体需求选择合适的视图类型来优化数据库查询性能和管理复杂查询。