百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT技术 > 正文

关于MySQL varchar类型最大值,原来一直都理解错了

wptr33 2024-12-26 17:07 32 浏览

我是架构精进之路,点击上方“关注”,坚持每天为你分享技术干货,私信我回复“01”,送你一份程序员成长进阶大礼包。

写在前面

关于MySQL varchar字段类型的最大值计算,也许我们一直都理解错误了,本文从问题出发,经实践验证得出一些实用经验,希望对大家的开发工作有些帮助~

背景描述

最近同事在做技术方案设计时候,考虑到一个表设计时希望利用varchar类型进行存储,而不是采用text,那就需要确定下varchar允许的最大长度是多少,用来评估下后期是否会遇到存储长度瓶颈。

那问题来了:MySQL 数据库的varchar字段类型最大存储长度到底是多少?

问题分析

一切以官方文档为准,翻了下官方描述如下:

In MySQL 4.1 the length is always 1 byte. In MySQL 5.0 the length may be either 1 byte (for up to 255) or 2 bytes (for 256 to 65535).

大概意思就是说:

  • 在MySQL 4.1以前,长度总是1个字节(varchar(20),指的是20字节)
  • 在MySQL 5.0以后,长度可以是1字节(最多255个字节)或2个字节(256到65535)

  • 按照官网说法最大值是65535bytes,utf8mb4编码情况下每个字符占4个bytes,最大值应该为16383.75

    65535/4=16383.75

    实践验证

    到此貌似已经有了结论了,但实际情况真的是这样的么?

    我们来实验下试试看?

    mysql 版本:
    select version(); // 5.7

    1、若一个表只有一个varchar类型

    定义如下:

    CREATETABLE`t1` (  
      `c`varchar(N) DEFAULTNULL
    ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

    那表 t1 的`c`字段的最大长度N为多少呢?

    (65535?1?2)/4=16383

    备注:

    · 减1的原因是实际行存储从第二个字节开始;

    · 减2的原因是varchar头部的2个字节表示长度;

    · 除4的原因是字符编码是utf8mb4。

    2)若表中包含其他多种类型的情况呢

    定义如下:

    CREATETABLE`t2` (
      `c1`int(10) DEFAULTNULL,
      `c2`char(32) DEFAULTNULL,
      `c3`varchar(N) DEFAULTNULL
    ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

    那表 t2 的`c`字段的最大长度N为多少呢?

    (65535?1?2?4?32*4)/4=16350

    备注:

    · 减1、减2的原因同上;

    · 减4的原因是int类型占用4个字节;

    · 减32*4的原因是utf8mb4编码的char类型占用4个字节(长度32)


    我们来验证一下是否如上述推断计算所述:

    1)修改t2表c3字段长度为16350

    alter table `t2` modify column `c3` varchar(16350);

    执行成功。

    2)修改t2表c3字段长度为16351

    alter table `t2` modify column `c3` varchar(16351);

    执行失败,报错信息如下:

    Row size too large. The maximum row size for the used table type, not counting BLOBs, is 65535. This includes storage overhead, check the manual. You have to change some columns to TEXT or BLOBs.

    总结一下

    Q:varchar到底能存多少个字符?

    A:这与表使用的字符集相关,latin1、gbk、utf8、utf8mb4编码存放一个字符分别需要占1、2、3、4个字节,同时还要考虑到去除其他字段的占用影响。


    实践出真知,可以简单试一下之后再下结论,希望本文对你有所帮助~


    - END -


    作者:架构精进之路,专注软件架构研究,技术学习与个人成长,关注并私信我回复“01”,送你一份程序员成长进阶大礼包。



    Thanks for reading!

    相关推荐

    [常用工具] git基础学习笔记_git工具有哪些

    添加推送信息,-m=messagegitcommit-m“添加注释”查看状态...

    centos7安装部署gitlab_centos7安装git服务器

    一、Gitlab介1.1gitlab信息GitLab是利用RubyonRails一个开源的版本管理系统,实现一个自托管的Git项目仓库,可通过Web界面进行访问公开的或者私人项目。...

    太高效了!玩了这么久的Linux,居然不知道这7个终端快捷键

    作为Linux用户,大家肯定在Linux终端下敲过无数的命令。有的命令很短,比如:ls、cd、pwd之类,这种命令大家毫无压力。但是,有些命令就比较长了,比如:...

    提高开发速度还能保证质量的10个小窍门

    养成坏习惯真是分分钟的事儿,而养成好习惯却很难。我发现,把那些对我有用的习惯写下来,能让我坚持住已经花心思养成的好习惯。...

    版本管理最好用的工具,你懂多少?

    版本控制(Revisioncontrol)是一种在开发的过程中用于管理我们对文件、目录或工程等内容的修改历史,方便查看更改历史记录,备份以便恢复以前的版本的软件工程技术。...

    Git回退到某个版本_git回退到某个版本详细步骤

    在开发过程,有时会遇到合并代码或者合并主分支代码导致自己分支代码冲突等问题,这时我们需要回退到某个commit_id版本1,查看所有历史版本,获取git的某个历史版本id...

    Kubernetes + Jenkins + Harbor 全景实战手册

    Kubernetes+Jenkins+Harbor全景实战手册在现代企业级DevOps体系中,Kubernetes(K8s)、Jenkins和Harbor组成的CI/CD流水...

    git常用命令整理_git常见命令

    一、Git仓库完整迁移完整迁移,就是指,不仅将所有代码移植到新的仓库,而且要保留所有的commit记录1.随便找个文件夹,从原地址克隆一份裸版本库...

    第三章:Git分支管理(多人协作基础)

    3.1分支基本概念分支是Git最强大的功能之一,它允许你在主线之外创建独立的开发线路,互不干扰。理解分支的工作原理是掌握Git的关键。核心概念:HEAD:指向当前分支的指针...

    云效Codeup怎么创建分支并进行分支管理

    云效Codeup怎么创建分支并进行分支管理,分支是为了将修改记录分叉备份保存,不受其他分支的影响,所以在同一个代码库里可以同时进行多个修改。创建仓库时,会自动创建Master分支作为默认分支,后续...

    git 如何删除本地和远程分支?_git怎么删除远程仓库

    Git分支对于开发人员来说是一项强大的功能,但要维护干净的存储库,就需要知道如何删除过时的分支。本指南涵盖了您需要了解的有关本地和远程删除Git分支的所有信息。了解Git分支...

    git 实现一份代码push到两个git地址上

    一直以来想把自己的博客代码托管到github和coding上想一次更改一次push两个地址一起更新今天有空查资料实践了下本博客的github地址coding的git地址如果是Gi...

    git操作:cherry-pick和rebase_git cherry-pick bad object

    在编码中经常涉及到分支之间的代码同步问题,那就需要cherry-pick和rebase命令问题:如何将某个分支的多个commit合并到另一个分支,并在另一个分支只保留一个commit记录解答:假设有两...

    模型文件硬塞进 Git,GitHub 直接打回原形:使用Git-LFS管理大文件

    前言最近接手了一个计算机视觉项目代码是屎山就不说了,反正我也不看代码主要就是构建一下docker镜像,测试一下部署的兼容性这本来不难但是,国内服务器的网络环境实在是恶劣,需要配置各种镜像(dock...

    防弹少年团田柾国《Euphoria》2周年 获世界实时趋势榜1位 恭喜呀

    当天韩国时间凌晨3时左右,该曲在Twitter上以“2YearsWithEuphoria”的HashTag登上了世界趋势1位。在韩国推特实时趋势中,从上午开始到现在“Euphoria2岁”的Has...