哪个更快/更好?SELECT*或SELECT column1,colum2,column3等

Dan*_*ert 159 sql database

我听说SELECT *在编写SQL命令时使用通常是不好的做法,因为它对SELECT您特别需要的列更有效.

如果我需要SELECT表中的每一列,我应该使用

SELECT * FROM TABLE
Run Code Online (Sandbox Code Playgroud)

要么

SELECT column1, colum2, column3, etc. FROM TABLE
Run Code Online (Sandbox Code Playgroud)

在这种情况下,效率真的重要吗?SELECT *如果你确实需要所有的数据,我认为内部会更优化,但我说的是没有真正了解数据库.

我很想知道在这种情况下最佳做法是什么.

更新:我可能应该指定我真正想要做的唯一情况SELECT *是当我从一个表中选择数据时我知道所有列总是需要被检索,即使添加了新列.

鉴于我所看到的反应,这仍然是一个坏主意,SELECT *永远不应该用于更多技术原因,我曾经考虑过.

Jon*_*way 163

选择特定列的更好的一个原因是它提高了SQL Server可以从索引访问数据而不是查询表数据的可能性.

这是我写的关于它的帖子: 选择查询的真正原因是索引覆盖率差

它的变化也不那么脆弱,因为任何使用数据的代码都将获得相同的数据结构,而不管您将来对表模式所做的更改.

  • 这不是他的问题的答案 - "如果我需要选择表格中的每一列,......" - 在这种情况下,*vs col1,..,coln无关紧要(但它适用于程序员时间,因为*更短!). (21认同)
  • 虽然乔恩所说的是完全正确的,并且是一个非常有效的观点,但我必须同意AS ASKED的问题是关于他们是否已经要求所有专栏.由于这部分问题,真正的问题是面对架构变化时的脆弱性. (4认同)
  • +1为此.如果引用的所有列都存在于单个索引中("覆盖索引"),那么您已经获得了金牌. (3认同)
  • 它仍然很重要,因为选择列表是一种契约形式,特别是如果SQL在存储过程中. (3认同)

IDi*_*ble 57

鉴于你的规范,你选择所有列,几乎没有什么差别 此时.但是,要意识到数据库模式确实会发生变化.如果您使用,SELECT *您将获得添加到表中的任何新列,即使很可能,您的代码也不准备使用或显示该新数据.这意味着您将系统暴露给意外的性能和功能更改.

您可能愿意将此视为次要成本,但要意识到您不需要的列仍必须是:

  1. 从数据库中读取
  2. 通过网络发送
  3. 编入你的过程
  4. (对于ADO类型的技术)保存在内存中的数据表中
  5. 忽略并丢弃/垃圾收集

项目#1有许多隐藏成本,包括消除一些潜在的覆盖索引,导致数据页面加载(以及服务器缓存颠簸),导致可能以其他方式避免的行/页面/表锁定.

将此与指定列与a *的潜在节省相平衡,唯一可能的节省是:

  1. 程序员不需要重新访问SQL来添加列
  2. SQL的网络传输更小/更快
  3. SQL Server查询解析/验证时间
  4. SQL Server查询计划缓存

对于第1项,实际情况是您要添加/更改代码以使用您可能添加的任何新列,因此它是一个清洗.

对于第2项,差异很小,无法将您推入不同的数据包大小或网络数据包数量.如果达到SQL语句传输时间是主要问题的程度,则可能需要首先降低语句的速率.

对于第3项,没有任何节省,因为*无论如何必须进行扩展,这意味着无论如何都要咨询表格模式.实际上,列出列将产生相同的成本,因为它们必须根据模式进行验证.换句话说,这是完全洗涤.

对于第4项,当您指定特定列,查询计划缓存可以得到更大的,但只有当你正在处理不同的列集合(这是不是您所指定的).在这种情况下,您需要不同的缓存条目,因为您需要根据需要使用不同的计划.

因此,由于您指定问题的方式,以及面对最终架构修改时的问题弹性,这一切都归结为.如果您正在将此架构刻录到ROM(它发生),那么*完全可以接受.

但是,我的一般准则是您应该只选择所需的列,这意味着有时它看起来像是要求所有这些列,但DBA和模式演变意味着可能会出现一些可能会极大地影响查询的新列.

我的建议是你应该总是选择特定的列.请记住,你一遍又一遍地擅长做什么,所以要养成做正确的习惯.

如果您想知道为什么架构可能在没有代码更改的情况下发生更改,请考虑审计日志记录,有效/到期日期以及DBA为系统性地添加合规性问题而添加的其他类似事项.另一个不良变化的来源是系统中其他地方或用户定义字段的性能的非规范化.

  • "实际情况是你要添加/更改代码以使用你可能添加的任何新列,所以这是一个清洗." - 只有在代码中按名称手动读取每列时才会这样.如果您正在使用自动映射,则情况并非如此,并且此问题变得非常重要. (3认同)

Gio*_*rgi 34

您应该只选择所需的列.即使您需要所有列,最好列出列名,以便sql server不必查询列的系统表.

此外,如果有人向表中添加列,您的应用程序可能会中断.您的程序将获得它不期望的列,它可能不知道如何处理它们.

除此之外,如果表具有二进制列,则查询将更慢并使用更多网络资源.

  • 啊啊所以通过使用*你正在为数据库添加额外的工作.好吧,这是我没想过的一个原因. (6认同)
  • SQL服务器是否需要验证或检查"col1"是否仍在指定的表中,即查询系统表? (6认同)
  • 最大的性能影响可能与索引有关.如果您要查找的列是用于查找数据的索引的一部分,则服务器将在那里获取数据,如果您执行select*,则很可能必须执行所谓的书签查找,这需要额外的扫描以找到您可能甚至不需要的其他基础数据. (3认同)
  • @Patrick - 现货.很多理由要避免*但这不是其中之一. (3认同)

pkh*_*pkh 30

有四个重要原因select *是坏事:

  1. 最重要的实际原因是它迫使用户神奇地知道返回列的顺序.最好是明确的,这也可以保护你免受表格的变化,这很好地分为......

  2. 如果您正在使用的列名称发生更改,则最好及早捕获它(在SQL调用时),而不是在尝试使用不再存在的列时(或更改其名称等). )

  3. 列出列名使您的代码更加自我记录,因此可能更具可读性.

  4. 如果您通过网络进行转移(或者即使您不是),您不需要的列也只是浪费.

  • "最重要的实际原因是它迫使用户神奇地知道返回列的顺序." 我不明白这是怎么回事.在任何现代数据库客户端中,您按名称而不是订单读取列. (5认同)

小智 10

指定列列表通常是最佳选择,因为如果有人向表中添加/插入列,则应用程序不会受到影响.


Her*_*ill 7

为服务器指定列名绝对更快。但是如果

  1. 性能并不是一个大问题(例如,这是一个网站内容数据库,每个表中有数百行,也许是数千行,但没有数百万行);和
  2. 您的工作是使用一个通用框架创建许多小型的类似应用程序(例如,面向公众的内容管理网站),而不是创建一个复杂的一次性应用程序;和
  3. 灵活性很重要(为每个站点定制大量的数据库模式);

那么最好还是坚持使用SELECT *。在我们的框架中,大量使用SELECT *可以使我们向表中引入新的网站托管内容字段,从而为它提供CMS的所有优势(版本,工作流/批准等),而只需触摸代码即可。几分,而不是几分。

我知道数据库专家会为此而讨厌我-继续,请投下我的票-但是在我的世界中,开发人员时间很少,CPU周期也很充裕,所以我要相应地调整我所保存和浪费的东西。


Vla*_*adV 6

即使查询不是通过网络发送的,SELECT*也是一种不好的做法.

  1. 选择多于您需要的数据会使查询效率降低 - 服务器必须读取和传输额外数据,因此需要花费时间并在系统上产生不必要的负载(不仅是网络,如其他人提到的,还有磁盘,CPU等). ).此外,服务器无法优化查询(例如,使用覆盖索引查询).
  2. 一段时间后,您的表结构可能会发生变化,因此SELECT*将返回一组不同的列.因此,您的应用程序可能会获得意外结构的数据集并在下游的某处中断.明确说明列可确保您获得已知结构的数据集,或者在数据库级别上获得明确的错误(例如"未找到列").

当然,对于一个小而简单的系统来说,这一切并不重要.


Chr*_*eld 5

到目前为止,这里已经回答了很多充分的理由,还有另一个尚未提及的理由。

明确命名列将帮助您进行后续维护。在某些时候,您将要进行更改或排除故障,并发现自己在问“该列到底用在哪里”。

如果您已经明确列出了名称,那么通过所有存储过程、视图等查找对该列的每个引用就很简单。只需为您的数据库模式转储一个 CREATE 脚本,然后通过它进行文本搜索。