CSS之BEM命名规范

什么是BEM

BEM是Block(块)、Element(元素)、Modifier(修饰符)的简写,是一种组件化的 CSS 命名方法和规范,由俄罗斯 Yandex 团队所提出。使用BEM主要是为了将用户界面划分成独立的块,使开发更为简单和快速,有利于团队协作,方便维护。

为什么要选择BEM

很多新手在开始写网页时,在命名方面可能都比较随心所欲。但是在一个正式的项目中,会有很多开发人员同时进行开发,如果每个开发人员都用自己的一套命名,这样会造成命名的识别度和一致性成为很大的问题,还会造成命名污染。这时使用BEM命名方法就可以很好的解决这个问题。

当然使用BEM还有很多其他的好处,例如每个块之间都是独立的,因此不会遇到层叠带来的问题。且这些块可以多次重用,可以减少必须维护的css代码量等。

BEM命名规则

  • 块名称为其元素和修饰符定义了命名空间。
  • 块名称与元素名称之间用双连字符--分隔。
  • 块名称与修饰符或元素与修饰符之间用双下划线__分隔。
  • 命名一般使用小写字母。
  • 单词之间可以使用-分隔。

命名约定的模式有如下几种:

.block{}
.block__element{}
.block--modifier{}
.block__element{}--modifier{}

其中block代表了更高级别的抽象或组件,block__element代表.block的后代,用于形成一个完整的block的整体,block--modifier代表block的不同状态或不同版本。使用两个连字符和下划线时为了让自己定义的块可以用单个连字符来界定。

Block

Block是一个功能独立的页面组件,可重复使用,也支持嵌套。我们平时浏览的网页,都是由”块“构成的。

每个块的块名必须是唯一的,用于明确指出它所描述的是哪个块。例如某个块的名字为head,那我们就能够根据这个块名推测出这应该是一个头部块,位于网页的头部。

在使用块时,块不应影响其环境,也就是不应设置块的外部几何形状或位置。需要注意的是块应该是独立的,当在页面中添加,删除,或者是移动某个块时,不需要对块进行修改。

示例:

下面一共有3个块,分别是top、search-form、bottom,块之间可以嵌套:

<div class="top">
    <form class="search-form">搜索</form>
</div>
<div class="bottom">底部</div>

我们在实际应用中,需要保证每个块都是独立的。举个例子说明,例如我们将top块中的search-form块移动到bottom块中,想要保证整体不会乱,就需要search-form块的CSS必须是独立的。如果我们在写CSS时,直接将search-form块嵌套在top块下是不好的:

/*错误写法*/
.top .search-form{...}  /*表示只有在top块内的search-form块才会应用此CSS样式*/
.bottom{}

这样的话,后续如果我们要在HTML代码中移动search-form块(或重用)到bottom块中时,页面样式肯定是会乱的,这样又增加了我们的工作量。我们一开始就应该考虑好后续可能会出现的情况,所以不会这样写CSS。

正确写法如下所示:

/*正确写法*/
.top{...}
.bottom{}
.search-form{...}

Element

元素是块的组成部分,是依赖上下文的。元素的名称用于描述它是什么,而不是它的状态。元素在所属的块中指定位置时,才能表现出应有的功能。

元素之间可以彼此嵌套,一个元素总是一个模块的一部分,而不是另一个元素的一部分,这意味着元素的名称不能被定义为 block__elem1__elem2 这样的层次结构。

示例:
<div class="top">
    <!--top块中的search-for块-->
    <form class="search-form">
        <!--在search-form块中的input元素-->
        <input class="search-form__input"> 
        <!--在search-form块中的button元素-->
        <button class="search-form__button">搜索按钮</button>
    </form>
</div>

此时我们编写CSS代码时,可以将search-form__inputsearch-form__button放在search-form块中。

.search-form .search-form__input{...}
.search-form .search-form__button{...}

Modifier

修饰符可以与块、元素一起工作。我们经常需要在已经定义好的块或者元素上,做一些小调整来满足特定的小功能。通常是外观或行为有些许改变,这时可以使用修饰符来处理。注意,修饰符不能单独使用,而且必须绑定在对应的块或元素上,不能混搭。

为什么修饰符一定要包含块名或元素名呢?

  • 因为每一个块都会建立独立的命名空间,可以有效的减少命名冲突。
  • 可以显示说明该修饰符所对应的块或元素,防止混淆。
  • 不包含块名或元素名,使用修饰符就必须使用组合选择器,这会增加样式权重使其难以覆盖。
示例:

例如当我们添加多个按钮时,一个红色按钮、一个绿色按钮、一个蓝色按钮、一个黄色按钮:

<div class="banner__btn">
    <button class=".button .banner__btn--red"></button>  
    <button class=".button .banner__btn--green"></button>
    <button class=".button .banner__btn--blue"></button>   
    <button class=".button .banner__btn--yellow"></button>       
</div>

CSS代码如下所示:

.banner__btn--red { background-color: red; }
.banner__btn--green { background-color: green; }
.banner__btn--blue { background-color: blue; }
.banner__btn--yellow { background-color: yellow; }

什么时候使用BEM

当我们使用BEM方法命名时,我们要知道哪些东西是应该使用BEM格式的。因为并不是每个地方都应该使用BEM命名方式,只有当需要明确关联性的模块关系时,才需要使用 BEM 格式。

例如仅仅是单独的公共样式,像定义字体大小:

.f16{
    font-size:16px;
}
.f18{
    font-size:18px;
}

使用混合拆分样式

在BEM中,位置和布局样式通过父级块来进行设置。这就需要通过混合组合块与元素,组合多个实体(块、元素、修饰符都被称作 BEM实体)的表现与样式,同时不耦合代码。

示例:
<!-- top 块 -->
<div class="top">
    <!-- search-form块混合top块的search-form元素 -->
    <form class="search-form top__search-form">搜索</form>
</div>

这样就通过混合的方式把位置样式从块中剥离了,可以在.top__search-form中设置表单的位置或浮动等样式,保持了 search-form块的样式独立,对其完整样式代码进行了解耦。在传统的命名方式中,我们经常通过嵌套的方式,如.top .search-form来对局部样式进行调整。但是这样做会改变选择器的权重。在 BEM 的思想中,保持选择器扁平和低权重是一个准则。

因此在使用 BEM 命名时需要格外注意遵循它的工作方式:

  • 不在块里设置位置、布局相关的样式,只设置基本样式。
  • 通过混合的方式,在作为父级块的元素时设置布局样式。
  • 适时拆分元素为独立的块,解耦样式并形成新的命名空间。

适时使用嵌套选择器

那是不是说,我们使用BEM命名方式时,不能嵌套选择器呢?也不完全时这样。BEM只是建议尽量保持嵌套层级越低越好,因为CSS 的权重问题很多人处理不好,最终很可能导致不停的嵌套或为了增加权重而进行的无意义嵌套和 !important,这无疑增加了代码的耦合。

可能你会想问,那到底什么时候用嵌套,当你需要通过块状态对内部元素进行调整时,就非常适合使用嵌套啦。

示例:
.search-form .search-form__input{}
.nav_content_dark .nav_item{}

开放封闭原则

开放封闭原则是所有面向对象原则的核心,是说软件实体应该是可扩展,而不可修改的。即对扩展是开放的,而对修改是封闭的。如果将这个原则应用到BEM的使用上,就是说我们应该使用modifier去拓展block或element的样式,而不应该去修改block或element的基础样式。

示例:

例如有两个按钮,如下所示:

<div>
    <button class="btn">正确</button>
    <button class="btn">错误</button>
</div>

.btn{
    font-size:16px;
    color:white;
    border:none;
    padding:10px 8px;
    display: inline-block;
    background-color:green;
}

如果要将名为错误的按钮的背景颜色改为红色,我们需要给.btn加一个modifier,而不是直接去修改.btn的样式:

<div>
    <button class="btn">正确</button>
    <button class="btn btn--error">错误</button>
</div>

.btn{
    font-size:16px;
    color:white;
    border:none;
    padding:10px 8px;
    display: inline-block;
    background-color:green;
}
.btn--error{
    background-color:red;
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,816评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,729评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,300评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,780评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,890评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,084评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,151评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,912评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,355评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,666评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,809评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,504评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,150评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,121评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,628评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,724评论 2 351

推荐阅读更多精彩内容

  • BEM 基于组件方式的web开发方法,基本思想是将用户界面分成独立的模块。 Block(块) 通常指模块,组件 B...
    xwwawj阅读 934评论 0 2
  • BEM 基于组件方式的web开发方法,基本思想是将用户界面分成独立的模块。 Block(块) (通常指模块,组件)...
    mocobk阅读 3,868评论 0 1
  • BEM 是对 CSS 命名的一种规范,推崇将 WEB 页面模块化,从而提高代码的重用度,减少后期维护的成本。原文 ...
    Jadyn阅读 5,866评论 0 12
  • 转自BEM —— 源自Yandex的CSS 命名方法论 BEM的意思就是块(block)、元素(element)、...
    while1love阅读 610评论 0 8
  • 转自BEM —— 源自Yandex的CSS 命名方法论 BEM的意思就是块(block)、元素(element)、...
    wdapp阅读 124评论 0 0