引言:数据库领域的 “最佳桥梁”
在当今多元化的数据库技术栈中,企业常常面临一个两难选择:是拥抱开源、生态丰富、性能卓越的 PostgreSQL,还是沿用商业强大、生态成熟但成本高昂的 Oracle 数据库?迁移的复杂性、高昂的许可费用以及对现有应用的兼容性担忧,往往让企业在技术选型和迁移上举步维艰。
IvorySQL 正是在这一背景下应运而生。它并非一个简单的数据库,而是一个旨在架起 PostgreSQL 与 Oracle 之间桥梁的革新性开源项目。它基于 PostgreSQL 开发,在保持与原版数据库 100% 语法兼容的同时,深度兼容 Oracle 语法,为企业提供了一条平滑、低成本的迁移与融合路径。
核心定位:生于 PostgreSQL,优于 Oracle 兼容
IvorySQL 的核心哲学可以概括为:以 PostgreSQL 为基石,以 Oracle 兼容性为核心。
坚守开源与 PostgreSQL 生态
IvorySQL 社区明确承诺,始终与 PostgreSQL 数据库保持 100% 兼容,并可直接替换最新版本的 PostgreSQL。这意味着 PostgreSQL 强大的扩展生态系统(如 PostGIS、pgvector、pg_cron 等)在 IvorySQL 中得以完整保留,用户无需担心失去 PostgreSQL 带来的灵活性、完整性和可靠性。
强大的 Oracle 兼容性
这是 IvorySQL 区别于其他基于 PostgreSQL 的分支的核心竞争力。它不仅仅是对少量 Oracle 函数和语法的简单映射,而是提供了一个系统性、深层次的兼容方案。其关键在于 ivorysql.compatible_mode 这个 GUC 参数。通过将其设置为 oracle 或 pg,用户可以在同一套数据库系统内,无缝切换 Oracle 兼容模式和原生 PostgreSQL 模式,这在数据库产品中是极为创新的设计。
深度技术解析:如何实现 “无缝兼容”?
IvorySQL 的 Oracle 兼容性并非空中楼阁,而是通过一系列精心设计的技术特性实现的。
1. GUC 参数驱动的双模式与初始化
ivorysql.compatible_mode 是贯穿 IvorySQL 的灵魂参数。
初始化时指定模式
初始化为纯 PostgreSQL 模式
initdb -D /usr/ivory-5/data -m pg
如果使用 -m pg 模式,ivorysql.compatible_mode 参数将失效,系统行为与原生 PostgreSQL 完全一致。
初始化为 Oracle 兼容模式(默认)
initdb -D /usr/ivory-5/data -m oracle
运行时动态切换
初始化后,可以在会话级别动态切换模式,实现 “一库两用”:
-- 默认是Oracle模式
SHOW ivorysql.compatible_mode;
-- 输出: oracle
-- 临时切换到纯PG模式,处理原生PG语法
SET ivorysql.compatible_mode TO pg;
SHOW ivorysql.compatible_mode;
-- 输出: pg
2. PL/iSQL:Oracle PL/SQL 的无缝兼容
这是 IvorySQL 最具亮点的特性。PL/iSQL 过程语言可以完整兼容 Oracle 的 PL/SQL 语法。
匿名块兼容
DECLARE
i integer := 10;
grade CHAR(1) := 'B';
BEGIN
CASE grade
WHEN 'A' THEN RAISE NOTICE 'Excellent';
WHEN 'B' THEN RAISE NOTICE 'Very Good';
END CASE;
EXCEPTION
WHEN CASE_NOT_FOUND THEN
RAISE NOTICE 'No such grade';
END;
/
-- 输出: NOTICE: Very Good
函数兼容
-- 创建Oracle风格函数
CREATE OR REPLACE FUNCTION add_numbers(p_a NUMBER, p_b NUMBER) RETURN NUMBER IS
BEGIN
RETURN p_a + p_b;
END;
/
-- 调用(可省略括号,Oracle风格)
SELECT add_numbers FROM dual;
包(PACKAGE)兼容
sql
-- 创建包规范
CREATE OR REPLACE PACKAGE my_pkg IS
var1 INTEGER;
FUNCTION get_double(p_id INTEGER) RETURN INTEGER;
PROCEDURE print_message(p_msg VARCHAR2);
END my_pkg;
/
-- 创建包体
CREATE OR REPLACE PACKAGE BODY my_pkg IS
FUNCTION get_double(p_id INTEGER) RETURN INTEGER IS
BEGIN
RETURN p_id * 2;
END;
PROCEDURE print_message(p_msg VARCHAR2) IS
BEGIN
DBMS_OUTPUT.PUT_LINE(p_msg);
END;
-- 初始化块
BEGIN
var1 := 100;
END my_pkg;
/
-- 调用包中的函数和过程
SELECT my_pkg.get_double(5) FROM dual;
-- 输出: 10
CALL my_pkg.print_message('Hello from IvorySQL Package!');
3. 数据类型与函数
IvorySQL 支持大量 Oracle 独有的数据类型和函数,大幅降低迁移成本。
VARCHAR2 类型
CREATE TABLE customers (
cust_id NUMBER PRIMARY KEY,
cust_name VARCHAR2(100) -- 非标准PG类型,IvorySQL兼容
);
常用内置函数
-- SYSDATE
SELECT SYSDATE FROM dual;
-- ADD_MONTHS
SELECT ADD_MONTHS(SYSDATE, 3) FROM dual;
-- NVL (与PG的COALESCE类似)
SELECT NVL(NULL, 'Default Value') FROM dual;
-- 输出: Default Value
-- DECODE
SELECT DECODE(1, 1, 'One', 2, 'Two', 'Other') FROM dual;
-- 输出: One
-- INSTRB, SUBSTRB (字节定位)
SELECT INSTRB('IvorySQL数据库', 'SQL') FROM dual;
-- 输出: 5 (字节位置)
RowID 伪列
-- 开启ROWID功能
SET ivorysql.default_with_rowids TO on;
CREATE TABLE test_rowid (id INT, data TEXT);
INSERT INTO test_rowid VALUES (1, 'Row 1');
-- 查询ROWID
SELECT rowid, id, data FROM test_rowid;
-- 输出: 一个类似 '01000000' 的唯一标识符
4. 其他关键特性代码演示
Sequence 序列
CREATE SEQUENCE my_seq START WITH 1 INCREMENT BY 1;
SELECT my_seq.NEXTVAL FROM DUAL;
-- 输出: 1
SELECT my_seq.CURRVAL FROM DUAL;
-- 输出: 1
不可见列(INVISIBLE COLUMN)
CREATE TABLE employee (
emp_id INT PRIMARY KEY,
emp_name TEXT,
salary DECIMAL(10,2) INVISIBLE -- 对旧查询透明
);
INSERT INTO employee (emp_id, emp_name, salary) VALUES (1, 'Alice', 10000);
SELECT * FROM employee;
-- 只显示 emp_id, emp_name
% TYPE 和 % ROWTYPE 属性
CREATE TABLE users (id INT, name VARCHAR2(100));
CREATE OR REPLACE FUNCTION get_user_info(p_id users.id%TYPE) RETURN users.name%TYPE IS
v_name users.name%TYPE;
BEGIN
SELECT name INTO v_name FROM users WHERE id = p_id;
RETURN v_name;
END;
CALL INTO 子句
VARIABLE v_result VARCHAR2(100);
CALL my_func() INTO :v_result;
PRINT v_result;
5. 标识符大小写转换
IvorySQL 通过 identifier_case_switch 参数提供三种转换模式,解决 Oracle 与 PG 标识符大小写差异问题:
- normal:保持双引号标识符大小写不变
- interchange(默认):全大写转小写,全小写转大写,适配 Oracle 与 PG 的大小写规则
- lowercase:全大写转小写,大小写混合保持不变
SET ivorysql.identifier_case_switch = interchange;
CREATE TABLE "EMPLOYEE_TABLE" (id INT); -- Oracle风格大写
-- 实际创建表名为 employee_table,符合PG习惯
安装部署:从单机到云原生的全景图
IvorySQL 安装方式丰富,覆盖开发、测试、生产全场景。
1. 快速起步:Yum 一键安装
配置 yum 源
sudo tee /etc/yum.repos.d/ivorysql.repo <<EOF
[ivorysql5]
name=IvorySQL Server 5 $releasever - $basearch
baseurl=https://yum.highgo.com/dists/ivorysql-rpms/5/redhat/rhel-$releasever-$basearch
enabled=1
gpgcheck=0
EOF
安装
sudo dnf install -y ivorysql5-5.3
配置环境变量
echo 'PATH=/usr/ivory-5/bin:$PATH' >> ~/.bash_profile
echo 'PGDATA=/usr/ivory-5/data' >> ~/.bash_profile
source ~/.bash_profile
初始化并启动
initdb -D $PGDATA
pg_ctl -D $PGDATA -l logfile start
2. Docker 部署
快速搭建开发、测试环境:
拉取镜像
docker pull ivorysql/ivorysql:5.3-ubi8
运行容器,同时暴露 PG 端口(5432)和 Oracle 端口(1521)
docker run --name ivorysql \
-p 5432:5432 \
-p 1521:1521 \
-e IVORYSQL_PASSWORD=mysecretpassword \
-d ivorysql/ivorysql:5.3-ubi8
通过 PG 端口连接
psql -h 127.0.0.1 -p 5432 -U ivorysql -d ivorysql
通过 Oracle 端口连接(默认 Oracle 兼容模式)
psql -h 127.0.0.1 -p 1521 -U ivorysql -d ivorysql
关键点:容器默认同时监听 5432(PG 模式)和 1521(Oracle 模式)双端口,可按需切换。
3. K8s/Harbor Operator 云原生部署
生产环境可使用 IvorySQL Operator,基于 Kubernetes 实现高可用、备份、监控自动化:
apiVersion: ivory-operator.ivorysql.org/v1beta1
kind: IvoryCluster
metadata:
name: hippo
spec:
image: ivorysql/ivorysql:5.3-ubi8
postgresVersion: 18
instances:
- name: instance1
replicas: 2 # 高可用副本
dataVolumeClaimSpec:
accessModes: ["ReadWriteOnce"]
resources:
requests:
storage: 1Gi
backups:
pgbackrest:
image: ivorysql/pgbackrest:ubi8-2.56.0-5.0-1
repos:
- name: repo1
volume:
volumeClaimSpec:
accessModes: ["ReadWriteOnce"]
resources:
requests:
storage: 1Gi
patroni:
dynamicConfiguration:
synchronous_mode: true # 开启同步复制
执行部署命令:
kubectl apply -k examples/kustomize/ivory Operator 自动完成 Pod、密钥、服务、证书、备份策略等配置,实现一键部署。
运维生态:让管理更简单,让性能更卓越
高可用集群搭建,基于流复制搭建主备集群:
主库 postgresql.conf
listen_addresses = '*'
wal_level = replica
max_wal_senders = 5
hot_standby = on
备库初始化
pg_basebackup -F p -P -X fetch -R -h <master_ip> -p <master_port> -U republic_user -D $PGDATA
性能监控与 SQL 分析
结合 pgMonitor + Grafana 搭建监控大屏,同时提供常用诊断 SQL:
-- 查看执行计划
EXPLAIN (ANALYZE, BUFFERS) SELECT * FROM large_table WHERE id = 100;
-- 查看当前活跃会话
SELECT pid, state, query, wait_event_type FROM pg_stat_activity WHERE state = 'active';
-- 诊断锁冲突
SELECT blocked_locks.pid AS blocked_pid,
blocked_activity.query AS blocked_query,
blocking_locks.pid AS blocking_pid,
blocking_activity.query AS blocking_query
FROM pg_locks blocked_locks
JOIN pg_stat_activity blocked_activity ON blocked_activity.pid = blocked_locks.pid
JOIN pg_locks blocking_locks ON blocking_locks.locktype = blocked_locks.locktype
AND blocking_locks.database IS NOT DISTINCT FROM blocked_locks.database
AND blocking_locks.relation IS NOT DISTINCT FROM blocked_locks.relation
AND blocking_locks.page IS NOT DISTINCT FROM blocked_locks.page
AND blocking_locks.tuple IS NOT DISTINCT FROM blocked_locks.tuple
AND blocking_locks.virtualxid IS NOT DISTINCT FROM blocked_locks.virtualxid
AND blocking_locks.transactionid IS NOT DISTINCT FROM blocked_locks.transactionid
AND blocking_locks.classid IS NOT DISTINCT FROM blocked_locks.classid
AND blocking_locks.objid IS NOT DISTINCT FROM blocked_locks.objid
AND blocking_locks.objsubid IS NOT DISTINCT FROM blocked_locks.objsubid
AND blocking_locks.pid != blocked_locks.pid
JOIN pg_stat_activity blocking_activity ON blocking_activity.pid = blocking_locks.pid
WHERE NOT blocked_locks.granted;
数据迁移
使用 pg_dump 完成同架构数据迁移:
pg_dump -h old_oracle_compatible_server -U old_user old_db | psql -h new_ivorysql_server -U new_user new_db
结论:面向未来的数据基础设施
IvorySQL 不仅仅是一个基于 PostgreSQL 的再发行版,它是一个思考深、设计巧、价值高的开源数据库项目。它成功地解决了 Oracle 用户向开源和云原生生态迁移的核心障碍 —— 兼容性。
对于希望摆脱 Oracle 高昂许可束缚、拥抱 PostgreSQL 开放生态的企业,IvorySQL 提供了一条近乎完美的平滑过渡方案。它不仅降低了迁移的门槛和风险,更重要的是,它将 Oracle 的强大与 PostgreSQL 的灵活结合在了一起,为用户在未来的数据结构、混合多模态数据处理上提供了无限的想象空间。
IvorySQL 的出现,标志着数据库领域 “融合” 趋势的进一步发展。它不是去颠覆谁,而是在做一件更有意义的事:让历史与未来,在一个高效、可靠的平台上共存共生。 对于任何正在规划数据库现代化或面临迁移决策的组织而言,深入研究并积极探索 IvorySQL,无疑是一个极具前瞻性和战略价值的明智之举。
作者:杨宇,PG ACE,第八届 PG 数据库生态大会 2025 年度金牌讲师,PG 分会哈尔滨用户组核心成员,负责本公司数据库、云计算等相关运维工作。
曾主持对公司 PostgreSQL 数据库进行架构设计、容灾系统搭建及报表优化工作。闲暇之余致力于 PostgreSQL 数据库的传播,发表多篇架构、优化相关的原创文章。目前拥有 ORACLE 11G/12C OCM、MYSQL 5.7/8.0 OCP、TIDB PCTP、OBCP V4、电科金仓 KCP 等资质认证。
876

被折叠的 条评论
为什么被折叠?



