1. 結構(Structure):html 語言:XHTML[可擴展超文本標識語言]和XML[可擴展標記語言] 2. 表現(Preasentation):css 3. 行爲(Behavior):JS[DOM+ES]
其中一些小的具體要求:css
1. 易於維護 2. 頁面響應快 3. 可訪問性高 4. 提升設備兼容性 5. 易被解析(搜索引擎)
Ps:html
> - IE:Trident內核(多稱:IE內核) > - Chrome:Webkit內核 ==> Blink內核 > - Firefox:Gecko內核(多稱:IE內核) > - Safari:Webkit內核 > - Opear:Preato====>Webkit內核 ====> Blink內核
當渲染樹中的一部分(或所有)由於元素的規模尺寸,佈局,隱藏等改變而須要從新構建, 這就稱爲迴流(reflow)。== 每一個頁面至少須要一次迴流,就是在頁面第一次加載的時候(第一次佈局也稱做迴流)==。reflow 幾乎是沒法避免的。如今界面上流行的一些效果,好比樹狀目錄的摺疊、展開(實質上是元素的顯示與隱藏:display:none和visibility:hidden)等,都將引發瀏覽器的 reflow。鼠標滑過、點擊……只要這些行爲引發了頁面上某些元素的佔位面積、定位方式、邊距等屬性的變化,都會引發它內部、周圍甚至整個頁面的從新渲染。一般咱們都沒法預估瀏覽器到底會 reflow 哪一部分的代碼,它們都彼此相互影響着。前端
改變某個元素的背景色、文字顏色、邊框顏色等等== (外觀屬性)==不影響它周圍或內部佈局的屬性時,屏幕的一部分要重畫,可是元素的幾何尺寸== (佈局)==沒有變。
在迴流(重排)的時候,瀏覽器會使佈局發生變化。完成迴流將變化後的新頁面繪製到頁面上,則觸發重繪。反之,重繪時不會不必定會使佈局發生變化,這不必定觸發重排(迴流)jquery
有些狀況下,好比修改了元素的樣式,瀏覽器並不會馬上reflow 或 repaint 一次,而是會把這樣的操做積攢一批,而後作一次 reflow,這又叫==異步 reflow 或增量異步 reflow==。可是在有些狀況下,好比==resize== 窗口,改變了頁面默認的字體等。對於這些操做,==瀏覽器會立刻進行 reflow==。程序員
==重排(迴流)一定會引起重繪,但重繪不必定會引起重排(迴流)==web
==重繪重排的代價:耗時,致使瀏覽器卡慢==chrome
優化:express
* CommonJs用在服務器端,AMD和CMD用在瀏覽器環境 * AMD 是 RequireJS 在推廣過程當中對模塊定義的規範化產出。 * MD 是 SeaJS 在推廣過程當中對模塊定義的規範化產出。 * AMD:提早執行(異步加載:依賴先執行)+延遲執行 * CMD:延遲執行(運行到需加載,根據順序執行)
模塊就是實現特定功能的文件,把幾個函數放在一個文件裏就構成了一個模塊。 須要的時候加載這個文件,調用其中的函數便可。
(1)函數寫法編程
function f1(){ //... } function f2(){ //... }
==缺點==:
這樣作會==污染全局變量==,沒法保證不與其餘模塊發生==變量名衝突==,並且==模塊成員之間沒什麼關係==。後端
(2)對象寫法
var module = { star : 0, f1 : function (){ //... }, f2 : function (){ //... } }; module.f1(); module.star = 1;
==優勢==:模塊寫成一個對象,模塊成員都封裝在對象裏,經過調用對象屬性,訪問使用模塊成員。
==缺點==:==暴露了模塊成員==,==外部能夠修改模塊內部狀態==。
(3)當即執行函數
var module = (function(){ var star = 0; var f1 = function (){ console.log('ok'); }; var f2 = function (){ //... }; return { f1:f1, f2:f2 }; })(); module.f1(); //ok console.log(module.star) //undefined
==優勢==:外部沒法訪問內部私有變量
CommonJS是服務器端模塊的規範,由Node推廣使用。
因爲服務端編程的複雜性,若是==沒有模塊很難與操做系統及其餘應用程序互動==。
CommonJS規範:
使用方法以下:
//math.js exports.add = function() { var sum = 0, i = 0, args = arguments, l = args.length; while (i < l) { sum += args[i++]; } return sum; }; //increment.js var add = require('math').add; exports.increment = function(val) { return add(val, 1); }; //index.js var increment = require('increment').increment; var a = increment(1); //2
問題:
require 是同步的。模塊系統須要同步讀取模塊文件內容,並編譯執行以獲得模塊接口。 但在瀏覽器端問題多多。
緣由:
瀏覽器端,加載 JavaScript 最佳、最容易的方式是在 document 中插入script標籤。 但腳本標籤天生異步,傳統 CommonJS 模塊在瀏覽器環境中沒法正常加載。
解決思路
define(function(require, exports, module) { // The module code goes here });
這套模板代碼爲模塊加載器提供了機會,使其能在模塊代碼執行以前,對模塊代碼進行靜態分析,並動態生成==依賴列表==。
//math.js define(function(require, exports, module) { exports.add = function() { var sum = 0, i = 0, args = arguments, l = args.length; while (i < l) { sum += args[i++]; } return sum; }; }); //increment.js define(function(require, exports, module) { var add = require('math').add; exports.increment = function(val) { return add(val, 1); }; }); //index.js define(function(require, exports, module) { var inc = require('increment').increment; inc(1); // 2 });
* AMD是"Asynchronous Module Definition"的縮寫。 * 因爲不是JavaScript原生支持,使用AMD規範進行頁面開發須要用到對應的庫函數RequireJS * AMD 是 RequireJS 在推廣過程當中對模塊定義的規範化的產出 * 採用異步方式加載模塊,模塊的加載不影響它後面語句的運行。全部依賴這個模塊的語句,都定義在一個回調函數中,等到加載完成以後,這個回調函數纔會運行。
==RequireJS主要解決兩個問題:==
==RequireJs採用require()語句加載模塊,不一樣於CommonJS,它要求兩個參數:==
require([module], callback); require([increment'], function (increment) { increment.add(1); });
==define()函數==
==參數說明:==
依賴參數是可選的,若是忽略此參數,它應該默認爲["require", "exports", "module"]。然而,若是工廠方法的長度屬性小於3,加載器會選擇以函數的長度屬性指定的參數個數調用工廠方法。
define("alpha", ["require", "exports", "beta"], function (require, exports, beta) { exports.verb = function() { return beta.verb(); //Or: return require("beta").verb(); } });
==RequireJs使用例子==
main.js //別名配置 requirejs.config({ paths: { jquery: 'jquery.min' //能夠省略.js } }); //引入模塊,用變量$表示jquery模塊 requirejs(['jquery'], function ($) { $('body').css('background-color','red'); });
引入模塊也能夠只寫require()。requirejs經過define()定義模塊,定義的參數上同。在==此模塊內的方法和變量外部是沒法訪問的==,只有經過return返回才行.
//將該模塊命名爲math.js保存。 //math.js define('math',['jquery'], function ($) {//引入jQuery模塊 return { add: function(x,y){ return x + y; } }; });
//main.js引入模塊方法 require(['jquery','math'], function ($,math) { console.log(math.add(10,100));//110 });
* CMD 即Common Module Definition通用模塊定義 * CMD規範是國內發展出來的 * CMD有個瀏覽器的實現SeaJS * SeaJS要解決的問題和requireJS同樣,只在🌟模塊定義方式和模塊加載(能夠說運行、解析時機上有所不一樣。
==CMD規範:==
(1)一個模塊就是一個文件。代碼的書寫格式以下:
define(function(require, exports, module) { // 模塊代碼 });
🌟🌟==AMD是依賴關係前置,在定義模塊的時候就要聲明其依賴的模塊;==
🌟🌟==CMD是按需加載依賴就近,只有在用到某個模塊的時候再去require:==
//CMD(seajs)使用例子 // 定義模塊 myModule.js define(function(require, exports, module) { var $ = require('jquery.js') $('div').addClass('active'); exports.data = 1; }); // 加載模塊 seajs.use(['myModule.js'], function(my){ var star= my.data; console.log(star); //1 }); // CMD define(function(require, exports, module) { var a = require('./a') a.doSomething() // 此處略去 100 行 var b = require('./b') // 依賴能夠就近書寫 b.doSomething() // ... }) // AMD define(['./a', './b'], function(a, b) { // 依賴必須一開始就寫好 a.doSomething() // 此處略去 100 行 b.doSomething() ... })
🌟==以上明顯的不一樣就是依賴加載方式不一樣,具體表如今define的參數構成上,但AMD和CMD都是異步加載==
1. HTML對象獲取問題
==解決辦法:統一使用document.getElementById("idName");==
2. const問題
==解決方法:統一使用var關鍵字來定義常量==
3. event.x與event.y問題
==解決方法:使用mX(mX = event.x ? event.x : event.pageX;)來代替IE下的event.x或者Firefox下的event.pageX==
4. window.location.href問題
==解決方法:使用window.location來代替window.location.href.==
5. frame問題
如下面的frame爲例:
(1)訪問frame對象:
==另外,在IE和Firefox中均可以使用window.document.getElementById("frameId")來訪問這個frame對象.==
(2)切換frame內容:
在 IE和Firefox中均可以使用window.document.getElementById("testFrame").src = "xxx.html"或window.frameName.location = "xxx.html"來切換frame的內容.
若是須要將frame中的參數傳回父窗口(注意不是opener,而是parent frame),能夠在frame中使用parent來訪問父窗口。例如:parent.document.form1.filename.value="Aqing";
6. 模態和非模態窗口問題
==解決方法:直接使用window.open(pageURL,name,parameters)方式打開新窗口。
若是須要將子窗口中的參數傳遞迴父窗口,能夠在子窗口中使用window.opener來訪問父窗口.==
例如:var parWin = window.opener; parWin.document.getElementById("Aqing").value = "Aqing";
7. firefox與IE的父元素(parentElement)的區別
IE:obj.parentElement
firefox:obj.parentNode
==解決方法: 由於firefox與IE都支持DOM,所以使用obj.parentNode是不錯選擇.==
8. document.formName.item(」itemName」) 問題
問題說明:IE下,可使用document.formName.item(」itemName」) 或document.formName.elements ["elementName"];Firefox 下,只能使用document.formName.elements["elementName"]。
==解決方法:統一使用document.formName.elements["elementName"]。==
9. 集合類對象問題
問題說明:IE下,可使用 () 或 [] 獲取集合類對象;Firefox下,只能使用 [ ]獲取集合類對象。
==解決方法:統一使用 [] 獲取集合類對象==。
10. 自定義屬性問題
==解決方法:統一經過getAttribute() 獲取自定義屬性。==
11. input.type屬性問題
問題說明:IE下input.type屬性爲只讀;可是Firefox下input.type屬性爲讀寫。
==解決辦法:不修改input.type屬性。若是必需要修改,能夠先隱藏原來的input,而後在一樣的位置再插入一個新的input元素。==
12. event.srcElement問題
==解決方法:使用srcObj = event.srcElement ?event.srcElement : event.target;==
若是考慮第8條問題,就改用myEvent代替event便可。
13. body載入問題
Ps: 經驗證,IE六、Opera9以及FireFox2中不存在上述問題,單純的JS腳本能夠訪問在腳本以前已經載入的全部對象和元素,即便這個元素尚未載入完成。
14. 事件委託方法
==解決方法:統一使用document.body.onload=new Function(’inject()’); 或者document.body.onload = function(){}==
[注意] Function和function的區別。
15. Table操做問題
問題說明:ie、firefox以及其它瀏覽器對於 table 標籤的操做都各不相同,在ie中不容許對table和tr的innerHTML賦值,使用js增長一個tr時,使用appendChild方法也無論用。
==解決方法://向table追加一個空行:==
var row = otable.insertRow(-1); var cell = document.createElement("td"); cell.innerHTML = ""; cell.className = "XXXX"; row.appendChild(cell);
[注] 建議使用JS框架集來操做table,如JQuery。
16. 對象寬高賦值問題
firefox不支持hand,但ie支持pointer
==解決方法: 統一使用pointer==
17. innerText在IE中能正常工做,但在FireFox中卻不行.需用textContent。
解決方法:
if(navigator.appName.indexOf("Explorer") > -1){ document.getElementById('element').innerText = "my text"; } else{ document.getElementById('element').textContent = "my text"; }
18. CSS透明
filter:progid:DXImageTransform.Microsoft.Alpha(style=0,opacity=60)
==解決方式:
一、使用 background:rgba(0,0,0,0.6) //IE8及如下無效果。
二、使用定位,背景色與子元素處於同級關係。==
19. css中的width和padding
==在IE7和FF中width寬度不包括padding,在Ie6中包括padding.==
20. FF和IEBOX模型解釋不一致致使相差2px
box.style{width:100;border 1px;}
==解決方法:p{margin:30px!important;margin:28px;}==
Ps:這兩個margin的順序必定不能寫反, IE不能識別!important這個屬性,但別的瀏覽器能夠識別。因此在IE下其實解釋成這樣:p{maring:30px;margin:28px}
重複定義的話按照最後一個來執行,因此不能夠只寫margin:XXpx!important;
21. IE5 和IE6的BOX解釋不一致
下p{width:300px;margin:0 10px 0 10px;}
p 的寬度會被解釋爲300px-10px(右填充)-10px(左填充),最終p的寬度爲280px
==修改 p{width:300px!important;width :340px;margin:0 10px 0 10px}
==
22. ul和ol列表縮進問題
list-style:none;margin:0px;padding:0px;
經驗證,
==在IE中僅僅設置margin:0px便可達到最終效果,而在Firefox中必須同時設置margin:0px、 padding:0px以及list-style:none三項才能達到最終效果。==
23. 元素水平居中問題
24. p的垂直居中問題
vertical-align:middle; line-height:200px; //將行距增長到和整個p同樣高
而後插入文字,就垂直居中了。缺點是要控制內容不要換行。
25. margin加倍的問題
==解決方案是在這個p裏面加上display:inline;塊變行==
例如:
相應的css爲
#imfloat{ float:left; margin:5px; display:inline; }
26. IE與寬度和高度的問題
==解決:設置在html上==
#box{ width: 80px; height: 35px; } html>body #box{ width: auto; height: auto; min-width: 80px; min-height: 35px; }
27. 頁面的最小寬度
#container{ min-width: 600px; width:expression(document.body.clientWidth< 600? "600px": "auto" ); }
==第一個min-width是正常的;但第2行的width使用了Javascript,這隻有IE才認得,這也會讓你的HTML文檔不太正規。它實際上經過Javascript的判斷來實現最小寬度。==
28. p浮動IE文本產生3象素的bug
#box{ float:left; width:800px;} #left{ float:left; width:50%;} #right{ width:50%;} 🌟html #left{ margin-right:-3px; //這句是關鍵}
29. IE捉迷藏的問題
當p應用複雜的時候每一個欄中又有一些連接,p等這個時候容易發生捉迷藏的問題。
有些內容顯示不出來,當鼠標選擇這個區域是發現內容確實在頁面。
解決辦法:對#layout使用line-height屬性或者給#layout使用固定高和寬。頁面結構儘可能簡單。
30. float的p閉合;清除浮動;自適應高度
?(1) 例如:
這裏的NOTfloatC並不但願繼續平移,而是但願往下排。(其中floatA、floatB的屬性已經設置爲float:left;)
這段代碼在IE中毫無問題,問題出在FF。緣由是NOTfloatC並不是float標籤,必須將float標籤閉合。在
之間加上
這個p必定要注意位置,並且必須與兩個具備float屬性的p同級,之間不能存在嵌套關係,不然會產生異常。而且將clear這種樣式定義爲爲以下便可:==.clear{clear:both;}==
(2)做爲外部 wrapper 的 p 不要定死高度,爲了讓高度能自適應,要在wrapper裏面加上overflow:hidden; 當包含float的box的時候,高度自適應在IE下無效,這時候應該觸發==IE的layout私有屬性(萬惡的IE啊!)用zoom:1;==能夠作到,這樣就達到了兼容。
例如某一個wrapper以下定義:
.colwrapper{overflow:hidden; zoom:1; margin:5px auto;}
(3)對於排版,咱們用得最多的css描述可能就是float:left.有的時候咱們須要在n欄的float p後面作一個統一的背景,譬如:
好比咱們要將page的背景設置成藍色,以達到全部三欄的背景顏色是藍色的目的,可是咱們會發現隨着left center right的向下拉長,而page竟然保存高度不變,問題來了,緣由在於page不是float屬性,而咱們的page因爲要居中,不能設置成float,因此咱們應該這樣解決:
再嵌入一個float left而寬度是100%的p解決之。
或者另外一種方法:用選擇器(:after)在page以後插入一個空標籤,並清除浮動
.page:after {content: ""; display: table; clear: both; }
(4)萬能float 閉合(很是重要!)
關於 clear float 的原理可參見 [How To Clear Floats Without Structural Markup],將如下代碼加入Global CSS 中,給須要閉合的p加上class="clearfix" 便可,屢試不爽。
.clearfix:after { content:"."; display:block; height:0; clear:both; visibility:hidden; } .clearfix { display:inline-block; } .clearfix {display:block;}
或者這樣設置:`.hackbox{ display:table; //將對象做爲塊元素級的表格顯示}
`
31. 高度不適應
高度不適應是當內層對象的高度發生變化時外層高度不能自動進行調節,特別是當內層對象使用margin 或padding時。
例:
#box {background-color:#eee; } #box p {margin-top: 20px;margin-bottom: 20px; text-align:center; }
p對象中的內容
==解決技巧:在P對象上下各加2個空的p對象CSS代碼{height:0px;overflow:hidden;}
或者爲p加上border屬性。==
32. IE6下圖片下有空隙產生
解決這個BUG的技巧有不少,能夠是
==改變html的排版
或者設置img爲display:block
或者設置vertical-align屬性爲vertical-align:top/bottom/middle/text-bottom 均可以解決.==
33. 對齊文本與文本輸入框
==加上vertical-align:middle;==
Ps:經驗證,在IE下任一版本都不適用,而ff、opera、safari、chrome均OK!
34. li中內容超過長度後以省略號顯示
此技巧適用與IE、Opera、safari、chrom瀏覽器,FF暫不支持。
35. 爲何web標準中IE沒法設置滾動條顏色了
==解決辦法是將body換成html==
39. FORM標籤
這個標籤
==css中通常首先都使用這樣的樣式ul,form{margin:0;padding:0;}。==
40. 屬性選擇器(這個不能算是兼容,是隱藏css的一個bug)
p[id]{}p[id]{}
41. 爲何FF下文本沒法撐開容器的高度
==去掉height設置min-height:200px; 這裏爲了照顧不認識min-height的IE6 能夠這樣定義:==
{ height:auto!important; height:200px; min-height:200px; }
43. IE和FireFox 對空格的尺寸解釋不一樣
==解決方法是在img外面套li,而且對li定義margin: 0; 避免方式:在必要的時候不要無視 list 標籤)==
44. 條件註釋
lte -- 小於等於
lt-- 小於
gte --大於等於
gt--大於
! --不等於
45.強制渲染
//這句話的意思是強制使用IE7模式來解析網頁代碼!
//Google Chrome Frame也可讓IE用上Chrome的引擎
或者//強制IE8使用IE7模式來解析
//強制IE8使用IE6或IE5模式來解析
//一個特定版本的IE支持所要求的兼容性模式多於一種
46.js兼容文件
使IE5,IE6兼容到IE7模式(推薦)
使IE5,IE6,IE7兼容到IE8模式
使IE5,IE6,IE7,IE8兼容到IE9模式
47.瀏覽器識別符
p{ _color:red; } IE6 專用
*html p{ color:#red; }IE6 專用
p{ +color:red; } IE6,7 專用
p{ *color:red; } IE6,7 專用
*html p{ color:red; } IE6,7 專用
p{*+color: red;} IE7 專用
Body> p{ color: red; }屏蔽 IE6
p{ color:red9; } IE8
Firefox: -moz-
Safari: -webkit-
Opera: -o-
IE: -ms-
整理時有一些用到了別人的文章,須要標註版權的請聯繫