SQL Server常见问题及解决方法分享

作者:Double_K 时间:2024-01-15 01:45:30 

写在前面

在QQ群,微信群,论坛中经常帮助使用SQL Server数据库的朋友解决问题,但是有一些最常见最基本的问题,每天都有人问,回答多了也不想再解答了,索性把这些问题整理一下,再有人问到直接发链接。

一时想法而写这篇文章,问题可能不全面,后续会一直更新。

基础问题收集资源下载

描述:XX版本数据库操作系统在哪里下载?

答:http://www.itellyou.cn/ 里面很多东西,有兴趣的自己看吧

连接问题

描述:数据库连接不上

SQL Server常见问题及解决方法分享

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

SQL Server常见问题及解决方法分享

日志问题

描述:系统日志LDF满了 或 日志文件非常大 如何收缩?

答:简单恢复模式下SQL Server会自动截断日志文件,完整模式下需要日志备份

恢复模式查看

SQL Server常见问题及解决方法分享

日志备份的方式

SQL Server常见问题及解决方法分享

收缩日志

SQL Server常见问题及解决方法分享

查询很久\很慢

描述:查询很久都查不出数据,很慢!

答:这样的情况出现一般是查询语句被其他语句阻塞。在查询中添加 select * from table with (nolock)如果能查出来说明阻塞

具体的阻塞情况 可以使用sp_who2 或者 sys.dm_exec_requests 视图查询

具体脚本(查看语句运行情况)


WITH sess AS
(
SELECT
es.session_id,
database_name = DB_NAME(er.database_id),
er.cpu_time,
er.reads,
er.writes,
er.logical_reads,
login_name,
er.status,
blocking_session_id,
wait_type,
wait_resource,
wait_time,
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),
parent_query = qt.text,
program_name,
host_name,
nt_domain,
start_time,
DATEDIFF(MS,er.start_time,GETDATE()) as duration,
(SELECT query_plan FROM sys.dm_exec_query_plan(er.plan_handle)) AS query_plan
FROM
sys.dm_exec_requests er
INNER JOIN sys.dm_exec_sessions es ON er.session_id = es.session_id
CROSS APPLY sys.dm_exec_sql_text(er.sql_handle)as qt
WHERE
es.session_id > 50
AND es.session_Id NOT IN (@@SPID)
)
SELECT
*
FROM
sess
UNION ALL SELECT
es.session_id,
database_name = '',
0,
0,
0,
0,
login_name,
es.status,
0,
'',
'',
'',
qt.text,
parent_query = qt.text,
program_name,
host_name,
nt_domain,
es.last_request_start_time,
DATEDIFF(MS,es.last_request_start_time,GETDATE()) as duration,
NULL AS query_plan
FROM
sys.dm_exec_sessions es
INNER JOIN sys.dm_exec_connections ec ON es.session_id = ec.session_id
CROSS APPLY sys.dm_exec_sql_text(ec.most_recent_sql_handle)as qt
WHERE
ec.most_recent_session_id IN
(
SELECT blocking_session_id FROM sess WHERE blocking_session_id NOT IN(SELECT DISTINCT session_id FROM sess)
)
ORDER BY
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

SQL Server常见问题及解决方法分享

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

SQL Server常见问题及解决方法分享

3.查询刚才创建“登录名”的脚本(此脚本也可以用于升级或迁移数据库还原后,登录名同步的问题)


CREATE PROCEDURE #sp_hexadecimal
@binvalue varbinary(256),
@hexvalue varchar (514) OUTPUT
AS
DECLARE @charvalue varchar (514)
DECLARE @i int
DECLARE @length int
DECLARE @hexstring char(16)

SELECT @charvalue = '0x'
SELECT @i = 1
SELECT @length = DATALENGTH (@binvalue)
SELECT @hexstring = '0123456789ABCDEF'
WHILE (@i <= @length)
BEGIN
DECLARE @tempint int
DECLARE @firstint int
DECLARE @secondint int
SELECT @tempint = CONVERT(int, SUBSTRING(@binvalue,@i,1))
SELECT @firstint = FLOOR(@tempint/16)
SELECT @secondint = @tempint - (@firstint*16)
SELECT @charvalue = @charvalue + SUBSTRING(@hexstring, @firstint+1, 1) + SUBSTRING(@hexstring, @secondint+1, 1)
SELECT @i = @i + 1
END
SELECT @hexvalue = @charvalue
GO

DECLARE @name sysname
DECLARE @type varchar (1)
DECLARE @hasaccess int
DECLARE @denylogin int
DECLARE @is_disabled int
DECLARE @PWD_varbinary varbinary (256)
DECLARE @PWD_string varchar (514)
DECLARE @Principal_id int
DECLARE @SID_varbinary varbinary (85)
DECLARE @SID_string varchar (514)
DECLARE @tmpstr varchar (1024)
DECLARE @is_policy_checked varchar (3)
DECLARE @is_expiration_checked varchar (3)
DECLARE @defaultdb sysname
DECLARE @language sysname
DECLARE @rolename sysname
DECLARE login_curs CURSOR FOR SELECT
p.principal_id,
p.sid,
p.name,
p.type,
p.is_disabled,
p.default_database_name,
p.default_language_name,
l.hasaccess,
l.denylogin
FROM
sys.server_principals p
LEFT JOIN
sys.syslogins l ON ( l.name = p.name )
WHERE
p.type IN ( 'S', 'G', 'U' ) AND
p.name <> 'sa'

OPEN login_curs

FETCH NEXT FROM login_curs INTO @Principal_id, @SID_varbinary, @name, @type, @is_disabled, @defaultdb, @language, @hasaccess, @denylogin
IF (@@fetch_status = -1)
BEGIN
PRINT 'No login(s) found.'
CLOSE login_curs
DEALLOCATE login_curs
RETURN
END
SET @tmpstr = '** Generated ' + CONVERT (varchar, GETDATE()) + ' on ' + @@SERVERNAME + ' */'
PRINT @tmpstr
PRINT ''
WHILE (@@fetch_status <> -1)
BEGIN
IF (@@fetch_status <> -2)
BEGIN
PRINT ''
SET @tmpstr = '-- Login: ' + @name
PRINT @tmpstr
IF (@type IN ( 'G', 'U'))
BEGIN -- NT authenticated account/group
 SET @tmpstr = 'CREATE LOGIN ' + QUOTENAME( @name ) + ' FROM WINDOWS WITH DEFAULT_DATABASE = [' + @defaultdb + '], DEFAULT_LANGUAGE = [' + @language + ']'
END
ELSE
BEGIN -- SQL Server authentication
 -- obtain password and sid
 SET @PWD_varbinary = CAST( LOGINPROPERTY( @name, 'PasswordHash' ) AS varbinary (256) )
 EXEC #sp_hexadecimal @PWD_varbinary, @PWD_string OUT
 EXEC #sp_hexadecimal @SID_varbinary,@SID_string OUT

-- obtain password policy state
 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
 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

SET @tmpstr = 'CREATE LOGIN ' + QUOTENAME( @name ) + ' WITH PASSWORD = ' + @PWD_string + ' HASHED, SID = ' + @SID_string + ', DEFAULT_DATABASE = [' + @defaultdb + '], DEFAULT_LANGUAGE = [' + @language + ']'

IF ( @is_policy_checked IS NOT NULL )
 BEGIN
 SET @tmpstr = @tmpstr + ', CHECK_POLICY = ' + @is_policy_checked
 END
 IF ( @is_expiration_checked IS NOT NULL )
 BEGIN
 SET @tmpstr = @tmpstr + ', CHECK_EXPIRATION = ' + @is_expiration_checked
 END
END
IF (@denylogin = 1)
BEGIN -- login is denied access
 SET @tmpstr = @tmpstr + '; DENY CONNECT SQL TO ' + QUOTENAME( @name )
END
ELSE IF (@hasaccess = 0)
BEGIN -- login exists but does not have access
 SET @tmpstr = @tmpstr + '; REVOKE CONNECT SQL TO ' + QUOTENAME( @name )
END
IF (@is_disabled = 1)
BEGIN -- login is disabled
 SET @tmpstr = @tmpstr + '; ALTER LOGIN ' + QUOTENAME( @name ) + ' DISABLE'
END
PRINT @tmpstr
PRINT 'GO'
DECLARE server_role_members_curs CURSOR FOR
 SELECT
 (SELECT [name] FROM sys.server_principals WHERE principal_id = role_principal_id) AS rolename
 FROM
 sys.server_role_members
 WHERE
 member_principal_id = @Principal_id
OPEN server_role_members_curs

FETCH NEXT FROM server_role_members_curs INTO @rolename
WHILE (@@fetch_status <> -1)
BEGIN
 SELECT @tmpstr = 'EXEC master..sp_addsrvrolemember @loginame = N''' + @name + ''', @rolename = N''' + @rolename + ''''
 PRINT @tmpstr
 PRINT 'GO'
 FETCH NEXT FROM server_role_members_curs INTO @rolename
END
CLOSE server_role_members_curs
DEALLOCATE server_role_members_curs
END
FETCH NEXT FROM login_curs INTO @Principal_id, @SID_varbinary, @name, @type, @is_disabled, @defaultdb, @language, @hasaccess, @denylogin
END
CLOSE login_curs
DEALLOCATE login_curs
GO

DROP PROCEDURE #sp_hexadecimal
GO

4.找到查询出的脚本,在辅助节点运行(其中主要的就是SID)

SQL Server常见问题及解决方法分享

进阶问题请大家点击原文阅读。

来源:http://www.cnblogs.com/double-K/archive/2017/01/09/6264814.html

标签:SQL,Server,问题
0
投稿

猜你喜欢

  • Javascript 阻止javascript事件冒泡,获取控件ID值

    2023-07-02 05:25:57
  • Mysql事务特性和级别原理解析

    2024-01-25 11:00:04
  • Python设计模式之代理模式实例详解

    2022-11-20 05:24:29
  • Python如何实现强制数据类型转换

    2022-10-18 10:08:02
  • mssql2005,2008导出数据字典实现方法

    2023-07-23 19:11:30
  • Python NLP开发之实现聊天机器人

    2021-12-21 18:04:48
  • Vue路由模式中的hash和history模式详细介绍

    2024-06-07 15:20:16
  • pytorch 如何使用float64训练

    2023-08-12 00:19:45
  • 使用Python写一个小游戏

    2023-07-20 17:31:07
  • python基础之多态

    2022-04-05 10:06:06
  • python3中rank函数的用法

    2022-09-26 07:00:45
  • Asp 操作Cookies(包括设置[赋值]、读取、删除[设置过期时间])

    2011-03-10 11:06:00
  • Laravel框架数据库CURD操作、连贯操作总结

    2023-11-17 07:22:26
  • Mysql全文搜索match against的用法

    2024-01-19 16:30:57
  • Mysql主键和唯一键的区别点总结

    2024-01-15 10:36:36
  • JavaScript 字符串连接性能优化

    2024-05-11 09:35:53
  • python训练数据时打乱训练数据与标签的两种方法小结

    2021-11-17 11:15:12
  • Python文件操作实战案例之用户登录

    2023-07-12 18:03:37
  • mysql8创建、删除用户以及授权、消权操作详解

    2024-01-13 22:46:34
  • 如何处理好网页色彩搭配

    2007-08-10 13:22:00
  • asp之家 网络编程 m.aspxhome.com