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.
1 2 3 |
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.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 |
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.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 |
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.

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.
1 2 |
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.
1 |
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.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 |
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.
1 2 |
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.
