頁每個Element元素都有style和classname屬性,可以通過CSS直接設置,也允許腳本指定文檔元素的CSS樣式,或修改應用到元素上的css類名,設置這些css相關的屬性會改變文檔元素的呈現。
class 屬性是在標簽上引用樣式表的方法之一,它的值是一個樣式表的選擇符,如果改變了 class 屬性的值,標簽所引用的樣式表也就更換了。
更改一個標簽的 class 屬性的代碼是:
document.getElementById( id ).className=字符串;
document.getElementById( id ) 用于獲取標簽對應的 DOM 對象,你也可以用其它方法獲取。className 是 DOM 對象的一個屬性,它對應于標簽的 class 屬性。字符串 是 class 屬性的新值,它應該是一個已定義的CSS選擇符。
利用這種辦法可以把標簽的CSS樣式表替換成另外一個,也可以讓一個沒有應用CSS樣式的標簽應用指定的樣式。
舉例1,代碼如下:
<style type="text/css">
.txt {
font-size: 30px; font-weight: bold; color: red; }
</style>
<div id="tt">歡迎光臨!</div>
<p><button on click="setClass()">更改樣式</button></p>
<script type="text/javas cript">
function setClass()
{
document.getElementById( "tt" ).className="txt";
}
</script>
舉例2,代碼如下:
<script type="text/javascript">
<!--
var h=document.getElementById("tab").getElementsByTagName("h3");
var d=document.getElementById("tab").getElementsByTagName("div");
function go_to(ao){
for(var i=0;i<h.length;i++){
if(ao-1==i){
h[i].className+=" up";
d[i].className+=" block";
}
else {
h[i].className=" ";
d[i].className=" ";
}
}
}
//-->
</script>
元素的style屬性可以直接設置元素的外觀屬性。Style 對象的屬性和 CSS 屬性是一一對應的,可以直接設置,也可以通過JS動態更改。
更改一個標簽的 style 屬性的代碼是:
document.getElementById( id ).style.屬性名=值;
document.getElementById( id ) 用于獲取標簽對應的 DOM 對象,你也可以用其它方法獲取。style 是 DOM 對象的一個屬性,它本身也是一個對象。屬性名 是 Style 對象的屬性名,它和某個CSS屬性是相對應的。
說明:這種方法修改的單一的一個CSS屬性,它不影響標簽上其它CSS屬性值。
舉例,代碼如下:
div id="t2">歡迎光臨!</div>
<p><button on click="setSize()">大小</button>
<button on click="setColor()">顏色</button>
<button on click="setbgColor()">背景</button>
<button on click="setBd()">邊框</button>
</p>
<script type="text/java script">
function setSize()
{
document.getElementById( "t2" ).style.fontSize="30px";
}
function setColor()
{
document.getElementById( "t2" ).style.color="red";
}
function setbgColor()
{
document.getElementById( "t2" ).style.backgroundColor="blue";
}
function setBd()
{
document.getElementById( "t2" ).style.border="3px solid #FA8072";
}
</script>
function clk(event) {
var parent=document.getElementById("parent");
//改變className
var child0=document.createElement("div");
child0.innerHTML="child0";
child0.className="newDiv";
parent.appendChild(child0);
//改變cssText
var child1=document.createElement("div");
child1.innerHTML="child1";
child1.style.cssText="color:red;";
parent.appendChild(child1);
//改變直接樣式
var child2=document.createElement("div");
child2.innerHTML="child2";
child2.style.fontSize="28px";
parent.appendChild(child2);
}
盒子標簽和屬性對照 | |
---|---|
CSS語法(不區分大小寫) | JavaScript語法(區分大小寫) |
border | border |
border-bottom | borderBottom |
border-bottom-color | borderBottomColor |
border-bottom-style | borderBottomStyle |
border-bottom-width | borderBottomWidth |
border-color | borderColor |
border-left | borderLeft |
border-left-color | borderLeftColor |
border-left-style | borderLeftStyle |
border-left-width | borderLeftWidth |
border-right | borderRight |
border-right-color | borderRightColor |
border-right-style | borderRightStyle |
border-right-width | borderRightWidth |
border-style | borderStyle |
border-top | borderTop |
border-top-color | borderTopColor |
border-top-style | borderTopStyle |
border-top-width | borderTopWidth |
border-width | borderWidth |
clear | clear |
float | floatStyle |
margin | margin |
margin-bottom | marginBottom |
margin-left | marginLeft |
margin-right | marginRight |
margin-top | marginTop |
padding | padding |
padding-bottom | paddingBottom |
padding-left | paddingLeft |
padding-right | paddingRight |
padding-top | paddingTop |
顏色和背景標簽和屬性對照 | |
CSS 語法(不區分大小寫) | JavaScript 語法(區分大小寫) |
background | background |
background-attachment | backgroundAttachment |
background-color | backgroundColor |
background-image | backgroundImage |
background-position | backgroundPosition |
background-repeat | backgroundRepeat |
color | color |
樣式標簽和屬性對照 | |
CSS語法(不區分大小寫) | JavaScript 語法(區分大小寫) |
display | display |
list-style-type | listStyleType |
list-style-image | listStyleImage |
list-style-position | listStylePosition |
list-style | listStyle |
white-space | whiteSpace |
文字樣式標簽和屬性對照 | |
CSS 語法(不區分大小寫) | JavaScript 語法(區分大小寫) |
font | font |
font-family | fontFamily |
font-size | fontSize |
font-style | fontStyle |
font-variant | fontVariant |
font-weight | fontWeight |
文本標簽和屬性對照 | |
CSS 語法(不區分大小寫) | JavaScript 語法(區分大小寫) |
letter-spacing | letterSpacing |
line-break | lineBreak |
line-height | lineHeight |
text-align | textAlign |
text-decoration | textDecoration |
text-indent | textIndent |
text-justify | textJustify |
text-transform | textTransform |
vertical-align | verticalAlign |
從上表可以看出,兩者的寫法區別在于CSS是不區分大小寫的,所以兩個英文字中間用了一個橫杠"-",用JS區分大小寫,所以不用橫杠"-",而用英文的小駝峰式命名法(lower camel case,第一個單字以小寫字母開始,第二個單字的首字母大寫)。
-End-
ss是一種用來為Html文檔添加樣式(字體、間距、位置、顏色、角度等顯示效果)的計算機語言。學習網站W3school。
css的引用樣式:
一:style標簽(內聯樣式)
通過在head標簽中間新建一個style標簽,該標簽內部存放的就是網頁文件中的css代碼。
二:外部引用css文件(外聯樣式)
新建一個css文件
在head標簽中間新建一個link標簽,通過href屬性設置外部的css文件地址。rel=“stylesheet”表示我們引用的一個樣式表(css文件)。
三:標簽內部style屬性(行內樣式)
在開始標簽的內部可以設置一個叫做style的屬性,屬性的雙引號存放該元素代碼的css樣式(不推薦使用)。
一般用的就是通過link標簽來引入外部css文件來修改樣式,一般修改樣式有字體,顏色,大小,文本居中,間距等。
們經常寫 HTML 、 CSS 和 JavaScript ,寫好這些之后,我們就會在瀏覽器中看到頁面,那瀏覽器究竟在這背后做了一些什么事情呢?本篇文章將揭曉答案!
了解瀏覽器的渲染原理是我們在通往更深層次的前端開發中不可缺少的,它可以讓我們從更深層次、角度去考慮性能優化等~
下面進入正文~
瀏覽器會分配一個線程“自上而下,從左到右”依次解析和渲染代碼,那么進程和線程是什么,它們之間有著怎樣的關系呢?
一個進程就是一個程序運行的實例。當啟動一個程序的時候,操作系統會為該程序創建一塊內存,用來存放代碼,運行中的數據和一個執行任務的主線程,這樣的一個運行環境就叫進程
線程不能單獨存在,它是由進程來啟動和管理的。線程依附于進程,進程中使用多線程并行處理能提升運算效率
1、進程中的任意一線程執行出錯,都會導致整個進程的崩潰
2、線程之間可以共享數據
3、當一個進程關閉后,操作系統會回收進程所占用的內存
4、進程之間的內容相互隔離
了解瀏覽器的渲染原理,我們就要從理解 HTML 、 CSS 和 JavaScrip 開始,我們先來看一張圖
HTML (超文本標記語言),顧名思義,由標記(標簽)和文本組成,每個標簽都有自己的語意,瀏覽器會根據標簽和文本展示對應的內容。
CSS (層疊樣式表),由選擇器和屬性組成,它可以改變 HTML 的樣式,比如上圖中,我們改變了 span 的顏色由藍色為綠色。
JavaScript ,我們可以通過 JS 完成很多事情,例如上圖中修改樣式。
下面開始分析渲染的原理
渲染模塊由于渲染的機制的復雜,被劃分為了很多子階段,輸入的 HTML 經過這些子階段,最后會輸出為像素。這樣的處理流程就叫做 渲染流水線
按照渲染的時間順序,流水線可分為幾個子階段:構建 DOM 樹、樣式計算、布局階段、分層、繪制、分塊、光柵化和合成
由于瀏覽器無法直接理解和使用 HTML ,所以需要將 HTML 轉換為瀏覽器能夠理解的結構( DOM 樹)
我們來分析一下下面這段代碼會構建出一棵什么樣的 DOM 樹
我們先將上面的代碼運行,然后在瀏覽器控制臺輸入 document ,看看會有什么效果
我們一層級一層級的打開就會看到如上圖的效果,我們可以根據這每一層級展開的效果,繪制出一棵 DOM 樹結構,如下:
接下來,我們試一下利用 JS 修改一下內容,看有什么改變:
我們可以看到“瀏覽器”的文字變成了“chrome”
再來看一下 DOM 樹是否有改變
我們看到在“瀏覽器”的位置換成了“chrome”,那么如何讓 DOM 節點擁有樣式?
樣式計算,顧名思義,就是 計算出 DOM 節點中每個元素的具體樣式 ,這個階段會分為三部分:
瀏覽器會新開辟一個線程,去服務器獲取對應的資源文件(不阻礙主線程的渲染)
從上到下解析,解析完繼續解析 DOM 結構。在真實項目中,如果 css 代碼不是很多,或是移動端項目,我們應該使用內嵌式,以此來減少 http 資源的請求,提高頁面渲染速度
它是同步的,不會開辟新線程去加載資源文件,而是讓主線程去獲取,這阻礙 DOM 結構的繼續渲染;只有把外部樣式導入進來,并且解析后,才會繼續渲染 DOM 結構
瀏覽器就像不能理解 HTML 一樣,不理解 CSS ,所以當渲染引擎接收到 CSS 文件時,會執行轉換操作,將 CSS 文本轉換為瀏覽器可以理解的 styleSheets 結構。
在 HTML 中,在瀏覽器中輸入 document 可以查看 html 的結構。在 css 中,可以輸入 document.styleSheets 看到 css 的結構
現在的結構是空的,我們來加一些樣式,看看效果
屬性值標準化就是將所有值轉換為渲染引擎容易理解的、標準化的計算值。我們大致看一下效果:
body {
font-size: 2em;
color: black;
font-weight: bold;
...
}
復制代碼
body {
font-size: 16px;
color: rgb(0, 0, 0);
font-weight: 700;
...
}
復制代碼
樣式計算有兩個CSS的規則:繼承規則和層疊規則
CSS 繼承就是每個 DOM 節點都包含有父節點的樣式。我們來看一下下面這段代碼中如何應用到 DOM 節點上
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Document</title>
<style>
h1 {
color: red;
}
div {
color: blue;
}
span {
font-size: 16px;
}
</style>
</head>
<body>
<h1>掘金</h1>
<div>
<span>瀏覽器</span>
<span>渲染原理</span>
構建DOM樹
</div>
</body>
</html>
復制代碼
子節點會擁有父節點的樣式,由此我們可以畫出這樣一張圖
我們還可以打開控制臺,看一下選中 span 標簽,都會看到哪些內容
通過上圖,我們可看到一個元素的樣式、繼承過程等, userAgent 樣式是瀏覽器默認的內置樣式,如果我們不提供任何樣式,就會使用此樣式。
層疊在 CSS 處于核心地位,它是 CSS 的一個基本特征,它定義了如何合并來自多個源的屬性值的算法。
樣式計算階段最終輸出的內容是每個 DOM 節點的樣式,并且保存在了 ComputedStyle 中。我們可以通過控制臺看到某個 DOM 元素最終的計算樣式
現在我們不知道 DOM 元素的幾何位置信息,所以現在我們需要計算出 DOM 樹中可見元素的幾何位置,這個計算過程就叫做布局。布局階段有兩個過程:
創建布局樹的意思就是創建一棵只包含可見元素的樹。我們來看下面一段代碼創建布局樹的過程
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Document</title>
<style>
h1 {
color: red;
}
div {
color: blue;
}
div span {
font-size: 16px;
}
div span:last-child {
display: none;
}
</style>
</head>
<body>
<h1>掘金</h1>
<div>
<span>瀏覽器</span>
<span>渲染原理</span>
構建DOM樹
</div>
</body>
</html>
復制代碼
構建布局樹的過程中, DOM 樹中所有不可見的節點都不會包含在這棵樹中。瀏覽器會遍歷 DOM 樹中所有能看見的節點,然后把這些節點加入到布局中;不可見的節點就會被忽略, head 標簽下面的內容、 div 下最后一個 span 節點都不會在布局樹中,我們看一下這個過程圖感受一下~
布局計算就是計算布局樹節點的坐標位置。這個計算過程極為復雜。
渲染引擎會為特定的節點生成專用的圖層,并生成一棵對應的圖層樹。這樣做是因為頁面中可能含有很多復雜的效果,我們可以打開控制臺看一下頁面的分層情況
我們可以看到,渲染引擎給頁面分了很多圖層,這些圖層會按照一定順序疊加在一起,形成最終的頁面
那么圖層的來源有哪些?
1、擁有層疊上下文屬性的元素會被提升為單獨的一層
層疊上下文可以使能夠使 HTML 元素具有三維的概念,這些 HTML 元素按照自身屬性的優先級分布在垂直于這個二維平面的 z 軸上。哪些元素具有層疊上下文屬性?
2、需要剪裁的地方會被創建為圖層
當我們創建一個有寬度和高度的 div 時,里面的文字內容可能會超出這個區域,這時候渲染引擎會把裁剪文字內容的一部分用于顯示在 div 區域,例如
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
<style>
div {
width: 100px;
height: 100px;
background: yellow;
overflow: auto;
}
</style>
</head>
<body>
<div>
我們經常寫`HTML`、`CSS`和`JavaScript`,寫好這些之后,我們就會在瀏覽器中看到頁面,那瀏覽器究竟在這背后做了一些什么事情呢?本篇文章將揭曉答案!
了解瀏覽器的渲染原理是我們在通往更深層次的前端開發中不可缺少的,它可以讓我們從更深層次、角度去考慮性能優化等~
</div>
</body>
</html>
復制代碼
運行結果
我們再打開控制臺的 layers 看一下效果
可以看到渲染引擎為文字部分單獨創建了一個圖層。
在布局樹中的節點如果擁有對應的圖層,這個節點就是一個圖層,如果沒有,這個節點就屬于父節點的圖層,如下圖:
創建好圖層樹后,渲染引擎會繪制圖層樹中的每個圖層。渲染引擎會將圖層繪制分解為很多小的繪制指令,然后將這些指令按照順序組成待繪制列表,我們可以打開控制臺的 layers ,選擇 document 層,看一下效果
柵格化就是將圖塊轉換位位圖,圖塊是柵格化執行的最小單位。渲染進程維護了一個柵格化的線程池,所有圖塊的柵格化都是在線程池內執行的。
圖層繪制列表準備好之后,主線程會把這個繪制列表提交給合成線程,繪制操作由渲染引擎中的合成線程來完成。
合成線程將圖層劃分為圖塊,然后合成線程會按照視口(可見區域)附近的圖塊優先生成位圖。
所有的圖塊都被光柵化后,合成線程會生成一個繪制圖塊的命令( DrawQuad ),然后將該命令提交給瀏覽器進程。瀏覽器進程里面 viz 組件用來接收 DrawQuad 命令,將其頁面內容繪制到內存中,最后將內存顯示到屏幕。這個時候,我們就看到了頁面
根據上文中描述,我們可以畫出這樣一張圖
我還在網上找到了另外一張圖
這兩張圖都是描述瀏覽器的渲染流程的。
*請認真填寫需求信息,我們會在24小時內與您取得聯系。