Warning: error_log(/data/www/wwwroot/hmttv.cn/caches/error_log.php): failed to open stream: Permission denied in /data/www/wwwroot/hmttv.cn/phpcms/libs/functions/global.func.php on line 537 Warning: error_log(/data/www/wwwroot/hmttv.cn/caches/error_log.php): failed to open stream: Permission denied in /data/www/wwwroot/hmttv.cn/phpcms/libs/functions/global.func.php on line 537 欧美在线一区二区三区欧美,一本色道久久爱88av,视频一区二区三区蜜桃麻豆

          整合營銷服務商

          電腦端+手機端+微信端=數據同步管理

          免費咨詢熱線:

          Jsx(javascript xml)

          eact定義的一種類似于XML的JS擴展語法,用來簡化創建虛擬DOM

          <!DOCTYPE html>
          <html lang="en">
              <head>
                  <meta charset="UTF-8">
                  <meta name="viewport" content="width=device-width, initial-scale=1.0">
                  <title>jsx</title>
                  <style>
                      .title{
                          background-color: orange:
                          width: 200px;
                      }
                  </style><!--定義個樣式,也可以寫在CSS文件里,引入進來-->
              </head>
          
              <body>
                  <div id="test"></div>
                  <script type="text/javascript" src="../js/react.development.js"></script>
                  <script type="text/javascript" src="../js/react-dom.development.js"></script>
                  <script type="text/javascript" src="../js/babel.min.js"></script>
              </body>
              <script type="text/babel">
                  const myid= "lOVE you"
                  const mydata= "hellO jsx"
                  const VDOM = ( // <!--h2標簽引入樣式,用className,span標簽引入內聯樣式的時候,不是用雙引號,而是雙花括號,-->
                  <div>
                      <h2 className="title" id={myid.toLowerCase()}> 
                          <span style={{color:'red',fontSize:'29px'}}>{mydata.toLowerCase()}</span>
                      </h2>)// <!--標簽中混入JS表達式時要用花括號{},如這里mydata取值 -->
                      //jsx中只能有一個根標簽,比如這里的h2,可以在h2的外面包一層div,就可以寫兩個h2
                      <h2 className="title" id={myid.toUpperCase()}> 
                          <span style={{color:'red',fontSize:'29px'}}>{mydata.toLowerCase()}</span>
                      </h2>)
                      //標簽首字母若是小寫字母,則將該標簽轉為html中同名元素,若html中無該標簽對應的同名元素,則報錯
                      //標簽首字母若是大寫字母,react就去渲染對應的組件,若組件沒有定義,則報錯
                  </div>
          
                  ReactDOM.render(VDOM,document.getElementById('test'))
              </script>
          </html>
          

          一個jsx的小練習

           <script type="text/babel">
                  const data=['A','B','C']
                  const VDOM = (
                      <div>   //js表達式會產生一個值。js語句(代碼),有if,for,switch判斷,
                          <h1>jsx框架</h1>
                          <ul>
                            {
                              data.map((item,index)=>{
                                  return <li key={index}>{item}</li>
                              }) //item拿到data里面對應的值,map遍歷的第二個值是索引值,
                            }
                          </ul>
                      </div>
                  )
                  ReactDOM.render(VDOM,document.getElementById('test'))
              </script>

          對應的網頁如下:

          模塊是向外提供特定功能的js程序,一般就是一個js文件。

          組件比模塊更高一級,比如實現一個網頁的頭部的html,字體,css,js,圖像這些元素組合在一起,就形成了頭部這個組件。

          組件分為函數式組件和類式組件:

          函數式組件:

          <script type="text/babel">
          function MyComponent(){
          return <h2>show the function component</h2>
          }
          ReactDOM.render(<MyComponent/>,document.getElementById('test'))
          //<MyComponent/>要寫上標簽,函數定義首字母需要大寫
          </script>

          執行了ReactDOM.render(<MyComponent/>。。之后,React解析組件標簽,找到MyComponent組件,發現組件是使用函數定義的,隨后調用該函數,將返回的虛擬DOM轉為真是DOM,隨后呈現在頁面中。

          類式組件:

          讀目錄

          • 利用表單實現文件上傳
          • 表單異步上傳(jquery.form插件)
          • 模擬表單數據上傳(FormData)
          • 分片上傳
          • 使用HTML5 拖拽、粘貼上傳
          • 上傳插件(WebUploader)
          • 總結

          作為程序員的我們,經常要用到文件的上傳和下載功能。到了需要用的時候,各種查資料。有木有..有木有...。為了方便下次使用,這里來做個總結和備忘。

          利用表單實現文件上傳

          最原始、最簡單、最粗暴的文件上傳。
          前端代碼:

          //方式1
          <form action="/Home/SaveFile1" method="post" enctype="multipart/form-data">
               <input type="file" class="file1" name="file1" />
               <button type="submit" class="but1">上傳</button>
          </form>
          
          

          【注意】

          • 1、需要post提交
          • 2、enctype="multipart/form-data" (傳輸文件)
          • 3、需要提交的表單元素需要設置 name 屬性

          后臺代碼:

          public ActionResult SaveFile1()
          {
              if (Request.Files.Count > 0)
              {
                  Request.Files[0].SaveAs(Server.MapPath("~/App_Data/") + Request.Files[0].FileName);
                  return Content("保存成功");
              }
              return Content("沒有讀到文件");
          }
          

          表單異步上傳(jquery.form插件)

          雖然上面的方式簡單粗暴,但是不夠友好。頁面必然會刷新。難以實現停留在當前頁面,并給出文件上傳成功的提示。
          隨著時間的流逝,技術日新月異。ajax的出現,使得異步文件提交變得更加容易。
          下面我們利用jquery.form插件來實現文件的異步上傳。
          首先我們需要導入
          jquery.jsjquery.form.js
          前端代碼:

          <form id="form2" action="/Home/SaveFile2" method="post" enctype="multipart/form-data">
              <input type="file" class="file1" name="file1" />
              <button type="submit" class="but1">上傳1</button>
              <button type="button" class="but2">上傳2</button>
          </form>
          
          
          //方式2(通過ajaxForm綁定ajax操作)
          $(function () {
              $('#form2').ajaxForm({
                  success: function (responseText) {
                      alert(responseText);
                  }
              });
          });
          
          //方式3(通過ajaxSubmit直接執行ajax操作)
          $(function () {
              $(".but2").click(function () {
                  $('#form2').ajaxSubmit({
                      success: function (responseText) {
                          alert(responseText);
                      }
                  });
              });
          });
          
          

          后臺代碼:

          public string SaveFile2()
          {
              if (Request.Files.Count > 0)
              {                
                  Request.Files[0].SaveAs(Server.MapPath("~/App_Data/") + Path.GetFileName(Request.Files[0].FileName));
                  return "保存成功";
              }
              return "沒有讀到文件";
          }
          

          原理:
          我們很多時候使用了插件,就不管其他三七二十一呢。
          如果有點好奇心,想想這個插件是怎么實現的。隨便看了看源碼一千五百多行。我的媽呀,不就是個異步上傳的嗎,怎么這么復雜。
          難以看出個什么鬼來,直接斷點調試下吧。


          原來插件內部有iframe和FormData不同方式來上傳,來適應更多版本瀏覽器。

          模擬表單數據上傳(FormData)

          iframe這東西太惡心。我們看到上面可以利用FormData來上傳文件,這個是Html 5 才有的。下面我們自己也來試試吧。
          前端代碼:

          <input id="fileinfo" type="file" class="notFormFile" />
          <button type="button" class="btnNotForm">上傳4</button>
          
          //方式4
          $(".btnNotForm").click(function () {
              var formData = new FormData();//初始化一個FormData對象
              formData.append("files", $(".notFormFile")[0].files[0]);//將文件塞入FormData
              $.ajax({
                  url: "/Home/SaveFile2",
                  type: "POST",
                  data: formData,
                  processData: false,  // 告訴jQuery不要去處理發送的數據
                  contentType: false,   // 告訴jQuery不要去設置Content-Type請求頭
                  success: function (responseText) {
                      alert(responseText);
                  }
              });
          });
          
          

          后面的代碼:(不變,還是上例代碼)

          public string SaveFile2()
          {
              if (Request.Files.Count > 0)
              {                
                  Request.Files[0].SaveAs(Server.MapPath("~/App_Data/") + Path.GetFileName(Request.Files[0].FileName));
                  return "保存成功";
              }
              return "沒有讀到文件";
          }
          

          我們看到,FormData對象也只是在模擬一個原始的表單格式的數據。那有沒有可能利用表單或表單格式來上傳文件呢?答案是肯定的。(下面馬上揭曉)
          前端代碼:

          <input type="file"  id="file5" multiple>
          <button type="button" class="btnFile5">上傳5</button>    
          
          
          //方式5
          $(".btnFile5").click(function () {
              $.ajax({
                  url: "/Home/SaveFile4",
                  type: "POST",
                  data: $("#file5")[0].files[0],
                  processData: false,  // 告訴jQuery不要去處理發送的數據
                  contentType: false,   // 告訴jQuery不要去設置Content-Type請求頭
                  success: function (responseText) {
                      alert(responseText);
                  }
              });;       
          });       
               
          

          后臺代碼:

          public string SaveFile4()
          {
              //這里發現只能得到一個網絡流,沒有其他信息了。(比如,文件大小、文件格式、文件名等)
              Request.SaveAs(Server.MapPath("~/App_Data/SaveFile4.data") + "", false);
              return "保存成功";
          }
          

          細心的你發現沒有了表單格式,我們除了可以上傳文件流數據外,不能再告訴后臺其他信息了(如文件格式)。
          到這里,我似乎明白了以前上傳文件為什么非得要用個form包起來,原來這只是和后臺約定的一個傳輸格式而已。
          其實我們單純地用jq的ajax傳輸文本數據的時候,最后也是組裝成了form格式的數據,如:

           $.ajax({
              data: { "userName": "張三" } 
          

          分片上傳

          在知道了上面的各種上傳之后,我們是不是就滿于現狀了呢?no,很多時候我們需要傳輸大文件,一般服務器都會有一定的大小限制。
          某天,你發現了一個激情小電影想要分享給大家。無奈,高清文件太大傳不了,怎么辦?我們可以化整為零,一部分一部分的傳嘛,也就是所謂的分片上傳。
          前端代碼:

          <input type="file" id="file6" multiple>
          <button type="button" class="btnFile6">分片上傳6</button>
          <div class="result"></div>
          
          
          //方式6
           $(".btnFile6").click(function () { 
               var upload = function (file, skip) {
                   var formData = new FormData();//初始化一個FormData對象
                   var blockSize = 1000000;//每塊的大小
                   var nextSize = Math.min((skip + 1) * blockSize, file.size);//讀取到結束位置             
                   var fileData = file.slice(skip * blockSize, nextSize);//截取 部分文件 塊
                   formData.append("file", fileData);//將 部分文件 塞入FormData
                   formData.append("fileName", file.name);//保存文件名字
                   $.ajax({
                       url: "/Home/SaveFile6",
                       type: "POST",
                       data: formData,
                       processData: false,  // 告訴jQuery不要去處理發送的數據
                       contentType: false,   // 告訴jQuery不要去設置Content-Type請求頭
                       success: function (responseText) {
                           $(".result").html("已經上傳了" + (skip + 1) + "塊文件");
                           if (file.size <= nextSize) {//如果上傳完成,則跳出繼續上傳
                               alert("上傳完成");
                               return;
                           }
                           upload(file, ++skip);//遞歸調用
                       }
                   });
               };
          
               var file = $("#file6")[0].files[0];
               upload(file, 0);
           }); 
          
          

          后臺代碼:

          public string SaveFile6()
          {
              //保存文件到根目錄 App_Data + 獲取文件名稱和格式
              var filePath = Server.MapPath("~/App_Data/") + Request.Form["fileName"];
              //創建一個追加(FileMode.Append)方式的文件流
              using (FileStream fs = new FileStream(filePath, FileMode.Append, FileAccess.Write))
              {
                  using (BinaryWriter bw = new BinaryWriter(fs))
                  {
                      //讀取文件流
                      BinaryReader br = new BinaryReader(Request.Files[0].InputStream);
                      //將文件留轉成字節數組
                      byte[] bytes = br.ReadBytes((int)Request.Files[0].InputStream.Length);
                      //將字節數組追加到文件
                      bw.Write(bytes);
                  }
              }
              return "保存成功";
          }
          

          相對而言,代碼量多了一點,復雜了一點。不過相對于網上的其他分片上傳的代碼應該要簡單得多(因為這里沒有考慮多文件塊同時上傳、斷點續傳。那樣就需要在后臺把文件塊排序,然后上傳完成按序合并,然后刪除原來的臨時文件。有興趣的同學可以自己試試,稍候在分析上傳插件webuploader的時候也會實現)。
          效果圖:


          【說明】:如果我們想要上傳多個文件怎么辦?其實H5中也提供了非常簡單的方式。直接在
          input里面標記multiple<input type="file" id="file6" multiple>,然后我們后臺接收的也是一個數組Request.Files

          使用HTML5 拖拽、粘貼上傳

          只能說H5真是強大啊,權限越來越大,操作越來越牛逼。
          前端代碼(拖拽上傳):

          <textarea class="divFile7" style="min-width:800px;height:150px" placeholder="請將文件拖拽或直接粘貼到這里"></textarea>
          
          //方式7
           $(".divFile7")[0].ondrop = function (event) {
          
               event.preventDefault();//不要執行與事件關聯的默認動作
               var files = event.dataTransfer.files;//獲取拖上來的文件
          
               //以下代碼不變
               var formData = new FormData();//初始化一個FormData對象
               formData.append("files", files[0]);//將文件塞入FormData
               $.ajax({
                   url: "/Home/SaveFile2",
                   type: "POST",
                   data: formData,
                   processData: false,  // 告訴jQuery不要去處理發送的數據
                   contentType: false,   // 告訴jQuery不要去設置Content-Type請求頭
                   success: function (responseText) {
                       alert(responseText);
                   }
               });
           };
          
          

          后臺代碼:
          略(和之前的SaveFile2一樣)

          前端代碼(粘貼上傳 限圖片格式):

          //方式8
          $(".divFile7")[0].onpaste = function (event) {
              event.preventDefault();//不要執行與事件關聯的默認動作
              var clipboard = event.clipboardData.items[0];//剪貼板數據
              if (clipboard.kind == 'file' || clipboard.type.indexOf('image') > -1) {//判斷是圖片格式
                  var imageFile = clipboard.getAsFile();//獲取文件
          
                  //以下代碼不變
                  var formData = new FormData;
                  formData.append('files', imageFile);
                  formData.append('fileName', "temp.png");//這里給文件命個名(或者直接在后臺保存的時候命名)
                  $.ajax({
                      url: "/Home/SaveFile8",
                      type: "POST",
                      data: formData,
                      processData: false,  // 告訴jQuery不要去處理發送的數據
                      contentType: false,   // 告訴jQuery不要去設置Content-Type請求頭
                      success: function (responseText) {
                          alert(responseText);
                      }
                  });
              }
          };
          
          

          后臺代碼:

          public string SaveFile8()
          {
              //保存文件到根目錄 App_Data + 獲取文件名稱和格式
              var filePath = Server.MapPath("~/App_Data/") + Request.Form["fileName"];      
              if (Request.Files.Count > 0)
              {
                  Request.Files[0].SaveAs(filePath);
                  return "保存成功";
              }
              return "沒有讀到文件";
          }
          
          

          效果圖:

          上傳插件(WebUploader)

          已經列舉分析了多種上傳文件的方式,我想總有一種適合你。不過,上傳這個功能比較通用,而我們自己寫的可能好多情況沒有考慮到。接下來簡單介紹下百度的WebUploader插件。
          比起我們自己寫的簡單上傳,它的優勢:穩定、兼容性好(有flash切換,所以支持IE)、功能多、并發上傳、斷點續傳(主要還是靠后臺配合)。
          官網:http://fex.baidu.com/webuploader/
          插件下載:https://github.com/fex-team/webuploader/releases/download/0.1.5/webuploader-0.1.5.zip
          下面開始對WebUploader的使用
          第一種,簡單粗暴
          前端代碼:

          <div id="picker">選擇文件</div>
          <button id="ctlBtn" class="btn btn-default">開始上傳</button>
          
          
          <!--引用webuploader的js和css-->
          <link href="~/Scripts/webuploader-0.1.5/webuploader.css" rel="stylesheet" />
          <script src="~/Scripts/webuploader-0.1.5/webuploader.js"></script>
          <script type="text/javascript">
              var uploader = WebUploader.create({
          
                  // (如果是新瀏覽器 可以不用 flash)
                  //swf: '/Scripts/webuploader-0.1.5/Uploader.swf',
          
                  // 文件接收服務端。
                  server: '/Webuploader/SaveFile',
          
                  // 選擇文件的按鈕。可選。
                  // 內部根據當前運行是創建,可能是input元素,也可能是flash.
                  pick: '#picker'
              });
          
              $("#ctlBtn").click(function () {
                  uploader.upload();
              });
          
              uploader.on('uploadSuccess', function (file) {
                  alert("上傳成功");
              });
          
          </script>
          
          

          后臺代碼:

          public string SaveFile()
          {
              if (Request.Files.Count > 0)
              {
                  Request.Files[0].SaveAs(Server.MapPath("~/App_Data/") + Path.GetFileName(Request.Files[0].FileName));
                  return "保存成功";
              }
              return "沒有讀到文件";
          }
          

          第二種,分片上傳。和我們之前自己寫的效果差不多。
          前端代碼:

          var uploader = WebUploader.create({ 
              //兼容老版本IE
              swf: '/Scripts/webuploader-0.1.5/Uploader.swf', 
              // 文件接收服務端。
              server: '/Webuploader/SveFile2', 
              // 開起分片上傳。
              chunked: true, 
              //分片大小
              chunkSize: 1000000, 
              //上傳并發數
              threads: 1,
              // 選擇文件的按鈕。 
              pick: '#picker'
          });
          
          // 點擊觸發上傳
          $("#ctlBtn").click(function () {
              uploader.upload();
          });
          
          uploader.on('uploadSuccess', function (file) {
              alert("上傳成功");
          });
          
          

          后臺代碼:

          public string SveFile2()
          {
              //保存文件到根目錄 App_Data + 獲取文件名稱和格式
              var filePath = Server.MapPath("~/App_Data/") + Path.GetFileName(Request.Files[0].FileName);
              //創建一個追加(FileMode.Append)方式的文件流
              using (FileStream fs = new FileStream(filePath, FileMode.Append, FileAccess.Write))
              {
                  using (BinaryWriter bw = new BinaryWriter(fs))
                  {
                      //讀取文件流
                      BinaryReader br = new BinaryReader(Request.Files[0].InputStream);
                      //將文件留轉成字節數組
                      byte[] bytes = br.ReadBytes((int)Request.Files[0].InputStream.Length);
                      //將字節數組追加到文件
                      bw.Write(bytes);
                  }
              }
              return "保存成功";
          }
          

          我們看到了有個參數threads: 1上傳并發數,如果我們改成大于1會怎樣?前端會同時發起多個文件片上傳。后臺就會報錯,多個進程同時操作一個文件。
          那如果我們想要多線程上傳怎么辦?改代碼吧(主要是后臺邏輯)。
          前端代碼:

          //并發上傳(多線程上傳)
          var uploader = WebUploader.create({
              //兼容老版本IE
              swf: '/Scripts/webuploader-0.1.5/Uploader.swf',
              // 文件接收服務端。
              server: '/Webuploader/SveFile3',
              // 開起分片上傳。
              chunked: true,
              //分片大小
              chunkSize: 1000000,
              //上傳并發數
              threads: 10,
              // 選擇文件的按鈕。
              pick: '#picker'
          });
          
          // 點擊觸發上傳
          $("#ctlBtn").click(function () {
              uploader.upload();
          });
          
          uploader.on('uploadSuccess', function (file) {
              //上傳完成后,給后臺發送一個合并文件的命令
              $.ajax({
                  url: "/Webuploader/FileMerge",
                  data: { "fileName": file.name },
                  type: "post",
                  success: function () {
                      alert("上傳成功");
                  }
              });
          });
          
          

          后臺代碼:

          public string SveFile3()
          {
              var chunk = Request.Form["chunk"];//當前是第多少片 
          
              var path = Server.MapPath("~/App_Data/") + Path.GetFileNameWithoutExtension(Request.Files
              if (!Directory.Exists(path))//判斷是否存在此路徑,如果不存在則創建
              {
                  Directory.CreateDirectory(path);
              }
              //保存文件到根目錄 App_Data + 獲取文件名稱和格式
              var filePath = path + "/" + chunk;
              //創建一個追加(FileMode.Append)方式的文件流
              using (FileStream fs = new FileStream(filePath, FileMode.Append, FileAccess.Write))
              {
                  using (BinaryWriter bw = new BinaryWriter(fs))
                  {
                      //讀取文件流
                      BinaryReader br = new BinaryReader(Request.Files[0].InputStream);
                      //將文件留轉成字節數組
                      byte[] bytes = br.ReadBytes((int)Request.Files[0].InputStream.Length);
                      //將字節數組追加到文件
                      bw.Write(bytes);
                  }
              }           
              return "保存成功";
          }
          
          /// <summary>
          /// 合并文件
          /// </summary>
          /// <param name="path"></param>
          /// <returns></returns>
          public bool FileMerge()
          {
              var fileName = Request.Form["fileName"];
              var path = Server.MapPath("~/App_Data/") + Path.GetFileNameWithoutExtension(fileName);
          
              //這里排序一定要正確,轉成數字后排序(字符串會按1 10 11排序,默認10比2小)
              foreach (var filePath in Directory.GetFiles(path).OrderBy(t => int.Parse(Path.GetFileNameWithoutExtension(t))))
              {
                  using (FileStream fs = new FileStream(Server.MapPath("~/App_Data/") + fileName, FileMode.Append, FileAccess.Write))
                  {
                      byte[] bytes = System.IO.File.ReadAllBytes(filePath);//讀取文件到字節數組
                      fs.Write(bytes, 0, bytes.Length);//寫入文件
                  }
                  System.IO.File.Delete(filePath);
              }
              Directory.Delete(path);
              return true;
          }
          

          到這里你以為就結束了嗎?錯,還有好多情況沒有考慮到。如果多個用戶上傳的文件名字一樣會怎樣?如何實現斷點續傳?還沒實現選擇多個文件?不過,這里不打算繼續貼代碼了(再貼下去,代碼量越來越多了),自己也來練習練習吧。
          提供一個思路,上傳前先往數據庫插入一條數據。數據包含文件要存的路徑、文件名(用GUID命名,防止同名文件沖突)、文件MD5(用來識別下次續傳和秒傳)、臨時文件塊存放路徑、文件是否完整上傳成功等信息。
          然后如果我們斷網后再傳,首先獲取文件MD5值,看數據庫里面有沒上傳完成的文件,如果有就實現秒傳。如果沒有,看是不是有上傳了部分的。如果有接著傳,如果沒有則重新傳一個新的文件。

          總結

          之前我一直很疑惑,為什么上傳文件一定要用form包起來,現在算是大概明白了。
          最開始在javascript還不流行時,我們就可以直接使用submit按鈕提交表單數據了。表單里面可以包含文字和文件。然后隨著js和ajax的流行,可以利用ajax直接異步提交部分表單數據。這里開始我就糾結了,為什么ajax可以提交自己組裝的數據。那為什么不能直接提交文件呢。這里我錯了,ajax提交的并不是隨意的數據,最后還是組裝成了表單格式(因為后臺技術對表單格式數據的支持比較普及)。但是現有的技術還不能通過js組裝一個文件格式的表單數據。直到H5中的FormData出現,讓前端js組裝一個包含文件的表單格式數據成為了可能。所以說表單只是為了滿足和后臺“約定”的數據格式而已。

          相關推薦

          • http://www.cnblogs.com/fish-li/archive/2011/07/17/2108884.html
          • http://javascript.ruanyifeng.com/htmlapi/file.html

          demo

          • https://github.com/zhaopeiym/BlogDemoCode/tree/master/上傳下載

          TML配置

          • 文件應以“”首行頂格開始,推薦使用“”。
          • 必須申明文檔的編碼charset,并且使用UTF-8編碼。
          • 移動端必須使用 viewport 適配
          • 頁面title是極為重要的不可缺少的一項。

          以下是一個標準的html結構

          html復制代碼<!DOCTYPE html>
          <html>
            <head>
              <!-- 必須聲明 utf-8 編碼格式 -->
              <meta charset="utf-8">
              <!-- 頁面標題不能為空 -->
              <title>京東商城:商家后臺</title>
              <!-- 移動端必須使用 viewport 適配需要, PC 看場景需要使用 -->
              <meta name="viewport" content="width=device-width,initial-scale=1.0,user-scalable=no">
               
              <!-- 樣式文件 以外鏈形式在此處引入 -->
              <link href="//static.360buyimg.com/bus/test/shop1_3/UI2.0update.css" rel="stylesheet">
            </head>
            <body>
              <!-- 頁面主體 -->
              <div class="warp">
                <header>頂部內容區域,比如:菜單頭</header>
                <div class="content">主內容區域</div>
                <footer>頂部頁尾區域,比如:備案號</footer>
              </div>
               
              <!-- js文件 在此處引入 -->
              <script type="text/javascript" src="//wl.jd.com/joya.js"></script>
            </body>
          </html>
          

          主框架(jQuery,vue,react,angular等) 避免混合使用

          避免多個框架混合使用,在同一個工程里面要保持主框架的一致性,多框架同時使用會造成代碼混亂,后期會變得越來越難維護。

          Css開發規范

          1、全局樣式禁止使用!important

          2、避免使用導入式引入css樣式文件;

          css復制代碼<style type="text/css">
            @import url(./demo.css);
          </style>
          

          未使用的js/css禁止引用

          歷史頁面會有一些功能下線,在去除頁面初始化的js語句的同時,也要將依賴的js文件刪除,避免資源加載浪費

          文件必須使用gzip壓縮

          開啟Gzip壓縮功能, 可以使網站的css、js 、xml、html 文件在傳輸時進行壓縮,提高訪問速度

          api接口中,無用字段占傳輸比例30%以上時,刪除無用字段

          在開發過程中,發現無用字段占傳輸比例30%以上時,請進行返回數據的刪減,加快數據請求速度

          關于圖片壓縮

          上傳圖片之前一定要做圖片的無損壓縮,節省網絡流量,推薦網站tinypng

          HTML的標準結構

          html復制代碼<!DOCTYPE html>
          <html>
            <head>
              <!-- 必須聲明 utf-8 編碼格式 -->
              <meta charset="utf-8">
              <!-- keywords 關鍵詞 和 Description 頁面描述 便于搜索引擎檢索,不強制使用 -->
              <meta name="Keywords" Content=”關鍵詞1,關鍵詞2,關鍵詞3,關鍵詞4″>
              <meta name="Description" Content=”頁面描述″>
              <!-- 移動端必須使用 viewport 適配需要, PC 看場景需要使用 -->
              <meta name="viewport" content="width=device-width,initial-scale=1.0,user-scalable=no">
              <!-- 推薦手動引入指定路徑 favicon  -->
              <link rel="shortcut icon" href="path/to/favicon.ico">
              <!-- 樣式文件必須以外鏈形式在此處引入 -->
              <link href="//static.360buyimg.com/bus/test/shop1_3/UI2.0update.css" rel="stylesheet">
              <!-- 內聯樣式 -->
              <style type="text/css">
                .warp {
                  margin: 0;
                  padding: 0;
                }
              </style> -->
              <!-- 頁面標題不能為空 -->
              <title>頁面標題</title>
            </head>
            <body>
            <!-- 頁面主體 -->
              <div class="warp">
              <header>頂部內容區域,比如:菜單頭</header>
              <div class="content">主內容區域</div>
              <footer>頂部頁尾區域,比如:備案號</footer>
              </div>
           
              <!-- JavaScript 文件在此處引入 -->
              <script type="text/javascript" src="//wl.jd.com/joya.js"></script>
            </body>
          </html>
          
          • 不同模塊區域可使用簡要備注標識模塊內容
          • HTML 標簽語義化使結構清晰,比如:
          • CSS 外鏈引入必須放在 中
          • JS 外鏈引入必須放在中
          • 省略圖像、媒體文件、樣式表和腳本等嵌入式資源 URL 協議頭部聲明 ( http:// , https:// ),用//代替。
          • favicon 在未指定 favicon 時,大多數瀏覽器會請求 Web Server 根目錄下的 favicon.ico 。為了保證 favicon 可訪問,避免404,必須遵循以下兩種方法之一:1。在 Web Server 根目錄放置 favicon.ico 文件,2。使用 link 指定 favicon

          HTML標簽使用守則

          標簽小寫

          原生 HTML 標簽元素小寫使用,自定義組件使用小駝峰命名(自定義標簽避免和原生標簽同名)

          html復制代碼<!-- 錯誤 ? -->
          <DIV></DIV>
          <SPAN></SPAN>  
           
          <!-- 正確 ? -->
          <div></div>  
          <p></p> 
           
          <!-- 自定義組件 -->
          <shareDialog><shareDialog/>
          

          套用規則

          禁止在行內元素中嵌套塊級元素??

          html復制代碼<!-- 錯誤的嵌套 ? -->
          <span><div></div></span>  
          <i><p></p></i>     
           
          <!-- 正確的嵌套 ? -->
          <div><span></span></div>  
          <p><i></i></p>   
          

          更多標簽使用規則介紹請查看尾部附錄

          屬性定義

          定義屬性賦值時,使用雙引號,禁止單雙引號混用

          html復制代碼<!-- 錯誤的定義 ? -->
          <input id="formTitle" type='text' placeholder="請輸入標題">   
           
          <!-- 正確的定義 ? -->
          <input id="formTitle" type="text" placeholder="請輸入標題" data="formTitle">   
          

          正確閉合

          除自閉合標簽外,所有標簽都需正確的編寫閉合標簽

          常用自閉合標簽: <br />、<col />、<img />、<input />、<link />、<meta />、<keygen />

          ID、Class 命名使用規范

          ID 和 Class 取通用且有意義的名字;

          使用連字符 - 作為 ID、Class 名稱界定符 ,不要駝峰命名法和下劃線;

          避免選擇器嵌套層級過多,盡量少于 3 級;

          使用命名空間防止命名沖突,利于維護;

          css復制代碼/* 不推薦: 無意義 不易理解 */
          #yee-1901 {}
          #meunBtn {}
          .login_input {}
           
          /* 推薦: 明確詳細 */
          #gallery {}
          #login {}
          .login-input {}
          .meun-btn {}
          
          • 避免選擇器和 Class、ID 疊加使用
          • 聲明順序,相關屬性應為一組,建議遵循: 定位布局屬性 -> 盒模型屬性 -> 文本屬性 -> 視覺屬性 -> 其他屬性
          • 對于 JS 操作相關選擇器 前面需加上 特殊應用標識前綴
          css復制代碼/* 標準的聲明順序 */
          .declaration-order {
            /* 布局屬性 */
            position: absolute;
            top: 0;
            right: 0;
            bottom: 0;
            left: 0;
            z-index: 100;
           
            /* 盒模型屬性 */
            display: block;
            box-sizing: border-box;
            width: 100px;
            height: 100px;
            padding: 10px;
            border: 1px solid #e5e5e5;
            border-radius: 3px;
            margin: 10px;
            float: right;
            overflow: hidden;
           
            /* 文本屬性 */
            font: normal 13px "Helvetica Neue", sans-serif;
            line-height: 1.5;
            text-align: center;
           
            /* 視覺屬性 */
            background-color: #f5f5f5;
            color: #fff;
            opacity: .8;
           
            /* 其他屬性 */
            cursor: pointer;
          }
          

          圖片相關使用規范

          <img/>標簽守則

          • src 屬性不能為空,需添加默認值(托底圖片路徑)
          • 為重要圖片添加 alt 屬性,當無法顯示圖像時,瀏覽器將顯示替代文本,便于 SEO 及用戶閱讀
          • 為圖片標簽 增加 width 和 height 屬性,避免圖片加載從無到有,導致頁面抖動
          • 大量圖片流資源,使用懶加載技術按需加載
          html復制代碼<!-- 禁止 src 取值為空 -->
          <img src="" />
           
          <!-- 推薦 -->
          <img src="xxxx" alt="xxx" style="width:100px;height:100px;" />
          

          圖片資源守則

          大小

          尺寸 小于 50 ×?50 的請使用 字體圖標(iconfont)或者 將多個圖標合成一張大圖使用(雪碧圖) 尺寸大于 50 ×?50 的圖片,引入前先使用 在線壓縮 對其進行壓縮 ,圖片大小盡量控制在 300kb 以內

          倍率

          PC端、移動端 推薦使用 2倍圖 (@2x),避免設備分辨率過高,圖片顯示失真模糊 如有兼容 iPad 場景等特殊場景,可使用 3倍圖(@3x)

          html復制代碼<img src="//logo@200x200.jpg" alt="xx" style="width:100px;height:100px;"/>
          

          2倍圖: 圖片展示區域大小 : 圖片實際尺寸 = 1 :2

          無論使用幾倍圖,圖片大小都需遵守上條限制

          如遇圖片倍圖問題,可咨詢 @UI童鞋

          性能守則

          • 避免不必要的 DOM 操作瀏覽器遍歷 DOM 元素的代價是昂貴的。最簡單優化 DOM 樹查詢的方案是,當一個元素出現多次時,將它保存在一個變量中,來避免多次查詢 DOM 樹。
          js復制代碼/* 推薦用變量保存頻繁使用的 DOM 元素*/
          var ul = document.getElementsByTagName('ul')[0],
            lis = document.getElementsByTagName('ul')[0].getElementsByTagName('p')
             
          /* 推薦用變量保存頻繁使用的 DOM 元素*/
          var ul = document.getElementsByTagName('ul')[0],
            lis = ul.getElementsByTagName('li')
          
          • 異步加載第三方內容 當你無法保證嵌入第三方內容 比如 埋點、調研等資源正常工作的時候,你需要考慮用異步加載這些代碼,避免阻塞整個頁面加載。
          • 減少標簽的數量 編寫 HTML 代碼時,盡量避免多余的標簽嵌套,避免 DOM樹的冗余
          • 控制靜態資源數量

          1個頁面中建議CSS文件不要超過3個(1個組件庫樣式文件、1個項目公共樣式、1個頁面樣式); JS文件不要超過5個(1個框架文件、1個組件庫文件、1個項目公共庫文件、1個頁面腳本文件、1個埋點文件)。

          編碼建議

          縮進使用 2個空格 樣式建議引入 CSS reset 重置各瀏覽器自帶樣式差異 github.com/necolas/nor… JS 中字符串定義 使用 '' 單引號

          其他建議

          保持一定的代碼潔癖,尤其在大型項目中

          • 代碼邏輯應當直截了當,清晰易讀,保持與業務邏輯與代碼一一對應關系,減少邏輯錯誤的可能性,降低二次開發成本;
          • 刪除過期無用代碼,減少代碼過大,造成不必要的維護成本
          • 代碼盡量復用,擁有組件化思想
          • 代碼命名合理化,即使不寫注釋情況下,也可以讓其他開發人員一眼明白意思
          • 性能調至最優,降低耦合度,避免別人做不合理的優化時而造成的混亂
          • 整潔的模塊邊界,明確的劃分模塊之間的邏輯邊界,盡量保證職責單一,避免功能交叉混雜
          • 讓代碼沒有改進的余地,在開發時把各種情況都想到了,如果有人企圖改進它,總是會回到原點

          性能檢測工具 - Lighthouse簡介 Lighthouse是一個Google開源的自動化工具,主要用于改進網絡應用的質量。Lighthouse會對各個測試項的結果打分,并給出優化建議,這些打分標準和優化建議可以視為Google的網頁最佳實踐。

          Chrome拓展程序 谷歌商店安裝lighthouse后,在右上角或者菜單里點擊圖中圖標,Options可以配置測試項目,點擊Generate report即可測試。

          命令行

          js復制代碼<!-- 安裝 lighthouse 組件(Node.js V5.0 或以上版本) -->
          npm install -g lighthouse
          <!-- 啟動測試 (測試過程中會自動打開 Chrome 完成后會在當前目錄生成一個靜態HTML文件) -->
          lighthouse https://www.baidu.com/ --view
          <!-- 幫助-->
          lighthouse --help
          

          檢測結果

          【附錄】常用的標簽規范

          標簽

          語義

          嵌套常見錯誤

          常用屬性

          <a></a>

          超鏈接/錨

          a不可嵌套a

          href,name,title,rel,target

          <div></div>

          塊級容器



          <p></p>

          段落

          不能嵌套塊級元素


          <span></span>

          內聯容器(行內元素)

          不可嵌套塊級容器


          <form></form>

          表單


          action,target,method,name

          <input />

          輸入框

          不可嵌套元素

          type,name,value,checked,disabled,maxlength,readonly,accesskey

          <textarea></textarea>

          多行文本輸入控件


          name,accesskey,disabled,readonly,rows,cols

          <img />

          圖像

          不可嵌套元素

          alt,src,width,height

          <label></label>

          標簽(常用input元素定義標注)

          不可嵌套塊級容器

          for

          <table></table>

          表格

          只可嵌套表格子元素

          width,align,background,cellpadding,cellspacing,summary,border

          <tbody></tbody>

          表格主體

          只能嵌套在table內


          <thead></thead>

          表頭

          只能嵌套在table內


          <tr></tr>

          表格行

          嵌套于table或thead、tbody、tfoot


          <td></td>

          表格中的單元格

          只用于tr

          colspan,rowspan

          <th></th>

          表格中的標題單元格

          只用于tr

          colspan,rowspan

          <tfoot></tfoot>

          表格表尾

          只用于table


          <button></button>

          按鈕

          不可嵌套表單、表格等塊級元素

          type,disabled

          <select></select>

          列表框或下拉框

          只能嵌套option或optgroup

          name,disabled,multiple

          <option></option>

          select中的一個選項

          只能嵌套在select內

          value,selected,disabled

          <ol></ol>

          有序列表

          只能嵌套li


          <ul></ul>

          無序列表

          只能嵌套li


          <li></li>

          無序列表項

          只能嵌套在 ul 或 ol 內


          <iframe></iframe>

          內嵌一個網頁


          frameborder,width,height,src,scrolling,name

          <br />

          換行



          <link />

          引用樣式或icon

          不可嵌套任何元素

          type,rel,href

          <meta />

          文檔信息

          只用于head內

          content,http-equiv,name

          <script></script>

          引用腳本

          不可嵌套任何元素

          type,src

          <style></style>

          引用樣式

          不可嵌套任何元素

          type,media

          <title></title>

          文檔標題

          只用于head內


          點贊收藏支持、手留余香、與有榮焉,動動你發財的小手喲,感謝各位大佬能留下您的足跡。


          作者:StriveToY
          鏈接:https://juejin.cn/post/7262378982255394873


          主站蜘蛛池模板: 一区二区三区免费视频观看| 国模大尺度视频一区二区| 99久久精品费精品国产一区二区| 国产婷婷色一区二区三区| 国产精品成人免费一区二区| 性色A码一区二区三区天美传媒| 免费视频精品一区二区| 在线播放偷拍一区精品| 无码少妇一区二区三区浪潮AV | 国产成人精品一区二区三区无码 | 精品一区二区三区在线视频观看| 无码国产精品一区二区免费式芒果| 一区二区免费国产在线观看| 相泽南亚洲一区二区在线播放| 97人妻无码一区二区精品免费| 一区二区三区国产精品| 国产高清视频一区三区| 久久精品道一区二区三区| 亚洲日韩精品一区二区三区| 精品国产一区二区三区AV性色 | 精品国产不卡一区二区三区| 国产在线第一区二区三区| 国产伦精品一区二区三区免.费| 国产精品久久久久久一区二区三区| 国产精品视频一区二区三区| 亚洲av无一区二区三区| 一本大道在线无码一区| 欧美日韩精品一区二区在线视频| 一本久久精品一区二区| 激情久久av一区av二区av三区| 日本在线视频一区二区三区| 国产精品综合一区二区| 亚洲成av人片一区二区三区| 变态拳头交视频一区二区 | 精品爆乳一区二区三区无码av| 久久久无码一区二区三区| 精品亚洲AV无码一区二区三区 | 成人欧美一区二区三区在线视频| 亚洲欧洲专线一区| 国产一区二区三区免费视频| 国产福利电影一区二区三区,亚洲国模精品一区 |