İleri Seviye SQL Server Sql Server

SQL Server’da İki Veritabanı Arasında Veri Aktarımı

SQL Server Eğitimleri Kanalına Abone Olun

Herkese merhaba,

Bu yazıda SQL Server’da iki veritabanı arasında, veri aktarım işleminin nasıl yapılacağından bahsedeceğim.

Yazının çıkış noktası bir öğrenci. Bu soru kendisine sorulmuş. Nasıl çözebileceği hakkında bana sordu. Sonrasında makaleye çevirmeye karar verdim.

Yapılmak istenen işlem ise şu.

INSERT INTO table1 (column1 )
SELECT  col1
FROM  table2  

Bu işlemi veritabanı seçerek (Parametrik) tüm tablolar için yapmak. (Prosedür kullanarak)

Kısaca iki veritabanının tablo ve şema isimleri aynı olacak ve birinci veritabanındaki dolu verilerin ikinci veritabanındaki boş tablolar içine aktarılması. (Tüm tablolar için yapılacak)

Bu işlem için örnek bir veritabanı oluşturup yapılacak işlemi anlatacağım.

Birinci veritabanımız ve tablolar içindeki veriler şöyle olsun.

CREATE DATABASE DB1
USE DB1

CREATE TABLE [dbo].[Tablo1](
	[id] [INT] IDENTITY(1,1) NOT NULL,
	[Ad] [VARCHAR](50) NULL,
	[Soyad] [VARCHAR](50) NULL,
 CONSTRAINT [PK_Tablo1] PRIMARY KEY CLUSTERED 
(
	[id] ASC
))


CREATE TABLE [dbo].[Tablo2](
	[id] [INT] IDENTITY(1,1) NOT NULL,
	[il] [VARCHAR](50) NULL,
 CONSTRAINT [PK_Tablo2] PRIMARY KEY CLUSTERED 
(
	[id] ASC
))

INSERT INTO Tablo1
VALUES
(  'Yavuz', 'Selim' ), 
( 'Hasan', 'Ali' )


INSERT INTO Tablo2
VALUES
( 'İstanbul' ), 
( 'Ankara' ), 
( 'İzmir' )


Veritabanı ve tabloları oluşturup sonrasında ilgili verileri içine ekleyelim.

Şimdi aynı işlemi ikinci veritabanı için yapalım. Fakat bu sefer tablo içine veri eklemiyoruz.

CREATE DATABASE DB2
USE DB2

CREATE TABLE [dbo].[Tablo1](
	[id] [INT] IDENTITY(1,1) NOT NULL,
	[Ad] [VARCHAR](50) NULL,
	[Soyad] [VARCHAR](50) NULL,
 CONSTRAINT [PK_Tablo1] PRIMARY KEY CLUSTERED 
(
	[id] ASC
))


CREATE TABLE [dbo].[Tablo2](
	[id] [INT] IDENTITY(1,1) NOT NULL,
	[il] [VARCHAR](50) NULL,
 CONSTRAINT [PK_Tablo2] PRIMARY KEY CLUSTERED 
(
	[id] ASC
))

Görüldüğü üzere ikinci veritabanı içine herhangi bir veri eklemedik.

Tablolara dikkat ederseniz iki tabloda da otomatik artan id değerleri var. Bu değerlerin insert into ile diğer tabloya aktarılması durumunda aşağıdaki gibi bir hata alacaksınız.

Cannot insert explicit value for identity column in table ‘Tablonuz’ when identity_insert is set to off.

Bunu kapatmanın yolunu öğrenmek için aşağıdaki makaleyi okuyabilirsiniz.

[related-post id=”2796″]

Makaleyi okuduysanız çözümü artık biliyorsunuz.

Şimdi gelelim esas konumuza.

Benim yapmak istediğim işlem kısaca dinamik olarak her tablo için INSERT INTO işleminin yapılması.

Aşağıdaki gibi neden yapmıyorsunuz diyenler olacak.

INSERT INTO tablo2
SELECT * FROM tablo1

SQL Server’da sorguları * kullanarak kolonları alıp diğer tablonun içine ekleyemiyorsunuz.

İsterseniz deneyebilirsiniz. Sadece tek tek kolon adlarını yazarak böyle bir işlemi yapabilirsiniz.

INSERT INTO tablo2 (kolon1, kolon2 ....) SELECT digerkolon1, digerkolon2... FROM tablo1;

Ayrıca burada her tablo için aşağıdaki sıkıntı da mevcut.

IDENTITY_INSERT is Set to Off Hatası.

İşte bu işlemlerin de dinamik olarak yapılması ve tablodaki verilerin diğer tabloya otomatik olarak eklenmesi için aşağıdaki gibi bir prosedür yazdım. Belki başka şekilde çözümler olacaktır. Bu konuda tarafıma her zaman yazabilirsiniz.

CREATE PROC DBTabloVeriAktar
(
    @AktarilacakDB VARCHAR(100),
    @AktarilanDB VARCHAR(100)
)
AS
BEGIN

    DECLARE @TabloAdi VARCHAR(100);
    DECLARE @VeritabaniAdi2 VARCHAR(100) = @AktarilanDB;
    DECLARE @KolonAdi VARCHAR(100);
    DECLARE @KolonAdiListesiSon NVARCHAR(4000);
    DECLARE @SQLSorgu NVARCHAR(4000);
    DECLARE CursorYapisi CURSOR FOR
    SELECT DISTINCT
           c.TABLE_NAME
    FROM INFORMATION_SCHEMA.COLUMNS c
        LEFT OUTER JOIN INFORMATION_SCHEMA.TABLES t
            ON c.TABLE_CATALOG = t.TABLE_CATALOG
               AND c.TABLE_SCHEMA = t.TABLE_SCHEMA
               AND c.TABLE_NAME = t.TABLE_NAME
    WHERE t.TABLE_TYPE = 'BASE TABLE'
          AND c.TABLE_CATALOG = @AktarilacakDB
          AND c.TABLE_NAME NOT IN ( 'sysdiagrams' );


    OPEN CursorYapisi;
    FETCH NEXT FROM CursorYapisi
    INTO @TabloAdi;

    WHILE @@FETCH_STATUS = 0
    BEGIN

        CREATE TABLE #tabloadlari
        (
            tabload VARCHAR(25)
        );


        DECLARE CursorYapisi2 CURSOR FOR
        SELECT DISTINCT
               c.COLUMN_NAME
        FROM INFORMATION_SCHEMA.COLUMNS c
            LEFT OUTER JOIN INFORMATION_SCHEMA.TABLES t
                ON c.TABLE_CATALOG = t.TABLE_CATALOG
                   AND c.TABLE_SCHEMA = t.TABLE_SCHEMA
                   AND c.TABLE_NAME = t.TABLE_NAME
        WHERE t.TABLE_TYPE = 'BASE TABLE'
              AND c.TABLE_CATALOG = @AktarilacakDB
              AND c.TABLE_NAME NOT IN ( 'sysdiagrams' )
              AND c.TABLE_NAME = @TabloAdi;

        OPEN CursorYapisi2;
        FETCH NEXT FROM CursorYapisi2
        INTO @KolonAdi;

        WHILE @@FETCH_STATUS = 0
        BEGIN


            -- PRINT @KolonAdi

            INSERT INTO #tabloadlari
            (
                tabload
            )
            VALUES
            (@KolonAdi -- tabload - varchar(25)
                );



            FETCH NEXT FROM CursorYapisi2
            INTO @KolonAdi;

        END;

        CLOSE CursorYapisi2;
        DEALLOCATE CursorYapisi2;


        DECLARE @KolonAdiListesi VARCHAR(4000);
        SET @KolonAdiListesi = '';
        SELECT @KolonAdiListesi = @KolonAdiListesi + tabload + ', '
        FROM #tabloadlari;

        SELECT @KolonAdiListesiSon = SUBSTRING(@KolonAdiListesi, 0, LEN(@KolonAdiListesi));
        --PRINT @KolonAdiListesi

        DROP TABLE #tabloadlari;


        SET @SQLSorgu
            = N'SET IDENTITY_INSERT  [' + @VeritabaniAdi2 + N'].[dbo].[' + @TabloAdi + N']  ON INSERT INTO ['
              + @AktarilanDB + N'].[dbo].[' + @TabloAdi + N']	( ' + @KolonAdiListesiSon + N' ) 			
				SELECT ' + @KolonAdiListesiSon + N' FROM [' + @AktarilacakDB + N'].[dbo].[' + @TabloAdi
              + N']  SET IDENTITY_INSERT  [' + @VeritabaniAdi2 + N'].[dbo].[' + @TabloAdi + N']  OFF;';

        EXEC (@SQLSorgu);

        --PRINT @SQLSorgu;


        FETCH NEXT FROM CursorYapisi
        INTO @TabloAdi;
    END;

    CLOSE CursorYapisi;
    DEALLOCATE CursorYapisi;


END;

Prosedürü oluşturalım.

Şimdi test edelim.

EXEC dbo.DBTabloVeriAktar @AktarilacakDB = 'DB1', -- varchar(100)
                          @AktarilanDB = 'DB2';   -- varchar(100)

Sorguyu çalıştırdığınızda ikinci veritabanı içine verilerin başarıyla eklendiğini göreceksiniz.

Görüldüğü üzere veriler başarıyla ikinci veritabanında ki iki tabloya da kopyalandı.

Herkese çalışma hayatında ve yaşamında başarılar kolaylıklar.

SQL Server Eğitimleri Kanalına Abone Olun

Yazar hakkında

Yavuz Selim Kart

SQL Server tarafında kendini ilerletmeye çalışırken bildiklerini sürekli başkalarına aktaran, yazılım dünyasında (C#, Java, Python vb.) kendini güncel tutmaya çalışan, araştırma yapmaktan bıkmayan, meraklı, bilginin üzerine bilgi katmaktan mutluluk duyan bir yazılım eğitmeni ve kod sever.
Grafik tasarım dünyasında Indesign'ı anlatacak kadar iyi kullanabilen, Photoshop ve Illustrator'e de orta düzey hakimiyeti bulunan, internet üzerinde online kimya dergisi kurmuş ve bunu yüzbinlerce kişiye ulaştırmayı başaran bir grafik sever.
WordPress tarafında 5 adet web sitesi kurmuş, bu sistemler üzerine çalışan bir WordPress sever.
Sosyal medya üzerinde okumalara ve araştırmalara devam eden, kendi kurmuş olduğu kimya dergisini daha çok kişiye nasıl ulaştırırım diye düşünen ve bu konuda SEO çalışmaları üzerine kafa yoran bir Sosyal Medya sever.
Kısaca çok çalışmayı seven, azimli biriyim.

Kopyalamak Yasaktır!