সাবগ্যুরি ব্যবহার করে পোস্টগ্রিতে টেবিল সারিগুলি আপডেট করা


301

পোস্টগ্রেস 8.4 ব্যবহার করে, আমার লক্ষ্য বিদ্যমান টেবিলটি আপডেট করা:

CREATE TABLE public.dummy
(
  address_id SERIAL,
  addr1 character(40),
  addr2 character(40),
  city character(25),
  state character(2),
  zip character(5),
  customer boolean,
  supplier boolean,
  partner boolean

)
WITH (
  OIDS=FALSE
);

প্রাথমিকভাবে আমি আমার জিজ্ঞাসা সন্নিবেশ বিবৃতি ব্যবহার করে পরীক্ষা করেছি:

insert into address customer,supplier,partner
SELECT  
    case when cust.addr1 is not null then TRUE else FALSE end customer, 
    case when suppl.addr1 is not null then TRUE else FALSE end supplier,
    case when partn.addr1 is not null then TRUE else FALSE end partner
from (
    SELECT *
        from address) pa
    left outer join cust_original cust
        on (pa.addr1=cust.addr1 and pa.addr2=cust.addr2 and pa.city=cust.city 
            and pa.state=cust.state and substring(cust.zip,1,5) = pa.zip  )
    left outer join supp_original suppl 
        on (pa.addr1=suppl.addr1 and pa.addr2=suppl.addr2 and pa.city=suppl.city 
                and pa.state=suppl.state and pa.zip = substring(suppl.zip,1,5))
    left outer join partner_original partn
        on (pa.addr1=partn.addr1 and pa.addr2=partn.addr2 and pa.city=partn.city
                  and pa.state=partn.state and pa.zip = substring(partn.zip,1,5) )
where pa.address_id = address_id

নবাবি হওয়াতে আমি বিবৃতিটি আপডেট করতে রূপান্তর করতে ব্যর্থ হয়েছি select কোন সাহায্যের অত্যন্ত প্রশংসা করা হয়।


আপনার ঠিকানা টেবিলে কোনও ধরণের আইডি রয়েছে, যা সারিটি বিদ্যমান তা নির্ধারণ করতে ব্যবহার করা যেতে পারে?
আন্দ্রে আদোমোভিচ

হ্যাঁ আমি করি তবে এর সিস্ট তৈরি হয়েছে।
স্ট্যাকওভার

উত্তর:


681

পোস্টগ্রাগেস অনুমতি দেয়:

UPDATE dummy
SET customer=subquery.customer,
    address=subquery.address,
    partn=subquery.partn
FROM (SELECT address_id, customer, address, partn
      FROM  /* big hairy SQL */ ...) AS subquery
WHERE dummy.address_id=subquery.address_id;

এই সিনট্যাক্সটি স্ট্যান্ডার্ড এসকিউএল নয়, তবে স্ট্যান্ডার্ড এসকিউএল এর চেয়ে এই ধরণের প্রশ্নের জন্য এটি অনেক বেশি সুবিধাজনক। আমি বিশ্বাস করি ওরাকল (কমপক্ষে) অনুরূপ কিছু গ্রহণ করে।


দেখে মনে হচ্ছে যে আমি উদাহরণস্বরূপ কিছুটা আলাদা জিনিস চেষ্টা করছি। যদি 3 টি বুল কলাম থাকে তবে c1, c2, c3 শুরুতে মিথ্যাতে সেট করা আছে। তবে সাবকিউরিয়ের উপর ভিত্তি করে সত্যকে সেট করা হয়েছে। আপডেট সেট c1 = সত্য যেখানে আইডি ইন (subquery1), সেট সি 2 = সত্য যেখানে আইডি ইন (subquery2), সেট সি 3 = সত্য যেখানে আইডি ইন (subquery3)। আমি যখন এটিকে 3 আপডেট হিসাবে বিভক্ত করি তখন আমি সফল হয়েছিলাম তবে একক আপডেটের সাথে কীভাবে ফলাফল অর্জন করা যায় তা আমি নিশ্চিত নই। আশা করি এটি বোধগম্য হবে।
স্ট্যাকওভার

3
এফডাব্লুআইডাব্লু, ওরাকল সেই বুনিয়াদী গঠনটি স্বীকার করে না, তবে টেবিলগুলি বড় হওয়ার সাথে সাথে আপডেটের কর্মক্ষমতা মারাত্মকভাবে হ্রাস পেতে থাকে। এটি ঠিক আছে যদিও ওরাকল এছাড়াও মার্জ স্টেটমেন্টকে সমর্থন করে।
gsiems

3
এটি পুরোপুরি ERROR: 42P01: relation "dummy" does not exist
পোস্টগ্র্যাস্কল

72
dummyআপনি যে টেবিলটি আপডেট করার চেষ্টা করছেন তার নামটি প্রতিস্থাপন করতে হবে। প্রয়োগ করার চেষ্টা করার আগে দয়া করে প্রশ্নোত্তর বুঝতে পারেন।
অ্যান্ড্রু লাজার

1
এটি উল্লেখ করার মতো হতে পারে যে ক্যোয়ারির শুরুতে কেবল বাম পাশের কলামের পথটি নির্দিষ্ট করা প্রয়োজন হয় না, অন্যথায় ডিবি ERROR এর সাথে অভিযোগ করবে: কলাম রেফারেন্স "ঠিকানা_id" অস্পষ্ট
ওজেভিএম

125

আপনি UPDATE FROMসিনট্যাক্স পরে ।

UPDATE 
  table T1  
SET 
  column1 = T2.column1 
FROM 
  table T2 
  INNER JOIN table T3 USING (column2) 
WHERE 
  T1.column2 = T2.column2;

তথ্যসূত্র


2
নির্বাচিত উত্তর হওয়া উচিত
জোশুয়া কিফার

50

যদি কোনও জয়েন ব্যবহার করে কোনও পারফরম্যান্স লাভ হয় না, তবে আমি পঠনযোগ্যতার জন্য সাধারণ টেবিল এক্সপ্রেশন (সিটিই) পছন্দ করি:

WITH subquery AS (
    SELECT address_id, customer, address, partn
    FROM  /* big hairy SQL */ ...
)
UPDATE dummy
SET customer = subquery.customer,
    address  = subquery.address,
    partn    = subquery.partn
FROM subquery
WHERE dummy.address_id = subquery.address_id;

আইএমএইচও আরও কিছুটা আধুনিক।


1
সিনট্যাক্স Postgres পুরোনো সংস্করণের সাথে সামঞ্জস্যপূর্ণ নয়, v9.1 আগে, (দেখুন postgresql.org/docs/9.1/static/sql-update.html এবং পূর্ববর্তী সংস্করণগুলির) আমি v8.2 উপর আছি, তাই আপনার আছে FROM কীওয়ার্ডের পরে বন্ধনীর ভিতরে পুরো সিটিই / উইথ স্টেটমেন্ট রাখার জন্য এটি কার্যকর হবে।
দ্বিতীয়

9

সারিগুলি আপডেট করার বিভিন্ন উপায় রয়েছে।

UPDATEসাবকিউরিগুলি ব্যবহার করে যখন সারিগুলিতে আসে তখন আপনি এই পদ্ধতির কোনও ব্যবহার করতে পারেন।

  1. পন্থা -১ [সরাসরি টেবিলের রেফারেন্স ব্যবহার করে]
UPDATE
  <table1>
SET
  customer=<table2>.customer,
  address=<table2>.address,
  partn=<table2>.partn
FROM
  <table2>
WHERE
  <table1>.address_id=<table2>.address_i;

ব্যাখ্যা: table1আমরা যে টেবিলটি আপডেট করতে চাই তা table2 হ'ল সেই টেবিলটি, যা থেকে আমরা মানটি প্রতিস্থাপন / আপডেট করার জন্য পেয়ে যাব। আমরা FROMক্লজটি ব্যবহার করছি , table2এর ডেটা আনতে । WHERE ধারাটি সঠিক ডেটা ম্যাপিং সেট করতে সহায়তা করবে।

  1. অ্যাপ্রোচ -২ [সাবকিউরিজ ব্যবহার করে]
UPDATE
  <table1>
SET
  customer=subquery.customer,
  address=subquery.address,
  partn=subquery.partn
FROM
  (
    SELECT
      address_id, customer, address, partn
    FROM  /* big hairy SQL */ ...
  ) AS subquery
WHERE
  dummy.address_id=subquery.address_id;

ব্যাখ্যা: এখানে আমরা FROMধারাটির অভ্যন্তরে সাবকিরি ব্যবহার করছি এবং এটির জন্য একটি উপাধি দিচ্ছি । যাতে এটি টেবিলের মতো কাজ করবে।

  1. অ্যাপ্রোচ -৩ [একাধিক যোগদানকারী টেবিলগুলি ব্যবহার করে]
UPDATE
  <table1>
SET
  customer=<table2>.customer,
  address=<table2>.address,
  partn=<table2>.partn
FROM
  <table2> as t2
  JOIN <table3> as t3
  ON
    t2.id = t3.id
WHERE
  <table1>.address_id=<table2>.address_i;

ব্যাখ্যা: কখনও কখনও আমরা সেই টেবিলের জয়েন্টটির পরিস্থিতির মুখোমুখি হয়ে থাকি আপডেটের জন্য সঠিক ডেটা পেতে important এটি করার জন্য, পোস্টগ্রিস আমাদের FROMক্লজের অভ্যন্তরে একাধিক টেবিলগুলিতে যোগদানের অনুমতি দেয় ।

  1. পন্থা -4 [বিবৃতি দিয়ে ব্যবহার করা]

    • ৪.১ [সাধারণ জিজ্ঞাসা ব্যবহার করা]
WITH subquery AS (
    SELECT
      address_id,
      customer,
      address,
      partn
    FROM
      <table1>;
)
UPDATE <table-X>
SET customer = subquery.customer,
    address  = subquery.address,
    partn    = subquery.partn
FROM subquery
WHERE <table-X>.address_id = subquery.address_id;
  • ৪.২ [জটিল জোনের সাথে ক্যোয়ারী ব্যবহার করা]
WITH subquery AS (
    SELECT address_id, customer, address, partn
    FROM
      <table1> as t1
    JOIN
      <table2> as t2
    ON
      t1.id = t2.id;
    -- You can build as COMPLEX as this query as per your need.
)
UPDATE <table-X>
SET customer = subquery.customer,
    address  = subquery.address,
    partn    = subquery.partn
FROM subquery
WHERE <table-X>.address_id = subquery.address_id;

ব্যাখ্যা: পোস্টগ্রিস ৯.১ থেকে এই ( WITH) ধারণাটি প্রবর্তিত হয়েছে। এটি ব্যবহার করে আমরা যে কোনও জটিল প্রশ্ন তৈরি করতে পারি এবং আকাঙ্ক্ষার ফলাফল উত্পন্ন করতে পারি। টেবিল আপডেট করার জন্য আমরা এখানে এই পদ্ধতির ব্যবহার করছি।

আমি আশা করি, এটি সহায়ক হবে 😊



1

@Mayur "4.2 [জটিল এর মাধ্যমে ব্যবহার ক্যোয়ারী JOIN]" সঙ্গে সাধারণ টেবিল প্রকাশ (CTEs) আমার জন্য কৌতুক করেনি।

WITH cte AS (
SELECT e.id, e.postcode
FROM employees e
LEFT JOIN locations lc ON lc.postcode=cte.postcode
WHERE e.id=1
)
UPDATE employee_location SET lat=lc.lat, longitude=lc.longi
FROM cte
WHERE employee_location.id=cte.id;

আশা করি এটি সাহায্য করবে ...: ডি

আমাদের সাইট ব্যবহার করে, আপনি স্বীকার করেছেন যে আপনি আমাদের কুকি নীতি এবং গোপনীয়তা নীতিটি পড়েছেন এবং বুঝতে পেরেছেন ।
Licensed under cc by-sa 3.0 with attribution required.