写在前面
在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_plan24 FROM25 sys.dm_exec_requests er26 INNER JOIN sys.dm_exec_sessions es ON er.session_id = es.session_id27 CROSS APPLY sys.dm_exec_sql_text(er.sql_handle)as qt28 WHERE29 es.session_id > 5030 AND es.session_Id NOT IN (@@SPID)31 )32 SELECT33 *34 FROM35 sess36 UNION ALL SELECT37 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_plan57 FROM58 sys.dm_exec_sessions es59 INNER JOIN sys.dm_exec_connections ec ON es.session_id = ec.session_id60 CROSS APPLY sys.dm_exec_sql_text(ec.most_recent_sql_handle)as qt61 WHERE62 ec.most_recent_session_id IN63 (64 SELECT blocking_session_id FROM sess WHERE blocking_session_id NOT IN(SELECT DISTINCT session_id FROM sess)65 )66 ORDER BY67 1, 2
分区表问题
描述:数据量千万级别了使用分区表提升性能
答:分区表的使用场景主要是管理数据,而提升性能主要是靠IO并行,需要合理规划多块物理磁盘,大多数的场景下几千万数据单一的模式查询只需要添加正确的索引即可。
高可用的选择
答:SQL自带的高可用或读写分离技术主要有:故障转移群集、发布订阅、镜像、日志传送、AlwaysON可用组(具体可以在进阶问题的资料中详细查看)
一般选用读写分离需要根据不同的场景和要求,比如同步的实时性,读写分离功能的需要情况
主要列出几个优缺点:
故障转移群集:主备模式,单活(辅助机不可读),硬件资源浪费,主要场景是数据库的高可用。
发布订阅:读写分离常用方式,配置灵活,副本节点可以多个,可以发布订阅部分数据(即可以对数据筛选),并提供多种发布订阅模式,缺点:维护比较麻烦,一般不能用作高可用。
镜像:主备模式,单活(辅助机不可读),硬件资源浪费,主要场景是数据库的高可用。相对于故障转移群集镜像是数据库级别的高可用。在镜像中可以使用快照的方式实现读写分离。
日志传送:主要用于灾备,在备用机上可读,但缺点是日志还原时不能读,读时不能还原。
AlwaysON可用组:综合性方案,满足高可用、读写分离等需要,要求:SQL Server2012 以上版本
第三方产品:moebius负载均衡集群,实现双活,读负载均衡、读写分离等。缺点实时同步不适合类似采集系统的大规模写入系统。
服务无法启动
答:服务无法启动有很多原因,需要具体问题具体定位,如果遇到此类问题要首先查看日志定位问题,日志主要两部分,SQL启动日志和windows日志,下面给出两篇经典解析SQL启动的文章:
数据库设计,表设计的问题
大多数这样的问题,在QQ群里问是根本得不到答案的,很多业务场景不是几句话可以描述清楚的。
SQL语句问题
描述:SQL语句增加或者减少一个条件就变得很慢
答:SQL语句的运行变化很微妙,需要理解执行计划,几句话或者贴个图无法解决,一些语句的习惯是需要养成的,请参见:
AlwaysOn配置问题
AlwaysOn配置问题请参见桦仔的几篇非常细致的文章:
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 BEGIN101 SET @tmpstr = @tmpstr + ', CHECK_POLICY = ' + @is_policy_checked102 END103 IF ( @is_expiration_checked IS NOT NULL )104 BEGIN105 SET @tmpstr = @tmpstr + ', CHECK_EXPIRATION = ' + @is_expiration_checked106 END107 END108 IF (@denylogin = 1)109 BEGIN -- login is denied access110 SET @tmpstr = @tmpstr + '; DENY CONNECT SQL TO ' + QUOTENAME( @name )111 END112 ELSE IF (@hasaccess = 0)113 BEGIN -- login exists but does not have access114 SET @tmpstr = @tmpstr + '; REVOKE CONNECT SQL TO ' + QUOTENAME( @name )115 END116 IF (@is_disabled = 1)117 BEGIN -- login is disabled118 SET @tmpstr = @tmpstr + '; ALTER LOGIN ' + QUOTENAME( @name ) + ' DISABLE'119 END120 PRINT @tmpstr121 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 rolename125 FROM 126 sys.server_role_members 127 WHERE 128 member_principal_id = @Principal_id129 OPEN server_role_members_curs130 131 FETCH NEXT FROM server_role_members_curs INTO @rolename132 WHILE (@@fetch_status <> -1)133 BEGIN134 SELECT @tmpstr = 'EXEC master..sp_addsrvrolemember @loginame = N''' + @name + ''', @rolename = N''' + @rolename + ''''135 PRINT @tmpstr136 PRINT 'GO'137 FETCH NEXT FROM server_role_members_curs INTO @rolename138 END139 CLOSE server_role_members_curs140 DEALLOCATE server_role_members_curs 141 END142 FETCH NEXT FROM login_curs INTO @Principal_id, @SID_varbinary, @name, @type, @is_disabled, @defaultdb, @language, @hasaccess, @denylogin143 END144 CLOSE login_curs145 DEALLOCATE login_curs146 GO147 148 DROP PROCEDURE #sp_hexadecimal149 GO
4.找到查询出的脚本,在辅助节点运行(其中主要的就是SID)
进阶问题
进阶问题中需要对数据库知识有一定的积累,无法几句话概括,所以下面给出一些经典文章的链接:
数据库优化问题
整体思路:
具体细节:
数据库巡检及指标
巡检系列:
高可用技术
负载均衡集群
常用优化工具平台
SQL专家云平台 :
profiler与性能计数器:
语句的分析工具:
运维脚本
--------------博客地址---------------------------------------------------------------------------------------
博客地址
欢迎转载,请注明出处,谢谢
-----------------------------------------------------------------------------------------------------
总结 : 遇到的问题很多,一时间很多想不起来,我会慢慢整理,慢慢补充,争取让此篇变成对看官们很有帮助的一边总结。
遇到的常见问题,希望大家给予补充,一起完善这篇文章。
----------------------------------------------------------------------------------------------------
注:此文章为原创,欢迎转载,请在文章页面明显位置给出此文链接!
若您觉得这篇文章还不错请点击下右下角的推荐,非常感谢!