寫在前面 在QQ群,微信群,論壇中經常幫助使用SQL Server資料庫的朋友解決問題,但是有一些最常見最基本的問題,每天都有人問,回答多了也不想再解答了,索性把這些問題整理一下,再有人問到直接發鏈接。 一時想法而寫這篇文章,問題可能不全面,後續會一直更新。 基礎問題收集 資源下載 描述:XX版本數 ...
寫在前面
在QQ群,微信群,論壇中經常幫助使用SQL Server資料庫的朋友解決問題,但是有一些最常見最基本的問題,每天都有人問,回答多了也不想再解答了,索性把這些問題整理一下,再有人問到直接發鏈接。
一時想法而寫這篇文章,問題可能不全面,後續會一直更新。
基礎問題收集
資源下載
描述:XX版本資料庫操作系統在哪裡下載?
答:http://www.itellyou.cn/ 裡面很多東西,有興趣的自己看吧
連接問題
描述:資料庫連接不上
答:請確認SQL服務是否啟動,用戶密碼是否正確,連接的實例名稱,埠是否正確
日誌問題
描述:系統日誌LDF滿了 或 日誌文件非常大 如何收縮?
答:簡單恢復模式下SQL Server會自動截斷日誌文件,完整模式下需要日誌備份
恢復模式查看
日誌備份的方式
收縮日誌
查詢很久慢
描述:查詢很久都查不出數據,很慢!
答:這樣的情況出現一般是查詢語句被其他語句阻塞。在查詢中添加 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啟動的文章:
你所不知道的SQL Server資料庫啟動過程(用戶資料庫載入過程的疑難雜症)
你所不知道的SQL Server資料庫啟動過程,以及啟動不起來的各種問題的分析及解決技巧
資料庫設計,表設計的問題
大多數這樣的問題,在QQ群里問是根本得不到答案的,很多業務場景不是幾句話可以描述清楚的。
SQL語句問題
描述:SQL語句增加或者減少一個條件就變得很慢
答:SQL語句的運行變化很微妙,需要理解執行計劃,幾句話或者貼個圖無法解決,一些語句的習慣是需要養成的,請參見:
AlwaysOn配置問題
AlwaysOn配置問題請參見樺仔的幾篇非常細緻的文章:
從0開始搭建SQL Server AlwaysOn 第一篇(配置域控)
從0開始搭建SQL Server AlwaysOn 第二篇(配置故障轉移集群)
從0開始搭建SQL Server AlwaysOn 第三篇(配置AlwaysOn)
從0開始搭建SQL Server AlwaysOn 第四篇(配置異地機房節點)
2016的AlwaysOn 搭建:SQL SERVER 2016 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)
進階問題
進階問題中需要對資料庫知識有一定的積累,無法幾句話概括,所以下麵給出一些經典文章的鏈接:
資料庫優化問題
整體思路:SQL SERVER全面優化-------Expert for SQL Server 診斷系列
具體細節:SQL Server性能調優系列
資料庫巡檢及指標
巡檢系列:輕鬆精通資料庫管理之道——運維巡檢系列
高可用技術
負載均衡集群
大數據時代下的SQL Server第三方負載均衡方案----Moebius測試
SQL SERVER 2016 AlwaysOn 無域集群+負載均衡搭建與簡測
常用優化工具平臺
SQL專家雲平臺 : 30分鐘帶你熟練性能優化的那點兒事兒(案例說明)
profiler與性能計數器:性能計數器與profiler的組合性能診斷
語句的分析工具:一款好用且免費的語句分析工具
運維腳本
SQL Server自動化運維繫列——監控性能指標腳本(Power Shell)
--------------博客地址---------------------------------------------------------------------------------------
博客地址 http://www.cnblogs.com/double-K/
歡迎轉載,請註明出處,謝謝
-----------------------------------------------------------------------------------------------------
總結 : 遇到的問題很多,一時間很多想不起來,我會慢慢整理,慢慢補充,爭取讓此篇變成對看官們很有幫助的一邊總結。
遇到的常見問題,希望大家給予補充,一起完善這篇文章。
----------------------------------------------------------------------------------------------------
註:此文章為原創,歡迎轉載,請在文章頁面明顯位置給出此文鏈接!
若您覺得這篇文章還不錯請點擊下右下角的推薦,非常感謝!