热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

mysql数据类型哪个效率更高_MySQL数据类型选择

当一个列可以选择多种数据类型时,应该优先选择数字类型,其次是日期或二进制类型,最后是字符类型。越简单的数据类型,需要的处理资

当一个列可以选择多种数据类型时,应该优先选择数字类型,其次是日期或二进制类型,最后是字符类型。越简单的数据类型,需要的处理资源就越少。

对于相同级别的数据类型,应该优先选择占用空间小的数据类型。因为在数据库中,数据处理是以页为单位的,每个页存储的数据量是一定的(Innodb一页是16K),列的长度越小,单页能容纳的行数就越多。这有利于减少磁盘IO,提高数据库性能。

最好指定列为NOT NULL。可为NULL的列对MySQL来说很难优化,因为可为NULL的列使得索引、索引统计和值比较都更复杂。可为NULL的列也会占用更多的存储空间。

整数类型

187d69649c7c303d0a322f61ac3e8a34.png

注意,MySql虽然可以为整数指定宽度,例如int(11),但是这没有任何意义,它并不会限制值的合法范围

实数类型

2247e1e697099b9a96f49e6ef91dff96.png

可以利用DECIMAL存储比BIGINT还大的整数。

CPU不支持对DECIMAL的直接计算,是MySQL自身实现了对DECIMAL的高精度计算。MySQL将DECIMAL类型的数字用二进制字符串存储,每4个字节存9个数字。例如,DECIMAL(18,9)小数点两边各存储9个数字,所以小数点两边各占用4个字节,再加上小数点本身占用1个字节,总共占用9个字节。DECIMAL类型的计算效率没有DOUBLE和FLOAT高。

MySQL使用DOUBLE作为内部浮点计算类型。

因为DECIMAL类型需要比较大的空间和计算开销,所以应该只在对小数进行精确计算时才使用DECIMAL。

VARCHAR和CHAR类型

varchar类型用于存储可变长字符串,它只占用必要的存储空间。

varchar需要使用1或2个额外字节记录字符串的长度:如果字符串的最大长度小于255则只占用1个字节用于记录字符串长度,如果字符串的长度大于255则要占用2个字节用于记录字符串的长度。这就代表varchar的最大长度就是65535。但是在实际中varchar列不能达到65535这么大,因为对innodb来说,65535是一行中所有varchar列共享的长度。

下面的这些情况使用varchar类型是合适的:

字符串的最大长度比平均长度大很多

列的更新很少。这是因为行是变长的,在update时可能使行变的比原来更长。如果行占用的空间增长,并且在页内没有更多的空间可以存储,就需要存储引擎特别的处理。例如,Innodb存储引擎需要分裂页使行放入页内。这会导致增加内存碎片,降低数据库性能。

char类型是定长的。它的最大宽度为255。

char类型字符串末尾的空格会被删除。

下面的这些情况使用char类型是合适的:

存储很短的字符串,或者所有字符串长度都比较接近。短字符串用char类型存储比用varchar类型效率要高。

经常变更的数据。因为char类型是定长的,对其变更不会产生内存碎片。

varchar和char的宽度都是以字符为单位的。两种类型相比,char类型会去掉字符串末尾的空格,而varchar类型不会。

VARCHAR(5)和VARCHAR(20)存储hello的空间开销是一样的,那么使用更短的列有什么优势吗?

有很大的优势。MySql通常会分配固定大小的内存块存储内部值。虽然varchar类型在磁盘上是采用变长的方式存储,但在内存中存储varchar类型字符串使用的是其最大宽度。所以更长的列会消耗更大的内存。应该只分配真正需要的空间。

日期和时间类型

MySql能存储的最小时间粒度为秒,但是MySql可以使用微妙级别的粒度进行临时运算。

DATETIME类型:

DATETIME可以保存从1001年到9999年之间的时间值,精度为秒,占用8个字节的存储空间。它将时间值保存到格式为YYYYMMDDHHMMSS的整数中,保存的时间值与时区无关。

TIMESTAMP类型:

TIMESTAMP就是时间戳,它保存了从1970年1月1日零点以来的秒数,与UNIX时间戳值相同。TIMESTAMP只占用4个字节的存储空间,表示的时间范围只能从1970年到2038年。

TIMESTAMP显示的值依赖于时区,下面例子展示了在不同的时区下TIMESTAMP类型的值会有变化。

ed7605e72ea50b2d3c734ceb6406b4e0.png

创建的test_date表分别包含了timestamp类型的列和datatime类型的列。通过set time_zone='+10:00'命令设置时区。向表中插入一行记录,两个列的值都是当前时间。通过select语句查看结果,两种不同类型列的值是相同的。再通过set time_zone='-10:00'命令修改时区,从第二次select语句的执行结果可以看到,timestamp类型列的值已经改变了,而datetime类型列的值没有改变。

timestamp还有datetime没有的特殊属性,默认情况下,如果insert时没有指定第一个timestamp类型列的值时,MySql会自动设置这个列的值为当前时间。在更新一行记录时,MySql也会自动更新timestamp类型列的值。

8be039841d1bdece6cdd56c332b78785.png

可以从例子中看到,MySql会自动给timestamp类型的列用当前时间赋值。而datetime类型的列不会。

70661e41299dd850823586851c61aad0.png

当更新列时,timestamp类型的列也会自动更新成当前时间。

标识列的类型选择

标识列可能会和其它值进行比较(如在关联操作中),或者根据标识列寻找其它列。标识列也可能作为其它表的外键使用。所以为标识列选择恰当的数据类型非常重要。

整数类型是标识列最好的选择,因为整数计算很快并且可以AUTO_INCREMENT。

应该尽可能避免采用字符串类型作为标识列。因为字符串类型很消耗空间,并且计算比数字类型要慢。



推荐阅读
  • MyBatis多表查询与动态SQL使用
    本文介绍了MyBatis多表查询与动态SQL的使用方法,包括一对一查询和一对多查询。同时还介绍了动态SQL的使用,包括if标签、trim标签、where标签、set标签和foreach标签的用法。文章还提供了相关的配置信息和示例代码。 ... [详细]
  • 本文讨论了在openwrt-17.01版本中,mt7628设备上初始化启动时eth0的mac地址总是随机生成的问题。每次随机生成的eth0的mac地址都会写到/sys/class/net/eth0/address目录下,而openwrt-17.01原版的SDK会根据随机生成的eth0的mac地址再生成eth0.1、eth0.2等,生成后的mac地址会保存在/etc/config/network下。 ... [详细]
  • CSS3选择器的使用方法详解,提高Web开发效率和精准度
    本文详细介绍了CSS3新增的选择器方法,包括属性选择器的使用。通过CSS3选择器,可以提高Web开发的效率和精准度,使得查找元素更加方便和快捷。同时,本文还对属性选择器的各种用法进行了详细解释,并给出了相应的代码示例。通过学习本文,读者可以更好地掌握CSS3选择器的使用方法,提升自己的Web开发能力。 ... [详细]
  • 本文介绍了如何使用php限制数据库插入的条数并显示每次插入数据库之间的数据数目,以及避免重复提交的方法。同时还介绍了如何限制某一个数据库用户的并发连接数,以及设置数据库的连接数和连接超时时间的方法。最后提供了一些关于浏览器在线用户数和数据库连接数量比例的参考值。 ... [详细]
  • 本文详细介绍了Linux中进程控制块PCBtask_struct结构体的结构和作用,包括进程状态、进程号、待处理信号、进程地址空间、调度标志、锁深度、基本时间片、调度策略以及内存管理信息等方面的内容。阅读本文可以更加深入地了解Linux进程管理的原理和机制。 ... [详细]
  • 计算机存储系统的层次结构及其优势
    本文介绍了计算机存储系统的层次结构,包括高速缓存、主存储器和辅助存储器三个层次。通过分层存储数据可以提高程序的执行效率。计算机存储系统的层次结构将各种不同存储容量、存取速度和价格的存储器有机组合成整体,形成可寻址存储空间比主存储器空间大得多的存储整体。由于辅助存储器容量大、价格低,使得整体存储系统的平均价格降低。同时,高速缓存的存取速度可以和CPU的工作速度相匹配,进一步提高程序执行效率。 ... [详细]
  • 本文介绍了一个在线急等问题解决方法,即如何统计数据库中某个字段下的所有数据,并将结果显示在文本框里。作者提到了自己是一个菜鸟,希望能够得到帮助。作者使用的是ACCESS数据库,并且给出了一个例子,希望得到的结果是560。作者还提到自己已经尝试了使用"select sum(字段2) from 表名"的语句,得到的结果是650,但不知道如何得到560。希望能够得到解决方案。 ... [详细]
  • 本文详细介绍了Spring的JdbcTemplate的使用方法,包括执行存储过程、存储函数的call()方法,执行任何SQL语句的execute()方法,单个更新和批量更新的update()和batchUpdate()方法,以及单查和列表查询的query()和queryForXXX()方法。提供了经过测试的API供使用。 ... [详细]
  • 高质量SQL书写的30条建议
    本文提供了30条关于优化SQL的建议,包括避免使用select *,使用具体字段,以及使用limit 1等。这些建议是基于实际开发经验总结出来的,旨在帮助读者优化SQL查询。 ... [详细]
  • 本文讨论了clone的fork与pthread_create创建线程的不同之处。进程是一个指令执行流及其执行环境,其执行环境是一个系统资源的集合。在调用系统调用fork创建一个进程时,子进程只是完全复制父进程的资源,这样得到的子进程独立于父进程,具有良好的并发性。但是二者之间的通讯需要通过专门的通讯机制,另外通过fork创建子进程系统开销很大。因此,在某些情况下,使用clone或pthread_create创建线程可能更加高效。 ... [详细]
  • 本文介绍了Windows操作系统的版本及其特点,包括Windows 7系统的6个版本:Starter、Home Basic、Home Premium、Professional、Enterprise、Ultimate。Windows操作系统是微软公司研发的一套操作系统,具有人机操作性优异、支持的应用软件较多、对硬件支持良好等优点。Windows 7 Starter是功能最少的版本,缺乏Aero特效功能,没有64位支持,最初设计不能同时运行三个以上应用程序。 ... [详细]
  • 本文介绍了设计师伊振华受邀参与沈阳市智慧城市运行管理中心项目的整体设计,并以数字赋能和创新驱动高质量发展的理念,建设了集成、智慧、高效的一体化城市综合管理平台,促进了城市的数字化转型。该中心被称为当代城市的智能心脏,为沈阳市的智慧城市建设做出了重要贡献。 ... [详细]
  • 向QTextEdit拖放文件的方法及实现步骤
    本文介绍了在使用QTextEdit时如何实现拖放文件的功能,包括相关的方法和实现步骤。通过重写dragEnterEvent和dropEvent函数,并结合QMimeData和QUrl等类,可以轻松实现向QTextEdit拖放文件的功能。详细的代码实现和说明可以参考本文提供的示例代码。 ... [详细]
  • Tomcat/Jetty为何选择扩展线程池而不是使用JDK原生线程池?
    本文探讨了Tomcat和Jetty选择扩展线程池而不是使用JDK原生线程池的原因。通过比较IO密集型任务和CPU密集型任务的特点,解释了为何Tomcat和Jetty需要扩展线程池来提高并发度和任务处理速度。同时,介绍了JDK原生线程池的工作流程。 ... [详细]
  • 本文介绍了机器学习手册中关于日期和时区操作的重要性以及其在实际应用中的作用。文章以一个故事为背景,描述了学童们面对老先生的教导时的反应,以及上官如在这个过程中的表现。同时,文章也提到了顾慎为对上官如的恨意以及他们之间的矛盾源于早年的结局。最后,文章强调了日期和时区操作在机器学习中的重要性,并指出了其在实际应用中的作用和意义。 ... [详细]
author-avatar
mobiledu2502862267
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有