دليل PostgreSQL السريع: مرجع مطورين

مرجع سريع لـ postgresql

Page content

مراجع سريعة للاستخدام اليومي PostgreSQL العمل: الاتصالات، تركيب SQL، الأوامر الميتا لـ psql، الأداء، JSON، وظائف النافذة، وغيرها.

قائمة SQL مختصرة تغطي الأوامر والأعمال الأساسية المهمة متوفرة أيضًا.

شعار postgresql

الاتصال والأساسيات

# الاتصال
psql -h HOST -p 5432 -U USER -d DB
psql $DATABASE_URL

# داخل psql
\conninfo             -- عرض الاتصال
\l[+]                 -- قائمة قواعد البيانات
\c DB                 -- الاتصال بـ DB
\dt[+] [schema.]pat   -- قائمة الجداول
\dv[+]                -- قائمة العروض
\ds[+]                -- قائمة التسلسلات
\df[+] [pat]          -- قائمة الدوال
\d[S+] name           -- وصف الجدول/العرض/التسلسل
\dn[+]                -- قائمة المخططات
\du[+]                -- قائمة الأدوار
\timing               -- تفعيل/تعطيل عرض وقت الاستعلام
\x                    -- عرض موسّع
\e                    -- تحرير المخزن المؤقت في $EDITOR
\i file.sql           -- تشغيل الملف
\copy ...             -- نسخ من الطرف العميل
\! shell_cmd          -- تشغيل أمر في الطرفية

أنواع البيانات (شائعة)

  • عددية: smallint, integer, bigint, decimal(p,s), numeric, real, double precision, serial, bigserial
  • نص: text, varchar(n), char(n)
  • منطقي: boolean
  • زمني: timestamp [مع/بدون منطقة زمنية], date, time, interval
  • UUID: uuid
  • JSON: json, jsonb (المفضل)
  • مصفوفة: type[] مثال: text[]
  • شبكة: inet, cidr, macaddr
  • هندسي: point, line, polygon, وغيرها

التحقق من إصدار PostgreSQL

SELECT version();

إصدار خادم PostgreSQL:

pg_config --version

إصدار عميل PostgreSQL:

psql --version

DDL (إنشاء / تعديل)

-- إنشاء مخطط وجدول
CREATE SCHEMA IF NOT EXISTS app;
CREATE TABLE app.users (
  id           bigserial PRIMARY KEY,
  email        text NOT NULL UNIQUE,
  name         text,
  active       boolean NOT NULL DEFAULT true,
  created_at   timestamptz NOT NULL DEFAULT now(),
  profile      jsonb,
  tags         text[]
);

-- تعديل
ALTER TABLE app.users ADD COLUMN last_login timestamptz;
ALTER TABLE app.users ALTER COLUMN name SET NOT NULL;
ALTER TABLE app.users DROP COLUMN tags;

-- القيود
ALTER TABLE app.users ADD CONSTRAINT email_lower_uk UNIQUE (lower(email));

-- الفهارس
CREATE INDEX ON app.users (email);
CREATE UNIQUE INDEX CONCURRENTLY users_email_uidx ON app.users (lower(email));
CREATE INDEX users_profile_gin ON app.users USING gin (profile);
CREATE INDEX users_created_at_idx ON app.users (created_at DESC);

DML (إدراج / تحديث / إدراج أو تحديث / حذف)

INSERT INTO app.users (email, name) VALUES
  ('a@x.com','A'), ('b@x.com','B')
RETURNING id;

-- إدراج أو تحديث (ON CONFLICT)
INSERT INTO app.users (email, name)
VALUES ('a@x.com','Alice')
ON CONFLICT (email)
DO UPDATE SET name = EXCLUDED.name, updated_at = now();

UPDATE app.users SET active = false WHERE last_login < now() - interval '1 year';
DELETE FROM app.users WHERE active = false AND last_login IS NULL;

أساسيات الاستعلام

SELECT * FROM app.users ORDER BY created_at DESC LIMIT 20 OFFSET 40; -- التصفح

-- التصفية
SELECT * FROM app.users WHERE email ILIKE '%@example.%' AND active;

-- التجميع والتصنيف حسب
SELECT active, count(*) AS n
FROM app.users
GROUP BY active
HAVING count(*) > 10;

-- الربط
SELECT o.id, u.email, o.total
FROM app.orders o
JOIN app.users u ON u.id = o.user_id
LEFT JOIN app.discounts d ON d.id = o.discount_id;

-- DISTINCT ON (مميز لـ Postgres)
SELECT DISTINCT ON (user_id) user_id, status, created_at
FROM app.events
ORDER BY user_id, created_at DESC;

-- CTEs
WITH recent AS (
  SELECT * FROM app.orders WHERE created_at > now() - interval '30 days'
)
SELECT count(*) FROM recent;

-- CTE تكراري
WITH RECURSIVE t(n) AS (
  SELECT 1
  UNION ALL
  SELECT n+1 FROM t WHERE n < 10
)
SELECT sum(n) FROM t;

وظائف النافذة

SELECT
  user_id,
  created_at,
  sum(total)      OVER (PARTITION BY user_id ORDER BY created_at) AS running_total,
  row_number()    OVER (PARTITION BY user_id ORDER BY created_at) AS rn,
  lag(total, 1)   OVER (PARTITION BY user_id ORDER BY created_at) AS prev_total
FROM app.orders;

JSON / JSONB

-- استخراج
SELECT profile->>'company' AS company FROM app.users;
SELECT profile->'address'->>'city' FROM app.users;

-- فهرس لاستعلامات jsonb
CREATE INDEX users_profile_company_gin ON app.users USING gin ((profile->>'company'));

-- وجود / احتواء
SELECT * FROM app.users WHERE profile ? 'company';              -- المفتاح موجود
SELECT * FROM app.users WHERE profile @> '{"role":"admin"}'; -- يحتوي

-- تحديث jsonb
UPDATE app.users
SET profile = jsonb_set(COALESCE(profile,'{}'::jsonb), '{prefs,theme}', '"dark"', true);

المصفوفات

-- العضوية / الاحتواء
SELECT * FROM app.users WHERE 'vip' = ANY(tags);
SELECT * FROM app.users WHERE tags @> ARRAY['beta'];

-- إضافة
UPDATE app.users SET tags = array_distinct(tags || ARRAY['vip']);

الوقت والتاريخ

SELECT now() AT TIME ZONE 'Australia/Melbourne';
SELECT date_trunc('day', created_at) AS d, count(*)
FROM app.users GROUP BY d ORDER BY d;

-- الفترات
SELECT now() - interval '7 days';

المعاملات والقفل

BEGIN;
UPDATE app.accounts SET balance = balance - 100 WHERE id = 1;
UPDATE app.accounts SET balance = balance + 100 WHERE id = 2;
COMMIT; -- أو ROLLBACK

-- مستوى العزل
SET TRANSACTION ISOLATION LEVEL REPEATABLE READ;

-- التحقق من القفل
SELECT * FROM pg_locks l JOIN pg_stat_activity a USING (pid);

الأدوار والصلاحيات

-- إنشاء دور/مستخدم
CREATE ROLE app_user LOGIN PASSWORD 'secret';
GRANT USAGE ON SCHEMA app TO app_user;
GRANT SELECT, INSERT, UPDATE, DELETE ON ALL TABLES IN SCHEMA app TO app_user;
ALTER DEFAULT PRIVILEGES IN SCHEMA app GRANT SELECT, INSERT, UPDATE, DELETE ON TABLES TO app_user;

الاستيراد / التصدير

-- من الطرف الخادم (يتطلب مستخدمًا رئيسيًا أو صلاحيات مناسبة)
COPY app.users TO '/tmp/users.csv' CSV HEADER;
COPY app.users(email,name) FROM '/tmp/users.csv' CSV HEADER;

-- من الطرف العميل (psql)
\copy app.users TO 'users.csv' CSV HEADER
\copy app.users(email,name) FROM 'users.csv' CSV HEADER

الأداء والمراقبة

EXPLAIN (ANALYZE, BUFFERS, VERBOSE) SELECT ...;  -- وقت التشغيل الفعلي

-- عرض الإحصائيات
SELECT * FROM pg_stat_user_tables;
SELECT * FROM pg_stat_statements ORDER BY total_time DESC LIMIT 20; -- يتطلب امتدادًا

-- الصيانة
VACUUM [FULL] [VERBOSE] table_name;
ANALYZE table_name;
REINDEX TABLE table_name;

تفعيل الامتدادات حسب الحاجة:

CREATE EXTENSION IF NOT EXISTS pg_stat_statements;
CREATE EXTENSION IF NOT EXISTS btree_gin;
CREATE EXTENSION IF NOT EXISTS pg_trgm; -- بحث التريجرام

البحث النصي الكامل (مختصر)

ALTER TABLE app.docs ADD COLUMN tsv tsvector;
UPDATE app.docs SET tsv = to_tsvector('english', coalesce(title,'') || ' ' || coalesce(body,''));
CREATE INDEX docs_tsv_idx ON app.docs USING gin(tsv);
SELECT id FROM app.docs WHERE tsv @@ plainto_tsquery('english', 'quick brown fox');

إعدادات مفيدة لـ psql

\pset pager off       -- تعطيل المتصفح
\pset null '∅'
\pset format aligned  -- أخرى: unaligned, csv
\set ON_ERROR_STOP on
\timing on

استعلامات فهرسة مفيدة

-- حجم الجدول
SELECT relname, pg_size_pretty(pg_total_relation_size(relid)) AS size
FROM pg_catalog.pg_statio_user_tables ORDER BY pg_total_relation_size(relid) DESC;

-- تضخم الفهارس (تقريبي)
SELECT schemaname, relname, indexrelname, idx_scan
FROM pg_stat_user_indexes ORDER BY idx_scan ASC NULLS FIRST LIMIT 20;

النسخ الاحتياطي / الاستعادة

# نسخ احتياطي منطقي
pg_dump -h HOST -U USER -d DB -F c -f db.dump        # التنسيق المخصص
pg_restore -h HOST -U USER -d NEW_DB -j 4 db.dump

# نص SQL عادي
pg_dump -h HOST -U USER -d DB > dump.sql
psql -h HOST -U USER -d DB -f dump.sql

التكرار (مستوى عالٍ)

  • ملفات WAL تتدفق من الرئيسيةالنسخ الاحتياطية
  • الإعدادات المهمة: wal_level, max_wal_senders, hot_standby, primary_conninfo
  • الأدوات: pg_basebackup, standby.signal (PG ≥12)

الأخطاء الشائعة والنصائح

  • استخدم دائمًا jsonb، وليس json، للفهارس والعمليات.
  • تفضّل timestamptz (الوقت المدرك للمنطقة الزمنية).
  • DISTINCT ON هو ذهب Postgres لـ “أعلى N لكل مجموعة”.
  • استخدم GENERATED ALWAYS AS IDENTITY بدلًا من serial.
  • تجنب SELECT * في الاستعلامات الإنتاجية.
  • أنشئ فهارس للفلاتر المحددة ومفاتيح الربط.
  • قم بقياس باستخدام EXPLAIN (ANALYZE) قبل التحسين.

المزايا الخاصة بالإصدار (≥v12+)

-- هوية مولدة
CREATE TABLE t (id bigINT GENERATED ALWAYS AS IDENTITY, ...);

-- إدراج أو تحديث مع فهرس جزئي
CREATE UNIQUE INDEX ON t (key) WHERE is_active;

روابط مفيدة