1. 程式人生 > >SQL Server常見問題總結

SQL Server常見問題總結

寫在前面

  在QQ群,微信群,論壇中經常幫助使用SQL Server資料庫的朋友解決問題,但是有一些最常見最基本的問題,每天都有人問,回答多了也不想再解答了,索性把這些問題整理一下,再有人問到直接發連結。

   一時想法而寫這篇文章,問題可能不全面,後續會一直更新。

基礎問題收集

資源下載

  描述:XX版本資料庫作業系統在哪裡下載?

連線問題

  描述:資料庫連線不上

  

  答:請確認SQL服務是否啟動,使用者密碼是否正確,連線的例項名稱,埠是否正確

  

日誌問題

  描述:系統日誌LDF滿了 或 日誌檔案非常大 如何收縮?

  答:簡單恢復模式下SQL Server會自動截斷日誌檔案,完整模式下需要日誌備份

  恢復模式檢視

  

  日誌備份的方式

  

  收縮日誌

  

  注:很多人使用簡單模式習慣了,或者根本不知道自己用的什麼模式,但是如果做的映象,AlwaysOn這類方案日誌必定是完整模式。

  日誌不能收縮有較多的原因,常見的是沒有備份和Replication 也就是使用映象、AlwaysOn、cdc這些技術的時候日誌同步中除了問題或這沒有同步完成。

  一般正規軍解決方式: 

  • 檢視 sys.databases 裡面 log_reuse_wait_desc欄位 如果是nothing才能收縮 
  • log_reuse_wait_desc 為 backup 需要備份日誌
  • Replication 則需要檢視映象、AlwaysOn、cdc這些技術狀態是否正常,如果不正常,必須拆除或者調整為正常
  • 依次處理直到nothing才能收縮

查詢很久慢

  描述:查詢很久都查不出資料,很慢!

  答:這樣的情況出現一般是查詢語句被其他語句阻塞。在查詢中新增 select * from table with (nolock)如果能查出來說明阻塞

  具體的阻塞情況 可以使用sp_who2 或者 sys.dm_exec_requests 檢視查詢

  具體指令碼(檢視語句執行情況)

 1 WITH sess AS
 2 (
 3     SELECT
4 es.session_id, 5 database_name = DB_NAME(er.database_id), 6 er.cpu_time, 7 er.reads, 8 er.writes, 9 er.logical_reads, 10 login_name, 11 er.status, 12 blocking_session_id, 13 wait_type, 14 wait_resource, 15 wait_time, 16 individual_query = SUBSTRING (qt.text, (er.statement_start_offset/2)+1, ((CASE WHEN er.statement_end_offset = -1 THEN LEN(CONVERT(NVARCHAR(MAX), qt.text)) * 2 ELSE er.statement_end_offset END - er.statement_start_offset)/2)+1), 17 parent_query = qt.text, 18 program_name, 19 host_name, 20 nt_domain, 21 start_time, 22 DATEDIFF(MS,er.start_time,GETDATE()) as duration, 23 (SELECT query_plan FROM sys.dm_exec_query_plan(er.plan_handle)) AS query_plan 24 FROM 25 sys.dm_exec_requests er 26 INNER JOIN sys.dm_exec_sessions es ON er.session_id = es.session_id 27 CROSS APPLY sys.dm_exec_sql_text(er.sql_handle)as qt 28 WHERE 29 es.session_id > 50 30 AND es.session_Id NOT IN (@@SPID) 31 ) 32 SELECT 33 * 34 FROM 35 sess 36 UNION ALL SELECT 37 es.session_id, 38 database_name = '', 39 0, 40 0, 41 0, 42 0, 43 login_name, 44 es.status, 45 0, 46 '', 47 '', 48 '', 49 qt.text, 50 parent_query = qt.text, 51 program_name, 52 host_name, 53 nt_domain, 54 es.last_request_start_time, 55 DATEDIFF(MS,es.last_request_start_time,GETDATE()) as duration, 56 NULL AS query_plan 57 FROM 58 sys.dm_exec_sessions es 59 INNER JOIN sys.dm_exec_connections ec ON es.session_id = ec.session_id 60 CROSS APPLY sys.dm_exec_sql_text(ec.most_recent_sql_handle)as qt 61 WHERE 62 ec.most_recent_session_id IN 63 ( 64 SELECT blocking_session_id FROM sess WHERE blocking_session_id NOT IN(SELECT DISTINCT session_id FROM sess) 65 ) 66 ORDER BY 67 1, 2

分割槽表問題

  描述:資料量千萬級別了使用分割槽表提升效能

   答:分割槽表的使用場景主要是管理資料,而提升效能主要是靠IO並行,需要合理規劃多塊物理磁碟,大多數的場景下幾千萬資料單一的模式查詢只需要新增正確的索引即可。

高可用的選擇

  答:SQL自帶的高可用或讀寫分離技術主要有:故障轉移群集、釋出訂閱、映象、日誌傳送、AlwaysON可用組(具體可以在進階問題的資料中詳細檢視)

  一般選用讀寫分離需要根據不同的場景和要求,比如同步的實時性,讀寫分離功能的需要情況

  主要列出幾個優缺點:

  故障轉移群集:主備模式,單活(輔助機不可讀),硬體資源浪費,主要場景是資料庫的高可用。

  釋出訂閱:讀寫分離常用方式,配置靈活,副本節點可以多個,可以釋出訂閱部分資料(即可以對資料篩選),並提供多種釋出訂閱模式,缺點:維護比較麻煩,一般不能用作高可用。

  映象:主備模式,單活(輔助機不可讀),硬體資源浪費,主要場景是資料庫的高可用。相對於故障轉移群集映象是資料庫級別的高可用。在映象中可以使用快照的方式實現讀寫分離。

  日誌傳送:主要用於災備,在備用機上可讀,但缺點是日誌還原時不能讀,讀時不能還原。

  AlwaysON可用組:綜合性方案,滿足高可用、讀寫分離等需要,要求:SQL Server2012 以上版本

  第三方產品:moebius負載均衡叢集,實現雙活,讀負載均衡、讀寫分離等。缺點實時同步不適合類似採集系統的大規模寫入系統。

服務無法啟動

  答:服務無法啟動有很多原因,需要具體問題具體定位,如果遇到此類問題要首先檢視日誌定位問題,日誌主要兩部分,SQL啟動日誌和windows日誌,下面給出兩篇經典解析SQL啟動的文章:

資料庫設計,表設計的問題

  大多數這樣的問題,在QQ群裡問是根本得不到答案的,很多業務場景不是幾句話可以描述清楚的。

SQL語句問題

  描述:SQL語句增加或者減少一個條件就變得很慢

  答:SQL語句的執行變化很微妙,需要理解執行計劃,幾句話或者貼個圖無法解決,一些語句的習慣是需要養成的,請參見:

AlwaysOn配置問題

   AlwaysOn配置問題請參見樺仔的幾篇非常細緻的文章:

AlwaysOn新建使用者 

  首先要明白AlwaysOn可用組中:

  1.只有主節點是可以寫入的,輔助節點只讀

  2.許可權分成兩部分,例項級別“登入名”和資料庫級別“使用者”

  3.在主節點建立登入名稱並選擇資料庫許可權後,因為資料同步,所以從庫上已經有了新建立使用者的資料庫許可權,但是沒有登入名。

  4.不能在輔助節點同樣的方式建立登入名,這樣就是“使用者孤立”問題

  解決方法:  

  1.在主節點上直接新增的是“登入名”,比如建立一個登入名 KK

  

  2.選擇資料庫許可權及使用者對映

  

  3.查詢剛才建立“登入名”的指令碼(此指令碼也可以用於升級或遷移資料庫還原後,登入名同步的問題)

  1 CREATE PROCEDURE #sp_hexadecimal
  2     @binvalue varbinary(256),
  3     @hexvalue varchar (514) OUTPUT
  4 AS
  5     DECLARE @charvalue varchar (514)
  6     DECLARE @i int
  7     DECLARE @length int
  8     DECLARE @hexstring char(16)
  9 
 10     SELECT @charvalue = '0x'
 11     SELECT @i = 1
 12     SELECT @length = DATALENGTH (@binvalue)
 13     SELECT @hexstring = '0123456789ABCDEF'
 14     WHILE (@i <= @length)
 15     BEGIN
 16         DECLARE @tempint int
 17         DECLARE @firstint int
 18         DECLARE @secondint int
 19         SELECT @tempint = CONVERT(int, SUBSTRING(@binvalue,@i,1))
 20         SELECT @firstint = FLOOR(@tempint/16)
 21         SELECT @secondint = @tempint - (@firstint*16)
 22         SELECT @charvalue = @charvalue + SUBSTRING(@hexstring, @firstint+1, 1) + SUBSTRING(@hexstring, @secondint+1, 1)
 23         SELECT @i = @i + 1
 24     END
 25     SELECT @hexvalue = @charvalue
 26 GO
 27 
 28 DECLARE @name sysname
 29 DECLARE @type varchar (1)
 30 DECLARE @hasaccess int
 31 DECLARE @denylogin int
 32 DECLARE @is_disabled int
 33 DECLARE @PWD_varbinary  varbinary (256)
 34 DECLARE @PWD_string  varchar (514)
 35 DECLARE @Principal_id int
 36 DECLARE @SID_varbinary varbinary (85)
 37 DECLARE @SID_string varchar (514)
 38 DECLARE @tmpstr  varchar (1024)
 39 DECLARE @is_policy_checked varchar (3)
 40 DECLARE @is_expiration_checked varchar (3)
 41 DECLARE @defaultdb sysname
 42 DECLARE @language sysname
 43 DECLARE @rolename sysname
 44 DECLARE login_curs CURSOR FOR SELECT 
 45     p.principal_id,
 46     p.sid, 
 47     p.name, 
 48     p.type, 
 49     p.is_disabled, 
 50     p.default_database_name, 
 51     p.default_language_name,
 52     l.hasaccess, 
 53     l.denylogin 
 54 FROM 
 55     sys.server_principals p 
 56 LEFT JOIN 
 57     sys.syslogins l ON ( l.name = p.name ) 
 58 WHERE 
 59     p.type IN ( 'S', 'G', 'U' ) AND 
 60     p.name <> 'sa'
 61 
 62 OPEN login_curs
 63 
 64 FETCH NEXT FROM login_curs INTO @Principal_id, @SID_varbinary, @name, @type, @is_disabled, @defaultdb, @language, @hasaccess, @denylogin
 65 IF (@@fetch_status = -1)
 66 BEGIN
 67   PRINT 'No login(s) found.'
 68   CLOSE login_curs
 69   DEALLOCATE login_curs
 70   RETURN
 71 END
 72 SET @tmpstr = '** Generated ' + CONVERT (varchar, GETDATE()) + ' on ' + @@SERVERNAME + ' */'
 73 PRINT @tmpstr
 74 PRINT ''
 75 WHILE (@@fetch_status <> -1)
 76 BEGIN
 77     IF (@@fetch_status <> -2)
 78     BEGIN
 79         PRINT ''
 80         SET @tmpstr = '-- Login: ' + @name
 81         PRINT @tmpstr
 82         IF (@type IN ( 'G', 'U'))
 83         BEGIN -- NT authenticated account/group
 84             SET @tmpstr = 'CREATE LOGIN ' + QUOTENAME( @name ) + ' FROM WINDOWS WITH DEFAULT_DATABASE = [' + @defaultdb + '], DEFAULT_LANGUAGE = [' + @language + ']'
 85         END
 86         ELSE 
 87         BEGIN -- SQL Server authentication
 88             -- obtain password and sid
 89             SET @PWD_varbinary = CAST( LOGINPROPERTY( @name, 'PasswordHash' ) AS varbinary (256) )
 90             EXEC #sp_hexadecimal @PWD_varbinary, @PWD_string OUT
 91             EXEC #sp_hexadecimal @SID_varbinary,@SID_string OUT
 92 
 93             -- obtain password policy state
 94             SELECT @is_policy_checked = CASE is_policy_checked WHEN 1 THEN 'ON' WHEN 0 THEN 'OFF' ELSE NULL END FROM sys.sql_logins WHERE name = @name
 95             SELECT @is_expiration_checked = CASE is_expiration_checked WHEN 1 THEN 'ON' WHEN 0 THEN 'OFF' ELSE NULL END FROM sys.sql_logins WHERE name = @name
 96 
 97             SET @tmpstr = 'CREATE LOGIN ' + QUOTENAME( @name ) + ' WITH PASSWORD = ' + @PWD_string + ' HASHED, SID = ' + @SID_string + ', DEFAULT_DATABASE = [' + @defaultdb + '], DEFAULT_LANGUAGE = [' + @language + ']'
 98 
 99             IF ( @is_policy_checked IS NOT NULL )
100             BEGIN
101                 SET @tmpstr = @tmpstr + ', CHECK_POLICY = ' + @is_policy_checked
102             END
103             IF ( @is_expiration_checked IS NOT NULL )
104             BEGIN
105                 SET @tmpstr = @tmpstr + ', CHECK_EXPIRATION = ' + @is_expiration_checked
106             END
107         END
108         IF (@denylogin = 1)
109         BEGIN -- login is denied access
110             SET @tmpstr = @tmpstr + '; DENY CONNECT SQL TO ' + QUOTENAME( @name )
111         END
112         ELSE IF (@hasaccess = 0)
113         BEGIN -- login exists but does not have access
114             SET @tmpstr = @tmpstr + '; REVOKE CONNECT SQL TO ' + QUOTENAME( @name )
115         END
116         IF (@is_disabled = 1)
117         BEGIN -- login is disabled
118             SET @tmpstr = @tmpstr + '; ALTER LOGIN ' + QUOTENAME( @name ) + ' DISABLE'
119         END
120         PRINT @tmpstr
121         PRINT 'GO'
122         DECLARE server_role_members_curs CURSOR FOR 
123             SELECT 
124                 (SELECT [name] FROM sys.server_principals WHERE principal_id = role_principal_id) AS rolename
125             FROM 
126                 sys.server_role_members 
127             WHERE 
128                 member_principal_id = @Principal_id
129         OPEN server_role_members_curs
130 
131         FETCH NEXT FROM server_role_members_curs INTO @rolename
132         WHILE (@@fetch_status <> -1)
133         BEGIN
134             SELECT @tmpstr = 'EXEC master..sp_addsrvrolemember @loginame = N''' + @name + ''', @rolename = N''' + @rolename + ''''
135             PRINT @tmpstr
136             PRINT 'GO'
137             FETCH NEXT FROM server_role_members_curs INTO @rolename
138         END
139         CLOSE server_role_members_curs
140         DEALLOCATE server_role_members_curs        
141     END
142     FETCH NEXT FROM login_curs INTO @Principal_id, @SID_varbinary, @name, @type, @is_disabled, @defaultdb, @language, @hasaccess, @denylogin
143 END
144 CLOSE login_curs
145 DEALLOCATE login_curs
146 GO
147 
148 DROP PROCEDURE #sp_hexadecimal
149 GO

  4.找到查詢出的指令碼,在輔助節點執行(其中主要的就是SID)

  

進階問題

  進階問題中需要對資料庫知識有一定的積累,無法幾句話概括,所以下面給出一些經典文章的連結:

資料庫優化問題

資料庫巡檢及指標

 高可用技術

負載均衡叢集

常用優化工具平臺

運維指令碼

--------------部落格地址---------------------------------------------------------------------------------------

 歡迎轉載,請註明出處,謝謝

-----------------------------------------------------------------------------------------------------

  總結 : 遇到的問題很多,一時間很多想不起來,我會慢慢整理,慢慢補充,爭取讓此篇變成對看官們很有幫助的一邊總結。

  遇到的常見問題,希望大家給予補充,一起完善這篇文章。

 ----------------------------------------------------------------------------------------------------

注:此文章為原創,歡迎轉載,請在文章頁面明顯位置給出此文連結!
若您覺得這篇文章還不錯請點選下右下角的推薦,非常感謝!