一篇通俗易懂的CSS層疊順序與層疊上下文研究

来源:http://www.cnblogs.com/pssp/archive/2016/10/12/5948356.html
-Advertisement-
Play Games

網上有很多這方面的教程,但不是苦澀難懂就是從哪copy過來的,反正很長一段時間我是沒看懂,時間長了也沒打算去研究了,主要原因是,基本上很少會遇到那些問題(所以說啊,要是沒有研究精神的才懶得管它)。但自從開始研究CSS以後就一發不可收拾,所以打算把CSS一系列的東西都給研究一遍,當然能研究懂自然是好的 ...


網上有很多這方面的教程,但不是苦澀難懂就是從哪copy過來的,反正很長一段時間我是沒看懂,時間長了也沒打算去研究了,主要原因是,基本上很少會遇到那些問題(所以說啊,要是沒有研究精神的才懶得管它)。但自從開始研究CSS以後就一發不可收拾,所以打算把CSS一系列的東西都給研究一遍,當然能研究懂自然是好的,不能就當自娛自樂了。話說這個層疊順序和堆棧上下文沒啥用對吧,你看我就是不學它,平時也沒出什麼問題。 …… 想讓它出問題也很簡單,不過既然你都說沒用了,也就不打算告訴你了。

本篇屬於短話長說型,前半部分比較簡單,但不看會影響後面你是否有勇氣看下去,建議全看,但如只是想瞭解一下,可以只看總結部分,但都看你自己。

層疊順序和層疊上下文是兩個概念,但它們又有著密不可分的關係,層疊順序很簡單^_^,認真思考即可,而堆棧上下文更是不值一提,我只需要遷根紅線你就懂了,所有說,都太簡單。第一段說難懂,主要是有了這篇文章,才化解位移。廢話少說,看劍。

在考慮到兩個元素可能重合的情況下,W3C提出了層疊這個概念,層疊是指如何去層疊另一個元素,比如兩個元素重合的時候應該讓誰在前面,誰在後面。那它們的規則又是什麼?先來試水。

層疊順序篇

當兩個元素都是塊級元素時,預設情況越後面的元素層級越高
<style>
    div{
        width:100px;
        height:100px;
    }
    .item1{
        color:red;
        background-color:pink;
    }
    .item2{
        margin-top:-100px;
        background-color:orange;
    }
</style>
<div class="item1">item11111</div>
<div class="item2">item22222</div>

可以看到item2item1給蓋住了,說明越後面的元素層級越高,另外item1的文字還是隱隱約約的可以看到,這說明瞭背景的層級比文字小。

另外我們可以這樣

<style>
    div{
        width:100px;
        height:100px;
    }
    .item1{
        color:red;
    }
    .item2{
        margin-top:-100px;
    }
</style>
<div class="item1">item11111</div>
<div class="item2"></div>

可以看到好像沒有item2這個元素一樣,主要原因是,在沒有設置背景的情況下,元素的背景是透明的(transparent),並且允許後面的元素透上來。

在這種條件下,你會發現一個很搞笑的事,如下:

<style>
    div{
        width:100px;
        height:100px;
    }
    .item1{
        color:red;
    }
    .item2{
        background-color:orange;
        margin-top:-100px;
    }
</style>
<div class="item1">我是item2</div>
<div class="item2"></div>

如果不看代碼,就好像這段代碼真的是item2的。

當兩個元素為行內塊時
<style>
    div{
        display:inline-block;
        width:100px;
        height:100px;
    }
    .item1{
        background-color:pink;
    }
    .item2{
        background-color:orange;
        margin-left:-108px;
    }
</style>
<div class="item1">item1111</div>
<div class="item2">item2</div>

也是後一個元素的層級比前一個元素的層級高,不過和兩個塊級元素不同的是行內塊元素的背景層級比文字高。

當兩個元素為行內元素時
<style>
    div{
        display:inline;
        width:100px;
        height:100px;
    }
    .item1{
        background-color:pink;
    }
    .item2{
        margin-left:-48px;
        background-color:orange;
    }
</style>
<div class="item1">item1</div>
<div class="item2">item2</div>

和行內塊的行為一樣,背景層級比文字高,並且也是後一個元素比前一個元素層級高。

小總結

  • 當兩個元素為正常流時,預設情況下後一個元素比前一個元素層級高,並且允許後面的元素透上來。
  • 如果兩個元素是塊級元素,文字比背景層級高(因此不管是否設置背景文字始終會透上來)。
  • 如果是行內或行內塊,背景比文字層級高(因此只要設置背景,後一個元素將透不上來)。
當一個元素為塊級元素,另一個為行內塊時
<style>
    div{
        width:100px;
        height:100px;
    }
    .item1{
        display:inline-block;
        background-color:pink;
    }
    .item2{
        margin-top:-100px;
        background-color:orange;
    }
</style>
<div class="item1">item1</div>
<div class="item2">item2</div>

行內塊比塊級元素層級高,依然是文字比背景層級高。

如果另外一個元素是行內元素和這個效果也是一樣的,代碼如下:

<style>
    div{
        width:100px;
        height:100px;
    }
    .item1{
        display:inline;
        background-color:pink;
    }
    .item2{
        margin-top:-30px;
        background-color:orange;
    }
</style>
<div class="item1">item1</div>
<div class="item2">item2</div>

當一個元素是行內塊另一個是行內元素時
<style>
    div{
        width:100px;
        height:100px;
    }
    .item1{
        display:inline-block;
        background-color:pink;
    }
    .item2{
        display:inline;
        margin-left:-100px;
        background-color:orange;
    }
</style>
<div class="item1">item1</div>
<div class="item2">item2</div>

行內元素層級比行內塊元素高,背景比文字層級高。

浮動系列
  • 浮動和浮動,後一個比前一個層級高。
  • 浮動和塊元素,浮動層級高。
  • 浮動和行內塊,行內塊層級高。
  • 浮動和行內,行內層級高。

效果如下:

定位系列
  • 絕對定位和絕對定位,後一個比前一個層級高。
  • 絕對定位和相對定位,後一個比前一個層級高。
  • 絕對定位和固定定位,後一個比前一個層級高。
  • 固定定位和相對定位,後一個比前一個層級高。
  • 絕對定位和塊元素,絕對定位層級高。
  • 決定定位和行內塊,絕對定位層級高。
  • 絕對定位和行內元素,絕對定位層級高。
  • 絕對定位和浮動,絕對定位層級高。
  • 其他定位和絕對定位效果一樣。

層疊順序總結

  • 當兩個元素類型一樣時,預設情況下後一個元素層級比前一個元素層級高。
  • 在沒有設置背景的情況下,元素的背景是透明的,並且允許後面的元素透上來。
  • 塊元素和其他任意除定位元素以外,文字層級比背景層級高。
  • 浮動和塊元素,浮動層級高。
  • 浮動和行內塊,行內塊層級高。
  • 浮動和行內,行內層級高。
  • 定位和定位,後一個元素層級高。
  • 定位比所有元素層級高。

它們的前後順序:小於0的z-index < 塊 < 浮動 < 行內塊 < 行內 < 定位 < 大於0的z-index

層疊上下文

如果你認真看完上一節,會不會奇怪一個問題,那就是在無特殊情況下為什麼定位元素總是比普通元素層級高,另外一點就是,大部分情況下為什麼總是後一個元素比前一個元素層級高,而罪魁禍首就是層疊上下文。

在HTML中有一個三維概念,也就是我們面向電腦屏幕的這一端為Z軸。

而凡是擁有層疊上下文的元素,將離用戶最近,也就是越靠在Z軸前面。預設情況下只有根元素HTML會產生一個層疊上下文,並且元素一旦使用了一些屬性也將會產生一個層疊上下文,如我們常用的定位屬性。如兩個層疊上下文相遇時,總是後一個層疊前一個,除非使用z-index來改變。如下:

<style>
    .box1{
        width:100px;
        height:100px;
        background-color:red;
    }
    .box1 .item{
        position:relative;
        height:100px;
    }
    .box2{
        margin-top:-50px;
        width:100px;
        height:100px;
        background-color:orange;
    }
</style>
<div class="box1">box1
    <div class="item"></div>
</div>
<div class="box2">box2</div>

雖然item產生了一個層疊上下文,但並不影響它父元素。它的父元素依然被box2層疊了。另外上面還說只要是產生層疊上下文的元素總是比其他元素層高,如下:

只需要給item加上一個背景即可,上一個案例只所以沒看到item元素是因為背景預設是透明的,並且允許後面的元素透上來。

除了定位元素可以創建層疊上下文以外,還有如下幾個屬性也可以做到。以下來自MDN

  • 根元素 (HTML),
  • z-index 值不為 "auto"的 絕對/相對定位,
  • 一個 z-index 值不為 "auto"的 flex 項目 (flex item),即:父元素 display: flex|inline-flex,
  • opacity 屬性值小於 1 的元素
  • transform 屬性值不為 "none"的元素,
  • mix-blend-mode 屬性值不為 "normal"的元素,
  • filter值不為“none”的元素,
  • perspective值不為“none”的元素,
  • isolation 屬性被設置為 "isolate"的元素,
  • position: fixed
  • 在 will-change 中指定了任意 CSS 屬性,即便你沒有直接指定這些屬性的值
  • -webkit-overflow-scrolling 屬性被設置 "touch"的元素

這裡再拿opacity試水。

代碼如下:

<style>
    .box1{
        opacity:.9;
        width:100px;
        height:100px;
        background-color:red;
    }
    .box2{
        margin-top:-50px;
        width:100px;
        height:100px;
        background-color:orange;
    }
</style>
<div class="box1">opacity</div>
<div class="box2">box2</div>

原本應該是box2層疊box1的,但因為box1創建了一個層疊上下文,所以把box2層疊了。

知道層疊上下文有什麼用?

最大的好處就是當你想要改變元素的層級又不想用定位時,你還可以用很多其他的方法。如下:

原本浮動比塊元素層級高,但當使用了opacity之後,塊元素層級比浮動層級高,代碼如下:

<style>
    .box1{
        float:left;
        width:100px;
        height:100px;
        margin-right:-20px;
        background-color:red;
    }
    .box2{
        overflow:hidden;
        width:100px;
        height:100px;
        background-color:orange;
        opacity:.99;
    }
</style>
<div class="box1">float</div>
<div class="box2">opacity</div>

總結

  • 創建了層疊上下文的元素比其他元素層級高。
  • 兩個層疊上下文相遇時,後一個層級高。如果想改變層級可以使用z-index

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 作為一個Java 初學者,對Java的理解可能有些片面,甚至有些錯誤的理解,對於觀看此處的您,希望您選擇性觀看!!! 天知道我為什麼選擇學習編程,我不愛編程,但是我既然學習了,還是會努力學習的,在此分享一些學習經驗。 訪問修飾符: 1.常用訪問修飾符: public 共有的 private 私有的 ...
  • 1.新建項目 > 選擇 web 應用程式 選擇 webApi 2. 創建一個httpmodeule類 放到app_data文件夾下 public class MyHttpModule : IHttpModule { #region IHttpModule 成員 public void Dispose ...
  • 從2005年畢業至今,畢業已經11年,正好現處於離職狀態,所以理一理些年的一些技術,一些想法,在此也分享給大家,不對的地方也請大家儘管打臉,贊同的話呢,那就點個贊吧。:) 先說一下風格,一不照本宣科,二不複製粘貼,純手打,三必須是結合實際開發的經驗或理解來整理。 ...
  • HP-Socket 是一套通用的高性能 TCP/UDP/HTTP 通信框架,包含服務端組件、客戶端組件和 Agent 組件,廣泛適用於各種不同應用場景的 TCP/UDP/HTTP 通信系統,提供 C/C++、C#、Delphi、E(易語言)、Java、Python 等編程語言介面。HP-Socket... ...
  • 一、簡單工廠模式 簡單工廠模式,提供了一種創建對象的便捷方法。創建者通過不同的類型參數,創建相對應的對象。 實現代碼如下: 優點: 在創建固定類型的對象時,比較簡單好用。 缺點: 1. 在添加對象類型時,需要更改Creator類裡面的函數,違反了軟體設計中的開閉原則。 2. Creator類依賴於具 ...
  • Atitit. Atiposter 發帖機 新特性 poster new feature v7 q39 V8 重構iocutilV4,use def iocFact...jettyUtil V8 gui 獨立的gui..使用jetty.. V1 初步實現sina csdn cnblogs V2 實現 ...
  • 在DOM事件深入淺出(一)中,我主要給大家講解了不同DOM級別下的事件處理程式,同時介紹了事件冒泡和捕獲的觸發原理和方法。本文將繼續介紹DOM事件中的知識點,主要側重於DOM事件中Event對象的屬性和方法。 那麼什麼是DOM事件中Event對象呢?事件對象(event object)指的是與特定事 ...
  • 本人是軟體開發的初學者,總結了一點點日常所學,記錄在此,主要目的是鼓勵自己堅持學習,怕有一天忘記了,還能複習曾經學過的知識點。 如有大神路過為我指點迷津,糾正改錯更是感激不盡,但請不要噴我這個菜鳥!謝謝 jQuery Mobile的基本使用方法和簡單的知識點 安裝: <link rel="style ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...