设为首页 收藏本站
查看: 1512|回复: 1

[经验分享] SQL Server常见问题总结

[复制链接]

尚未签到

发表于 2017-12-7 14:18:35 | 显示全部楼层 |阅读模式
写在前面
  在QQ群,微信群,论坛中经常帮助使用SQL Server数据库的朋友解决问题,但是有一些最常见最基本的问题,每天都有人问,回答多了也不想再解答了,索性把这些问题整理一下,再有人问到直接发链接。
  一时想法而写这篇文章,问题可能不全面,后续会一直更新。

基础问题收集

资源下载
  描述:XX版本数据库操作系统在哪里下载?
  答:http://www.itellyou.cn/  里面很多东西,有兴趣的自己看吧

连接问题
  描述:数据库连接不上
DSC0000.png

  答:请确认SQL服务是否启动,用户密码是否正确,连接的实例名称,端口是否正确
DSC0001.png


日志问题
  描述:系统日志LDF满了 或 日志文件非常大 如何收缩?
  答:简单恢复模式下SQL Server会自动截断日志文件,完整模式下需要日志备份
  恢复模式查看
DSC0002.png

  日志备份的方式
DSC0003.png

  收缩日志
DSC0004.png


查询很久慢
  描述:查询很久都查不出数据,很慢!
  答:这样的情况出现一般是查询语句被其他语句阻塞。在查询中添加 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语句的运行变化很微妙,需要理解执行计划,几句话或者贴个图无法解决,一些语句的习惯是需要养成的,请参见:
  SQL SERVER全面优化-------写出好语句是习惯
  SQL SERVER全面优化-------索引有多重要?

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
DSC0005.png

  2.选择数据库权限及用户映射
DSC0006.png

  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)
DSC0007.png


进阶问题
  进阶问题中需要对数据库知识有一定的积累,无法几句话概括,所以下面给出一些经典文章的链接:

数据库优化问题
  整体思路:SQL SERVER全面优化-------Expert for SQL Server 诊断系列
  具体细节:SQL Server性能调优系列
    

数据库巡检及指标
    巡检系列:轻松精通数据库管理之道——运维巡检系列


高可用技术
  数据库集群技术漫谈
  SQL Server中的高可用性(1)----高可用性概览

负载均衡集群
  大数据时代下的SQL Server第三方负载均衡方案----Moebius测试
  SQL SERVER 2016 AlwaysOn 无域集群+负载均衡搭建与简测

常用优化工具平台
  SQL专家云平台 : 30分钟带你熟练性能优化的那点儿事儿(案例说明)   
  profiler与性能计数器:性能计数器与profiler的组合性能诊断
  语句的分析工具:一款好用且免费的语句分析工具

运维脚本
  数据库的运维策略(内附脚本,无私分享)
  SQL Server自动化运维系列——监控性能指标脚本(Power Shell)
  --------------博客地址---------------------------------------------------------------------------------------
  博客地址 http://www.cnblogs.com/double-K/
  
  欢迎转载,请注明出处,谢谢
  -----------------------------------------------------------------------------------------------------
  总结 : 遇到的问题很多,一时间很多想不起来,我会慢慢整理,慢慢补充,争取让此篇变成对看官们很有帮助的一边总结。
  遇到的常见问题,希望大家给予补充,一起完善这篇文章。
  ----------------------------------------------------------------------------------------------------
  注:此文章为原创,欢迎转载,请在文章页面明显位置给出此文链接!
若您觉得这篇文章还不错请点击下右下角的推荐,非常感谢!

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-421740-1-1.html 上篇帖子: SQL Server 2014忘记SA密码或禁用而且Windows身份验证也无法登录的解决办法 下篇帖子: Windows Server 2008 R2 报错事件ID:10之WMI报错
累计签到:16 天
连续签到:4 天
发表于 2017-12-7 14:28:57 | 显示全部楼层
感谢楼主分享~~~~~~~~~~~

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表