From d88a902e76d31fa40de4bd19885e231059aa0ccf Mon Sep 17 00:00:00 2001 From: enson0131 Date: Tue, 14 Jan 2025 07:18:32 +0000 Subject: [PATCH] deploy: 2360d8c79af8de0ca744724993528bbb0dda9cbe --- 404.html | 2 +- ...\256\276\350\256\241_index.md.d79253fb.js" | 2 +- ...276\350\256\241_index.md.d79253fb.lean.js" | 2 +- ...75\344\274\230\345\214\226.md.9993b631.js" | 1 + ...4\274\230\345\214\226.md.9993b631.lean.js" | 1 + ...04\346\246\202\350\277\260.md.5690116d.js" | 2 +- ...6\246\202\350\277\260.md.5690116d.lean.js" | 2 +- ...54\346\246\202\345\277\265.md.302ccee7.js" | 2 +- ...6\246\202\345\277\265.md.302ccee7.lean.js" | 2 +- ...01\347\274\251\346\224\276.md.140dd9a6.js" | 2 +- ...7\274\251\346\224\276.md.140dd9a6.lean.js" | 2 +- ...60\346\241\206\351\200\211.md.382e7c32.js" | 1 + ...6\241\206\351\200\211.md.382e7c32.lean.js" | 1 + ...11\347\211\251\344\275\223.md.01b61e2b.js" | 1 - ...11\347\211\251\344\275\223.md.4b2066fc.js" | 1 + ...7\211\251\344\275\223.md.4b2066fc.lean.js" | 2 +- ...60\347\273\204\351\200\211.md.3e1b07e1.js" | 1 - ...7\273\204\351\200\211.md.3e1b07e1.lean.js" | 1 - ...52\345\233\276\345\275\242.md.fc479c77.js" | 2 +- ...5\233\276\345\275\242.md.fc479c77.lean.js" | 2 +- ...\260\203\350\257\225Fabric.md.b88e0c09.js" | 2 +- ...203\350\257\225Fabric.md.b88e0c09.lean.js" | 2 +- ...3_\346\246\202\350\246\201.md.5e89b8cf.js" | 2 +- ...6\246\202\350\246\201.md.5e89b8cf.lean.js" | 2 +- ...e for Schedule about React.md.8a28d74e.js} | 2 +- ... Schedule about React.md.8a28d74e.lean.js} | 2 +- ...04\345\214\272\345\210\253.md.11b5926a.js" | 2 +- ...5\214\272\345\210\253.md.11b5926a.lean.js" | 2 +- ...66\346\234\272\345\210\266.md.d11dda49.js" | 2 +- ...6\234\272\345\210\266.md.d11dda49.lean.js" | 2 +- ...70\346\234\272\345\210\266.md.ac78472e.js" | 2 +- ...6\234\272\345\210\266.md.ac78472e.lean.js" | 2 +- ...75\344\274\230\345\214\226.md.780419b2.js" | 2 +- ...4\274\230\345\214\226.md.780419b2.lean.js" | 2 +- ...344\275\277\347\224\250JSX.md.40e22e85.js" | 2 +- ...75\277\347\224\250JSX.md.40e22e85.lean.js" | 2 +- ...57\344\273\200\344\271\210.md.ee0c3664.js" | 2 +- ...4\273\200\344\271\210.md.ee0c3664.lean.js" | 2 +- ...62\346\237\223\347\232\204.md.f058feee.js" | 2 +- ...6\237\223\347\232\204.md.f058feee.lean.js" | 2 +- ...32\344\277\241\347\232\204.md.a0c59c32.js" | 2 +- ...4\277\241\347\232\204.md.a0c59c32.lean.js" | 2 +- ...ff\347\256\227\346\263\225.md.4f2b0e59.js" | 2 +- ...7\256\227\346\263\225.md.4f2b0e59.lean.js" | 2 +- ...70\346\234\272\345\210\266.md.39386d10.js" | 2 +- ...6\234\272\345\210\266.md.39386d10.lean.js" | 2 +- ...02\346\255\245\347\232\204.md.10d004c3.js" | 2 +- ...6\255\245\347\232\204.md.10d004c3.lean.js" | 2 +- ...4\271\210\346\230\257Fiber.md.9a75b68a.js" | 2 +- ...\210\346\230\257Fiber.md.9a75b68a.lean.js" | 2 +- ...ct\347\273\204\344\273\266.md.b6076bcd.js" | 2 +- ...7\273\204\344\273\266.md.b6076bcd.lean.js" | 2 +- ...04\351\227\256\351\242\230.md.5afef26f.js" | 2 +- ...1\227\256\351\242\230.md.5afef26f.lean.js" | 2 +- ...01\350\247\243\350\257\273.md.e0e871fd.js" | 2 +- ...0\247\243\350\257\273.md.e0e871fd.lean.js" | 2 +- ...41\347\220\206\345\277\265.md.00f4264a.js" | 2 +- ...7\220\206\345\277\265.md.00f4264a.lean.js" | 2 +- ...75\346\225\260\346\215\256.md.6565d23d.js" | 2 +- ...6\225\260\346\215\256.md.6565d23d.lean.js" | 2 +- ...56\346\213\206\345\210\206.md.64da80d7.js" | 2 +- ...6\213\206\345\210\206.md.64da80d7.lean.js" | 2 +- ...26\350\276\223\345\207\272.md.1a2532c7.js" | 2 +- ...0\276\223\345\207\272.md.1a2532c7.lean.js" | 2 +- ...04\346\265\201\347\250\213.md.58fac56a.js" | 2 +- ...6\265\201\347\250\213.md.58fac56a.lean.js" | 2 +- ...26\346\226\271\345\274\217.md.a17d2dce.js" | 2 +- ...6\226\271\345\274\217.md.a17d2dce.lean.js" | 2 +- ...60\346\215\256\345\272\223.md.0af240f5.js" | 2 +- ...6\215\256\345\272\223.md.0af240f5.lean.js" | 2 +- ...37\345\205\245\351\227\250.md.e65bc2d1.js" | 2 +- ...5\205\245\351\227\250.md.e65bc2d1.lean.js" | 2 +- ...1\210\346\230\257langchain.md.487089c2.js" | 2 +- ...\346\230\257langchain.md.487089c2.lean.js" | 2 +- ...\347\232\204PromptTemplate.md.2d5ec969.js" | 2 +- ...232\204PromptTemplate.md.2d5ec969.lean.js" | 2 +- ...07\347\237\253\346\255\243.md.6671b487.js" | 2 +- ...7\237\253\346\255\243.md.6671b487.lean.js" | 2 +- ...31\346\200\247\350\203\275.md.1828bbfc.js" | 2 +- ...6\200\247\350\203\275.md.1828bbfc.lean.js" | 2 +- ...61\344\271\246\345\206\231.md.a9d3a092.js" | 2 +- ...4\271\246\345\206\231.md.a9d3a092.lean.js" | 2 +- ...31\346\200\247\350\203\275.md.74068538.js" | 2 +- ...6\200\247\350\203\275.md.74068538.lean.js" | 2 +- ...31\346\200\247\350\203\275.md.fc9344af.js" | 2 +- ...6\200\247\350\203\275.md.fc9344af.lean.js" | 2 +- ...31\346\200\247\350\203\275.md.86c812db.js" | 2 +- ...6\200\247\350\203\275.md.86c812db.lean.js" | 2 +- ...31\346\200\247\350\203\275.md.1acbc131.js" | 2 +- ...6\200\247\350\203\275.md.1acbc131.lean.js" | 2 +- ...47\233\270\345\205\263_BFC.md.fbec5d1d.js" | 2 +- ...3\270\345\205\263_BFC.md.fbec5d1d.lean.js" | 2 +- ...33\270\345\205\263_display.md.6d34245c.js" | 2 +- ...0\345\205\263_display.md.6d34245c.lean.js" | 2 +- ...04\345\205\263\347\263\273.md.f469d683.js" | 2 +- ...5\205\263\347\263\273.md.f469d683.lean.js" | 2 +- ...ty\345\214\272\345\210\253.md.2678884b.js" | 2 +- ...5\214\272\345\210\253.md.2678884b.lean.js" | 2 +- ...3\270\345\205\263_position.md.b4f4db42.js" | 2 +- ...\345\205\263_position.md.b4f4db42.lean.js" | 2 +- ...74\346\226\271\345\274\217.md.7f0ca022.js" | 2 +- ...6\226\271\345\274\217.md.7f0ca022.lean.js" | 2 +- ...3_\346\246\202\350\246\201.md.9e3e6a15.js" | 2 +- ...6\246\202\350\246\201.md.9e3e6a15.lean.js" | 2 +- ...04\346\226\271\346\263\225.md.3f129790.js" | 2 +- ...6\226\271\346\263\225.md.3f129790.lean.js" | 2 +- ...OM\347\233\270\345\205\263.md.b5647b3c.js" | 2 +- ...7\233\270\345\205\263.md.b5647b3c.lean.js" | 2 +- ...S6\347\233\270\345\205\263.md.a46321f6.js" | 2 +- ...7\233\270\345\205\263.md.a46321f6.lean.js" | 2 +- ...67\346\234\272\345\210\266.md.bd59bbe3.js" | 2 +- ...6\234\272\345\210\266.md.bd59bbe3.lean.js" | 2 +- ...14\346\234\272\345\210\266.md.b95eb7c2.js" | 2 +- ...6\234\272\345\210\266.md.b95eb7c2.lean.js" | 2 +- ...21\346\234\272\345\210\266.md.4e21f748.js" | 2 +- ...6\234\272\345\210\266.md.4e21f748.lean.js" | 2 +- ...33\270\345\205\263_Promise.md.9dcd3d04.js" | 2 +- ...0\345\205\263_Promise.md.9dcd3d04.lean.js" | 2 +- ...\233\270\345\205\263_index.md.6fa036d3.js" | 2 +- ...270\345\205\263_index.md.6fa036d3.lean.js" | 2 +- ...57\346\234\272\345\210\266.md.d507e357.js" | 2 +- ...6\234\272\345\210\266.md.d507e357.lean.js" | 2 +- ...66\346\234\272\345\210\266.md.41875062.js" | 2 +- ...6\234\272\345\210\266.md.41875062.lean.js" | 2 +- ...00\346\246\202\345\277\265.md.53cfc405.js" | 2 +- ...6\246\202\345\277\265.md.53cfc405.lean.js" | 2 +- ...350\247\201\347\232\204API.md.699074e1.js" | 2 +- ...47\201\347\232\204API.md.699074e1.lean.js" | 2 +- ...3_\345\256\232\344\271\211.md.ea2dc8ec.js" | 2 +- ...5\256\232\344\271\211.md.ea2dc8ec.lean.js" | 2 +- ...350\247\201\347\232\204API.md.6057b79b.js" | 2 +- ...47\201\347\232\204API.md.6057b79b.lean.js" | 2 +- ...56\347\261\273\345\236\213.md.77b639a5.js" | 2 +- ...7\261\273\345\236\213.md.77b639a5.lean.js" | 2 +- ...350\247\201\347\232\204API.md.452ffdf7.js" | 2 +- ...47\201\347\232\204API.md.452ffdf7.lean.js" | 2 +- ...20\347\256\227\347\254\246.md.c8bd4cf5.js" | 2 +- ...7\256\227\347\254\246.md.c8bd4cf5.lean.js" | 2 +- ...50\350\276\276\345\274\217.md.df2c1874.js" | 2 +- ...0\276\276\345\274\217.md.df2c1874.lean.js" | 2 +- ...77\347\233\270\345\205\263.md.86520b75.js" | 2 +- ...7\233\270\345\205\263.md.86520b75.lean.js" | 2 +- ...js => guide_webpack_Loader.md.8ee4eae1.js} | 2 +- ... guide_webpack_Loader.md.8ee4eae1.lean.js} | 2 +- ....js => guide_webpack_index.md.8c12b279.js} | 2 +- ...> guide_webpack_index.md.8c12b279.lean.js} | 2 +- ...guide_webpack_mini-webpack.md.7a56cd34.js} | 2 +- ..._webpack_mini-webpack.md.7a56cd34.lean.js} | 2 +- ...72\346\265\201\347\250\213.md.61545129.js" | 2 +- ...6\265\201\347\250\213.md.61545129.lean.js" | 2 +- ...27\350\201\224\351\202\246.md.a4b55cfe.js" | 2 +- ...0\201\224\351\202\246.md.a4b55cfe.lean.js" | 2 +- ...60\345\216\237\347\220\206.md.87ba4ad8.js" | 2 +- ...5\216\237\347\220\206.md.87ba4ad8.lean.js" | 2 +- ...3_\346\246\202\350\246\201.md.8b5cc56c.js" | 2 +- ...6\246\202\350\246\201.md.8b5cc56c.lean.js" | 2 +- ...50\345\206\205\346\240\270.md.6fdfde81.js" | 2 +- ...5\206\205\346\240\270.md.6fdfde81.lean.js" | 2 +- ...50\345\256\211\345\205\250.md.ff5d442d.js" | 2 +- ...5\256\211\345\205\250.md.ff5d442d.lean.js" | 2 +- ...23\346\265\201\347\250\213.md.08093b9d.js" | 2 +- ...6\265\201\347\250\213.md.08093b9d.lean.js" | 2 +- ...50\347\274\223\345\255\230.md.3b8afe38.js" | 2 +- ...7\274\223\345\255\230.md.3b8afe38.lean.js" | 2 +- ...13\346\236\266\346\236\204.md.4157b63e.js" | 2 +- ...6\236\266\346\236\204.md.4157b63e.lean.js" | 2 +- ...47\233\270\345\205\263_CDN.md.426d00a8.js" | 2 +- ...3\270\345\205\263_CDN.md.426d00a8.lean.js" | 2 +- ...47\233\270\345\205\263_DNS.md.dc1c7d42.js" | 2 +- ...3\270\345\205\263_DNS.md.dc1c7d42.lean.js" | 2 +- ...04\345\214\272\345\210\253.md.675258cb.js" | 2 +- ...5\214\272\345\210\253.md.675258cb.lean.js" | 2 +- ...7\233\270\345\205\263_HTTP.md.2fd00196.js" | 2 +- ...\270\345\205\263_HTTP.md.2fd00196.lean.js" | 2 +- ...\233\270\345\205\263_HTTPS.md.1a833174.js" | 2 +- ...270\345\205\263_HTTPS.md.1a833174.lean.js" | 2 +- ...24\346\212\245\346\226\207.md.b46d175f.js" | 2 +- ...6\212\245\346\226\207.md.b46d175f.lean.js" | 2 +- ...04\351\227\256\351\242\230.md.fee49d4c.js" | 2 +- ...1\227\256\351\242\230.md.fee49d4c.lean.js" | 2 +- ...3_\346\246\202\350\246\201.md.3fbc149d.js" | 2 +- ...6\246\202\350\246\201.md.3fbc149d.lean.js" | 2 +- ...02\346\212\245\346\226\207.md.dd8a47bc.js" | 2 +- ...6\212\245\346\226\207.md.dd8a47bc.lean.js" | 2 +- ...37\350\257\267\346\261\202.md.848f5d06.js" | 2 +- ...0\257\267\346\261\202.md.848f5d06.lean.js" | 2 +- ...ex.md.b3d414ec.js => index.md.dd3d0939.js} | 2 +- ...14ec.lean.js => index.md.dd3d0939.lean.js} | 2 +- ...37\345\205\211\345\272\246.md.1329ace6.js" | 2 +- ...5\205\211\345\272\246.md.1329ace6.lean.js" | 2 +- ...5_\345\205\211\345\234\210.md.5071eb5f.js" | 2 +- ...5\205\211\345\234\210.md.5071eb5f.lean.js" | 2 +- ...50\351\200\237\345\272\246.md.858c51e6.js" | 2 +- ...1\200\237\345\272\246.md.858c51e6.lean.js" | 2 +- ...\212\200\345\267\247_index.md.e5c3a073.js" | 2 +- ...200\345\267\247_index.md.e5c3a073.lean.js" | 2 +- ...03\344\272\272\345\203\217.md.55937770.js" | 2 +- ...4\272\272\345\203\217.md.55937770.lean.js" | 2 +- ...24\344\272\272\345\203\217.md.852b078d.js" | 2 +- ...4\272\272\345\203\217.md.852b078d.lean.js" | 2 +- ...21\350\203\214\346\231\257.md.5e8263dc.js" | 2 +- ...0\203\214\346\231\257.md.5e8263dc.lean.js" | 2 +- ...15\351\233\250\344\270\235.md.7e47cf94.js" | 2 +- ...1\233\250\344\270\235.md.7e47cf94.lean.js" | 2 +- ...73\345\205\211\346\226\221.md.8ca08bfb.js" | 2 +- ...5\205\211\346\226\221.md.8ca08bfb.lean.js" | 2 +- ...03\344\272\272\345\203\217.md.740aab34.js" | 2 +- ...4\272\272\345\203\217.md.740aab34.lean.js" | 2 +- ...11\344\272\272\345\203\217.md.57538260.js" | 2 +- ...4\272\272\345\203\217.md.57538260.lean.js" | 2 +- ...76\346\212\200\345\267\247.md.9d941cfa.js" | 2 +- ...6\212\200\345\267\247.md.9d941cfa.lean.js" | 2 +- ...37\346\221\204\345\275\261.md.abf24cee.js" | 2 +- ...6\221\204\345\275\261.md.abf24cee.lean.js" | 2 +- ...04\346\230\237\347\251\272.md.79082746.js" | 2 +- ...6\230\237\347\251\272.md.79082746.lean.js" | 2 +- ...04\346\234\210\344\272\256.md.6fbf4de5.js" | 2 +- ...6\234\210\344\272\256.md.6fbf4de5.lean.js" | 2 +- ...04\347\203\237\350\212\261.md.bff2e64b.js" | 2 +- ...7\203\237\350\212\261.md.bff2e64b.lean.js" | 2 +- .../index.html" | 6 ++-- ...\350\203\275\344\274\230\345\214\226.html" | 8 ++--- ...\346\236\204\346\246\202\350\277\260.html" | 6 ++-- ...\346\234\254\346\246\202\345\277\265.html" | 8 ++--- ...\343\200\201\347\274\251\346\224\276.html" | 8 ++--- ...\347\216\260\346\241\206\351\200\211.html" | 30 +++++++++++++++++++ ...\351\200\211\347\211\251\344\275\223.html" | 6 ++-- ...\344\270\252\345\233\276\345\275\242.html" | 8 ++--- ...75\225\350\260\203\350\257\225Fabric.html" | 6 ++-- .../\346\246\202\350\246\201.html" | 6 ++-- ...al Knowledge for Schedule about React.html | 8 ++--- ...\347\232\204\345\214\272\345\210\253.html" | 6 ++-- ...\344\273\266\346\234\272\345\210\266.html" | 6 ++-- ...\345\270\270\346\234\272\345\210\266.html" | 6 ++-- ...\350\203\275\344\274\230\345\214\226.html" | 8 ++--- ...0\246\201\344\275\277\347\224\250JSX.html" | 6 ++-- ...\346\230\257\344\273\200\344\271\210.html" | 8 ++--- ...\346\270\262\346\237\223\347\232\204.html" | 6 ++-- ...\351\200\232\344\277\241\347\232\204.html" | 6 ++-- .../React/diff\347\256\227\346\263\225.html" | 6 ++-- ...\345\270\270\346\234\272\345\210\266.html" | 6 ++-- ...\345\274\202\346\255\245\347\232\204.html" | 8 ++--- ...273\200\344\271\210\346\230\257Fiber.html" | 6 ++-- ...256\241React\347\273\204\344\273\266.html" | 6 ++-- ...\347\232\204\351\227\256\351\242\230.html" | 6 ++-- ...\347\240\201\350\247\243\350\257\273.html" | 6 ++-- ...\350\256\241\347\220\206\345\277\265.html" | 6 ++-- ...\350\275\275\346\225\260\346\215\256.html" | 8 ++--- ...\346\215\256\346\213\206\345\210\206.html" | 8 ++--- ...\345\214\226\350\276\223\345\207\272.html" | 8 ++--- ...\347\232\204\346\265\201\347\250\213.html" | 6 ++-- ...\345\214\226\346\226\271\345\274\217.html" | 8 ++--- ...\346\225\260\346\215\256\345\272\223.html" | 8 ++--- ...\351\200\237\345\205\245\351\227\250.html" | 8 ++--- ...200\344\271\210\346\230\257langchain.html" | 6 ++-- ...47\224\250\347\232\204PromptTemplate.html" | 8 ++--- ...\347\216\207\347\237\253\346\255\243.html" | 8 ++--- ...\345\206\231\346\200\247\350\203\275.html" | 8 ++--- ...\347\224\261\344\271\246\345\206\231.html" | 8 ++--- ...\345\206\231\346\200\247\350\203\275.html" | 8 ++--- ...\345\206\231\346\200\247\350\203\275.html" | 8 ++--- ...\345\206\231\346\200\247\350\203\275.html" | 8 ++--- ...\345\206\231\346\200\247\350\203\275.html" | 8 ++--- "guide/css\347\233\270\345\205\263/BFC.html" | 6 ++-- .../css\347\233\270\345\205\263/display.html" | 8 ++--- ...\347\232\204\345\205\263\347\263\273.html" | 8 ++--- ...0\201opacity\345\214\272\345\210\253.html" | 6 ++-- .../position.html" | 8 ++--- ...\345\200\274\346\226\271\345\274\217.html" | 6 ++-- .../\346\246\202\350\246\201.html" | 6 ++-- ...\347\232\204\346\226\271\346\263\225.html" | 6 ++-- .../DOM\347\233\270\345\205\263.html" | 8 ++--- .../ES6\347\233\270\345\205\263.html" | 8 ++--- ...\350\216\267\346\234\272\345\210\266.html" | 6 ++-- ...\350\241\214\346\234\272\345\210\266.html" | 6 ++-- ...\350\257\221\346\234\272\345\210\266.html" | 6 ++-- .../Promise.html" | 8 ++--- .../index.html" | 6 ++-- ...\347\216\257\346\234\272\345\210\266.html" | 6 ++-- ...\346\224\266\346\234\272\345\210\266.html" | 6 ++-- ...\347\241\200\346\246\202\345\277\265.html" | 8 ++--- ...5\270\270\350\247\201\347\232\204API.html" | 6 ++-- .../\345\256\232\344\271\211.html" | 6 ++-- ...5\270\270\350\247\201\347\232\204API.html" | 6 ++-- ...\346\215\256\347\261\273\345\236\213.html" | 8 ++--- ...5\270\270\350\247\201\347\232\204API.html" | 6 ++-- ...\350\277\220\347\256\227\347\254\246.html" | 6 ++-- ...\350\241\250\350\276\276\345\274\217.html" | 6 ++-- ...\346\211\277\347\233\270\345\205\263.html" | 8 ++--- guide/webpack/Loader.html | 8 ++--- guide/webpack/index.html | 6 ++-- guide/webpack/mini-webpack.html | 6 ++-- ...\345\273\272\346\265\201\347\250\213.html" | 8 ++--- ...\345\235\227\350\201\224\351\202\246.html" | 8 ++--- ...\346\226\260\345\216\237\347\220\206.html" | 6 ++-- .../\346\246\202\350\246\201.html" | 6 ++-- ...\345\231\250\345\206\205\346\240\270.html" | 6 ++-- ...\345\231\250\345\256\211\345\205\250.html" | 6 ++-- ...\346\237\223\346\265\201\347\250\213.html" | 6 ++-- ...\345\231\250\347\274\223\345\255\230.html" | 8 ++--- ...\347\250\213\346\236\266\346\236\204.html" | 6 ++-- .../CDN.html" | 6 ++-- .../DNS.html" | 6 ++-- ...\347\232\204\345\214\272\345\210\253.html" | 6 ++-- .../HTTP.html" | 6 ++-- .../HTTPS.html" | 6 ++-- ...\345\272\224\346\212\245\346\226\207.html" | 6 ++-- ...\347\232\204\351\227\256\351\242\230.html" | 6 ++-- .../\346\246\202\350\246\201.html" | 6 ++-- ...\346\261\202\346\212\245\346\226\207.html" | 6 ++-- ...\345\237\237\350\257\267\346\261\202.html" | 6 ++-- hashmap.json | 2 +- index.html | 6 ++-- ...\346\204\237\345\205\211\345\272\246.html" | 6 ++-- .../\345\205\211\345\234\210.html" | 6 ++-- ...\351\227\250\351\200\237\345\272\246.html" | 6 ++-- .../index.html" | 6 ++-- ...\350\260\203\344\272\272\345\203\217.html" | 6 ++-- ...\346\257\224\344\272\272\345\203\217.html" | 6 ++-- ...\351\273\221\350\203\214\346\231\257.html" | 6 ++-- ...\346\213\215\351\233\250\344\270\235.html" | 6 ++-- ...\345\271\273\345\205\211\346\226\221.html" | 6 ++-- ...\350\260\203\344\272\272\345\203\217.html" | 6 ++-- ...\345\205\211\344\272\272\345\203\217.html" | 6 ++-- ...\345\233\276\346\212\200\345\267\247.html" | 6 ++-- ...\350\277\237\346\221\204\345\275\261.html" | 6 ++-- ...\346\221\204\346\230\237\347\251\272.html" | 6 ++-- ...\346\221\204\346\234\210\344\272\256.html" | 6 ++-- ...\346\221\204\347\203\237\350\212\261.html" | 6 ++-- 329 files changed, 604 insertions(+), 572 deletions(-) rename "assets/guide_DDD \351\242\206\345\237\237\351\251\261\345\212\250\350\256\276\350\256\241_index.md.b9fd2eae.js" => "assets/guide_DDD \351\242\206\345\237\237\351\251\261\345\212\250\350\256\276\350\256\241_index.md.d79253fb.js" (92%) rename "assets/guide_DDD \351\242\206\345\237\237\351\251\261\345\212\250\350\256\276\350\256\241_index.md.b9fd2eae.lean.js" => "assets/guide_DDD \351\242\206\345\237\237\351\251\261\345\212\250\350\256\276\350\256\241_index.md.d79253fb.lean.js" (92%) create mode 100644 "assets/guide_Fabric.js_Fabric \344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.md.9993b631.js" create mode 100644 "assets/guide_Fabric.js_Fabric \344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.md.9993b631.lean.js" rename "assets/guide_Fabric.js_\345\206\205\351\203\250\347\273\223\346\236\204\346\246\202\350\277\260.md.be67fc71.js" => "assets/guide_Fabric.js_\345\206\205\351\203\250\347\273\223\346\236\204\346\246\202\350\277\260.md.5690116d.js" (98%) rename "assets/guide_Fabric.js_\345\206\205\351\203\250\347\273\223\346\236\204\346\246\202\350\277\260.md.be67fc71.lean.js" => "assets/guide_Fabric.js_\345\206\205\351\203\250\347\273\223\346\236\204\346\246\202\350\277\260.md.5690116d.lean.js" (90%) rename "assets/guide_Fabric.js_\345\237\272\346\234\254\346\246\202\345\277\265.md.711f70e5.js" => "assets/guide_Fabric.js_\345\237\272\346\234\254\346\246\202\345\277\265.md.302ccee7.js" (98%) rename "assets/guide_Fabric.js_\345\237\272\346\234\254\346\246\202\345\277\265.md.711f70e5.lean.js" => "assets/guide_Fabric.js_\345\237\272\346\234\254\346\246\202\345\277\265.md.302ccee7.lean.js" (89%) rename "assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\345\205\203\347\264\240\347\232\204\345\271\263\347\247\273\343\200\201\346\227\213\350\275\254\343\200\201\347\274\251\346\224\276.md.493fc2f7.js" => "assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\345\205\203\347\264\240\347\232\204\345\271\263\347\247\273\343\200\201\346\227\213\350\275\254\343\200\201\347\274\251\346\224\276.md.140dd9a6.js" (99%) rename "assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\345\205\203\347\264\240\347\232\204\345\271\263\347\247\273\343\200\201\346\227\213\350\275\254\343\200\201\347\274\251\346\224\276.md.493fc2f7.lean.js" => "assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\345\205\203\347\264\240\347\232\204\345\271\263\347\247\273\343\200\201\346\227\213\350\275\254\343\200\201\347\274\251\346\224\276.md.140dd9a6.lean.js" (98%) create mode 100644 "assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\346\241\206\351\200\211.md.382e7c32.js" create mode 100644 "assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\346\241\206\351\200\211.md.382e7c32.lean.js" delete mode 100644 "assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\347\202\271\351\200\211\347\211\251\344\275\223.md.01b61e2b.js" create mode 100644 "assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\347\202\271\351\200\211\347\211\251\344\275\223.md.4b2066fc.js" rename "assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\347\202\271\351\200\211\347\211\251\344\275\223.md.01b61e2b.lean.js" => "assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\347\202\271\351\200\211\347\211\251\344\275\223.md.4b2066fc.lean.js" (75%) delete mode 100644 "assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\347\273\204\351\200\211.md.3e1b07e1.js" delete mode 100644 "assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\347\273\204\351\200\211.md.3e1b07e1.lean.js" rename "assets/guide_Fabric.js_\345\246\202\344\275\225\347\273\230\345\210\266\344\270\200\344\270\252\345\233\276\345\275\242.md.78353ed8.js" => "assets/guide_Fabric.js_\345\246\202\344\275\225\347\273\230\345\210\266\344\270\200\344\270\252\345\233\276\345\275\242.md.fc479c77.js" (99%) rename "assets/guide_Fabric.js_\345\246\202\344\275\225\347\273\230\345\210\266\344\270\200\344\270\252\345\233\276\345\275\242.md.78353ed8.lean.js" => "assets/guide_Fabric.js_\345\246\202\344\275\225\347\273\230\345\210\266\344\270\200\344\270\252\345\233\276\345\275\242.md.fc479c77.lean.js" (91%) rename "assets/guide_Fabric.js_\345\246\202\344\275\225\350\260\203\350\257\225Fabric.md.e3e1288a.js" => "assets/guide_Fabric.js_\345\246\202\344\275\225\350\260\203\350\257\225Fabric.md.b88e0c09.js" (99%) rename "assets/guide_Fabric.js_\345\246\202\344\275\225\350\260\203\350\257\225Fabric.md.e3e1288a.lean.js" => "assets/guide_Fabric.js_\345\246\202\344\275\225\350\260\203\350\257\225Fabric.md.b88e0c09.lean.js" (88%) rename "assets/guide_Node\347\233\270\345\205\263_\346\246\202\350\246\201.md.6d2f3b80.js" => "assets/guide_Node\347\233\270\345\205\263_\346\246\202\350\246\201.md.5e89b8cf.js" (84%) rename "assets/guide_Node\347\233\270\345\205\263_\346\246\202\350\246\201.md.6d2f3b80.lean.js" => "assets/guide_Node\347\233\270\345\205\263_\346\246\202\350\246\201.md.5e89b8cf.lean.js" (84%) rename assets/{guide_React_Essential Knowledge for Schedule about React.md.5abf05d0.js => guide_React_Essential Knowledge for Schedule about React.md.8a28d74e.js} (99%) rename assets/{guide_React_Essential Knowledge for Schedule about React.md.5abf05d0.lean.js => guide_React_Essential Knowledge for Schedule about React.md.8a28d74e.lean.js} (93%) rename "assets/guide_React_React\344\270\216Vue\347\232\204\345\214\272\345\210\253.md.aefdc566.js" => "assets/guide_React_React\344\270\216Vue\347\232\204\345\214\272\345\210\253.md.11b5926a.js" (85%) rename "assets/guide_React_React\344\270\216Vue\347\232\204\345\214\272\345\210\253.md.aefdc566.lean.js" => "assets/guide_React_React\344\270\216Vue\347\232\204\345\214\272\345\210\253.md.11b5926a.lean.js" (85%) rename "assets/guide_React_React\344\270\255\347\232\204\344\272\213\344\273\266\346\234\272\345\210\266.md.4c13470e.js" => "assets/guide_React_React\344\270\255\347\232\204\344\272\213\344\273\266\346\234\272\345\210\266.md.d11dda49.js" (95%) rename "assets/guide_React_React\344\270\255\347\232\204\344\272\213\344\273\266\346\234\272\345\210\266.md.4c13470e.lean.js" => "assets/guide_React_React\344\270\255\347\232\204\344\272\213\344\273\266\346\234\272\345\210\266.md.d11dda49.lean.js" (86%) rename "assets/guide_React_React\344\270\255\347\232\204\345\274\202\345\270\270\346\234\272\345\210\266.md.b181d518.js" => "assets/guide_React_React\344\270\255\347\232\204\345\274\202\345\270\270\346\234\272\345\210\266.md.ac78472e.js" (85%) rename "assets/guide_React_React\344\270\255\347\232\204\345\274\202\345\270\270\346\234\272\345\210\266.md.b181d518.lean.js" => "assets/guide_React_React\344\270\255\347\232\204\345\274\202\345\270\270\346\234\272\345\210\266.md.ac78472e.lean.js" (85%) rename "assets/guide_React_React\344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.md.6ad45a79.js" => "assets/guide_React_React\344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.md.780419b2.js" (99%) rename "assets/guide_React_React\344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.md.6ad45a79.lean.js" => "assets/guide_React_React\344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.md.780419b2.lean.js" (87%) rename "assets/guide_React_React\344\270\272\344\273\200\344\271\210\350\246\201\344\275\277\347\224\250JSX.md.62407e1e.js" => "assets/guide_React_React\344\270\272\344\273\200\344\271\210\350\246\201\344\275\277\347\224\250JSX.md.40e22e85.js" (94%) rename "assets/guide_React_React\344\270\272\344\273\200\344\271\210\350\246\201\344\275\277\347\224\250JSX.md.62407e1e.lean.js" => "assets/guide_React_React\344\270\272\344\273\200\344\271\210\350\246\201\344\275\277\347\224\250JSX.md.40e22e85.lean.js" (94%) rename "assets/guide_React_React\346\230\257\344\273\200\344\271\210.md.471a44bf.js" => "assets/guide_React_React\346\230\257\344\273\200\344\271\210.md.ee0c3664.js" (98%) rename "assets/guide_React_React\346\230\257\344\273\200\344\271\210.md.471a44bf.lean.js" => "assets/guide_React_React\346\230\257\344\273\200\344\271\210.md.ee0c3664.lean.js" (86%) rename "assets/guide_React_React\346\230\257\345\246\202\344\275\225\346\270\262\346\237\223\347\232\204.md.ee09ffde.js" => "assets/guide_React_React\346\230\257\345\246\202\344\275\225\346\270\262\346\237\223\347\232\204.md.f058feee.js" (99%) rename "assets/guide_React_React\346\230\257\345\246\202\344\275\225\346\270\262\346\237\223\347\232\204.md.ee09ffde.lean.js" => "assets/guide_React_React\346\230\257\345\246\202\344\275\225\346\270\262\346\237\223\347\232\204.md.f058feee.lean.js" (94%) rename "assets/guide_React_React\347\273\204\344\273\266\346\230\257\345\246\202\344\275\225\351\200\232\344\277\241\347\232\204.md.a82d5c10.js" => "assets/guide_React_React\347\273\204\344\273\266\346\230\257\345\246\202\344\275\225\351\200\232\344\277\241\347\232\204.md.a0c59c32.js" (91%) rename "assets/guide_React_React\347\273\204\344\273\266\346\230\257\345\246\202\344\275\225\351\200\232\344\277\241\347\232\204.md.a82d5c10.lean.js" => "assets/guide_React_React\347\273\204\344\273\266\346\230\257\345\246\202\344\275\225\351\200\232\344\277\241\347\232\204.md.a0c59c32.lean.js" (91%) rename "assets/guide_React_diff\347\256\227\346\263\225.md.e12213ca.js" => "assets/guide_React_diff\347\256\227\346\263\225.md.4f2b0e59.js" (98%) rename "assets/guide_React_diff\347\256\227\346\263\225.md.e12213ca.lean.js" => "assets/guide_React_diff\347\256\227\346\263\225.md.4f2b0e59.lean.js" (86%) rename "assets/guide_React_react\345\274\202\345\270\270\346\234\272\345\210\266.md.95438d51.js" => "assets/guide_React_react\345\274\202\345\270\270\346\234\272\345\210\266.md.39386d10.js" (84%) rename "assets/guide_React_react\345\274\202\345\270\270\346\234\272\345\210\266.md.95438d51.lean.js" => "assets/guide_React_react\345\274\202\345\270\270\346\234\272\345\210\266.md.39386d10.lean.js" (84%) rename "assets/guide_React_setState\346\230\257\345\220\214\346\255\245\350\277\230\346\230\257\345\274\202\346\255\245\347\232\204.md.d2fb1cb9.js" => "assets/guide_React_setState\346\230\257\345\220\214\346\255\245\350\277\230\346\230\257\345\274\202\346\255\245\347\232\204.md.10d004c3.js" (99%) rename "assets/guide_React_setState\346\230\257\345\220\214\346\255\245\350\277\230\346\230\257\345\274\202\346\255\245\347\232\204.md.d2fb1cb9.lean.js" => "assets/guide_React_setState\346\230\257\345\220\214\346\255\245\350\277\230\346\230\257\345\274\202\346\255\245\347\232\204.md.10d004c3.lean.js" (88%) rename "assets/guide_React_\344\273\200\344\271\210\346\230\257Fiber.md.29e5a130.js" => "assets/guide_React_\344\273\200\344\271\210\346\230\257Fiber.md.9a75b68a.js" (96%) rename "assets/guide_React_\344\273\200\344\271\210\346\230\257Fiber.md.29e5a130.lean.js" => "assets/guide_React_\344\273\200\344\271\210\346\230\257Fiber.md.9a75b68a.lean.js" (86%) rename "assets/guide_React_\345\246\202\344\275\225\350\256\276\350\256\241React\347\273\204\344\273\266.md.7bfc2acc.js" => "assets/guide_React_\345\246\202\344\275\225\350\256\276\350\256\241React\347\273\204\344\273\266.md.b6076bcd.js" (95%) rename "assets/guide_React_\345\246\202\344\275\225\350\256\276\350\256\241React\347\273\204\344\273\266.md.7bfc2acc.lean.js" => "assets/guide_React_\345\246\202\344\275\225\350\256\276\350\256\241React\347\273\204\344\273\266.md.b6076bcd.lean.js" (88%) rename "assets/guide_React_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.681d5e24.js" => "assets/guide_React_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.5afef26f.js" (91%) rename "assets/guide_React_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.681d5e24.lean.js" => "assets/guide_React_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.5afef26f.lean.js" (91%) rename "assets/guide_Redux_\346\272\220\347\240\201\350\247\243\350\257\273.md.a19e3f7c.js" => "assets/guide_Redux_\346\272\220\347\240\201\350\247\243\350\257\273.md.e0e871fd.js" (89%) rename "assets/guide_Redux_\346\272\220\347\240\201\350\247\243\350\257\273.md.a19e3f7c.lean.js" => "assets/guide_Redux_\346\272\220\347\240\201\350\247\243\350\257\273.md.e0e871fd.lean.js" (89%) rename "assets/guide_Redux_\350\256\276\350\256\241\347\220\206\345\277\265.md.c290b375.lean.js" => "assets/guide_Redux_\350\256\276\350\256\241\347\220\206\345\277\265.md.00f4264a.js" (89%) rename "assets/guide_Redux_\350\256\276\350\256\241\347\220\206\345\277\265.md.c290b375.js" => "assets/guide_Redux_\350\256\276\350\256\241\347\220\206\345\277\265.md.00f4264a.lean.js" (89%) rename "assets/guide_ai_Embedding\344\271\213\345\212\240\350\275\275\346\225\260\346\215\256.md.a848af48.js" => "assets/guide_ai_Embedding\344\271\213\345\212\240\350\275\275\346\225\260\346\215\256.md.6565d23d.js" (99%) rename "assets/guide_ai_Embedding\344\271\213\345\212\240\350\275\275\346\225\260\346\215\256.md.a848af48.lean.js" => "assets/guide_ai_Embedding\344\271\213\345\212\240\350\275\275\346\225\260\346\215\256.md.6565d23d.lean.js" (89%) rename "assets/guide_ai_Embedding\344\271\213\345\244\247\350\247\204\346\250\241\346\225\260\346\215\256\346\213\206\345\210\206.md.8b24253c.js" => "assets/guide_ai_Embedding\344\271\213\345\244\247\350\247\204\346\250\241\346\225\260\346\215\256\346\213\206\345\210\206.md.64da80d7.js" (99%) rename "assets/guide_ai_Embedding\344\271\213\345\244\247\350\247\204\346\250\241\346\225\260\346\215\256\346\213\206\345\210\206.md.8b24253c.lean.js" => "assets/guide_ai_Embedding\344\271\213\345\244\247\350\247\204\346\250\241\346\225\260\346\215\256\346\213\206\345\210\206.md.64da80d7.lean.js" (88%) rename "assets/guide_ai_OutputParser\346\236\204\345\273\272\346\240\274\345\274\217\345\214\226\350\276\223\345\207\272.md.d681019e.js" => "assets/guide_ai_OutputParser\346\236\204\345\273\272\346\240\274\345\274\217\345\214\226\350\276\223\345\207\272.md.1a2532c7.js" (99%) rename "assets/guide_ai_OutputParser\346\236\204\345\273\272\346\240\274\345\274\217\345\214\226\350\276\223\345\207\272.md.d681019e.lean.js" => "assets/guide_ai_OutputParser\346\236\204\345\273\272\346\240\274\345\274\217\345\214\226\350\276\223\345\207\272.md.1a2532c7.lean.js" (88%) rename "assets/guide_ai_RAG \346\243\200\347\264\242\345\242\236\345\274\272\347\224\237\346\210\220\347\232\204\346\265\201\347\250\213.md.c97a9cbd.js" => "assets/guide_ai_RAG \346\243\200\347\264\242\345\242\236\345\274\272\347\224\237\346\210\220\347\232\204\346\265\201\347\250\213.md.58fac56a.js" (98%) rename "assets/guide_ai_RAG \346\243\200\347\264\242\345\242\236\345\274\272\347\224\237\346\210\220\347\232\204\346\265\201\347\250\213.md.c97a9cbd.lean.js" => "assets/guide_ai_RAG \346\243\200\347\264\242\345\242\236\345\274\272\347\224\237\346\210\220\347\232\204\346\265\201\347\250\213.md.58fac56a.lean.js" (88%) rename "assets/guide_ai_Retriever \345\270\270\350\247\201\347\232\204\344\274\230\345\214\226\346\226\271\345\274\217.md.86e88704.js" => "assets/guide_ai_Retriever \345\270\270\350\247\201\347\232\204\344\274\230\345\214\226\346\226\271\345\274\217.md.a17d2dce.js" (99%) rename "assets/guide_ai_Retriever \345\270\270\350\247\201\347\232\204\344\274\230\345\214\226\346\226\271\345\274\217.md.86e88704.lean.js" => "assets/guide_ai_Retriever \345\270\270\350\247\201\347\232\204\344\274\230\345\214\226\346\226\271\345\274\217.md.a17d2dce.lean.js" (89%) rename "assets/guide_ai_Retriever\344\271\213\345\220\221\351\207\217\346\225\260\346\215\256\345\272\223.md.6b2ddb25.js" => "assets/guide_ai_Retriever\344\271\213\345\220\221\351\207\217\346\225\260\346\215\256\345\272\223.md.0af240f5.js" (99%) rename "assets/guide_ai_Retriever\344\271\213\345\220\221\351\207\217\346\225\260\346\215\256\345\272\223.md.6b2ddb25.lean.js" => "assets/guide_ai_Retriever\344\271\213\345\220\221\351\207\217\346\225\260\346\215\256\345\272\223.md.0af240f5.lean.js" (88%) rename "assets/guide_ai_langchain\345\277\253\351\200\237\345\205\245\351\227\250.md.ab114f06.js" => "assets/guide_ai_langchain\345\277\253\351\200\237\345\205\245\351\227\250.md.e65bc2d1.js" (99%) rename "assets/guide_ai_langchain\345\277\253\351\200\237\345\205\245\351\227\250.md.ab114f06.lean.js" => "assets/guide_ai_langchain\345\277\253\351\200\237\345\205\245\351\227\250.md.e65bc2d1.lean.js" (89%) rename "assets/guide_ai_\344\273\200\344\271\210\346\230\257langchain.md.a8c31c8d.js" => "assets/guide_ai_\344\273\200\344\271\210\346\230\257langchain.md.487089c2.js" (98%) rename "assets/guide_ai_\344\273\200\344\271\210\346\230\257langchain.md.a8c31c8d.lean.js" => "assets/guide_ai_\344\273\200\344\271\210\346\230\257langchain.md.487089c2.lean.js" (86%) rename "assets/guide_ai_\346\236\204\345\273\272\345\217\257\345\244\215\347\224\250\347\232\204PromptTemplate.md.25d3a453.js" => "assets/guide_ai_\346\236\204\345\273\272\345\217\257\345\244\215\347\224\250\347\232\204PromptTemplate.md.2d5ec969.js" (99%) rename "assets/guide_ai_\346\236\204\345\273\272\345\217\257\345\244\215\347\224\250\347\232\204PromptTemplate.md.25d3a453.lean.js" => "assets/guide_ai_\346\236\204\345\273\272\345\217\257\345\244\215\347\224\250\347\232\204PromptTemplate.md.2d5ec969.lean.js" (87%) rename "assets/guide_canvas_Canvas\345\260\272\345\257\270\345\217\212\345\210\206\350\276\250\347\216\207\347\237\253\346\255\243.md.e573b0a5.js" => "assets/guide_canvas_Canvas\345\260\272\345\257\270\345\217\212\345\210\206\350\276\250\347\216\207\347\237\253\346\255\243.md.6671b487.js" (99%) rename "assets/guide_canvas_Canvas\345\260\272\345\257\270\345\217\212\345\210\206\350\276\250\347\216\207\347\237\253\346\255\243.md.e573b0a5.lean.js" => "assets/guide_canvas_Canvas\345\260\272\345\257\270\345\217\212\345\210\206\350\276\250\347\216\207\347\237\253\346\255\243.md.6671b487.lean.js" (90%) rename "assets/guide_canvas_\345\217\257\350\247\206\345\214\272\345\237\237\345\206\205\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.e953eb67.js" => "assets/guide_canvas_\345\217\257\350\247\206\345\214\272\345\237\237\345\206\205\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.1828bbfc.js" (99%) rename "assets/guide_canvas_\345\217\257\350\247\206\345\214\272\345\237\237\345\206\205\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.e953eb67.lean.js" => "assets/guide_canvas_\345\217\257\350\247\206\345\214\272\345\237\237\345\206\205\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.1828bbfc.lean.js" (92%) rename "assets/guide_canvas_\345\246\202\344\275\225\345\234\250Canvas\347\224\273\346\235\277\344\270\212\350\207\252\347\224\261\344\271\246\345\206\231.md.a3f6d57a.js" => "assets/guide_canvas_\345\246\202\344\275\225\345\234\250Canvas\347\224\273\346\235\277\344\270\212\350\207\252\347\224\261\344\271\246\345\206\231.md.a9d3a092.js" (99%) rename "assets/guide_canvas_\345\246\202\344\275\225\345\234\250Canvas\347\224\273\346\235\277\344\270\212\350\207\252\347\224\261\344\271\246\345\206\231.md.a3f6d57a.lean.js" => "assets/guide_canvas_\345\246\202\344\275\225\345\234\250Canvas\347\224\273\346\235\277\344\270\212\350\207\252\347\224\261\344\271\246\345\206\231.md.a9d3a092.lean.js" (91%) rename "assets/guide_canvas_\351\200\232\350\277\207 OffscreenCanvas _ Worker \346\217\220\351\253\230\344\271\246\345\206\231\346\200\247\350\203\275.md.09ce6c22.js" => "assets/guide_canvas_\351\200\232\350\277\207 OffscreenCanvas _ Worker \346\217\220\351\253\230\344\271\246\345\206\231\346\200\247\350\203\275.md.74068538.js" (99%) rename "assets/guide_canvas_\351\200\232\350\277\207 OffscreenCanvas _ Worker \346\217\220\351\253\230\344\271\246\345\206\231\346\200\247\350\203\275.md.09ce6c22.lean.js" => "assets/guide_canvas_\351\200\232\350\277\207 OffscreenCanvas _ Worker \346\217\220\351\253\230\344\271\246\345\206\231\346\200\247\350\203\275.md.74068538.lean.js" (90%) rename "assets/guide_canvas_\351\200\232\350\277\207\344\270\212\344\270\213\345\210\206\345\261\202\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.81bf21d6.js" => "assets/guide_canvas_\351\200\232\350\277\207\344\270\212\344\270\213\345\210\206\345\261\202\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.fc9344af.js" (99%) rename "assets/guide_canvas_\351\200\232\350\277\207\344\270\212\344\270\213\345\210\206\345\261\202\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.81bf21d6.lean.js" => "assets/guide_canvas_\351\200\232\350\277\207\344\270\212\344\270\213\345\210\206\345\261\202\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.fc9344af.lean.js" (80%) rename "assets/guide_canvas_\351\200\232\350\277\207\347\246\273\345\261\217\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.5cb7017a.js" => "assets/guide_canvas_\351\200\232\350\277\207\347\246\273\345\261\217\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.86c812db.js" (99%) rename "assets/guide_canvas_\351\200\232\350\277\207\347\246\273\345\261\217\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.5cb7017a.lean.js" => "assets/guide_canvas_\351\200\232\350\277\207\347\246\273\345\261\217\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.86c812db.lean.js" (89%) rename "assets/guide_canvas_\351\200\232\350\277\207\350\264\235\345\241\236\345\260\224\346\233\262\347\272\277\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.61a93128.js" => "assets/guide_canvas_\351\200\232\350\277\207\350\264\235\345\241\236\345\260\224\346\233\262\347\272\277\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.1acbc131.js" (99%) rename "assets/guide_canvas_\351\200\232\350\277\207\350\264\235\345\241\236\345\260\224\346\233\262\347\272\277\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.61a93128.lean.js" => "assets/guide_canvas_\351\200\232\350\277\207\350\264\235\345\241\236\345\260\224\346\233\262\347\272\277\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.1acbc131.lean.js" (90%) rename "assets/guide_css\347\233\270\345\205\263_BFC.md.97daca69.js" => "assets/guide_css\347\233\270\345\205\263_BFC.md.fbec5d1d.js" (96%) rename "assets/guide_css\347\233\270\345\205\263_BFC.md.97daca69.lean.js" => "assets/guide_css\347\233\270\345\205\263_BFC.md.fbec5d1d.lean.js" (85%) rename "assets/guide_css\347\233\270\345\205\263_display.md.3c174711.js" => "assets/guide_css\347\233\270\345\205\263_display.md.6d34245c.js" (99%) rename "assets/guide_css\347\233\270\345\205\263_display.md.3c174711.lean.js" => "assets/guide_css\347\233\270\345\205\263_display.md.6d34245c.lean.js" (85%) rename "assets/guide_css\347\233\270\345\205\263_display\343\200\201float\343\200\201position\347\232\204\345\205\263\347\263\273.md.0e4b5524.js" => "assets/guide_css\347\233\270\345\205\263_display\343\200\201float\343\200\201position\347\232\204\345\205\263\347\263\273.md.f469d683.js" (99%) rename "assets/guide_css\347\233\270\345\205\263_display\343\200\201float\343\200\201position\347\232\204\345\205\263\347\263\273.md.0e4b5524.lean.js" => "assets/guide_css\347\233\270\345\205\263_display\343\200\201float\343\200\201position\347\232\204\345\205\263\347\263\273.md.f469d683.lean.js" (88%) rename "assets/guide_css\347\233\270\345\205\263_display\343\200\201visibility\343\200\201opacity\345\214\272\345\210\253.md.66ff32b5.js" => "assets/guide_css\347\233\270\345\205\263_display\343\200\201visibility\343\200\201opacity\345\214\272\345\210\253.md.2678884b.js" (96%) rename "assets/guide_css\347\233\270\345\205\263_display\343\200\201visibility\343\200\201opacity\345\214\272\345\210\253.md.66ff32b5.lean.js" => "assets/guide_css\347\233\270\345\205\263_display\343\200\201visibility\343\200\201opacity\345\214\272\345\210\253.md.2678884b.lean.js" (88%) rename "assets/guide_css\347\233\270\345\205\263_position.md.839d8549.js" => "assets/guide_css\347\233\270\345\205\263_position.md.b4f4db42.js" (99%) rename "assets/guide_css\347\233\270\345\205\263_position.md.839d8549.lean.js" => "assets/guide_css\347\233\270\345\205\263_position.md.b4f4db42.lean.js" (99%) rename "assets/guide_css\347\233\270\345\205\263_\345\257\271 line-height \347\232\204\347\220\206\350\247\243\345\217\212\345\205\266\350\265\213\345\200\274\346\226\271\345\274\217.md.838b7cb9.js" => "assets/guide_css\347\233\270\345\205\263_\345\257\271 line-height \347\232\204\347\220\206\350\247\243\345\217\212\345\205\266\350\265\213\345\200\274\346\226\271\345\274\217.md.7f0ca022.js" (97%) rename "assets/guide_css\347\233\270\345\205\263_\345\257\271 line-height \347\232\204\347\220\206\350\247\243\345\217\212\345\205\266\350\265\213\345\200\274\346\226\271\345\274\217.md.838b7cb9.lean.js" => "assets/guide_css\347\233\270\345\205\263_\345\257\271 line-height \347\232\204\347\220\206\350\247\243\345\217\212\345\205\266\350\265\213\345\200\274\346\226\271\345\274\217.md.7f0ca022.lean.js" (89%) rename "assets/guide_css\347\233\270\345\205\263_\346\246\202\350\246\201.md.6367bf5b.js" => "assets/guide_css\347\233\270\345\205\263_\346\246\202\350\246\201.md.9e3e6a15.js" (87%) rename "assets/guide_css\347\233\270\345\205\263_\346\246\202\350\246\201.md.6367bf5b.lean.js" => "assets/guide_css\347\233\270\345\205\263_\346\246\202\350\246\201.md.9e3e6a15.lean.js" (87%) rename "assets/guide_css\347\233\270\345\205\263_\351\232\220\350\227\217\345\205\203\347\264\240\347\232\204\346\226\271\346\263\225.md.6677bfcf.js" => "assets/guide_css\347\233\270\345\205\263_\351\232\220\350\227\217\345\205\203\347\264\240\347\232\204\346\226\271\346\263\225.md.3f129790.js" (94%) rename "assets/guide_css\347\233\270\345\205\263_\351\232\220\350\227\217\345\205\203\347\264\240\347\232\204\346\226\271\346\263\225.md.6677bfcf.lean.js" => "assets/guide_css\347\233\270\345\205\263_\351\232\220\350\227\217\345\205\203\347\264\240\347\232\204\346\226\271\346\263\225.md.3f129790.lean.js" (87%) rename "assets/guide_javaScript\347\233\270\345\205\263_DOM\347\233\270\345\205\263.md.7690a9dd.js" => "assets/guide_javaScript\347\233\270\345\205\263_DOM\347\233\270\345\205\263.md.b5647b3c.js" (99%) rename "assets/guide_javaScript\347\233\270\345\205\263_DOM\347\233\270\345\205\263.md.7690a9dd.lean.js" => "assets/guide_javaScript\347\233\270\345\205\263_DOM\347\233\270\345\205\263.md.b5647b3c.lean.js" (86%) rename "assets/guide_javaScript\347\233\270\345\205\263_ES6\347\233\270\345\205\263.md.4615c5ea.js" => "assets/guide_javaScript\347\233\270\345\205\263_ES6\347\233\270\345\205\263.md.a46321f6.js" (99%) rename "assets/guide_javaScript\347\233\270\345\205\263_ES6\347\233\270\345\205\263.md.4615c5ea.lean.js" => "assets/guide_javaScript\347\233\270\345\205\263_ES6\347\233\270\345\205\263.md.a46321f6.lean.js" (87%) rename "assets/guide_javaScript\347\233\270\345\205\263_JS\345\274\202\345\270\270\346\215\225\350\216\267\346\234\272\345\210\266.md.b6816c6f.js" => "assets/guide_javaScript\347\233\270\345\205\263_JS\345\274\202\345\270\270\346\215\225\350\216\267\346\234\272\345\210\266.md.bd59bbe3.js" (86%) rename "assets/guide_javaScript\347\233\270\345\205\263_JS\345\274\202\345\270\270\346\215\225\350\216\267\346\234\272\345\210\266.md.b6816c6f.lean.js" => "assets/guide_javaScript\347\233\270\345\205\263_JS\345\274\202\345\270\270\346\215\225\350\216\267\346\234\272\345\210\266.md.bd59bbe3.lean.js" (86%) rename "assets/guide_javaScript\347\233\270\345\205\263_JavaScript\346\211\247\350\241\214\346\234\272\345\210\266.md.f9a311e1.js" => "assets/guide_javaScript\347\233\270\345\205\263_JavaScript\346\211\247\350\241\214\346\234\272\345\210\266.md.b95eb7c2.js" (92%) rename "assets/guide_javaScript\347\233\270\345\205\263_JavaScript\346\211\247\350\241\214\346\234\272\345\210\266.md.f9a311e1.lean.js" => "assets/guide_javaScript\347\233\270\345\205\263_JavaScript\346\211\247\350\241\214\346\234\272\345\210\266.md.b95eb7c2.lean.js" (92%) rename "assets/guide_javaScript\347\233\270\345\205\263_JavaScript\347\274\226\350\257\221\346\234\272\345\210\266.md.4fefe971.lean.js" => "assets/guide_javaScript\347\233\270\345\205\263_JavaScript\347\274\226\350\257\221\346\234\272\345\210\266.md.4e21f748.js" (92%) rename "assets/guide_javaScript\347\233\270\345\205\263_JavaScript\347\274\226\350\257\221\346\234\272\345\210\266.md.4fefe971.js" => "assets/guide_javaScript\347\233\270\345\205\263_JavaScript\347\274\226\350\257\221\346\234\272\345\210\266.md.4e21f748.lean.js" (92%) rename "assets/guide_javaScript\347\233\270\345\205\263_Promise.md.54f111a9.js" => "assets/guide_javaScript\347\233\270\345\205\263_Promise.md.9dcd3d04.js" (99%) rename "assets/guide_javaScript\347\233\270\345\205\263_Promise.md.54f111a9.lean.js" => "assets/guide_javaScript\347\233\270\345\205\263_Promise.md.9dcd3d04.lean.js" (90%) rename "assets/guide_javaScript\347\233\270\345\205\263_index.md.41467648.js" => "assets/guide_javaScript\347\233\270\345\205\263_index.md.6fa036d3.js" (88%) rename "assets/guide_javaScript\347\233\270\345\205\263_index.md.41467648.lean.js" => "assets/guide_javaScript\347\233\270\345\205\263_index.md.6fa036d3.lean.js" (88%) rename "assets/guide_javaScript\347\233\270\345\205\263_\344\272\213\344\273\266\345\276\252\347\216\257\346\234\272\345\210\266.md.0b3536a3.js" => "assets/guide_javaScript\347\233\270\345\205\263_\344\272\213\344\273\266\345\276\252\347\216\257\346\234\272\345\210\266.md.d507e357.js" (93%) rename "assets/guide_javaScript\347\233\270\345\205\263_\344\272\213\344\273\266\345\276\252\347\216\257\346\234\272\345\210\266.md.0b3536a3.lean.js" => "assets/guide_javaScript\347\233\270\345\205\263_\344\272\213\344\273\266\345\276\252\347\216\257\346\234\272\345\210\266.md.d507e357.lean.js" (93%) rename "assets/guide_javaScript\347\233\270\345\205\263_\345\236\203\345\234\276\345\233\236\346\224\266\346\234\272\345\210\266.md.7fe4a5e0.js" => "assets/guide_javaScript\347\233\270\345\205\263_\345\236\203\345\234\276\345\233\236\346\224\266\346\234\272\345\210\266.md.41875062.js" (95%) rename "assets/guide_javaScript\347\233\270\345\205\263_\345\236\203\345\234\276\345\233\236\346\224\266\346\234\272\345\210\266.md.7fe4a5e0.lean.js" => "assets/guide_javaScript\347\233\270\345\205\263_\345\236\203\345\234\276\345\233\236\346\224\266\346\234\272\345\210\266.md.41875062.lean.js" (95%) rename "assets/guide_javaScript\347\233\270\345\205\263_\345\237\272\347\241\200\346\246\202\345\277\265.md.3cedefb2.js" => "assets/guide_javaScript\347\233\270\345\205\263_\345\237\272\347\241\200\346\246\202\345\277\265.md.53cfc405.js" (99%) rename "assets/guide_javaScript\347\233\270\345\205\263_\345\237\272\347\241\200\346\246\202\345\277\265.md.3cedefb2.lean.js" => "assets/guide_javaScript\347\233\270\345\205\263_\345\237\272\347\241\200\346\246\202\345\277\265.md.53cfc405.lean.js" (88%) rename "assets/guide_javaScript\347\233\270\345\205\263_\345\255\227\347\254\246\344\270\262\345\270\270\350\247\201\347\232\204API.md.c4f598fd.js" => "assets/guide_javaScript\347\233\270\345\205\263_\345\255\227\347\254\246\344\270\262\345\270\270\350\247\201\347\232\204API.md.699074e1.js" (86%) rename "assets/guide_javaScript\347\233\270\345\205\263_\345\255\227\347\254\246\344\270\262\345\270\270\350\247\201\347\232\204API.md.c4f598fd.lean.js" => "assets/guide_javaScript\347\233\270\345\205\263_\345\255\227\347\254\246\344\270\262\345\270\270\350\247\201\347\232\204API.md.699074e1.lean.js" (86%) rename "assets/guide_javaScript\347\233\270\345\205\263_\345\256\232\344\271\211.md.05754f7e.js" => "assets/guide_javaScript\347\233\270\345\205\263_\345\256\232\344\271\211.md.ea2dc8ec.js" (94%) rename "assets/guide_javaScript\347\233\270\345\205\263_\345\256\232\344\271\211.md.05754f7e.lean.js" => "assets/guide_javaScript\347\233\270\345\205\263_\345\256\232\344\271\211.md.ea2dc8ec.lean.js" (94%) rename "assets/guide_javaScript\347\233\270\345\205\263_\345\257\271\350\261\241\345\270\270\350\247\201\347\232\204API.md.c8d02937.js" => "assets/guide_javaScript\347\233\270\345\205\263_\345\257\271\350\261\241\345\270\270\350\247\201\347\232\204API.md.6057b79b.js" (86%) rename "assets/guide_javaScript\347\233\270\345\205\263_\345\257\271\350\261\241\345\270\270\350\247\201\347\232\204API.md.c8d02937.lean.js" => "assets/guide_javaScript\347\233\270\345\205\263_\345\257\271\350\261\241\345\270\270\350\247\201\347\232\204API.md.6057b79b.lean.js" (86%) rename "assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\346\215\256\347\261\273\345\236\213.md.b73ffe80.js" => "assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\346\215\256\347\261\273\345\236\213.md.77b639a5.js" (99%) rename "assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\346\215\256\347\261\273\345\236\213.md.b73ffe80.lean.js" => "assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\346\215\256\347\261\273\345\236\213.md.77b639a5.lean.js" (87%) rename "assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\347\273\204\345\270\270\350\247\201\347\232\204API.md.9011f496.js" => "assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\347\273\204\345\270\270\350\247\201\347\232\204API.md.452ffdf7.js" (86%) rename "assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\347\273\204\345\270\270\350\247\201\347\232\204API.md.9011f496.lean.js" => "assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\347\273\204\345\270\270\350\247\201\347\232\204API.md.452ffdf7.lean.js" (86%) rename "assets/guide_javaScript\347\233\270\345\205\263_\346\226\260\347\232\204\350\277\220\347\256\227\347\254\246.md.c349c9ff.js" => "assets/guide_javaScript\347\233\270\345\205\263_\346\226\260\347\232\204\350\277\220\347\256\227\347\254\246.md.c8bd4cf5.js" (85%) rename "assets/guide_javaScript\347\233\270\345\205\263_\346\226\260\347\232\204\350\277\220\347\256\227\347\254\246.md.c349c9ff.lean.js" => "assets/guide_javaScript\347\233\270\345\205\263_\346\226\260\347\232\204\350\277\220\347\256\227\347\254\246.md.c8bd4cf5.lean.js" (85%) rename "assets/guide_javaScript\347\233\270\345\205\263_\346\255\243\345\210\231\350\241\250\350\276\276\345\274\217.md.e5d049d6.js" => "assets/guide_javaScript\347\233\270\345\205\263_\346\255\243\345\210\231\350\241\250\350\276\276\345\274\217.md.df2c1874.js" (85%) rename "assets/guide_javaScript\347\233\270\345\205\263_\346\255\243\345\210\231\350\241\250\350\276\276\345\274\217.md.e5d049d6.lean.js" => "assets/guide_javaScript\347\233\270\345\205\263_\346\255\243\345\210\231\350\241\250\350\276\276\345\274\217.md.df2c1874.lean.js" (85%) rename "assets/guide_javaScript\347\233\270\345\205\263_\347\273\247\346\211\277\347\233\270\345\205\263.md.7058e93b.js" => "assets/guide_javaScript\347\233\270\345\205\263_\347\273\247\346\211\277\347\233\270\345\205\263.md.86520b75.js" (99%) rename "assets/guide_javaScript\347\233\270\345\205\263_\347\273\247\346\211\277\347\233\270\345\205\263.md.7058e93b.lean.js" => "assets/guide_javaScript\347\233\270\345\205\263_\347\273\247\346\211\277\347\233\270\345\205\263.md.86520b75.lean.js" (86%) rename assets/{guide_webpack_Loader.md.daa9d3ad.js => guide_webpack_Loader.md.8ee4eae1.js} (99%) rename assets/{guide_webpack_Loader.md.daa9d3ad.lean.js => guide_webpack_Loader.md.8ee4eae1.lean.js} (87%) rename assets/{guide_webpack_index.md.fcc0bdf4.js => guide_webpack_index.md.8c12b279.js} (94%) rename assets/{guide_webpack_index.md.fcc0bdf4.lean.js => guide_webpack_index.md.8c12b279.lean.js} (94%) rename assets/{guide_webpack_mini-webpack.md.938949bc.js => guide_webpack_mini-webpack.md.7a56cd34.js} (91%) rename assets/{guide_webpack_mini-webpack.md.938949bc.lean.js => guide_webpack_mini-webpack.md.7a56cd34.lean.js} (91%) rename "assets/guide_webpack_\346\236\204\345\273\272\346\265\201\347\250\213.md.e14ee84d.js" => "assets/guide_webpack_\346\236\204\345\273\272\346\265\201\347\250\213.md.61545129.js" (99%) rename "assets/guide_webpack_\346\236\204\345\273\272\346\265\201\347\250\213.md.e14ee84d.lean.js" => "assets/guide_webpack_\346\236\204\345\273\272\346\265\201\347\250\213.md.61545129.lean.js" (86%) rename "assets/guide_webpack_\346\250\241\345\235\227\350\201\224\351\202\246.md.9176a16d.js" => "assets/guide_webpack_\346\250\241\345\235\227\350\201\224\351\202\246.md.a4b55cfe.js" (99%) rename "assets/guide_webpack_\346\250\241\345\235\227\350\201\224\351\202\246.md.9176a16d.lean.js" => "assets/guide_webpack_\346\250\241\345\235\227\350\201\224\351\202\246.md.a4b55cfe.lean.js" (96%) rename "assets/guide_webpack_\347\203\255\346\233\264\346\226\260\345\216\237\347\220\206.md.f90caa71.js" => "assets/guide_webpack_\347\203\255\346\233\264\346\226\260\345\216\237\347\220\206.md.87ba4ad8.js" (90%) rename "assets/guide_webpack_\347\203\255\346\233\264\346\226\260\345\216\237\347\220\206.md.f90caa71.lean.js" => "assets/guide_webpack_\347\203\255\346\233\264\346\226\260\345\216\237\347\220\206.md.87ba4ad8.lean.js" (90%) rename "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\246\202\350\246\201.md.053a8f89.js" => "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\246\202\350\246\201.md.8b5cc56c.js" (88%) rename "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\246\202\350\246\201.md.053a8f89.lean.js" => "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\246\202\350\246\201.md.8b5cc56c.lean.js" (88%) rename "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\206\205\346\240\270.md.8232bb8e.js" => "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\206\205\346\240\270.md.6fdfde81.js" (92%) rename "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\206\205\346\240\270.md.8232bb8e.lean.js" => "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\206\205\346\240\270.md.6fdfde81.lean.js" (92%) rename "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\256\211\345\205\250.md.73ccefca.js" => "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\256\211\345\205\250.md.ff5d442d.js" (99%) rename "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\256\211\345\205\250.md.73ccefca.lean.js" => "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\256\211\345\205\250.md.ff5d442d.lean.js" (90%) rename "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\346\270\262\346\237\223\346\265\201\347\250\213.md.f3c634f1.js" => "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\346\270\262\346\237\223\346\265\201\347\250\213.md.08093b9d.js" (99%) rename "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\346\270\262\346\237\223\346\265\201\347\250\213.md.f3c634f1.lean.js" => "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\346\270\262\346\237\223\346\265\201\347\250\213.md.08093b9d.lean.js" (90%) rename "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\347\274\223\345\255\230.md.4f6289a9.js" => "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\347\274\223\345\255\230.md.3b8afe38.js" (99%) rename "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\347\274\223\345\255\230.md.4f6289a9.lean.js" => "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\347\274\223\345\255\230.md.3b8afe38.lean.js" (87%) rename "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\350\277\233\347\250\213\346\236\266\346\236\204.md.5bd0b06d.js" => "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\350\277\233\347\250\213\346\236\266\346\236\204.md.4157b63e.js" (97%) rename "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\350\277\233\347\250\213\346\236\266\346\236\204.md.5bd0b06d.lean.js" => "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\350\277\233\347\250\213\346\236\266\346\236\204.md.4157b63e.lean.js" (87%) rename "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_CDN.md.0fd3f4e9.js" => "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_CDN.md.426d00a8.js" (98%) rename "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_CDN.md.0fd3f4e9.lean.js" => "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_CDN.md.426d00a8.lean.js" (86%) rename "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_DNS.md.584b66a4.js" => "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_DNS.md.dc1c7d42.js" (93%) rename "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_DNS.md.584b66a4.lean.js" => "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_DNS.md.dc1c7d42.lean.js" (70%) rename "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_GET\350\257\267\346\261\202\345\222\214POST\350\257\267\346\261\202\347\232\204\345\214\272\345\210\253.md.2f19dc13.js" => "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_GET\350\257\267\346\261\202\345\222\214POST\350\257\267\346\261\202\347\232\204\345\214\272\345\210\253.md.675258cb.js" (95%) rename "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_GET\350\257\267\346\261\202\345\222\214POST\350\257\267\346\261\202\347\232\204\345\214\272\345\210\253.md.2f19dc13.lean.js" => "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_GET\350\257\267\346\261\202\345\222\214POST\350\257\267\346\261\202\347\232\204\345\214\272\345\210\253.md.675258cb.lean.js" (95%) rename "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTP.md.5112c2aa.js" => "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTP.md.2fd00196.js" (98%) rename "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTP.md.5112c2aa.lean.js" => "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTP.md.2fd00196.lean.js" (77%) rename "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTPS.md.efe51aa4.js" => "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTPS.md.1a833174.js" (98%) rename "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTPS.md.efe51aa4.lean.js" => "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTPS.md.1a833174.lean.js" (90%) rename "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\223\215\345\272\224\346\212\245\346\226\207.md.cc269f40.js" => "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\223\215\345\272\224\346\212\245\346\226\207.md.b46d175f.js" (99%) rename "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\223\215\345\272\224\346\212\245\346\226\207.md.cc269f40.lean.js" => "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\223\215\345\272\224\346\212\245\346\226\207.md.b46d175f.lean.js" (98%) rename "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.0601ae85.js" => "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.fee49d4c.js" (85%) rename "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.0601ae85.lean.js" => "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.fee49d4c.lean.js" (85%) rename "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\346\246\202\350\246\201.md.a9f4d298.js" => "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\346\246\202\350\246\201.md.3fbc149d.js" (88%) rename "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\346\246\202\350\246\201.md.a9f4d298.lean.js" => "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\346\246\202\350\246\201.md.3fbc149d.lean.js" (88%) rename "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\257\267\346\261\202\346\212\245\346\226\207.md.c7f4f8d0.js" => "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\257\267\346\261\202\346\212\245\346\226\207.md.dd8a47bc.js" (98%) rename "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\257\267\346\261\202\346\212\245\346\226\207.md.c7f4f8d0.lean.js" => "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\257\267\346\261\202\346\212\245\346\226\207.md.dd8a47bc.lean.js" (88%) rename "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\267\250\345\237\237\350\257\267\346\261\202.md.bcf87cbb.js" => "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\267\250\345\237\237\350\257\267\346\261\202.md.848f5d06.js" (99%) rename "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\267\250\345\237\237\350\257\267\346\261\202.md.bcf87cbb.lean.js" => "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\267\250\345\237\237\350\257\267\346\261\202.md.848f5d06.lean.js" (90%) rename assets/{index.md.b3d414ec.js => index.md.dd3d0939.js} (92%) rename assets/{index.md.b3d414ec.lean.js => index.md.dd3d0939.lean.js} (92%) rename "assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_ISO\346\204\237\345\205\211\345\272\246.md.2f26bf17.js" => "assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_ISO\346\204\237\345\205\211\345\272\246.md.1329ace6.js" (91%) rename "assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_ISO\346\204\237\345\205\211\345\272\246.md.2f26bf17.lean.js" => "assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_ISO\346\204\237\345\205\211\345\272\246.md.1329ace6.lean.js" (91%) rename "assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\205\211\345\234\210.md.ed2aed3b.js" => "assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\205\211\345\234\210.md.5071eb5f.js" (92%) rename "assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\205\211\345\234\210.md.ed2aed3b.lean.js" => "assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\205\211\345\234\210.md.5071eb5f.lean.js" (92%) rename "assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\277\253\351\227\250\351\200\237\345\272\246.md.8bdd3100.js" => "assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\277\253\351\227\250\351\200\237\345\272\246.md.858c51e6.js" (93%) rename "assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\277\253\351\227\250\351\200\237\345\272\246.md.8bdd3100.lean.js" => "assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\277\253\351\227\250\351\200\237\345\272\246.md.858c51e6.lean.js" (93%) rename "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_index.md.5e1171b3.js" => "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_index.md.e5c3a073.js" (92%) rename "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_index.md.5e1171b3.lean.js" => "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_index.md.e5c3a073.lean.js" (92%) rename "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\344\272\256\350\260\203\344\272\272\345\203\217.md.1d135fd4.js" => "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\344\272\256\350\260\203\344\272\272\345\203\217.md.55937770.js" (94%) rename "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\344\272\256\350\260\203\344\272\272\345\203\217.md.1d135fd4.lean.js" => "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\344\272\256\350\260\203\344\272\272\345\203\217.md.55937770.lean.js" (86%) rename "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\206\267\346\232\226\345\257\271\346\257\224\344\272\272\345\203\217.md.70050c0c.js" => "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\206\267\346\232\226\345\257\271\346\257\224\344\272\272\345\203\217.md.852b078d.js" (94%) rename "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\206\267\346\232\226\345\257\271\346\257\224\344\272\272\345\203\217.md.70050c0c.lean.js" => "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\206\267\346\232\226\345\257\271\346\257\224\344\272\272\345\203\217.md.852b078d.lean.js" (87%) rename "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\350\212\261\347\232\204\351\273\221\350\203\214\346\231\257.md.fb63b890.js" => "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\350\212\261\347\232\204\351\273\221\350\203\214\346\231\257.md.5e8263dc.js" (93%) rename "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\350\212\261\347\232\204\351\273\221\350\203\214\346\231\257.md.fb63b890.lean.js" => "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\350\212\261\347\232\204\351\273\221\350\203\214\346\231\257.md.5e8263dc.lean.js" (93%) rename "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\351\233\250\344\270\235.md.e607e8e7.js" => "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\351\233\250\344\270\235.md.7e47cf94.js" (96%) rename "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\351\233\250\344\270\235.md.e607e8e7.lean.js" => "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\351\233\250\344\270\235.md.7e47cf94.lean.js" (87%) rename "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\213\215\346\221\204\346\242\246\345\271\273\345\205\211\346\226\221.md.498fd396.js" => "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\213\215\346\221\204\346\242\246\345\271\273\345\205\211\346\226\221.md.8ca08bfb.js" (94%) rename "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\213\215\346\221\204\346\242\246\345\271\273\345\205\211\346\226\221.md.498fd396.lean.js" => "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\213\215\346\221\204\346\242\246\345\271\273\345\205\211\346\226\221.md.8ca08bfb.lean.js" (94%) rename "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\232\227\350\260\203\344\272\272\345\203\217.md.09e0e873.js" => "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\232\227\350\260\203\344\272\272\345\203\217.md.740aab34.js" (95%) rename "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\232\227\350\260\203\344\272\272\345\203\217.md.09e0e873.lean.js" => "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\232\227\350\260\203\344\272\272\345\203\217.md.740aab34.lean.js" (86%) rename "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\351\200\206\345\205\211\344\272\272\345\203\217.md.0e2e6697.js" => "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\351\200\206\345\205\211\344\272\272\345\203\217.md.57538260.js" (95%) rename "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\351\200\206\345\205\211\344\272\272\345\203\217.md.0e2e6697.lean.js" => "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\351\200\206\345\205\211\344\272\272\345\203\217.md.57538260.lean.js" (87%) rename "assets/shoot_\346\236\204\345\233\276\345\275\242\345\274\217_\346\236\204\345\233\276\346\212\200\345\267\247.md.01544455.js" => "assets/shoot_\346\236\204\345\233\276\345\275\242\345\274\217_\346\236\204\345\233\276\346\212\200\345\267\247.md.9d941cfa.js" (98%) rename "assets/shoot_\346\236\204\345\233\276\345\275\242\345\274\217_\346\236\204\345\233\276\346\212\200\345\267\247.md.01544455.lean.js" => "assets/shoot_\346\236\204\345\233\276\345\275\242\345\274\217_\346\236\204\345\233\276\346\212\200\345\267\247.md.9d941cfa.lean.js" (94%) rename "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\345\273\266\350\277\237\346\221\204\345\275\261.md.7117a224.js" => "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\345\273\266\350\277\237\346\221\204\345\275\261.md.abf24cee.js" (97%) rename "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\345\273\266\350\277\237\346\221\204\345\275\261.md.7117a224.lean.js" => "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\345\273\266\350\277\237\346\221\204\345\275\261.md.abf24cee.lean.js" (89%) rename "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\230\237\347\251\272.md.cbaec0d7.js" => "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\230\237\347\251\272.md.79082746.js" (97%) rename "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\230\237\347\251\272.md.cbaec0d7.lean.js" => "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\230\237\347\251\272.md.79082746.lean.js" (88%) rename "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\234\210\344\272\256.md.155f2cd6.js" => "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\234\210\344\272\256.md.6fbf4de5.js" (92%) rename "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\234\210\344\272\256.md.155f2cd6.lean.js" => "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\234\210\344\272\256.md.6fbf4de5.lean.js" (92%) rename "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\347\203\237\350\212\261.md.481cbdad.js" => "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\347\203\237\350\212\261.md.bff2e64b.js" (93%) rename "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\347\203\237\350\212\261.md.481cbdad.lean.js" => "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\347\203\237\350\212\261.md.bff2e64b.lean.js" (93%) rename "guide/Fabric.js/\345\246\202\344\275\225\345\256\236\347\216\260\347\273\204\351\200\211.html" => "guide/Fabric.js/Fabric \344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.html" (69%) create mode 100644 "guide/Fabric.js/\345\246\202\344\275\225\345\256\236\347\216\260\346\241\206\351\200\211.html" diff --git a/404.html b/404.html index 8edf2c6e..67a45cf0 100644 --- a/404.html +++ b/404.html @@ -21,7 +21,7 @@
Skip to content

404

PAGE NOT FOUND

But if you don't change your direction, and if you keep looking, you may end up where you are heading.
- + \ No newline at end of file diff --git "a/assets/guide_DDD \351\242\206\345\237\237\351\251\261\345\212\250\350\256\276\350\256\241_index.md.b9fd2eae.js" "b/assets/guide_DDD \351\242\206\345\237\237\351\251\261\345\212\250\350\256\276\350\256\241_index.md.d79253fb.js" similarity index 92% rename from "assets/guide_DDD \351\242\206\345\237\237\351\251\261\345\212\250\350\256\276\350\256\241_index.md.b9fd2eae.js" rename to "assets/guide_DDD \351\242\206\345\237\237\351\251\261\345\212\250\350\256\276\350\256\241_index.md.d79253fb.js" index cb3f8f64..d8aaa13a 100644 --- "a/assets/guide_DDD \351\242\206\345\237\237\351\251\261\345\212\250\350\256\276\350\256\241_index.md.b9fd2eae.js" +++ "b/assets/guide_DDD \351\242\206\345\237\237\351\251\261\345\212\250\350\256\276\350\256\241_index.md.d79253fb.js" @@ -1 +1 @@ -import{_ as t,o as d,c as r,k as e,a}from"./chunks/framework.b6910bb2.js";const D=JSON.parse('{"title":"领域驱动设计","description":"","frontmatter":{},"headers":[],"relativePath":"guide/DDD 领域驱动设计/index.md","filePath":"guide/DDD 领域驱动设计/index.md","lastUpdated":1736065650000}'),s={name:"guide/DDD 领域驱动设计/index.md"},n=e("h1",{id:"领域驱动设计",tabindex:"-1"},[a("领域驱动设计 "),e("a",{class:"header-anchor",href:"#领域驱动设计","aria-label":'Permalink to "领域驱动设计"'},"​")],-1),o=e("h2",{id:"定义",tabindex:"-1"},[a("定义 "),e("a",{class:"header-anchor",href:"#定义","aria-label":'Permalink to "定义"'},"​")],-1),i=[n,o];function c(l,_,h,p,m,f){return d(),r("div",null,i)}const u=t(s,[["render",c]]);export{D as __pageData,u as default}; +import{_ as t,o as d,c as r,k as e,a}from"./chunks/framework.b6910bb2.js";const D=JSON.parse('{"title":"领域驱动设计","description":"","frontmatter":{},"headers":[],"relativePath":"guide/DDD 领域驱动设计/index.md","filePath":"guide/DDD 领域驱动设计/index.md","lastUpdated":1736839070000}'),s={name:"guide/DDD 领域驱动设计/index.md"},n=e("h1",{id:"领域驱动设计",tabindex:"-1"},[a("领域驱动设计 "),e("a",{class:"header-anchor",href:"#领域驱动设计","aria-label":'Permalink to "领域驱动设计"'},"​")],-1),o=e("h2",{id:"定义",tabindex:"-1"},[a("定义 "),e("a",{class:"header-anchor",href:"#定义","aria-label":'Permalink to "定义"'},"​")],-1),i=[n,o];function c(l,_,h,p,m,f){return d(),r("div",null,i)}const u=t(s,[["render",c]]);export{D as __pageData,u as default}; diff --git "a/assets/guide_DDD \351\242\206\345\237\237\351\251\261\345\212\250\350\256\276\350\256\241_index.md.b9fd2eae.lean.js" "b/assets/guide_DDD \351\242\206\345\237\237\351\251\261\345\212\250\350\256\276\350\256\241_index.md.d79253fb.lean.js" similarity index 92% rename from "assets/guide_DDD \351\242\206\345\237\237\351\251\261\345\212\250\350\256\276\350\256\241_index.md.b9fd2eae.lean.js" rename to "assets/guide_DDD \351\242\206\345\237\237\351\251\261\345\212\250\350\256\276\350\256\241_index.md.d79253fb.lean.js" index cb3f8f64..d8aaa13a 100644 --- "a/assets/guide_DDD \351\242\206\345\237\237\351\251\261\345\212\250\350\256\276\350\256\241_index.md.b9fd2eae.lean.js" +++ "b/assets/guide_DDD \351\242\206\345\237\237\351\251\261\345\212\250\350\256\276\350\256\241_index.md.d79253fb.lean.js" @@ -1 +1 @@ -import{_ as t,o as d,c as r,k as e,a}from"./chunks/framework.b6910bb2.js";const D=JSON.parse('{"title":"领域驱动设计","description":"","frontmatter":{},"headers":[],"relativePath":"guide/DDD 领域驱动设计/index.md","filePath":"guide/DDD 领域驱动设计/index.md","lastUpdated":1736065650000}'),s={name:"guide/DDD 领域驱动设计/index.md"},n=e("h1",{id:"领域驱动设计",tabindex:"-1"},[a("领域驱动设计 "),e("a",{class:"header-anchor",href:"#领域驱动设计","aria-label":'Permalink to "领域驱动设计"'},"​")],-1),o=e("h2",{id:"定义",tabindex:"-1"},[a("定义 "),e("a",{class:"header-anchor",href:"#定义","aria-label":'Permalink to "定义"'},"​")],-1),i=[n,o];function c(l,_,h,p,m,f){return d(),r("div",null,i)}const u=t(s,[["render",c]]);export{D as __pageData,u as default}; +import{_ as t,o as d,c as r,k as e,a}from"./chunks/framework.b6910bb2.js";const D=JSON.parse('{"title":"领域驱动设计","description":"","frontmatter":{},"headers":[],"relativePath":"guide/DDD 领域驱动设计/index.md","filePath":"guide/DDD 领域驱动设计/index.md","lastUpdated":1736839070000}'),s={name:"guide/DDD 领域驱动设计/index.md"},n=e("h1",{id:"领域驱动设计",tabindex:"-1"},[a("领域驱动设计 "),e("a",{class:"header-anchor",href:"#领域驱动设计","aria-label":'Permalink to "领域驱动设计"'},"​")],-1),o=e("h2",{id:"定义",tabindex:"-1"},[a("定义 "),e("a",{class:"header-anchor",href:"#定义","aria-label":'Permalink to "定义"'},"​")],-1),i=[n,o];function c(l,_,h,p,m,f){return d(),r("div",null,i)}const u=t(s,[["render",c]]);export{D as __pageData,u as default}; diff --git "a/assets/guide_Fabric.js_Fabric \344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.md.9993b631.js" "b/assets/guide_Fabric.js_Fabric \344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.md.9993b631.js" new file mode 100644 index 00000000..21ba5219 --- /dev/null +++ "b/assets/guide_Fabric.js_Fabric \344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.md.9993b631.js" @@ -0,0 +1 @@ +import{_ as a,o as e,c as t,Q as r}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"Fabric 中的性能优化","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Fabric.js/Fabric 中的性能优化.md","filePath":"guide/Fabric.js/Fabric 中的性能优化.md","lastUpdated":1736839070000}'),c={name:"guide/Fabric.js/Fabric 中的性能优化.md"},o=r('

Fabric 中的性能优化

参考文章

',3),i=[o];function s(_,l,n,d,h,u){return e(),t("div",null,i)}const p=a(c,[["render",s]]);export{m as __pageData,p as default}; diff --git "a/assets/guide_Fabric.js_Fabric \344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.md.9993b631.lean.js" "b/assets/guide_Fabric.js_Fabric \344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.md.9993b631.lean.js" new file mode 100644 index 00000000..2ff75a9c --- /dev/null +++ "b/assets/guide_Fabric.js_Fabric \344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.md.9993b631.lean.js" @@ -0,0 +1 @@ +import{_ as a,o as e,c as t,Q as r}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"Fabric 中的性能优化","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Fabric.js/Fabric 中的性能优化.md","filePath":"guide/Fabric.js/Fabric 中的性能优化.md","lastUpdated":1736839070000}'),c={name:"guide/Fabric.js/Fabric 中的性能优化.md"},o=r("",3),i=[o];function s(_,l,n,d,h,u){return e(),t("div",null,i)}const p=a(c,[["render",s]]);export{m as __pageData,p as default}; diff --git "a/assets/guide_Fabric.js_\345\206\205\351\203\250\347\273\223\346\236\204\346\246\202\350\277\260.md.be67fc71.js" "b/assets/guide_Fabric.js_\345\206\205\351\203\250\347\273\223\346\236\204\346\246\202\350\277\260.md.5690116d.js" similarity index 98% rename from "assets/guide_Fabric.js_\345\206\205\351\203\250\347\273\223\346\236\204\346\246\202\350\277\260.md.be67fc71.js" rename to "assets/guide_Fabric.js_\345\206\205\351\203\250\347\273\223\346\236\204\346\246\202\350\277\260.md.5690116d.js" index 29493fef..cb60154a 100644 --- "a/assets/guide_Fabric.js_\345\206\205\351\203\250\347\273\223\346\236\204\346\246\202\350\277\260.md.be67fc71.js" +++ "b/assets/guide_Fabric.js_\345\206\205\351\203\250\347\273\223\346\236\204\346\246\202\350\277\260.md.5690116d.js" @@ -1 +1 @@ -import{_ as a,o as e,c as s,Q as t}from"./chunks/framework.b6910bb2.js";const r="/vitePress-blob/assets/1.1634239e.png",i="/vitePress-blob/assets/4.84143987.png",n="/vitePress-blob/assets/2.8e8014c3.png",l="/vitePress-blob/assets/5.5d843167.png",o="/vitePress-blob/assets/6.e9563edc.png",P=JSON.parse('{"title":"内部结构概述","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Fabric.js/内部结构概述.md","filePath":"guide/Fabric.js/内部结构概述.md","lastUpdated":1736065650000}'),c={name:"guide/Fabric.js/内部结构概述.md"},p=t('

内部结构概述

使用场景上看

从使用场景上看,Fabric.js 通过对象的形式来创建图形 (源码上也充满了面向对象设计的身影)

对于使用者而言,不需要关心底层 Canvas 是如何实现的,只需要关心如何创建这些图像对象即可。

Canvas 开发原理

Fabric 底层是通过 Canvas 进行绘制的,因此需先简单了解下 Canvas 的开发原理。

Canvas 开发的本质其实很简单,可以想象下这种少儿画板:

初始状态图

Canvas 的绘制其实就是在不断在画布上绘制图形,然后擦出的过程。

Fabric.js 源码解析

基本原理

fabric.js 在初始化的时候会将你指定的 Canvas 元素(叫做 lowerCanvas)外面包裹上一 层 div 元素, 然后内部会插入另外一个上层的 Canvas 元素(叫做 upperCanvas),这两 个 Canvas 有如下区别

alt text

大体结构

当我们引入 fabric.js 后,再控制台打印 fabric ,可以看到如下结构:

初始状态图

可以看到,fabric 是一个对象,里面包含了多个属性,这些属性指向不同的模块。

Fabric的源码也是相当易读的,大部分情况下一个文件就代表一个模块,源码结构如下:

alt text

模块结构图

基于上述源码结构,可以很方便的绘制出 Fabric 的大概的一个模块结构图:

alt text

至此,我们大致了解了 Fabric 的内部结构,接下来,我们将通过一个简单的例子,来了解下 Fabric 是如何绘制一个图形。

参考

',26),h=[p];function b(d,_,u,m,f,v){return e(),s("div",null,h)}const g=a(c,[["render",b]]);export{P as __pageData,g as default}; +import{_ as a,o as e,c as s,Q as t}from"./chunks/framework.b6910bb2.js";const r="/vitePress-blob/assets/1.1634239e.png",i="/vitePress-blob/assets/4.84143987.png",n="/vitePress-blob/assets/2.8e8014c3.png",l="/vitePress-blob/assets/5.5d843167.png",o="/vitePress-blob/assets/6.e9563edc.png",P=JSON.parse('{"title":"内部结构概述","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Fabric.js/内部结构概述.md","filePath":"guide/Fabric.js/内部结构概述.md","lastUpdated":1736839070000}'),c={name:"guide/Fabric.js/内部结构概述.md"},p=t('

内部结构概述

使用场景上看

从使用场景上看,Fabric.js 通过对象的形式来创建图形 (源码上也充满了面向对象设计的身影)

对于使用者而言,不需要关心底层 Canvas 是如何实现的,只需要关心如何创建这些图像对象即可。

Canvas 开发原理

Fabric 底层是通过 Canvas 进行绘制的,因此需先简单了解下 Canvas 的开发原理。

Canvas 开发的本质其实很简单,可以想象下这种少儿画板:

初始状态图

Canvas 的绘制其实就是在不断在画布上绘制图形,然后擦出的过程。

Fabric.js 源码解析

基本原理

fabric.js 在初始化的时候会将你指定的 Canvas 元素(叫做 lowerCanvas)外面包裹上一 层 div 元素, 然后内部会插入另外一个上层的 Canvas 元素(叫做 upperCanvas),这两 个 Canvas 有如下区别

alt text

大体结构

当我们引入 fabric.js 后,再控制台打印 fabric ,可以看到如下结构:

初始状态图

可以看到,fabric 是一个对象,里面包含了多个属性,这些属性指向不同的模块。

Fabric的源码也是相当易读的,大部分情况下一个文件就代表一个模块,源码结构如下:

alt text

模块结构图

基于上述源码结构,可以很方便的绘制出 Fabric 的大概的一个模块结构图:

alt text

至此,我们大致了解了 Fabric 的内部结构,接下来,我们将通过一个简单的例子,来了解下 Fabric 是如何绘制一个图形。

参考

',26),h=[p];function b(d,_,u,m,f,v){return e(),s("div",null,h)}const g=a(c,[["render",b]]);export{P as __pageData,g as default}; diff --git "a/assets/guide_Fabric.js_\345\206\205\351\203\250\347\273\223\346\236\204\346\246\202\350\277\260.md.be67fc71.lean.js" "b/assets/guide_Fabric.js_\345\206\205\351\203\250\347\273\223\346\236\204\346\246\202\350\277\260.md.5690116d.lean.js" similarity index 90% rename from "assets/guide_Fabric.js_\345\206\205\351\203\250\347\273\223\346\236\204\346\246\202\350\277\260.md.be67fc71.lean.js" rename to "assets/guide_Fabric.js_\345\206\205\351\203\250\347\273\223\346\236\204\346\246\202\350\277\260.md.5690116d.lean.js" index cf4351ba..8a48819f 100644 --- "a/assets/guide_Fabric.js_\345\206\205\351\203\250\347\273\223\346\236\204\346\246\202\350\277\260.md.be67fc71.lean.js" +++ "b/assets/guide_Fabric.js_\345\206\205\351\203\250\347\273\223\346\236\204\346\246\202\350\277\260.md.5690116d.lean.js" @@ -1 +1 @@ -import{_ as a,o as e,c as s,Q as t}from"./chunks/framework.b6910bb2.js";const r="/vitePress-blob/assets/1.1634239e.png",i="/vitePress-blob/assets/4.84143987.png",n="/vitePress-blob/assets/2.8e8014c3.png",l="/vitePress-blob/assets/5.5d843167.png",o="/vitePress-blob/assets/6.e9563edc.png",P=JSON.parse('{"title":"内部结构概述","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Fabric.js/内部结构概述.md","filePath":"guide/Fabric.js/内部结构概述.md","lastUpdated":1736065650000}'),c={name:"guide/Fabric.js/内部结构概述.md"},p=t("",26),h=[p];function b(d,_,u,m,f,v){return e(),s("div",null,h)}const g=a(c,[["render",b]]);export{P as __pageData,g as default}; +import{_ as a,o as e,c as s,Q as t}from"./chunks/framework.b6910bb2.js";const r="/vitePress-blob/assets/1.1634239e.png",i="/vitePress-blob/assets/4.84143987.png",n="/vitePress-blob/assets/2.8e8014c3.png",l="/vitePress-blob/assets/5.5d843167.png",o="/vitePress-blob/assets/6.e9563edc.png",P=JSON.parse('{"title":"内部结构概述","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Fabric.js/内部结构概述.md","filePath":"guide/Fabric.js/内部结构概述.md","lastUpdated":1736839070000}'),c={name:"guide/Fabric.js/内部结构概述.md"},p=t("",26),h=[p];function b(d,_,u,m,f,v){return e(),s("div",null,h)}const g=a(c,[["render",b]]);export{P as __pageData,g as default}; diff --git "a/assets/guide_Fabric.js_\345\237\272\346\234\254\346\246\202\345\277\265.md.711f70e5.js" "b/assets/guide_Fabric.js_\345\237\272\346\234\254\346\246\202\345\277\265.md.302ccee7.js" similarity index 98% rename from "assets/guide_Fabric.js_\345\237\272\346\234\254\346\246\202\345\277\265.md.711f70e5.js" rename to "assets/guide_Fabric.js_\345\237\272\346\234\254\346\246\202\345\277\265.md.302ccee7.js" index f6080de6..5a039960 100644 --- "a/assets/guide_Fabric.js_\345\237\272\346\234\254\346\246\202\345\277\265.md.711f70e5.js" +++ "b/assets/guide_Fabric.js_\345\237\272\346\234\254\346\246\202\345\277\265.md.302ccee7.js" @@ -2,4 +2,4 @@ import{_ as s}from"./chunks/container.fad5294e.js";import{o as a,c as n,H as l,Q <canvas id="canvas"></canvas>
<!-- HTML -->
 <canvas id="canvas"></canvas>
js
// JavaScript
 const canvas = new fabric.Canvas('canvas');
// JavaScript
-const canvas = new fabric.Canvas('canvas');

添加图形到画布

js
canvas.add(new fabric.Rect({ left: 100, top: 100, fill: 'red', width: 100, height: 100 }));
canvas.add(new fabric.Rect({ left: 100, top: 100, fill: 'red', width: 100, height: 100 }));

通过案例可以感受到,如果你需要用 HTML Canvas 来绘制一些东西,并且这些东西可以响应用户的交互,比如:拖动、变形、旋转等 操作。 那用 fabric.js 是非常合适的,因为它内部不仅实现了 Canvas 对象模型,还将一些常用的交互操作封装好了,可以说是开箱即用。

`,10),y=JSON.parse('{"title":"简介","description":"","frontmatter":{"aside":false,"outline":false},"headers":[],"relativePath":"guide/Fabric.js/基本概念.md","filePath":"guide/Fabric.js/基本概念.md","lastUpdated":1736065650000}'),e={name:"guide/Fabric.js/基本概念.md"},d=Object.assign(e,{setup(t){return(c,r)=>(a(),n("div",null,[p,l(s,{url:"https://enson0131.github.io/mini-fabric-whiteboard/"})]))}});export{y as __pageData,d as default}; +const canvas = new fabric.Canvas('canvas');

添加图形到画布

js
canvas.add(new fabric.Rect({ left: 100, top: 100, fill: 'red', width: 100, height: 100 }));
canvas.add(new fabric.Rect({ left: 100, top: 100, fill: 'red', width: 100, height: 100 }));

通过案例可以感受到,如果你需要用 HTML Canvas 来绘制一些东西,并且这些东西可以响应用户的交互,比如:拖动、变形、旋转等 操作。 那用 fabric.js 是非常合适的,因为它内部不仅实现了 Canvas 对象模型,还将一些常用的交互操作封装好了,可以说是开箱即用。

`,10),y=JSON.parse('{"title":"简介","description":"","frontmatter":{"aside":false,"outline":false},"headers":[],"relativePath":"guide/Fabric.js/基本概念.md","filePath":"guide/Fabric.js/基本概念.md","lastUpdated":1736839070000}'),e={name:"guide/Fabric.js/基本概念.md"},d=Object.assign(e,{setup(t){return(c,r)=>(a(),n("div",null,[p,l(s,{url:"https://enson0131.github.io/mini-fabric-whiteboard/"})]))}});export{y as __pageData,d as default}; diff --git "a/assets/guide_Fabric.js_\345\237\272\346\234\254\346\246\202\345\277\265.md.711f70e5.lean.js" "b/assets/guide_Fabric.js_\345\237\272\346\234\254\346\246\202\345\277\265.md.302ccee7.lean.js" similarity index 89% rename from "assets/guide_Fabric.js_\345\237\272\346\234\254\346\246\202\345\277\265.md.711f70e5.lean.js" rename to "assets/guide_Fabric.js_\345\237\272\346\234\254\346\246\202\345\277\265.md.302ccee7.lean.js" index 60c43043..767f8d5b 100644 --- "a/assets/guide_Fabric.js_\345\237\272\346\234\254\346\246\202\345\277\265.md.711f70e5.lean.js" +++ "b/assets/guide_Fabric.js_\345\237\272\346\234\254\346\246\202\345\277\265.md.302ccee7.lean.js" @@ -1 +1 @@ -import{_ as s}from"./chunks/container.fad5294e.js";import{o as a,c as n,H as l,Q as o}from"./chunks/framework.b6910bb2.js";const p=o("",10),y=JSON.parse('{"title":"简介","description":"","frontmatter":{"aside":false,"outline":false},"headers":[],"relativePath":"guide/Fabric.js/基本概念.md","filePath":"guide/Fabric.js/基本概念.md","lastUpdated":1736065650000}'),e={name:"guide/Fabric.js/基本概念.md"},d=Object.assign(e,{setup(t){return(c,r)=>(a(),n("div",null,[p,l(s,{url:"https://enson0131.github.io/mini-fabric-whiteboard/"})]))}});export{y as __pageData,d as default}; +import{_ as s}from"./chunks/container.fad5294e.js";import{o as a,c as n,H as l,Q as o}from"./chunks/framework.b6910bb2.js";const p=o("",10),y=JSON.parse('{"title":"简介","description":"","frontmatter":{"aside":false,"outline":false},"headers":[],"relativePath":"guide/Fabric.js/基本概念.md","filePath":"guide/Fabric.js/基本概念.md","lastUpdated":1736839070000}'),e={name:"guide/Fabric.js/基本概念.md"},d=Object.assign(e,{setup(t){return(c,r)=>(a(),n("div",null,[p,l(s,{url:"https://enson0131.github.io/mini-fabric-whiteboard/"})]))}});export{y as __pageData,d as default}; diff --git "a/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\345\205\203\347\264\240\347\232\204\345\271\263\347\247\273\343\200\201\346\227\213\350\275\254\343\200\201\347\274\251\346\224\276.md.493fc2f7.js" "b/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\345\205\203\347\264\240\347\232\204\345\271\263\347\247\273\343\200\201\346\227\213\350\275\254\343\200\201\347\274\251\346\224\276.md.140dd9a6.js" similarity index 99% rename from "assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\345\205\203\347\264\240\347\232\204\345\271\263\347\247\273\343\200\201\346\227\213\350\275\254\343\200\201\347\274\251\346\224\276.md.493fc2f7.js" rename to "assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\345\205\203\347\264\240\347\232\204\345\271\263\347\247\273\343\200\201\346\227\213\350\275\254\343\200\201\347\274\251\346\224\276.md.140dd9a6.js" index 22fdba4f..bb2a51ff 100644 --- "a/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\345\205\203\347\264\240\347\232\204\345\271\263\347\247\273\343\200\201\346\227\213\350\275\254\343\200\201\347\274\251\346\224\276.md.493fc2f7.js" +++ "b/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\345\205\203\347\264\240\347\232\204\345\271\263\347\247\273\343\200\201\346\227\213\350\275\254\343\200\201\347\274\251\346\224\276.md.140dd9a6.js" @@ -1,4 +1,4 @@ -import{_ as s,o as a,c as n,Q as p}from"./chunks/framework.b6910bb2.js";const l="/vitePress-blob/assets/12.c019413f.png",o="/vitePress-blob/assets/13.591a51f7.png",e="",t="/vitePress-blob/assets/15.54a5c9c6.gif",d=JSON.parse('{"title":"如何实现元素的平移、旋转、缩放","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Fabric.js/如何实现元素的平移、旋转、缩放.md","filePath":"guide/Fabric.js/如何实现元素的平移、旋转、缩放.md","lastUpdated":1736065650000}'),c={name:"guide/Fabric.js/如何实现元素的平移、旋转、缩放.md"},r=p(`

如何实现元素的平移、旋转、缩放

在上一节中,我们知道了元素是如何渲染到 Canvas 的画布上,这一节我们将了解如何对元素进行平移、旋转、缩放。

如何平移、旋转、缩放元素

在上一节我们知道,元素都继承自 fabric.Object,而 fabric.Object 的 render 方法都由各个元素实现。那么对于元素的平移、旋转、缩放,我们又需要如何抽离出通用的逻辑呢?

不妨思考一下,这个案例

假设要在 (100, 100) 的地方绘制一个 50*50 的矩形,并将其放大 2 倍,之后旋转 45°,该怎么画呢?

正常逻辑是:

  1. 手动算下宽高 100 * 100

  2. 手动算下旋转之后各个顶点的坐标

  3. 连接四个顶点,绘制矩形

也就是我们先手动先计算出各个转化后的坐标,通过坐标绘制出图形。

但在 Canvas 中,要改掉这种绘制的思想,而是要通过并善用变换坐标系来绘制物体

思路如下: 1 绘制一个 50*50 的矩形

2 转化坐标系

代码如下:

js
ctx.save(); // 之前提到过了,你要修改 ctx 上的一些配置或者画一个物体,最好先 save 一下,这是个好习惯
+import{_ as s,o as a,c as n,Q as p}from"./chunks/framework.b6910bb2.js";const l="/vitePress-blob/assets/12.c019413f.png",o="/vitePress-blob/assets/13.591a51f7.png",e="",t="/vitePress-blob/assets/15.54a5c9c6.gif",d=JSON.parse('{"title":"如何实现元素的平移、旋转、缩放","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Fabric.js/如何实现元素的平移、旋转、缩放.md","filePath":"guide/Fabric.js/如何实现元素的平移、旋转、缩放.md","lastUpdated":1736839070000}'),c={name:"guide/Fabric.js/如何实现元素的平移、旋转、缩放.md"},r=p(`

如何实现元素的平移、旋转、缩放

在上一节中,我们知道了元素是如何渲染到 Canvas 的画布上,这一节我们将了解如何对元素进行平移、旋转、缩放。

如何平移、旋转、缩放元素

在上一节我们知道,元素都继承自 fabric.Object,而 fabric.Object 的 render 方法都由各个元素实现。那么对于元素的平移、旋转、缩放,我们又需要如何抽离出通用的逻辑呢?

不妨思考一下,这个案例

假设要在 (100, 100) 的地方绘制一个 50*50 的矩形,并将其放大 2 倍,之后旋转 45°,该怎么画呢?

正常逻辑是:

  1. 手动算下宽高 100 * 100

  2. 手动算下旋转之后各个顶点的坐标

  3. 连接四个顶点,绘制矩形

也就是我们先手动先计算出各个转化后的坐标,通过坐标绘制出图形。

但在 Canvas 中,要改掉这种绘制的思想,而是要通过并善用变换坐标系来绘制物体

思路如下: 1 绘制一个 50*50 的矩形

2 转化坐标系

代码如下:

js
ctx.save(); // 之前提到过了,你要修改 ctx 上的一些配置或者画一个物体,最好先 save 一下,这是个好习惯
 ctx.translate(100, 100); // 此时原点已经变到了 (100, 100) 的地方
 ctx.scale(2, 2); // 坐标系放大两倍
 ctx.rotate(Util.degreesToRadians(45)); // 注意 canvas 中用的都是弧度(弧度 / 2 * Math.PI = 角度 / 360),所以需要简单换算下
diff --git "a/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\345\205\203\347\264\240\347\232\204\345\271\263\347\247\273\343\200\201\346\227\213\350\275\254\343\200\201\347\274\251\346\224\276.md.493fc2f7.lean.js" "b/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\345\205\203\347\264\240\347\232\204\345\271\263\347\247\273\343\200\201\346\227\213\350\275\254\343\200\201\347\274\251\346\224\276.md.140dd9a6.lean.js"
similarity index 98%
rename from "assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\345\205\203\347\264\240\347\232\204\345\271\263\347\247\273\343\200\201\346\227\213\350\275\254\343\200\201\347\274\251\346\224\276.md.493fc2f7.lean.js"
rename to "assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\345\205\203\347\264\240\347\232\204\345\271\263\347\247\273\343\200\201\346\227\213\350\275\254\343\200\201\347\274\251\346\224\276.md.140dd9a6.lean.js"
index fd85c304..75c3ffe4 100644
--- "a/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\345\205\203\347\264\240\347\232\204\345\271\263\347\247\273\343\200\201\346\227\213\350\275\254\343\200\201\347\274\251\346\224\276.md.493fc2f7.lean.js"
+++ "b/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\345\205\203\347\264\240\347\232\204\345\271\263\347\247\273\343\200\201\346\227\213\350\275\254\343\200\201\347\274\251\346\224\276.md.140dd9a6.lean.js"
@@ -1 +1 @@
-import{_ as s,o as a,c as n,Q as p}from"./chunks/framework.b6910bb2.js";const l="/vitePress-blob/assets/12.c019413f.png",o="/vitePress-blob/assets/13.591a51f7.png",e="",t="/vitePress-blob/assets/15.54a5c9c6.gif",d=JSON.parse('{"title":"如何实现元素的平移、旋转、缩放","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Fabric.js/如何实现元素的平移、旋转、缩放.md","filePath":"guide/Fabric.js/如何实现元素的平移、旋转、缩放.md","lastUpdated":1736065650000}'),c={name:"guide/Fabric.js/如何实现元素的平移、旋转、缩放.md"},r=p("",37),E=[r];function y(A,i,C,g,F,h){return a(),n("div",null,E)}const m=s(c,[["render",y]]);export{d as __pageData,m as default};
+import{_ as s,o as a,c as n,Q as p}from"./chunks/framework.b6910bb2.js";const l="/vitePress-blob/assets/12.c019413f.png",o="/vitePress-blob/assets/13.591a51f7.png",e="",t="/vitePress-blob/assets/15.54a5c9c6.gif",d=JSON.parse('{"title":"如何实现元素的平移、旋转、缩放","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Fabric.js/如何实现元素的平移、旋转、缩放.md","filePath":"guide/Fabric.js/如何实现元素的平移、旋转、缩放.md","lastUpdated":1736839070000}'),c={name:"guide/Fabric.js/如何实现元素的平移、旋转、缩放.md"},r=p("",37),E=[r];function y(A,i,C,g,F,h){return a(),n("div",null,E)}const m=s(c,[["render",y]]);export{d as __pageData,m as default};
diff --git "a/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\346\241\206\351\200\211.md.382e7c32.js" "b/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\346\241\206\351\200\211.md.382e7c32.js"
new file mode 100644
index 00000000..c20c6c71
--- /dev/null
+++ "b/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\346\241\206\351\200\211.md.382e7c32.js"
@@ -0,0 +1 @@
+import{_ as a,o as e,c as t,Q as r}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"如何实现框选","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Fabric.js/如何实现框选.md","filePath":"guide/Fabric.js/如何实现框选.md","lastUpdated":1736839070000}'),o={name:"guide/Fabric.js/如何实现框选.md"},i=r('

如何实现框选

判断俩个图形是否相交

判断俩个线段是否相交

向量叉乘 的方式判断

参考文章

',6),l=[i];function c(n,s,_,d,h,u){return e(),t("div",null,l)}const f=a(o,[["render",c]]);export{m as __pageData,f as default}; diff --git "a/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\346\241\206\351\200\211.md.382e7c32.lean.js" "b/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\346\241\206\351\200\211.md.382e7c32.lean.js" new file mode 100644 index 00000000..4de9564c --- /dev/null +++ "b/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\346\241\206\351\200\211.md.382e7c32.lean.js" @@ -0,0 +1 @@ +import{_ as a,o as e,c as t,Q as r}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"如何实现框选","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Fabric.js/如何实现框选.md","filePath":"guide/Fabric.js/如何实现框选.md","lastUpdated":1736839070000}'),o={name:"guide/Fabric.js/如何实现框选.md"},i=r("",6),l=[i];function c(n,s,_,d,h,u){return e(),t("div",null,l)}const f=a(o,[["render",c]]);export{m as __pageData,f as default}; diff --git "a/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\347\202\271\351\200\211\347\211\251\344\275\223.md.01b61e2b.js" "b/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\347\202\271\351\200\211\347\211\251\344\275\223.md.01b61e2b.js" deleted file mode 100644 index 6c4448a8..00000000 --- "a/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\347\202\271\351\200\211\347\211\251\344\275\223.md.01b61e2b.js" +++ /dev/null @@ -1 +0,0 @@ -import{_ as a,o as e,c as t,Q as r}from"./chunks/framework.b6910bb2.js";const p=JSON.parse('{"title":"如何实现点选物体","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Fabric.js/如何实现点选物体.md","filePath":"guide/Fabric.js/如何实现点选物体.md","lastUpdated":1736065650000}'),o={name:"guide/Fabric.js/如何实现点选物体.md"},i=r('

如何实现点选物体

包围盒

常见的有 OBB、AABB、球模型包围盒

OBB 包围盒

AABB 包围盒

点射法

优化

1 记录最近的这个物体,下次再次判断这个物体。 2 判断是否是透明区域

',8),s=[i];function d(h,n,_,c,l,b){return e(),t("div",null,s)}const m=a(o,[["render",d]]);export{p as __pageData,m as default}; diff --git "a/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\347\202\271\351\200\211\347\211\251\344\275\223.md.4b2066fc.js" "b/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\347\202\271\351\200\211\347\211\251\344\275\223.md.4b2066fc.js" new file mode 100644 index 00000000..1a147473 --- /dev/null +++ "b/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\347\202\271\351\200\211\347\211\251\344\275\223.md.4b2066fc.js" @@ -0,0 +1 @@ +import{_ as a,o as e,c as t,Q as r}from"./chunks/framework.b6910bb2.js";const p=JSON.parse('{"title":"如何实现点选物体","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Fabric.js/如何实现点选物体.md","filePath":"guide/Fabric.js/如何实现点选物体.md","lastUpdated":1736839070000}'),o={name:"guide/Fabric.js/如何实现点选物体.md"},i=r('

如何实现点选物体

上一节,我们了解了

包围盒

常见的有 OBB、AABB、球模型包围盒

OBB 包围盒

AABB 包围盒

点射法

优化

1 记录最近的这个物体,下次再次判断这个物体。 2 判断是否是透明区域

',9),s=[i];function d(h,n,_,c,l,b){return e(),t("div",null,s)}const m=a(o,[["render",d]]);export{p as __pageData,m as default}; diff --git "a/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\347\202\271\351\200\211\347\211\251\344\275\223.md.01b61e2b.lean.js" "b/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\347\202\271\351\200\211\347\211\251\344\275\223.md.4b2066fc.lean.js" similarity index 75% rename from "assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\347\202\271\351\200\211\347\211\251\344\275\223.md.01b61e2b.lean.js" rename to "assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\347\202\271\351\200\211\347\211\251\344\275\223.md.4b2066fc.lean.js" index b2dc4d0c..aae0727a 100644 --- "a/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\347\202\271\351\200\211\347\211\251\344\275\223.md.01b61e2b.lean.js" +++ "b/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\347\202\271\351\200\211\347\211\251\344\275\223.md.4b2066fc.lean.js" @@ -1 +1 @@ -import{_ as a,o as e,c as t,Q as r}from"./chunks/framework.b6910bb2.js";const p=JSON.parse('{"title":"如何实现点选物体","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Fabric.js/如何实现点选物体.md","filePath":"guide/Fabric.js/如何实现点选物体.md","lastUpdated":1736065650000}'),o={name:"guide/Fabric.js/如何实现点选物体.md"},i=r("",8),s=[i];function d(h,n,_,c,l,b){return e(),t("div",null,s)}const m=a(o,[["render",d]]);export{p as __pageData,m as default}; +import{_ as a,o as e,c as t,Q as r}from"./chunks/framework.b6910bb2.js";const p=JSON.parse('{"title":"如何实现点选物体","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Fabric.js/如何实现点选物体.md","filePath":"guide/Fabric.js/如何实现点选物体.md","lastUpdated":1736839070000}'),o={name:"guide/Fabric.js/如何实现点选物体.md"},i=r("",9),s=[i];function d(h,n,_,c,l,b){return e(),t("div",null,s)}const m=a(o,[["render",d]]);export{p as __pageData,m as default}; diff --git "a/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\347\273\204\351\200\211.md.3e1b07e1.js" "b/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\347\273\204\351\200\211.md.3e1b07e1.js" deleted file mode 100644 index 0974a633..00000000 --- "a/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\347\273\204\351\200\211.md.3e1b07e1.js" +++ /dev/null @@ -1 +0,0 @@ -import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Fabric.js/如何实现组选.md","filePath":"guide/Fabric.js/如何实现组选.md","lastUpdated":1736065650000}'),r={name:"guide/Fabric.js/如何实现组选.md"};function s(c,o,_,i,d,n){return t(),a("div")}const f=e(r,[["render",s]]);export{m as __pageData,f as default}; diff --git "a/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\347\273\204\351\200\211.md.3e1b07e1.lean.js" "b/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\347\273\204\351\200\211.md.3e1b07e1.lean.js" deleted file mode 100644 index 0974a633..00000000 --- "a/assets/guide_Fabric.js_\345\246\202\344\275\225\345\256\236\347\216\260\347\273\204\351\200\211.md.3e1b07e1.lean.js" +++ /dev/null @@ -1 +0,0 @@ -import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Fabric.js/如何实现组选.md","filePath":"guide/Fabric.js/如何实现组选.md","lastUpdated":1736065650000}'),r={name:"guide/Fabric.js/如何实现组选.md"};function s(c,o,_,i,d,n){return t(),a("div")}const f=e(r,[["render",s]]);export{m as __pageData,f as default}; diff --git "a/assets/guide_Fabric.js_\345\246\202\344\275\225\347\273\230\345\210\266\344\270\200\344\270\252\345\233\276\345\275\242.md.78353ed8.js" "b/assets/guide_Fabric.js_\345\246\202\344\275\225\347\273\230\345\210\266\344\270\200\344\270\252\345\233\276\345\275\242.md.fc479c77.js" similarity index 99% rename from "assets/guide_Fabric.js_\345\246\202\344\275\225\347\273\230\345\210\266\344\270\200\344\270\252\345\233\276\345\275\242.md.78353ed8.js" rename to "assets/guide_Fabric.js_\345\246\202\344\275\225\347\273\230\345\210\266\344\270\200\344\270\252\345\233\276\345\275\242.md.fc479c77.js" index 30c1572a..b78f7d7c 100644 --- "a/assets/guide_Fabric.js_\345\246\202\344\275\225\347\273\230\345\210\266\344\270\200\344\270\252\345\233\276\345\275\242.md.78353ed8.js" +++ "b/assets/guide_Fabric.js_\345\246\202\344\275\225\347\273\230\345\210\266\344\270\200\344\270\252\345\233\276\345\275\242.md.fc479c77.js" @@ -1,4 +1,4 @@ -import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/7.9514225d.png",o="/vitePress-blob/assets/8.9aa69160.png",e="/vitePress-blob/assets/9.1e766c59.png",t="/vitePress-blob/assets/10.655eecbc.png",c="/vitePress-blob/assets/11.58a7f11b.png",r="/vitePress-blob/assets/16.787d12a1.png",A=JSON.parse('{"title":"如何绘制一个图形","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Fabric.js/如何绘制一个图形.md","filePath":"guide/Fabric.js/如何绘制一个图形.md","lastUpdated":1736065650000}'),E={name:"guide/Fabric.js/如何绘制一个图形.md"},y=l(`

如何绘制一个图形

上一节中,大致了解了 Fabric 的内部结构,并且知道 Fabric.js 拥有俩层 Canvas。

上层 Canvas 称之为动态层,用于处理交互相关以及事件绑定

下层 Canvas 称之为静态层,用于获取数据,通过数据绘制图形

之前有编写过 React 的同学应该对父子组件的单向数据流有一定的印象,而 Fabric.js 的分层结构设计就很类似于这种单向数据流思想,

动态层用于获取数据输出到静态层进行渲染。

在这一节中,我们将以绘制矩形为例,了解 Fabric.js 是如何实现矩形的绘制的?

Fabric.js 是如何绘制矩形的

在 Fabric.js 中绘制矩形也非常简单,只需要初始化画布,添加矩形对象即可,代码如下👇:

html
<!-- html -->
+import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/7.9514225d.png",o="/vitePress-blob/assets/8.9aa69160.png",e="/vitePress-blob/assets/9.1e766c59.png",t="/vitePress-blob/assets/10.655eecbc.png",c="/vitePress-blob/assets/11.58a7f11b.png",r="/vitePress-blob/assets/16.787d12a1.png",A=JSON.parse('{"title":"如何绘制一个图形","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Fabric.js/如何绘制一个图形.md","filePath":"guide/Fabric.js/如何绘制一个图形.md","lastUpdated":1736839070000}'),E={name:"guide/Fabric.js/如何绘制一个图形.md"},y=l(`

如何绘制一个图形

上一节中,大致了解了 Fabric 的内部结构,并且知道 Fabric.js 拥有俩层 Canvas。

上层 Canvas 称之为动态层,用于处理交互相关以及事件绑定

下层 Canvas 称之为静态层,用于获取数据,通过数据绘制图形

之前有编写过 React 的同学应该对父子组件的单向数据流有一定的印象,而 Fabric.js 的分层结构设计就很类似于这种单向数据流思想,

动态层用于获取数据输出到静态层进行渲染。

在这一节中,我们将以绘制矩形为例,了解 Fabric.js 是如何实现矩形的绘制的?

Fabric.js 是如何绘制矩形的

在 Fabric.js 中绘制矩形也非常简单,只需要初始化画布,添加矩形对象即可,代码如下👇:

html
<!-- html -->
 <canvas id="canvas"></canvas>
<!-- html -->
 <canvas id="canvas"></canvas>
js
// js
 const canvas = new fabric.Canvas('canvas');
diff --git "a/assets/guide_Fabric.js_\345\246\202\344\275\225\347\273\230\345\210\266\344\270\200\344\270\252\345\233\276\345\275\242.md.78353ed8.lean.js" "b/assets/guide_Fabric.js_\345\246\202\344\275\225\347\273\230\345\210\266\344\270\200\344\270\252\345\233\276\345\275\242.md.fc479c77.lean.js"
similarity index 91%
rename from "assets/guide_Fabric.js_\345\246\202\344\275\225\347\273\230\345\210\266\344\270\200\344\270\252\345\233\276\345\275\242.md.78353ed8.lean.js"
rename to "assets/guide_Fabric.js_\345\246\202\344\275\225\347\273\230\345\210\266\344\270\200\344\270\252\345\233\276\345\275\242.md.fc479c77.lean.js"
index 49c7be1d..ba6a5f4e 100644
--- "a/assets/guide_Fabric.js_\345\246\202\344\275\225\347\273\230\345\210\266\344\270\200\344\270\252\345\233\276\345\275\242.md.78353ed8.lean.js"
+++ "b/assets/guide_Fabric.js_\345\246\202\344\275\225\347\273\230\345\210\266\344\270\200\344\270\252\345\233\276\345\275\242.md.fc479c77.lean.js"
@@ -1 +1 @@
-import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/7.9514225d.png",o="/vitePress-blob/assets/8.9aa69160.png",e="/vitePress-blob/assets/9.1e766c59.png",t="/vitePress-blob/assets/10.655eecbc.png",c="/vitePress-blob/assets/11.58a7f11b.png",r="/vitePress-blob/assets/16.787d12a1.png",A=JSON.parse('{"title":"如何绘制一个图形","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Fabric.js/如何绘制一个图形.md","filePath":"guide/Fabric.js/如何绘制一个图形.md","lastUpdated":1736065650000}'),E={name:"guide/Fabric.js/如何绘制一个图形.md"},y=l("",36),i=[y];function d(F,h,b,C,v,u){return a(),n("div",null,i)}const _=s(E,[["render",d]]);export{A as __pageData,_ as default};
+import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/7.9514225d.png",o="/vitePress-blob/assets/8.9aa69160.png",e="/vitePress-blob/assets/9.1e766c59.png",t="/vitePress-blob/assets/10.655eecbc.png",c="/vitePress-blob/assets/11.58a7f11b.png",r="/vitePress-blob/assets/16.787d12a1.png",A=JSON.parse('{"title":"如何绘制一个图形","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Fabric.js/如何绘制一个图形.md","filePath":"guide/Fabric.js/如何绘制一个图形.md","lastUpdated":1736839070000}'),E={name:"guide/Fabric.js/如何绘制一个图形.md"},y=l("",36),i=[y];function d(F,h,b,C,v,u){return a(),n("div",null,i)}const _=s(E,[["render",d]]);export{A as __pageData,_ as default};
diff --git "a/assets/guide_Fabric.js_\345\246\202\344\275\225\350\260\203\350\257\225Fabric.md.e3e1288a.js" "b/assets/guide_Fabric.js_\345\246\202\344\275\225\350\260\203\350\257\225Fabric.md.b88e0c09.js"
similarity index 99%
rename from "assets/guide_Fabric.js_\345\246\202\344\275\225\350\260\203\350\257\225Fabric.md.e3e1288a.js"
rename to "assets/guide_Fabric.js_\345\246\202\344\275\225\350\260\203\350\257\225Fabric.md.b88e0c09.js"
index da398a69..85672369 100644
--- "a/assets/guide_Fabric.js_\345\246\202\344\275\225\350\260\203\350\257\225Fabric.md.e3e1288a.js"
+++ "b/assets/guide_Fabric.js_\345\246\202\344\275\225\350\260\203\350\257\225Fabric.md.b88e0c09.js"
@@ -1 +1 @@
-import{_ as s,o as a,c as l,Q as e}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/调试Fabric.2a8c097f.png",F=JSON.parse('{"title":"如何调试 Fabric.js","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Fabric.js/如何调试Fabric.md","filePath":"guide/Fabric.js/如何调试Fabric.md","lastUpdated":1736065650000}'),n={name:"guide/Fabric.js/如何调试Fabric.md"},o=e('

如何调试 Fabric.js

为了更好的学习 Fabric.js 的源码,我们需要了解如何调试 Fabric.js。

第一步:下载源码

首先,我们需要下载 Fabric.js 的源码。Fabric.js 的源码托管在 GitHub 上,我们可以通过 git clone 命令将源码下载到本地。

bash
git clone https://github.com/fabricjs/fabricjs.com.git
git clone https://github.com/fabricjs/fabricjs.com.git

第二步:安装依赖

Fabric.js 使用 Jekyll 来提供页面服务,因此我们需要安装 Jekyll

Jekyll 提供了多种安装方式,在下将以 macOS 为例

Step 1: Install Homebrew

bash
/bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)"
/bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)"

Step 2: Install chruby and the latest Ruby with ruby-install

bash
brew install chruby ruby-install xz
brew install chruby ruby-install xz

Step 3: Install Ruby

bash
ruby-install ruby 3.1.3
ruby-install ruby 3.1.3

Step 4: 检查 ruby 是否安装成功

bash
ruby -v
ruby -v

它应该显示 ruby​​ 3.1.3p185(2022-11-24 修订版 1a6b16756e)或更新版本。 alt text

Step 5: 安装 Jekyll

bash
gem install jekyll
gem install jekyll

Step 6: 运行环境 安装完成后,在项目仓库根目录中运行命令 jekyll serve 即可.在此命令的控制台输出中,您将看到 Server address: <base_url> ,在浏览器中访问此 base_url ,您将看到 fabricjs.com 上的内容以及本地更改应用。

',20),t=[o];function c(r,i,y,b,d,h){return a(),l("div",null,t)}const E=s(n,[["render",c]]);export{F as __pageData,E as default}; +import{_ as s,o as a,c as l,Q as e}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/调试Fabric.2a8c097f.png",F=JSON.parse('{"title":"如何调试 Fabric.js","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Fabric.js/如何调试Fabric.md","filePath":"guide/Fabric.js/如何调试Fabric.md","lastUpdated":1736839070000}'),n={name:"guide/Fabric.js/如何调试Fabric.md"},o=e('

如何调试 Fabric.js

为了更好的学习 Fabric.js 的源码,我们需要了解如何调试 Fabric.js。

第一步:下载源码

首先,我们需要下载 Fabric.js 的源码。Fabric.js 的源码托管在 GitHub 上,我们可以通过 git clone 命令将源码下载到本地。

bash
git clone https://github.com/fabricjs/fabricjs.com.git
git clone https://github.com/fabricjs/fabricjs.com.git

第二步:安装依赖

Fabric.js 使用 Jekyll 来提供页面服务,因此我们需要安装 Jekyll

Jekyll 提供了多种安装方式,在下将以 macOS 为例

Step 1: Install Homebrew

bash
/bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)"
/bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)"

Step 2: Install chruby and the latest Ruby with ruby-install

bash
brew install chruby ruby-install xz
brew install chruby ruby-install xz

Step 3: Install Ruby

bash
ruby-install ruby 3.1.3
ruby-install ruby 3.1.3

Step 4: 检查 ruby 是否安装成功

bash
ruby -v
ruby -v

它应该显示 ruby​​ 3.1.3p185(2022-11-24 修订版 1a6b16756e)或更新版本。 alt text

Step 5: 安装 Jekyll

bash
gem install jekyll
gem install jekyll

Step 6: 运行环境 安装完成后,在项目仓库根目录中运行命令 jekyll serve 即可.在此命令的控制台输出中,您将看到 Server address: <base_url> ,在浏览器中访问此 base_url ,您将看到 fabricjs.com 上的内容以及本地更改应用。

',20),t=[o];function c(r,i,y,b,d,h){return a(),l("div",null,t)}const E=s(n,[["render",c]]);export{F as __pageData,E as default}; diff --git "a/assets/guide_Fabric.js_\345\246\202\344\275\225\350\260\203\350\257\225Fabric.md.e3e1288a.lean.js" "b/assets/guide_Fabric.js_\345\246\202\344\275\225\350\260\203\350\257\225Fabric.md.b88e0c09.lean.js" similarity index 88% rename from "assets/guide_Fabric.js_\345\246\202\344\275\225\350\260\203\350\257\225Fabric.md.e3e1288a.lean.js" rename to "assets/guide_Fabric.js_\345\246\202\344\275\225\350\260\203\350\257\225Fabric.md.b88e0c09.lean.js" index c62e009f..5e5d6a25 100644 --- "a/assets/guide_Fabric.js_\345\246\202\344\275\225\350\260\203\350\257\225Fabric.md.e3e1288a.lean.js" +++ "b/assets/guide_Fabric.js_\345\246\202\344\275\225\350\260\203\350\257\225Fabric.md.b88e0c09.lean.js" @@ -1 +1 @@ -import{_ as s,o as a,c as l,Q as e}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/调试Fabric.2a8c097f.png",F=JSON.parse('{"title":"如何调试 Fabric.js","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Fabric.js/如何调试Fabric.md","filePath":"guide/Fabric.js/如何调试Fabric.md","lastUpdated":1736065650000}'),n={name:"guide/Fabric.js/如何调试Fabric.md"},o=e("",20),t=[o];function c(r,i,y,b,d,h){return a(),l("div",null,t)}const E=s(n,[["render",c]]);export{F as __pageData,E as default}; +import{_ as s,o as a,c as l,Q as e}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/调试Fabric.2a8c097f.png",F=JSON.parse('{"title":"如何调试 Fabric.js","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Fabric.js/如何调试Fabric.md","filePath":"guide/Fabric.js/如何调试Fabric.md","lastUpdated":1736839070000}'),n={name:"guide/Fabric.js/如何调试Fabric.md"},o=e("",20),t=[o];function c(r,i,y,b,d,h){return a(),l("div",null,t)}const E=s(n,[["render",c]]);export{F as __pageData,E as default}; diff --git "a/assets/guide_Node\347\233\270\345\205\263_\346\246\202\350\246\201.md.6d2f3b80.js" "b/assets/guide_Node\347\233\270\345\205\263_\346\246\202\350\246\201.md.5e89b8cf.js" similarity index 84% rename from "assets/guide_Node\347\233\270\345\205\263_\346\246\202\350\246\201.md.6d2f3b80.js" rename to "assets/guide_Node\347\233\270\345\205\263_\346\246\202\350\246\201.md.5e89b8cf.js" index cea2cc21..bfadf201 100644 --- "a/assets/guide_Node\347\233\270\345\205\263_\346\246\202\350\246\201.md.6d2f3b80.js" +++ "b/assets/guide_Node\347\233\270\345\205\263_\346\246\202\350\246\201.md.5e89b8cf.js" @@ -1 +1 @@ -import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Node相关/概要.md","filePath":"guide/Node相关/概要.md","lastUpdated":1736065650000}'),o={name:"guide/Node相关/概要.md"};function d(r,s,c,n,i,_){return t(),a("div")}const f=e(o,[["render",d]]);export{m as __pageData,f as default}; +import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Node相关/概要.md","filePath":"guide/Node相关/概要.md","lastUpdated":1736839070000}'),o={name:"guide/Node相关/概要.md"};function d(r,s,c,n,i,_){return t(),a("div")}const f=e(o,[["render",d]]);export{m as __pageData,f as default}; diff --git "a/assets/guide_Node\347\233\270\345\205\263_\346\246\202\350\246\201.md.6d2f3b80.lean.js" "b/assets/guide_Node\347\233\270\345\205\263_\346\246\202\350\246\201.md.5e89b8cf.lean.js" similarity index 84% rename from "assets/guide_Node\347\233\270\345\205\263_\346\246\202\350\246\201.md.6d2f3b80.lean.js" rename to "assets/guide_Node\347\233\270\345\205\263_\346\246\202\350\246\201.md.5e89b8cf.lean.js" index cea2cc21..bfadf201 100644 --- "a/assets/guide_Node\347\233\270\345\205\263_\346\246\202\350\246\201.md.6d2f3b80.lean.js" +++ "b/assets/guide_Node\347\233\270\345\205\263_\346\246\202\350\246\201.md.5e89b8cf.lean.js" @@ -1 +1 @@ -import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Node相关/概要.md","filePath":"guide/Node相关/概要.md","lastUpdated":1736065650000}'),o={name:"guide/Node相关/概要.md"};function d(r,s,c,n,i,_){return t(),a("div")}const f=e(o,[["render",d]]);export{m as __pageData,f as default}; +import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Node相关/概要.md","filePath":"guide/Node相关/概要.md","lastUpdated":1736839070000}'),o={name:"guide/Node相关/概要.md"};function d(r,s,c,n,i,_){return t(),a("div")}const f=e(o,[["render",d]]);export{m as __pageData,f as default}; diff --git a/assets/guide_React_Essential Knowledge for Schedule about React.md.5abf05d0.js b/assets/guide_React_Essential Knowledge for Schedule about React.md.8a28d74e.js similarity index 99% rename from assets/guide_React_Essential Knowledge for Schedule about React.md.5abf05d0.js rename to assets/guide_React_Essential Knowledge for Schedule about React.md.8a28d74e.js index 60fddf17..d753fb22 100644 --- a/assets/guide_React_Essential Knowledge for Schedule about React.md.5abf05d0.js +++ b/assets/guide_React_Essential Knowledge for Schedule about React.md.8a28d74e.js @@ -1,4 +1,4 @@ -import{_ as a,o as n,c as l,Q as o}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/9.659126c5.png",s="/vitePress-blob/assets/2.f0f92321.png",e="/vitePress-blob/assets/3.763bc96f.png",t="/vitePress-blob/assets/4.c5d9e973.png",r="/vitePress-blob/assets/5.17c05605.png",c="/vitePress-blob/assets/6.b913b10e.png",E="/vitePress-blob/assets/7.ad239368.png",y="/vitePress-blob/assets/8.cb09fdac.png",w=JSON.parse('{"title":"[React Scheduler] Essential Knowledge for Scheduling API","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/Essential Knowledge for Schedule about React.md","filePath":"guide/React/Essential Knowledge for Schedule about React.md","lastUpdated":1736065650000}'),i={name:"guide/React/Essential Knowledge for Schedule about React.md"},d=o('

[React Scheduler] Essential Knowledge for Scheduling API

Frame

Introduction

This article summarizes key points about Scheduling API (e.g., requestAnimationFrame, requestIdleCallback, setTimeout, MessageChannel, microTask ( Promise, MutationObserver )).

Screen Refresh Rate

The screen refresh rate is the number of times the screen is updated per second. The screen refresh rate is usually 60Hz, which means the screen is updated 60 times per second.

In The browser, it will try to match the screen refresh rate with the frame rate of the web page.

If the frame rate is higher than 60HZ, the browser will skip some frames.

If the frame rate is lower than the screen refresh rate, the browser will render the same frame multiple times, which will cause the screen to block.

The budgeted time what is rendering per frame is 16.66 ms (1 second /60), so when writing code, be careful to take lower than 16ms of work per frame. Within each frame, the browser does the following:

  1. Execute Macro Task、Micro Task、Event Handler and so on.
  2. Execute requestAnimationFrame.
  3. Update Render Tree
  4. Execute requestIdleCallback if there is time left.
html
<!DOCTYPE html>
+import{_ as a,o as n,c as l,Q as o}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/9.659126c5.png",s="/vitePress-blob/assets/2.f0f92321.png",e="/vitePress-blob/assets/3.763bc96f.png",t="/vitePress-blob/assets/4.c5d9e973.png",r="/vitePress-blob/assets/5.17c05605.png",c="/vitePress-blob/assets/6.b913b10e.png",E="/vitePress-blob/assets/7.ad239368.png",y="/vitePress-blob/assets/8.cb09fdac.png",w=JSON.parse('{"title":"[React Scheduler] Essential Knowledge for Scheduling API","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/Essential Knowledge for Schedule about React.md","filePath":"guide/React/Essential Knowledge for Schedule about React.md","lastUpdated":1736839070000}'),i={name:"guide/React/Essential Knowledge for Schedule about React.md"},d=o('

[React Scheduler] Essential Knowledge for Scheduling API

Frame

Introduction

This article summarizes key points about Scheduling API (e.g., requestAnimationFrame, requestIdleCallback, setTimeout, MessageChannel, microTask ( Promise, MutationObserver )).

Screen Refresh Rate

The screen refresh rate is the number of times the screen is updated per second. The screen refresh rate is usually 60Hz, which means the screen is updated 60 times per second.

In The browser, it will try to match the screen refresh rate with the frame rate of the web page.

If the frame rate is higher than 60HZ, the browser will skip some frames.

If the frame rate is lower than the screen refresh rate, the browser will render the same frame multiple times, which will cause the screen to block.

The budgeted time what is rendering per frame is 16.66 ms (1 second /60), so when writing code, be careful to take lower than 16ms of work per frame. Within each frame, the browser does the following:

  1. Execute Macro Task、Micro Task、Event Handler and so on.
  2. Execute requestAnimationFrame.
  3. Update Render Tree
  4. Execute requestIdleCallback if there is time left.
html
<!DOCTYPE html>
 <html lang="en">
   <head>
     <meta charset="utf-8" />
diff --git a/assets/guide_React_Essential Knowledge for Schedule about React.md.5abf05d0.lean.js b/assets/guide_React_Essential Knowledge for Schedule about React.md.8a28d74e.lean.js
similarity index 93%
rename from assets/guide_React_Essential Knowledge for Schedule about React.md.5abf05d0.lean.js
rename to assets/guide_React_Essential Knowledge for Schedule about React.md.8a28d74e.lean.js
index c9bd7099..8afd57d2 100644
--- a/assets/guide_React_Essential Knowledge for Schedule about React.md.5abf05d0.lean.js	
+++ b/assets/guide_React_Essential Knowledge for Schedule about React.md.8a28d74e.lean.js	
@@ -1 +1 @@
-import{_ as a,o as n,c as l,Q as o}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/9.659126c5.png",s="/vitePress-blob/assets/2.f0f92321.png",e="/vitePress-blob/assets/3.763bc96f.png",t="/vitePress-blob/assets/4.c5d9e973.png",r="/vitePress-blob/assets/5.17c05605.png",c="/vitePress-blob/assets/6.b913b10e.png",E="/vitePress-blob/assets/7.ad239368.png",y="/vitePress-blob/assets/8.cb09fdac.png",w=JSON.parse('{"title":"[React Scheduler] Essential Knowledge for Scheduling API","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/Essential Knowledge for Schedule about React.md","filePath":"guide/React/Essential Knowledge for Schedule about React.md","lastUpdated":1736065650000}'),i={name:"guide/React/Essential Knowledge for Schedule about React.md"},d=o("",70),u=[d];function m(F,h,C,b,g,k){return n(),l("div",null,u)}const A=a(i,[["render",m]]);export{w as __pageData,A as default};
+import{_ as a,o as n,c as l,Q as o}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/9.659126c5.png",s="/vitePress-blob/assets/2.f0f92321.png",e="/vitePress-blob/assets/3.763bc96f.png",t="/vitePress-blob/assets/4.c5d9e973.png",r="/vitePress-blob/assets/5.17c05605.png",c="/vitePress-blob/assets/6.b913b10e.png",E="/vitePress-blob/assets/7.ad239368.png",y="/vitePress-blob/assets/8.cb09fdac.png",w=JSON.parse('{"title":"[React Scheduler] Essential Knowledge for Scheduling API","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/Essential Knowledge for Schedule about React.md","filePath":"guide/React/Essential Knowledge for Schedule about React.md","lastUpdated":1736839070000}'),i={name:"guide/React/Essential Knowledge for Schedule about React.md"},d=o("",70),u=[d];function m(F,h,C,b,g,k){return n(),l("div",null,u)}const A=a(i,[["render",m]]);export{w as __pageData,A as default};
diff --git "a/assets/guide_React_React\344\270\216Vue\347\232\204\345\214\272\345\210\253.md.aefdc566.js" "b/assets/guide_React_React\344\270\216Vue\347\232\204\345\214\272\345\210\253.md.11b5926a.js"
similarity index 85%
rename from "assets/guide_React_React\344\270\216Vue\347\232\204\345\214\272\345\210\253.md.aefdc566.js"
rename to "assets/guide_React_React\344\270\216Vue\347\232\204\345\214\272\345\210\253.md.11b5926a.js"
index dca6d5dc..5e999615 100644
--- "a/assets/guide_React_React\344\270\216Vue\347\232\204\345\214\272\345\210\253.md.aefdc566.js"
+++ "b/assets/guide_React_React\344\270\216Vue\347\232\204\345\214\272\345\210\253.md.11b5926a.js"
@@ -1 +1 @@
-import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const u=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React与Vue的区别.md","filePath":"guide/React/React与Vue的区别.md","lastUpdated":1736065650000}'),c={name:"guide/React/React与Vue的区别.md"};function r(o,s,d,n,_,i){return t(),a("div")}const m=e(c,[["render",r]]);export{u as __pageData,m as default};
+import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const u=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React与Vue的区别.md","filePath":"guide/React/React与Vue的区别.md","lastUpdated":1736839070000}'),c={name:"guide/React/React与Vue的区别.md"};function r(o,s,d,n,_,i){return t(),a("div")}const m=e(c,[["render",r]]);export{u as __pageData,m as default};
diff --git "a/assets/guide_React_React\344\270\216Vue\347\232\204\345\214\272\345\210\253.md.aefdc566.lean.js" "b/assets/guide_React_React\344\270\216Vue\347\232\204\345\214\272\345\210\253.md.11b5926a.lean.js"
similarity index 85%
rename from "assets/guide_React_React\344\270\216Vue\347\232\204\345\214\272\345\210\253.md.aefdc566.lean.js"
rename to "assets/guide_React_React\344\270\216Vue\347\232\204\345\214\272\345\210\253.md.11b5926a.lean.js"
index dca6d5dc..5e999615 100644
--- "a/assets/guide_React_React\344\270\216Vue\347\232\204\345\214\272\345\210\253.md.aefdc566.lean.js"
+++ "b/assets/guide_React_React\344\270\216Vue\347\232\204\345\214\272\345\210\253.md.11b5926a.lean.js"
@@ -1 +1 @@
-import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const u=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React与Vue的区别.md","filePath":"guide/React/React与Vue的区别.md","lastUpdated":1736065650000}'),c={name:"guide/React/React与Vue的区别.md"};function r(o,s,d,n,_,i){return t(),a("div")}const m=e(c,[["render",r]]);export{u as __pageData,m as default};
+import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const u=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React与Vue的区别.md","filePath":"guide/React/React与Vue的区别.md","lastUpdated":1736839070000}'),c={name:"guide/React/React与Vue的区别.md"};function r(o,s,d,n,_,i){return t(),a("div")}const m=e(c,[["render",r]]);export{u as __pageData,m as default};
diff --git "a/assets/guide_React_React\344\270\255\347\232\204\344\272\213\344\273\266\346\234\272\345\210\266.md.4c13470e.js" "b/assets/guide_React_React\344\270\255\347\232\204\344\272\213\344\273\266\346\234\272\345\210\266.md.d11dda49.js"
similarity index 95%
rename from "assets/guide_React_React\344\270\255\347\232\204\344\272\213\344\273\266\346\234\272\345\210\266.md.4c13470e.js"
rename to "assets/guide_React_React\344\270\255\347\232\204\344\272\213\344\273\266\346\234\272\345\210\266.md.d11dda49.js"
index 1cd8b046..a0477f84 100644
--- "a/assets/guide_React_React\344\270\255\347\232\204\344\272\213\344\273\266\346\234\272\345\210\266.md.4c13470e.js"
+++ "b/assets/guide_React_React\344\270\255\347\232\204\344\272\213\344\273\266\346\234\272\345\210\266.md.d11dda49.js"
@@ -1 +1 @@
-import{_ as a,o as e,c as t,Q as c}from"./chunks/framework.b6910bb2.js";const p=JSON.parse('{"title":"React 事件机制","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React中的事件机制.md","filePath":"guide/React/React中的事件机制.md","lastUpdated":1736065650000}'),r={name:"guide/React/React中的事件机制.md"},o=c('

React 事件机制

概要

为了抹平各个浏览器之间的差异以及统一维护和管理事件,React 自行实现了合成事件,用来模拟原生事件的行为 在 React 16 之前,React 的合成事件放在了 document 的冒泡阶段 在 React 16 之后,为了兼容各个版本,React 的合成事件放在了 根节点 的捕获和冒泡阶段

具体实现

',4),_=[o];function d(i,n,s,l,h,R){return e(),t("div",null,_)}const m=a(r,[["render",d]]);export{p as __pageData,m as default}; +import{_ as a,o as e,c as t,Q as c}from"./chunks/framework.b6910bb2.js";const p=JSON.parse('{"title":"React 事件机制","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React中的事件机制.md","filePath":"guide/React/React中的事件机制.md","lastUpdated":1736839070000}'),r={name:"guide/React/React中的事件机制.md"},o=c('

React 事件机制

概要

为了抹平各个浏览器之间的差异以及统一维护和管理事件,React 自行实现了合成事件,用来模拟原生事件的行为 在 React 16 之前,React 的合成事件放在了 document 的冒泡阶段 在 React 16 之后,为了兼容各个版本,React 的合成事件放在了 根节点 的捕获和冒泡阶段

具体实现

',4),_=[o];function d(i,n,s,l,h,R){return e(),t("div",null,_)}const m=a(r,[["render",d]]);export{p as __pageData,m as default}; diff --git "a/assets/guide_React_React\344\270\255\347\232\204\344\272\213\344\273\266\346\234\272\345\210\266.md.4c13470e.lean.js" "b/assets/guide_React_React\344\270\255\347\232\204\344\272\213\344\273\266\346\234\272\345\210\266.md.d11dda49.lean.js" similarity index 86% rename from "assets/guide_React_React\344\270\255\347\232\204\344\272\213\344\273\266\346\234\272\345\210\266.md.4c13470e.lean.js" rename to "assets/guide_React_React\344\270\255\347\232\204\344\272\213\344\273\266\346\234\272\345\210\266.md.d11dda49.lean.js" index eb31198b..449fe7e0 100644 --- "a/assets/guide_React_React\344\270\255\347\232\204\344\272\213\344\273\266\346\234\272\345\210\266.md.4c13470e.lean.js" +++ "b/assets/guide_React_React\344\270\255\347\232\204\344\272\213\344\273\266\346\234\272\345\210\266.md.d11dda49.lean.js" @@ -1 +1 @@ -import{_ as a,o as e,c as t,Q as c}from"./chunks/framework.b6910bb2.js";const p=JSON.parse('{"title":"React 事件机制","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React中的事件机制.md","filePath":"guide/React/React中的事件机制.md","lastUpdated":1736065650000}'),r={name:"guide/React/React中的事件机制.md"},o=c("",4),_=[o];function d(i,n,s,l,h,R){return e(),t("div",null,_)}const m=a(r,[["render",d]]);export{p as __pageData,m as default}; +import{_ as a,o as e,c as t,Q as c}from"./chunks/framework.b6910bb2.js";const p=JSON.parse('{"title":"React 事件机制","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React中的事件机制.md","filePath":"guide/React/React中的事件机制.md","lastUpdated":1736839070000}'),r={name:"guide/React/React中的事件机制.md"},o=c("",4),_=[o];function d(i,n,s,l,h,R){return e(),t("div",null,_)}const m=a(r,[["render",d]]);export{p as __pageData,m as default}; diff --git "a/assets/guide_React_React\344\270\255\347\232\204\345\274\202\345\270\270\346\234\272\345\210\266.md.b181d518.js" "b/assets/guide_React_React\344\270\255\347\232\204\345\274\202\345\270\270\346\234\272\345\210\266.md.ac78472e.js" similarity index 85% rename from "assets/guide_React_React\344\270\255\347\232\204\345\274\202\345\270\270\346\234\272\345\210\266.md.b181d518.js" rename to "assets/guide_React_React\344\270\255\347\232\204\345\274\202\345\270\270\346\234\272\345\210\266.md.ac78472e.js" index dae56ae2..ed734527 100644 --- "a/assets/guide_React_React\344\270\255\347\232\204\345\274\202\345\270\270\346\234\272\345\210\266.md.b181d518.js" +++ "b/assets/guide_React_React\344\270\255\347\232\204\345\274\202\345\270\270\346\234\272\345\210\266.md.ac78472e.js" @@ -1 +1 @@ -import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React中的异常机制.md","filePath":"guide/React/React中的异常机制.md","lastUpdated":1736065650000}'),c={name:"guide/React/React中的异常机制.md"};function r(o,s,_,d,n,i){return t(),a("div")}const f=e(c,[["render",r]]);export{m as __pageData,f as default}; +import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React中的异常机制.md","filePath":"guide/React/React中的异常机制.md","lastUpdated":1736839070000}'),c={name:"guide/React/React中的异常机制.md"};function r(o,s,_,d,n,i){return t(),a("div")}const f=e(c,[["render",r]]);export{m as __pageData,f as default}; diff --git "a/assets/guide_React_React\344\270\255\347\232\204\345\274\202\345\270\270\346\234\272\345\210\266.md.b181d518.lean.js" "b/assets/guide_React_React\344\270\255\347\232\204\345\274\202\345\270\270\346\234\272\345\210\266.md.ac78472e.lean.js" similarity index 85% rename from "assets/guide_React_React\344\270\255\347\232\204\345\274\202\345\270\270\346\234\272\345\210\266.md.b181d518.lean.js" rename to "assets/guide_React_React\344\270\255\347\232\204\345\274\202\345\270\270\346\234\272\345\210\266.md.ac78472e.lean.js" index dae56ae2..ed734527 100644 --- "a/assets/guide_React_React\344\270\255\347\232\204\345\274\202\345\270\270\346\234\272\345\210\266.md.b181d518.lean.js" +++ "b/assets/guide_React_React\344\270\255\347\232\204\345\274\202\345\270\270\346\234\272\345\210\266.md.ac78472e.lean.js" @@ -1 +1 @@ -import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React中的异常机制.md","filePath":"guide/React/React中的异常机制.md","lastUpdated":1736065650000}'),c={name:"guide/React/React中的异常机制.md"};function r(o,s,_,d,n,i){return t(),a("div")}const f=e(c,[["render",r]]);export{m as __pageData,f as default}; +import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React中的异常机制.md","filePath":"guide/React/React中的异常机制.md","lastUpdated":1736839070000}'),c={name:"guide/React/React中的异常机制.md"};function r(o,s,_,d,n,i){return t(),a("div")}const f=e(c,[["render",r]]);export{m as __pageData,f as default}; diff --git "a/assets/guide_React_React\344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.md.6ad45a79.js" "b/assets/guide_React_React\344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.md.780419b2.js" similarity index 99% rename from "assets/guide_React_React\344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.md.6ad45a79.js" rename to "assets/guide_React_React\344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.md.780419b2.js" index dfb1f1be..81a46365 100644 --- "a/assets/guide_React_React\344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.md.6ad45a79.js" +++ "b/assets/guide_React_React\344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.md.780419b2.js" @@ -1,4 +1,4 @@ -import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const u=JSON.parse('{"title":"React 中的性能优化","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React中的性能优化.md","filePath":"guide/React/React中的性能优化.md","lastUpdated":1736065650000}'),p={name:"guide/React/React中的性能优化.md"},o=l(`

React 中的性能优化

在 React 中当我们的父组件更新时,将渲染整个子组件树,这样会造成很大的性能开销,所以我们需要对组件进行优化,避免不必要的渲染。

常见的性能优化手段有:

  1. 减少不必要的渲染

    1. 使用 useMemo 缓存数据、使用 useCallback 缓存函数
    2. 使用 React.memo 缓存组件
    3. 合理的使用 Key
    4. 使用 Fragment 避免额外标记
    5. 通过 Suspense 和 Lazy 拆分组件
  2. 通用方案

    1. 在组件销毁的时候清除定时器/事件
    2. 频繁切换时使用 display: none; 避免不必要的性能开销
    3. 为组件创建错误边界

1 使用 useMemo 缓存数据、使用 useCallback 缓存函数

jsx
/**
+import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const u=JSON.parse('{"title":"React 中的性能优化","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React中的性能优化.md","filePath":"guide/React/React中的性能优化.md","lastUpdated":1736839070000}'),p={name:"guide/React/React中的性能优化.md"},o=l(`

React 中的性能优化

在 React 中当我们的父组件更新时,将渲染整个子组件树,这样会造成很大的性能开销,所以我们需要对组件进行优化,避免不必要的渲染。

常见的性能优化手段有:

  1. 减少不必要的渲染

    1. 使用 useMemo 缓存数据、使用 useCallback 缓存函数
    2. 使用 React.memo 缓存组件
    3. 合理的使用 Key
    4. 使用 Fragment 避免额外标记
    5. 通过 Suspense 和 Lazy 拆分组件
  2. 通用方案

    1. 在组件销毁的时候清除定时器/事件
    2. 频繁切换时使用 display: none; 避免不必要的性能开销
    3. 为组件创建错误边界

1 使用 useMemo 缓存数据、使用 useCallback 缓存函数

jsx
/**
  * 使用 useMemo 缓存数据,类似于 Vue 的 computed 计算属性
  * 使用 useCallback 缓存函数
  * @returns
diff --git "a/assets/guide_React_React\344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.md.6ad45a79.lean.js" "b/assets/guide_React_React\344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.md.780419b2.lean.js"
similarity index 87%
rename from "assets/guide_React_React\344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.md.6ad45a79.lean.js"
rename to "assets/guide_React_React\344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.md.780419b2.lean.js"
index be766881..4c761625 100644
--- "a/assets/guide_React_React\344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.md.6ad45a79.lean.js"
+++ "b/assets/guide_React_React\344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.md.780419b2.lean.js"
@@ -1 +1 @@
-import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const u=JSON.parse('{"title":"React 中的性能优化","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React中的性能优化.md","filePath":"guide/React/React中的性能优化.md","lastUpdated":1736065650000}'),p={name:"guide/React/React中的性能优化.md"},o=l("",41),e=[o];function t(c,r,E,y,i,F){return n(),a("div",null,e)}const A=s(p,[["render",t]]);export{u as __pageData,A as default};
+import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const u=JSON.parse('{"title":"React 中的性能优化","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React中的性能优化.md","filePath":"guide/React/React中的性能优化.md","lastUpdated":1736839070000}'),p={name:"guide/React/React中的性能优化.md"},o=l("",41),e=[o];function t(c,r,E,y,i,F){return n(),a("div",null,e)}const A=s(p,[["render",t]]);export{u as __pageData,A as default};
diff --git "a/assets/guide_React_React\344\270\272\344\273\200\344\271\210\350\246\201\344\275\277\347\224\250JSX.md.62407e1e.js" "b/assets/guide_React_React\344\270\272\344\273\200\344\271\210\350\246\201\344\275\277\347\224\250JSX.md.40e22e85.js"
similarity index 94%
rename from "assets/guide_React_React\344\270\272\344\273\200\344\271\210\350\246\201\344\275\277\347\224\250JSX.md.62407e1e.js"
rename to "assets/guide_React_React\344\270\272\344\273\200\344\271\210\350\246\201\344\275\277\347\224\250JSX.md.40e22e85.js"
index 6cec0d37..abb895b9 100644
--- "a/assets/guide_React_React\344\270\272\344\273\200\344\271\210\350\246\201\344\275\277\347\224\250JSX.md.62407e1e.js"
+++ "b/assets/guide_React_React\344\270\272\344\273\200\344\271\210\350\246\201\344\275\277\347\224\250JSX.md.40e22e85.js"
@@ -1 +1 @@
-import{_ as t,o as a,c,k as e,a as s}from"./chunks/framework.b6910bb2.js";const X=JSON.parse('{"title":"React 为什么要使用 JSX ?","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React为什么要使用JSX.md","filePath":"guide/React/React为什么要使用JSX.md","lastUpdated":1736065650000}'),r={name:"guide/React/React为什么要使用JSX.md"},o=e("h1",{id:"react-为什么要使用-jsx",tabindex:"-1"},[s("React 为什么要使用 JSX ? "),e("a",{class:"header-anchor",href:"#react-为什么要使用-jsx","aria-label":'Permalink to "React 为什么要使用 JSX ?"'},"​")],-1),n=e("p",null,"JSX 是 JavaScript 语法拓展,结构类似于 XML,它可以在 JavaScript 中直接使用 HTML 标签,这样可以方便开发者编写组件,提高开发效率。",-1),_=e("p",null,"没有JSX时,也可以使用 React.createElement 实现一个组件,但是这样的代码可读性很差,而且不方便复用,所以 React 推荐使用 JSX。",-1),d=[o,n,_];function i(l,p,J,S,h,R){return a(),c("div",null,d)}const u=t(r,[["render",i]]);export{X as __pageData,u as default};
+import{_ as t,o as a,c,k as e,a as s}from"./chunks/framework.b6910bb2.js";const X=JSON.parse('{"title":"React 为什么要使用 JSX ?","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React为什么要使用JSX.md","filePath":"guide/React/React为什么要使用JSX.md","lastUpdated":1736839070000}'),r={name:"guide/React/React为什么要使用JSX.md"},o=e("h1",{id:"react-为什么要使用-jsx",tabindex:"-1"},[s("React 为什么要使用 JSX ? "),e("a",{class:"header-anchor",href:"#react-为什么要使用-jsx","aria-label":'Permalink to "React 为什么要使用 JSX ?"'},"​")],-1),n=e("p",null,"JSX 是 JavaScript 语法拓展,结构类似于 XML,它可以在 JavaScript 中直接使用 HTML 标签,这样可以方便开发者编写组件,提高开发效率。",-1),_=e("p",null,"没有JSX时,也可以使用 React.createElement 实现一个组件,但是这样的代码可读性很差,而且不方便复用,所以 React 推荐使用 JSX。",-1),d=[o,n,_];function i(l,p,J,S,h,R){return a(),c("div",null,d)}const u=t(r,[["render",i]]);export{X as __pageData,u as default};
diff --git "a/assets/guide_React_React\344\270\272\344\273\200\344\271\210\350\246\201\344\275\277\347\224\250JSX.md.62407e1e.lean.js" "b/assets/guide_React_React\344\270\272\344\273\200\344\271\210\350\246\201\344\275\277\347\224\250JSX.md.40e22e85.lean.js"
similarity index 94%
rename from "assets/guide_React_React\344\270\272\344\273\200\344\271\210\350\246\201\344\275\277\347\224\250JSX.md.62407e1e.lean.js"
rename to "assets/guide_React_React\344\270\272\344\273\200\344\271\210\350\246\201\344\275\277\347\224\250JSX.md.40e22e85.lean.js"
index 6cec0d37..abb895b9 100644
--- "a/assets/guide_React_React\344\270\272\344\273\200\344\271\210\350\246\201\344\275\277\347\224\250JSX.md.62407e1e.lean.js"
+++ "b/assets/guide_React_React\344\270\272\344\273\200\344\271\210\350\246\201\344\275\277\347\224\250JSX.md.40e22e85.lean.js"
@@ -1 +1 @@
-import{_ as t,o as a,c,k as e,a as s}from"./chunks/framework.b6910bb2.js";const X=JSON.parse('{"title":"React 为什么要使用 JSX ?","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React为什么要使用JSX.md","filePath":"guide/React/React为什么要使用JSX.md","lastUpdated":1736065650000}'),r={name:"guide/React/React为什么要使用JSX.md"},o=e("h1",{id:"react-为什么要使用-jsx",tabindex:"-1"},[s("React 为什么要使用 JSX ? "),e("a",{class:"header-anchor",href:"#react-为什么要使用-jsx","aria-label":'Permalink to "React 为什么要使用 JSX ?"'},"​")],-1),n=e("p",null,"JSX 是 JavaScript 语法拓展,结构类似于 XML,它可以在 JavaScript 中直接使用 HTML 标签,这样可以方便开发者编写组件,提高开发效率。",-1),_=e("p",null,"没有JSX时,也可以使用 React.createElement 实现一个组件,但是这样的代码可读性很差,而且不方便复用,所以 React 推荐使用 JSX。",-1),d=[o,n,_];function i(l,p,J,S,h,R){return a(),c("div",null,d)}const u=t(r,[["render",i]]);export{X as __pageData,u as default};
+import{_ as t,o as a,c,k as e,a as s}from"./chunks/framework.b6910bb2.js";const X=JSON.parse('{"title":"React 为什么要使用 JSX ?","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React为什么要使用JSX.md","filePath":"guide/React/React为什么要使用JSX.md","lastUpdated":1736839070000}'),r={name:"guide/React/React为什么要使用JSX.md"},o=e("h1",{id:"react-为什么要使用-jsx",tabindex:"-1"},[s("React 为什么要使用 JSX ? "),e("a",{class:"header-anchor",href:"#react-为什么要使用-jsx","aria-label":'Permalink to "React 为什么要使用 JSX ?"'},"​")],-1),n=e("p",null,"JSX 是 JavaScript 语法拓展,结构类似于 XML,它可以在 JavaScript 中直接使用 HTML 标签,这样可以方便开发者编写组件,提高开发效率。",-1),_=e("p",null,"没有JSX时,也可以使用 React.createElement 实现一个组件,但是这样的代码可读性很差,而且不方便复用,所以 React 推荐使用 JSX。",-1),d=[o,n,_];function i(l,p,J,S,h,R){return a(),c("div",null,d)}const u=t(r,[["render",i]]);export{X as __pageData,u as default};
diff --git "a/assets/guide_React_React\346\230\257\344\273\200\344\271\210.md.471a44bf.js" "b/assets/guide_React_React\346\230\257\344\273\200\344\271\210.md.ee0c3664.js"
similarity index 98%
rename from "assets/guide_React_React\346\230\257\344\273\200\344\271\210.md.471a44bf.js"
rename to "assets/guide_React_React\346\230\257\344\273\200\344\271\210.md.ee0c3664.js"
index aacf8ca3..dda2bede 100644
--- "a/assets/guide_React_React\346\230\257\344\273\200\344\271\210.md.471a44bf.js"
+++ "b/assets/guide_React_React\346\230\257\344\273\200\344\271\210.md.ee0c3664.js"
@@ -1,4 +1,4 @@
-import{_ as s,o as a,c as o,Q as n}from"./chunks/framework.b6910bb2.js";const _=JSON.parse('{"title":"React 是什么?","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React是什么.md","filePath":"guide/React/React是什么.md","lastUpdated":1736065650000}'),e={name:"guide/React/React是什么.md"},l=n(`

React 是什么?

React 是一个用于构建用户界面的 JavaScript 库, 通过组件化的方式解决视图层复用的问题。

它的核心思路是: 声明式、组件化、通用型

声明式

相比 JQ 的命令式编程: $(body).css('color', 'red');, 声明式编程更加直观、便于复用:

jsx
const Body = (props) => {
+import{_ as s,o as a,c as o,Q as n}from"./chunks/framework.b6910bb2.js";const _=JSON.parse('{"title":"React 是什么?","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React是什么.md","filePath":"guide/React/React是什么.md","lastUpdated":1736839070000}'),e={name:"guide/React/React是什么.md"},l=n(`

React 是什么?

React 是一个用于构建用户界面的 JavaScript 库, 通过组件化的方式解决视图层复用的问题。

它的核心思路是: 声明式、组件化、通用型

声明式

相比 JQ 的命令式编程: $(body).css('color', 'red');, 声明式编程更加直观、便于复用:

jsx
const Body = (props) => {
    return <div style={{color: 'red'}}>{props.children}</div>
 }
const Body = (props) => {
    return <div style={{color: 'red'}}>{props.children}</div>
diff --git "a/assets/guide_React_React\346\230\257\344\273\200\344\271\210.md.471a44bf.lean.js" "b/assets/guide_React_React\346\230\257\344\273\200\344\271\210.md.ee0c3664.lean.js"
similarity index 86%
rename from "assets/guide_React_React\346\230\257\344\273\200\344\271\210.md.471a44bf.lean.js"
rename to "assets/guide_React_React\346\230\257\344\273\200\344\271\210.md.ee0c3664.lean.js"
index 06221b88..0cf610e9 100644
--- "a/assets/guide_React_React\346\230\257\344\273\200\344\271\210.md.471a44bf.lean.js"
+++ "b/assets/guide_React_React\346\230\257\344\273\200\344\271\210.md.ee0c3664.lean.js"
@@ -1 +1 @@
-import{_ as s,o as a,c as o,Q as n}from"./chunks/framework.b6910bb2.js";const _=JSON.parse('{"title":"React 是什么?","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React是什么.md","filePath":"guide/React/React是什么.md","lastUpdated":1736065650000}'),e={name:"guide/React/React是什么.md"},l=n("",12),p=[l];function t(c,r,i,d,E,y){return a(),o("div",null,p)}const u=s(e,[["render",t]]);export{_ as __pageData,u as default};
+import{_ as s,o as a,c as o,Q as n}from"./chunks/framework.b6910bb2.js";const _=JSON.parse('{"title":"React 是什么?","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React是什么.md","filePath":"guide/React/React是什么.md","lastUpdated":1736839070000}'),e={name:"guide/React/React是什么.md"},l=n("",12),p=[l];function t(c,r,i,d,E,y){return a(),o("div",null,p)}const u=s(e,[["render",t]]);export{_ as __pageData,u as default};
diff --git "a/assets/guide_React_React\346\230\257\345\246\202\344\275\225\346\270\262\346\237\223\347\232\204.md.ee09ffde.js" "b/assets/guide_React_React\346\230\257\345\246\202\344\275\225\346\270\262\346\237\223\347\232\204.md.f058feee.js"
similarity index 99%
rename from "assets/guide_React_React\346\230\257\345\246\202\344\275\225\346\270\262\346\237\223\347\232\204.md.ee09ffde.js"
rename to "assets/guide_React_React\346\230\257\345\246\202\344\275\225\346\270\262\346\237\223\347\232\204.md.f058feee.js"
index 635b6007..896ee098 100644
--- "a/assets/guide_React_React\346\230\257\345\246\202\344\275\225\346\270\262\346\237\223\347\232\204.md.ee09ffde.js"
+++ "b/assets/guide_React_React\346\230\257\345\246\202\344\275\225\346\270\262\346\237\223\347\232\204.md.f058feee.js"
@@ -1 +1 @@
-import{_ as e,o as r,c as o,Q as t}from"./chunks/framework.b6910bb2.js";const a="/vitePress-blob/assets/3.9b1f9060.png",i="/vitePress-blob/assets/4.f4dd49d0.png",n="/vitePress-blob/assets/5.34ee565e.png",s="/vitePress-blob/assets/12.ba19c2aa.png",c="/vitePress-blob/assets/6.121ea6f5.png",l="/vitePress-blob/assets/7.3f678f52.png",p="/vitePress-blob/assets/8.9d04f238.png",d="/vitePress-blob/assets/9.bae86652.png",b="/vitePress-blob/assets/11.07a5666c.png",h="/vitePress-blob/assets/13.5bf5bc75.png",m="/vitePress-blob/assets/14.f1d70f78.png",u="/vitePress-blob/assets/16.4d6f740b.png",_="/vitePress-blob/assets/15.254fe341.png",f="/vitePress-blob/assets/18.09fcc54c.png",k="/vitePress-blob/assets/17.625ac807.png",g="/vitePress-blob/assets/19.77c3b0a6.png",C=JSON.parse('{"title":"React 是如何渲染的","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React是如何渲染的.md","filePath":"guide/React/React是如何渲染的.md","lastUpdated":1736065650000}'),R={name:"guide/React/React是如何渲染的.md"},P=t('

React 是如何渲染的

概要

React 的渲染过程可以分成 2 大阶段, 分别是调和阶段和提交阶段。 调和阶段可以分成 beginWork 阶段、 completeWork 阶段。 在 beginWork 阶段中,React 会根据新生成的 ReactElement 对象和旧的 Fiber 节点进行对比,判断是否可以复用旧的 Fiber 节点并对 Fiber 进行标记。 在 completeWork 节点中,会自底向上构建副作用链表,用来记录需要更新的节点,生成的 DOM 节点会挂载在 Fiber 的 stateNode 属性上。

提交阶段主要分成:操作 DOM 前阶段、操作 DOM 阶段、操作 DOM 后阶段。

React 16 以前

在浏览器中 js 线程与渲染线程是互斥的,如果 js 线程长期占用着浏览器的主线程,那么界面将长时间不更新,在动画等一些场景下会造成卡顿效果。

因为 Stack Reconciler 是一个同步的递归过程,随着业务复杂度增加,Stack Reconciler 需要的调和时间会变长,

这意味着 js 将长时间占用浏览器,进而导致页面卡顿

Stack Reconciler

React 16 以后

将同步执行的 Stack Reconciler 替换成了异步可中断的 Fiber Reconciler

Fiber Reconciler

在更新时,每个任务会被赋予一个优先级,当任务抵达调度器时,高优先级的任务会更快抵达协调器,如有新的更高优先级的任务进入调度器时,当前协调器的任务就会被中断,更高优先级的任务将进入 reconciler

Fiber Reconciler

新的架构会导致部分生命周期重复执行:

  • componentWillMount
  • componentWillUpdate
  • showComponentUpdate
  • componentWillReceiveProps

Render Performance

从首次渲染的调用栈来看,React 的渲染过程主要分为以下几个步骤:

    1. Mount 阶段
    1. Render 阶段
    1. Commit 阶段

Mount 阶段

当执行 ReactDOM.render 时会直接调用 legacyRenderSubtreeIntoContainer 方法

legacyRenderSubtreeIntoContainer 方法

会创建 reactRootContainer 对象(也就是挂载的容器对象), reactRootContainer 对象的 _internalRoot 会指向 fiberRoot (FiberRootNode 类),也就是根节点对象。

legacyRenderSubtreeIntoContainer 最终返回挂载组件( App组件 )的实例对象。

Mount 渲染过程

FiberRootNode对象的描述

在 fiberRoot 对象( FiberRootNode类 )中的 current 属性将指向 rootFiber 对象 (根节点Fiber,即 FiberNode 实例)

fiberRoot对象指向 rootFiber 根节点Fiber

挂载创建关系图

updateContainer 函数

updateContainer 函数主要有以下 3 件事:

    1. 获取当前节点的优先级 lane
    1. 结合 lane 创建当前 Fiber 节点 update 对象,并将其入队
    1. 调度当前节点 (rootFiber 节点)

updateContainer

scheduleUpdateOnFiber 函数

scheduleUpdateOnFiber 函数中会获取 Fiber 节点的mode属性判断是否走同步渲染还是异步渲染的逻辑,在 React17 中首次渲染走的是同步渲染的逻辑

scheduleUpdateOnFiber

这里可能有小伙伴会问,Fiber架构不就是异步渲染的么? 我想说的是,Fiber架构的设计初衷确实是为了异步渲染而设计的,但是 Fiber 架构并不能和异步渲染画上等号,我们不难发现,Fiber 架构同时兼容了同步渲染和异步渲染,如下图,决定同步还是异步取决于 mode

mode 决定同步异步

Render 阶段

performSyncWorkOnRoot 函数

核心逻辑在 renderRootSync 函数中

renderRootSync 函数

核心方法有俩个 prepareFreshStackworkLoopSync 函数

renderRootSync 逻辑

prepareFreshStack 函数

主要是有个方法 createWorkInProgress , 用来构建 workInProgress 双缓冲树,通过 alternate 相互指向

createWorkInProgress 构建 work-in-progress 树

当建立好双缓冲树的关系后,我们不难得到以下的关系图

双缓冲树关系图

workLoopSync 函数

当我们构建完 workInProgress Tree 的根节点时,建立 current tree 和 workInProgess Tree 的关联关系后,将进入 workLoopSync 调和阶段。

反复判断 workInProgress 是否为空,如果不为空,就执行 performUnitOfWork 方法 workLoopSync 逻辑

performUnitOfWork 函数

performUnitOfWork 函数的作用是 beginWork 优先创建子节点。 completeUnitOfWork 创建完子节点后判断是否有兄弟节点,有则创建兄弟节点,无则继续向上遍历父节点,直到遍历到根节点为止

performUnitOfWork 逻辑

beginWork 函数

参考文档

1 https://zhuanlan.zhihu.com/p/385319664 2

',58),q=[P];function F(S,y,W,v,w,x){return r(),o("div",null,q)}const U=e(R,[["render",F]]);export{C as __pageData,U as default}; +import{_ as e,o as r,c as o,Q as t}from"./chunks/framework.b6910bb2.js";const a="/vitePress-blob/assets/3.9b1f9060.png",i="/vitePress-blob/assets/4.f4dd49d0.png",n="/vitePress-blob/assets/5.34ee565e.png",s="/vitePress-blob/assets/12.ba19c2aa.png",c="/vitePress-blob/assets/6.121ea6f5.png",l="/vitePress-blob/assets/7.3f678f52.png",p="/vitePress-blob/assets/8.9d04f238.png",d="/vitePress-blob/assets/9.bae86652.png",b="/vitePress-blob/assets/11.07a5666c.png",h="/vitePress-blob/assets/13.5bf5bc75.png",m="/vitePress-blob/assets/14.f1d70f78.png",u="/vitePress-blob/assets/16.4d6f740b.png",_="/vitePress-blob/assets/15.254fe341.png",f="/vitePress-blob/assets/18.09fcc54c.png",k="/vitePress-blob/assets/17.625ac807.png",g="/vitePress-blob/assets/19.77c3b0a6.png",C=JSON.parse('{"title":"React 是如何渲染的","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React是如何渲染的.md","filePath":"guide/React/React是如何渲染的.md","lastUpdated":1736839070000}'),R={name:"guide/React/React是如何渲染的.md"},P=t('

React 是如何渲染的

概要

React 的渲染过程可以分成 2 大阶段, 分别是调和阶段和提交阶段。 调和阶段可以分成 beginWork 阶段、 completeWork 阶段。 在 beginWork 阶段中,React 会根据新生成的 ReactElement 对象和旧的 Fiber 节点进行对比,判断是否可以复用旧的 Fiber 节点并对 Fiber 进行标记。 在 completeWork 节点中,会自底向上构建副作用链表,用来记录需要更新的节点,生成的 DOM 节点会挂载在 Fiber 的 stateNode 属性上。

提交阶段主要分成:操作 DOM 前阶段、操作 DOM 阶段、操作 DOM 后阶段。

React 16 以前

在浏览器中 js 线程与渲染线程是互斥的,如果 js 线程长期占用着浏览器的主线程,那么界面将长时间不更新,在动画等一些场景下会造成卡顿效果。

因为 Stack Reconciler 是一个同步的递归过程,随着业务复杂度增加,Stack Reconciler 需要的调和时间会变长,

这意味着 js 将长时间占用浏览器,进而导致页面卡顿

Stack Reconciler

React 16 以后

将同步执行的 Stack Reconciler 替换成了异步可中断的 Fiber Reconciler

Fiber Reconciler

在更新时,每个任务会被赋予一个优先级,当任务抵达调度器时,高优先级的任务会更快抵达协调器,如有新的更高优先级的任务进入调度器时,当前协调器的任务就会被中断,更高优先级的任务将进入 reconciler

Fiber Reconciler

新的架构会导致部分生命周期重复执行:

  • componentWillMount
  • componentWillUpdate
  • showComponentUpdate
  • componentWillReceiveProps

Render Performance

从首次渲染的调用栈来看,React 的渲染过程主要分为以下几个步骤:

    1. Mount 阶段
    1. Render 阶段
    1. Commit 阶段

Mount 阶段

当执行 ReactDOM.render 时会直接调用 legacyRenderSubtreeIntoContainer 方法

legacyRenderSubtreeIntoContainer 方法

会创建 reactRootContainer 对象(也就是挂载的容器对象), reactRootContainer 对象的 _internalRoot 会指向 fiberRoot (FiberRootNode 类),也就是根节点对象。

legacyRenderSubtreeIntoContainer 最终返回挂载组件( App组件 )的实例对象。

Mount 渲染过程

FiberRootNode对象的描述

在 fiberRoot 对象( FiberRootNode类 )中的 current 属性将指向 rootFiber 对象 (根节点Fiber,即 FiberNode 实例)

fiberRoot对象指向 rootFiber 根节点Fiber

挂载创建关系图

updateContainer 函数

updateContainer 函数主要有以下 3 件事:

    1. 获取当前节点的优先级 lane
    1. 结合 lane 创建当前 Fiber 节点 update 对象,并将其入队
    1. 调度当前节点 (rootFiber 节点)

updateContainer

scheduleUpdateOnFiber 函数

scheduleUpdateOnFiber 函数中会获取 Fiber 节点的mode属性判断是否走同步渲染还是异步渲染的逻辑,在 React17 中首次渲染走的是同步渲染的逻辑

scheduleUpdateOnFiber

这里可能有小伙伴会问,Fiber架构不就是异步渲染的么? 我想说的是,Fiber架构的设计初衷确实是为了异步渲染而设计的,但是 Fiber 架构并不能和异步渲染画上等号,我们不难发现,Fiber 架构同时兼容了同步渲染和异步渲染,如下图,决定同步还是异步取决于 mode

mode 决定同步异步

Render 阶段

performSyncWorkOnRoot 函数

核心逻辑在 renderRootSync 函数中

renderRootSync 函数

核心方法有俩个 prepareFreshStackworkLoopSync 函数

renderRootSync 逻辑

prepareFreshStack 函数

主要是有个方法 createWorkInProgress , 用来构建 workInProgress 双缓冲树,通过 alternate 相互指向

createWorkInProgress 构建 work-in-progress 树

当建立好双缓冲树的关系后,我们不难得到以下的关系图

双缓冲树关系图

workLoopSync 函数

当我们构建完 workInProgress Tree 的根节点时,建立 current tree 和 workInProgess Tree 的关联关系后,将进入 workLoopSync 调和阶段。

反复判断 workInProgress 是否为空,如果不为空,就执行 performUnitOfWork 方法 workLoopSync 逻辑

performUnitOfWork 函数

performUnitOfWork 函数的作用是 beginWork 优先创建子节点。 completeUnitOfWork 创建完子节点后判断是否有兄弟节点,有则创建兄弟节点,无则继续向上遍历父节点,直到遍历到根节点为止

performUnitOfWork 逻辑

beginWork 函数

参考文档

1 https://zhuanlan.zhihu.com/p/385319664 2

',58),q=[P];function F(S,y,W,v,w,x){return r(),o("div",null,q)}const U=e(R,[["render",F]]);export{C as __pageData,U as default}; diff --git "a/assets/guide_React_React\346\230\257\345\246\202\344\275\225\346\270\262\346\237\223\347\232\204.md.ee09ffde.lean.js" "b/assets/guide_React_React\346\230\257\345\246\202\344\275\225\346\270\262\346\237\223\347\232\204.md.f058feee.lean.js" similarity index 94% rename from "assets/guide_React_React\346\230\257\345\246\202\344\275\225\346\270\262\346\237\223\347\232\204.md.ee09ffde.lean.js" rename to "assets/guide_React_React\346\230\257\345\246\202\344\275\225\346\270\262\346\237\223\347\232\204.md.f058feee.lean.js" index 5ecaf992..42625cb7 100644 --- "a/assets/guide_React_React\346\230\257\345\246\202\344\275\225\346\270\262\346\237\223\347\232\204.md.ee09ffde.lean.js" +++ "b/assets/guide_React_React\346\230\257\345\246\202\344\275\225\346\270\262\346\237\223\347\232\204.md.f058feee.lean.js" @@ -1 +1 @@ -import{_ as e,o as r,c as o,Q as t}from"./chunks/framework.b6910bb2.js";const a="/vitePress-blob/assets/3.9b1f9060.png",i="/vitePress-blob/assets/4.f4dd49d0.png",n="/vitePress-blob/assets/5.34ee565e.png",s="/vitePress-blob/assets/12.ba19c2aa.png",c="/vitePress-blob/assets/6.121ea6f5.png",l="/vitePress-blob/assets/7.3f678f52.png",p="/vitePress-blob/assets/8.9d04f238.png",d="/vitePress-blob/assets/9.bae86652.png",b="/vitePress-blob/assets/11.07a5666c.png",h="/vitePress-blob/assets/13.5bf5bc75.png",m="/vitePress-blob/assets/14.f1d70f78.png",u="/vitePress-blob/assets/16.4d6f740b.png",_="/vitePress-blob/assets/15.254fe341.png",f="/vitePress-blob/assets/18.09fcc54c.png",k="/vitePress-blob/assets/17.625ac807.png",g="/vitePress-blob/assets/19.77c3b0a6.png",C=JSON.parse('{"title":"React 是如何渲染的","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React是如何渲染的.md","filePath":"guide/React/React是如何渲染的.md","lastUpdated":1736065650000}'),R={name:"guide/React/React是如何渲染的.md"},P=t("",58),q=[P];function F(S,y,W,v,w,x){return r(),o("div",null,q)}const U=e(R,[["render",F]]);export{C as __pageData,U as default}; +import{_ as e,o as r,c as o,Q as t}from"./chunks/framework.b6910bb2.js";const a="/vitePress-blob/assets/3.9b1f9060.png",i="/vitePress-blob/assets/4.f4dd49d0.png",n="/vitePress-blob/assets/5.34ee565e.png",s="/vitePress-blob/assets/12.ba19c2aa.png",c="/vitePress-blob/assets/6.121ea6f5.png",l="/vitePress-blob/assets/7.3f678f52.png",p="/vitePress-blob/assets/8.9d04f238.png",d="/vitePress-blob/assets/9.bae86652.png",b="/vitePress-blob/assets/11.07a5666c.png",h="/vitePress-blob/assets/13.5bf5bc75.png",m="/vitePress-blob/assets/14.f1d70f78.png",u="/vitePress-blob/assets/16.4d6f740b.png",_="/vitePress-blob/assets/15.254fe341.png",f="/vitePress-blob/assets/18.09fcc54c.png",k="/vitePress-blob/assets/17.625ac807.png",g="/vitePress-blob/assets/19.77c3b0a6.png",C=JSON.parse('{"title":"React 是如何渲染的","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React是如何渲染的.md","filePath":"guide/React/React是如何渲染的.md","lastUpdated":1736839070000}'),R={name:"guide/React/React是如何渲染的.md"},P=t("",58),q=[P];function F(S,y,W,v,w,x){return r(),o("div",null,q)}const U=e(R,[["render",F]]);export{C as __pageData,U as default}; diff --git "a/assets/guide_React_React\347\273\204\344\273\266\346\230\257\345\246\202\344\275\225\351\200\232\344\277\241\347\232\204.md.a82d5c10.js" "b/assets/guide_React_React\347\273\204\344\273\266\346\230\257\345\246\202\344\275\225\351\200\232\344\277\241\347\232\204.md.a0c59c32.js" similarity index 91% rename from "assets/guide_React_React\347\273\204\344\273\266\346\230\257\345\246\202\344\275\225\351\200\232\344\277\241\347\232\204.md.a82d5c10.js" rename to "assets/guide_React_React\347\273\204\344\273\266\346\230\257\345\246\202\344\275\225\351\200\232\344\277\241\347\232\204.md.a0c59c32.js" index d3ff34e4..2ba5799e 100644 --- "a/assets/guide_React_React\347\273\204\344\273\266\346\230\257\345\246\202\344\275\225\351\200\232\344\277\241\347\232\204.md.a82d5c10.js" +++ "b/assets/guide_React_React\347\273\204\344\273\266\346\230\257\345\246\202\344\275\225\351\200\232\344\277\241\347\232\204.md.a0c59c32.js" @@ -1 +1 @@ -import{_ as a,o as t,c,k as e,a as r}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"React 组件是如何通信的","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React组件是如何通信的.md","filePath":"guide/React/React组件是如何通信的.md","lastUpdated":1736065650000}'),s={name:"guide/React/React组件是如何通信的.md"},o=e("h1",{id:"react-组件是如何通信的",tabindex:"-1"},[r("React 组件是如何通信的 "),e("a",{class:"header-anchor",href:"#react-组件是如何通信的","aria-label":'Permalink to "React 组件是如何通信的"'},"​")],-1),_=[o];function d(n,i,l,p,h,m){return t(),c("div",null,_)}const u=a(s,[["render",d]]);export{f as __pageData,u as default}; +import{_ as a,o as t,c,k as e,a as r}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"React 组件是如何通信的","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React组件是如何通信的.md","filePath":"guide/React/React组件是如何通信的.md","lastUpdated":1736839070000}'),s={name:"guide/React/React组件是如何通信的.md"},o=e("h1",{id:"react-组件是如何通信的",tabindex:"-1"},[r("React 组件是如何通信的 "),e("a",{class:"header-anchor",href:"#react-组件是如何通信的","aria-label":'Permalink to "React 组件是如何通信的"'},"​")],-1),_=[o];function d(n,i,l,p,h,m){return t(),c("div",null,_)}const u=a(s,[["render",d]]);export{f as __pageData,u as default}; diff --git "a/assets/guide_React_React\347\273\204\344\273\266\346\230\257\345\246\202\344\275\225\351\200\232\344\277\241\347\232\204.md.a82d5c10.lean.js" "b/assets/guide_React_React\347\273\204\344\273\266\346\230\257\345\246\202\344\275\225\351\200\232\344\277\241\347\232\204.md.a0c59c32.lean.js" similarity index 91% rename from "assets/guide_React_React\347\273\204\344\273\266\346\230\257\345\246\202\344\275\225\351\200\232\344\277\241\347\232\204.md.a82d5c10.lean.js" rename to "assets/guide_React_React\347\273\204\344\273\266\346\230\257\345\246\202\344\275\225\351\200\232\344\277\241\347\232\204.md.a0c59c32.lean.js" index d3ff34e4..2ba5799e 100644 --- "a/assets/guide_React_React\347\273\204\344\273\266\346\230\257\345\246\202\344\275\225\351\200\232\344\277\241\347\232\204.md.a82d5c10.lean.js" +++ "b/assets/guide_React_React\347\273\204\344\273\266\346\230\257\345\246\202\344\275\225\351\200\232\344\277\241\347\232\204.md.a0c59c32.lean.js" @@ -1 +1 @@ -import{_ as a,o as t,c,k as e,a as r}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"React 组件是如何通信的","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React组件是如何通信的.md","filePath":"guide/React/React组件是如何通信的.md","lastUpdated":1736065650000}'),s={name:"guide/React/React组件是如何通信的.md"},o=e("h1",{id:"react-组件是如何通信的",tabindex:"-1"},[r("React 组件是如何通信的 "),e("a",{class:"header-anchor",href:"#react-组件是如何通信的","aria-label":'Permalink to "React 组件是如何通信的"'},"​")],-1),_=[o];function d(n,i,l,p,h,m){return t(),c("div",null,_)}const u=a(s,[["render",d]]);export{f as __pageData,u as default}; +import{_ as a,o as t,c,k as e,a as r}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"React 组件是如何通信的","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/React组件是如何通信的.md","filePath":"guide/React/React组件是如何通信的.md","lastUpdated":1736839070000}'),s={name:"guide/React/React组件是如何通信的.md"},o=e("h1",{id:"react-组件是如何通信的",tabindex:"-1"},[r("React 组件是如何通信的 "),e("a",{class:"header-anchor",href:"#react-组件是如何通信的","aria-label":'Permalink to "React 组件是如何通信的"'},"​")],-1),_=[o];function d(n,i,l,p,h,m){return t(),c("div",null,_)}const u=a(s,[["render",d]]);export{f as __pageData,u as default}; diff --git "a/assets/guide_React_diff\347\256\227\346\263\225.md.e12213ca.js" "b/assets/guide_React_diff\347\256\227\346\263\225.md.4f2b0e59.js" similarity index 98% rename from "assets/guide_React_diff\347\256\227\346\263\225.md.e12213ca.js" rename to "assets/guide_React_diff\347\256\227\346\263\225.md.4f2b0e59.js" index e5057892..077815a5 100644 --- "a/assets/guide_React_diff\347\256\227\346\263\225.md.e12213ca.js" +++ "b/assets/guide_React_diff\347\256\227\346\263\225.md.4f2b0e59.js" @@ -1 +1 @@ -import{_ as e,o as a,c as t,Q as i}from"./chunks/framework.b6910bb2.js";const k=JSON.parse('{"title":"React 中的 Diff 算法","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/diff算法.md","filePath":"guide/React/diff算法.md","lastUpdated":1736065650000}'),d={name:"guide/React/diff算法.md"},r=i('

React 中的 Diff 算法

概要

React Diff 算法主要分单节点和多节点 Diff 算法 在单节点 Diff 算法中,会对比 key 和 tag 是否可以复用 如果 key 和 tag 都相同时,复用当前的 Fiber 节点,标记其他 Fiber 节点为删除 如果 key 不相同,标记删除当前的 Fiber 节点,对比下一个节点 如果 key 相同、tag 不同,则标记删除所有 Fiber 节点,生成新的 Fiber 节点

在多节点 Diff 中,有俩个 for 循环,第一个 for 循环判断可以复用的节点,记录最后可复用节点的 lastIndex 位置

通过 Map 建立节点和下标的对应关系,如果新节点的key可以在 Map 中找到,则复用旧节点,并判断新节点下标和 lastIndex 下标的关系, 如果新节点下标 > lastIndex 下标,说明只需要更新节点,不需要移动位置,更新 lastIndex。 如果新节点下标 < lastIndex,说明需要更新移动位置,不需要更新 lastIndex。

单节点diff算法 (render方法创建的element元素子节点只有一个)

1. 旧节点不存在

  1. 如果旧节点不存在,则直接新增节点

2. 旧节点存在

1 判断 key 和 tag 是否相同,相同则复用节点,更新属性 2 如果 key 相同但 tag 不相同,则删除所有节点 3 如果 key、tag 不同,则不需要复用,旧节点标记为删除

多节点diff算法 (render方法创建的element元素子节点有多个)

多节点diff有俩次 for 循环, 第一次 for 循环判断元素是否需要更新,第二次 for 循环判断元素是否需要移动位置

1 第一次 for 循环比较key和tag, 如果相同则复用, 并用 lastIndex 标记当前可以复用的节点位置

2 遇到不相同时, 跳出第一层 for 循环, 创建一个 Map 对象, 存储旧节点的 key 和 index

  • key不同导致不可复用,立即跳出整个遍历,第一轮遍历结束
  • key 相同 type 不同导致不可复用,会将 oldFiber 标记为 DELETION,并继续遍历

3 在 Map 对象中查找新节点的 key, 如果存在, 则说明新节点可以复用旧节点, 并且判断是否需要移动位置

4 如果 index > lastIndex, 则不需要移动位置, 更新 lastIndex

5 如果 index < lastIndex, 则需要移动位置, 不需要更新 lastIndex

',18),n=[r];function l(o,f,s,h,x,c){return a(),t("div",null,n)}const p=e(d,[["render",l]]);export{k as __pageData,p as default}; +import{_ as e,o as a,c as t,Q as i}from"./chunks/framework.b6910bb2.js";const k=JSON.parse('{"title":"React 中的 Diff 算法","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/diff算法.md","filePath":"guide/React/diff算法.md","lastUpdated":1736839070000}'),d={name:"guide/React/diff算法.md"},r=i('

React 中的 Diff 算法

概要

React Diff 算法主要分单节点和多节点 Diff 算法 在单节点 Diff 算法中,会对比 key 和 tag 是否可以复用 如果 key 和 tag 都相同时,复用当前的 Fiber 节点,标记其他 Fiber 节点为删除 如果 key 不相同,标记删除当前的 Fiber 节点,对比下一个节点 如果 key 相同、tag 不同,则标记删除所有 Fiber 节点,生成新的 Fiber 节点

在多节点 Diff 中,有俩个 for 循环,第一个 for 循环判断可以复用的节点,记录最后可复用节点的 lastIndex 位置

通过 Map 建立节点和下标的对应关系,如果新节点的key可以在 Map 中找到,则复用旧节点,并判断新节点下标和 lastIndex 下标的关系, 如果新节点下标 > lastIndex 下标,说明只需要更新节点,不需要移动位置,更新 lastIndex。 如果新节点下标 < lastIndex,说明需要更新移动位置,不需要更新 lastIndex。

单节点diff算法 (render方法创建的element元素子节点只有一个)

1. 旧节点不存在

  1. 如果旧节点不存在,则直接新增节点

2. 旧节点存在

1 判断 key 和 tag 是否相同,相同则复用节点,更新属性 2 如果 key 相同但 tag 不相同,则删除所有节点 3 如果 key、tag 不同,则不需要复用,旧节点标记为删除

多节点diff算法 (render方法创建的element元素子节点有多个)

多节点diff有俩次 for 循环, 第一次 for 循环判断元素是否需要更新,第二次 for 循环判断元素是否需要移动位置

1 第一次 for 循环比较key和tag, 如果相同则复用, 并用 lastIndex 标记当前可以复用的节点位置

2 遇到不相同时, 跳出第一层 for 循环, 创建一个 Map 对象, 存储旧节点的 key 和 index

  • key不同导致不可复用,立即跳出整个遍历,第一轮遍历结束
  • key 相同 type 不同导致不可复用,会将 oldFiber 标记为 DELETION,并继续遍历

3 在 Map 对象中查找新节点的 key, 如果存在, 则说明新节点可以复用旧节点, 并且判断是否需要移动位置

4 如果 index > lastIndex, 则不需要移动位置, 更新 lastIndex

5 如果 index < lastIndex, 则需要移动位置, 不需要更新 lastIndex

',18),n=[r];function l(o,f,s,h,x,c){return a(),t("div",null,n)}const p=e(d,[["render",l]]);export{k as __pageData,p as default}; diff --git "a/assets/guide_React_diff\347\256\227\346\263\225.md.e12213ca.lean.js" "b/assets/guide_React_diff\347\256\227\346\263\225.md.4f2b0e59.lean.js" similarity index 86% rename from "assets/guide_React_diff\347\256\227\346\263\225.md.e12213ca.lean.js" rename to "assets/guide_React_diff\347\256\227\346\263\225.md.4f2b0e59.lean.js" index d71da55b..1bccebc5 100644 --- "a/assets/guide_React_diff\347\256\227\346\263\225.md.e12213ca.lean.js" +++ "b/assets/guide_React_diff\347\256\227\346\263\225.md.4f2b0e59.lean.js" @@ -1 +1 @@ -import{_ as e,o as a,c as t,Q as i}from"./chunks/framework.b6910bb2.js";const k=JSON.parse('{"title":"React 中的 Diff 算法","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/diff算法.md","filePath":"guide/React/diff算法.md","lastUpdated":1736065650000}'),d={name:"guide/React/diff算法.md"},r=i("",18),n=[r];function l(o,f,s,h,x,c){return a(),t("div",null,n)}const p=e(d,[["render",l]]);export{k as __pageData,p as default}; +import{_ as e,o as a,c as t,Q as i}from"./chunks/framework.b6910bb2.js";const k=JSON.parse('{"title":"React 中的 Diff 算法","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/diff算法.md","filePath":"guide/React/diff算法.md","lastUpdated":1736839070000}'),d={name:"guide/React/diff算法.md"},r=i("",18),n=[r];function l(o,f,s,h,x,c){return a(),t("div",null,n)}const p=e(d,[["render",l]]);export{k as __pageData,p as default}; diff --git "a/assets/guide_React_react\345\274\202\345\270\270\346\234\272\345\210\266.md.95438d51.js" "b/assets/guide_React_react\345\274\202\345\270\270\346\234\272\345\210\266.md.39386d10.js" similarity index 84% rename from "assets/guide_React_react\345\274\202\345\270\270\346\234\272\345\210\266.md.95438d51.js" rename to "assets/guide_React_react\345\274\202\345\270\270\346\234\272\345\210\266.md.39386d10.js" index 8a931adb..ccafb5aa 100644 --- "a/assets/guide_React_react\345\274\202\345\270\270\346\234\272\345\210\266.md.95438d51.js" +++ "b/assets/guide_React_react\345\274\202\345\270\270\346\234\272\345\210\266.md.39386d10.js" @@ -1 +1 @@ -import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/react异常机制.md","filePath":"guide/React/react异常机制.md","lastUpdated":1736065650000}'),c={name:"guide/React/react异常机制.md"};function r(o,s,d,n,_,i){return t(),a("div")}const f=e(c,[["render",r]]);export{m as __pageData,f as default}; +import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/react异常机制.md","filePath":"guide/React/react异常机制.md","lastUpdated":1736839070000}'),c={name:"guide/React/react异常机制.md"};function r(o,s,d,n,_,i){return t(),a("div")}const f=e(c,[["render",r]]);export{m as __pageData,f as default}; diff --git "a/assets/guide_React_react\345\274\202\345\270\270\346\234\272\345\210\266.md.95438d51.lean.js" "b/assets/guide_React_react\345\274\202\345\270\270\346\234\272\345\210\266.md.39386d10.lean.js" similarity index 84% rename from "assets/guide_React_react\345\274\202\345\270\270\346\234\272\345\210\266.md.95438d51.lean.js" rename to "assets/guide_React_react\345\274\202\345\270\270\346\234\272\345\210\266.md.39386d10.lean.js" index 8a931adb..ccafb5aa 100644 --- "a/assets/guide_React_react\345\274\202\345\270\270\346\234\272\345\210\266.md.95438d51.lean.js" +++ "b/assets/guide_React_react\345\274\202\345\270\270\346\234\272\345\210\266.md.39386d10.lean.js" @@ -1 +1 @@ -import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/react异常机制.md","filePath":"guide/React/react异常机制.md","lastUpdated":1736065650000}'),c={name:"guide/React/react异常机制.md"};function r(o,s,d,n,_,i){return t(),a("div")}const f=e(c,[["render",r]]);export{m as __pageData,f as default}; +import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/react异常机制.md","filePath":"guide/React/react异常机制.md","lastUpdated":1736839070000}'),c={name:"guide/React/react异常机制.md"};function r(o,s,d,n,_,i){return t(),a("div")}const f=e(c,[["render",r]]);export{m as __pageData,f as default}; diff --git "a/assets/guide_React_setState\346\230\257\345\220\214\346\255\245\350\277\230\346\230\257\345\274\202\346\255\245\347\232\204.md.d2fb1cb9.js" "b/assets/guide_React_setState\346\230\257\345\220\214\346\255\245\350\277\230\346\230\257\345\274\202\346\255\245\347\232\204.md.10d004c3.js" similarity index 99% rename from "assets/guide_React_setState\346\230\257\345\220\214\346\255\245\350\277\230\346\230\257\345\274\202\346\255\245\347\232\204.md.d2fb1cb9.js" rename to "assets/guide_React_setState\346\230\257\345\220\214\346\255\245\350\277\230\346\230\257\345\274\202\346\255\245\347\232\204.md.10d004c3.js" index d0ba4108..f55cb502 100644 --- "a/assets/guide_React_setState\346\230\257\345\220\214\346\255\245\350\277\230\346\230\257\345\274\202\346\255\245\347\232\204.md.d2fb1cb9.js" +++ "b/assets/guide_React_setState\346\230\257\345\220\214\346\255\245\350\277\230\346\230\257\345\274\202\346\255\245\347\232\204.md.10d004c3.js" @@ -1,4 +1,4 @@ -import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/1.5e4dabab.png",h=JSON.parse('{"title":"setState 是同步还是异步的?","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/setState是同步还是异步的.md","filePath":"guide/React/setState是同步还是异步的.md","lastUpdated":1736065650000}'),o={name:"guide/React/setState是同步还是异步的.md"},e=l('

setState 是同步还是异步的?

概要

setState 用于状态变更,触发组件的重新渲染,更新视图。

在 React 18 以前,setState 同步还是异步取决于是否处于 isBatchUpdates 字段,true 的话就是异步,否则就是同步

在 React 18 版本所有的 setState 都是异步批量操作

setState

js
// React 15
+import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/1.5e4dabab.png",h=JSON.parse('{"title":"setState 是同步还是异步的?","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/setState是同步还是异步的.md","filePath":"guide/React/setState是同步还是异步的.md","lastUpdated":1736839070000}'),o={name:"guide/React/setState是同步还是异步的.md"},e=l('

setState 是同步还是异步的?

概要

setState 用于状态变更,触发组件的重新渲染,更新视图。

在 React 18 以前,setState 同步还是异步取决于是否处于 isBatchUpdates 字段,true 的话就是异步,否则就是同步

在 React 18 版本所有的 setState 都是异步批量操作

setState

js
// React 15
 function enqueueUpdate (component) {
   // isBatchingUpdates 判断当前是否处理批量更新操作
   if (!batchingStrategy.isBatchingUpdates) { // 锁管理器
diff --git "a/assets/guide_React_setState\346\230\257\345\220\214\346\255\245\350\277\230\346\230\257\345\274\202\346\255\245\347\232\204.md.d2fb1cb9.lean.js" "b/assets/guide_React_setState\346\230\257\345\220\214\346\255\245\350\277\230\346\230\257\345\274\202\346\255\245\347\232\204.md.10d004c3.lean.js"
similarity index 88%
rename from "assets/guide_React_setState\346\230\257\345\220\214\346\255\245\350\277\230\346\230\257\345\274\202\346\255\245\347\232\204.md.d2fb1cb9.lean.js"
rename to "assets/guide_React_setState\346\230\257\345\220\214\346\255\245\350\277\230\346\230\257\345\274\202\346\255\245\347\232\204.md.10d004c3.lean.js"
index be66787f..039bfed9 100644
--- "a/assets/guide_React_setState\346\230\257\345\220\214\346\255\245\350\277\230\346\230\257\345\274\202\346\255\245\347\232\204.md.d2fb1cb9.lean.js"
+++ "b/assets/guide_React_setState\346\230\257\345\220\214\346\255\245\350\277\230\346\230\257\345\274\202\346\255\245\347\232\204.md.10d004c3.lean.js"
@@ -1 +1 @@
-import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/1.5e4dabab.png",h=JSON.parse('{"title":"setState 是同步还是异步的?","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/setState是同步还是异步的.md","filePath":"guide/React/setState是同步还是异步的.md","lastUpdated":1736065650000}'),o={name:"guide/React/setState是同步还是异步的.md"},e=l("",35),t=[e];function c(r,E,y,i,F,d){return n(),a("div",null,t)}const C=s(o,[["render",c]]);export{h as __pageData,C as default};
+import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/1.5e4dabab.png",h=JSON.parse('{"title":"setState 是同步还是异步的?","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/setState是同步还是异步的.md","filePath":"guide/React/setState是同步还是异步的.md","lastUpdated":1736839070000}'),o={name:"guide/React/setState是同步还是异步的.md"},e=l("",35),t=[e];function c(r,E,y,i,F,d){return n(),a("div",null,t)}const C=s(o,[["render",c]]);export{h as __pageData,C as default};
diff --git "a/assets/guide_React_\344\273\200\344\271\210\346\230\257Fiber.md.29e5a130.js" "b/assets/guide_React_\344\273\200\344\271\210\346\230\257Fiber.md.9a75b68a.js"
similarity index 96%
rename from "assets/guide_React_\344\273\200\344\271\210\346\230\257Fiber.md.29e5a130.js"
rename to "assets/guide_React_\344\273\200\344\271\210\346\230\257Fiber.md.9a75b68a.js"
index 79e66764..aba6b388 100644
--- "a/assets/guide_React_\344\273\200\344\271\210\346\230\257Fiber.md.29e5a130.js"
+++ "b/assets/guide_React_\344\273\200\344\271\210\346\230\257Fiber.md.9a75b68a.js"
@@ -1 +1 @@
-import{_ as e,o as t,c as o,Q as c}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"对 Fiber 的理解","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/什么是Fiber.md","filePath":"guide/React/什么是Fiber.md","lastUpdated":1736065650000}'),r={name:"guide/React/什么是Fiber.md"},a=c('

对 Fiber 的理解

在 React 15 中通过 递归 的形式进行对比,找到需要更新的节点,并同步更新它,在这段时间一直占据着浏览器主线程,可能会给用户带来卡顿的感受(在渲染进程中,js线程和渲染线程是互斥的

在 React 15 以后引入了 Fiber 架构,将对比的过程变成了异步可中断的过程,让出浏览器的使用权,让浏览器处理更高优先级的事情

Fiber 的调和过程(Reconciler)由分成了 beginWork 阶段 和 completeUnitOfWork 阶段。

beginWork 阶段自顶向下,根据当前工作的 Fiber 节点最新的 React Element 子元素与旧 Fiber 节点进行对比,决定是否需要复用旧 Fiber 节点并标记 Fiber 节点是否有副作用。

compeleteUnitOfWork 阶段自底向上构建副作用链表,生成的 DOM 节点挂在 Fiber 的 stateNode 属性

',6),i=[a];function d(_,n,s,p,b,l){return t(),o("div",null,i)}const m=e(r,[["render",d]]);export{f as __pageData,m as default}; +import{_ as e,o as t,c as o,Q as c}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"对 Fiber 的理解","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/什么是Fiber.md","filePath":"guide/React/什么是Fiber.md","lastUpdated":1736839070000}'),r={name:"guide/React/什么是Fiber.md"},a=c('

对 Fiber 的理解

在 React 15 中通过 递归 的形式进行对比,找到需要更新的节点,并同步更新它,在这段时间一直占据着浏览器主线程,可能会给用户带来卡顿的感受(在渲染进程中,js线程和渲染线程是互斥的

在 React 15 以后引入了 Fiber 架构,将对比的过程变成了异步可中断的过程,让出浏览器的使用权,让浏览器处理更高优先级的事情

Fiber 的调和过程(Reconciler)由分成了 beginWork 阶段 和 completeUnitOfWork 阶段。

beginWork 阶段自顶向下,根据当前工作的 Fiber 节点最新的 React Element 子元素与旧 Fiber 节点进行对比,决定是否需要复用旧 Fiber 节点并标记 Fiber 节点是否有副作用。

compeleteUnitOfWork 阶段自底向上构建副作用链表,生成的 DOM 节点挂在 Fiber 的 stateNode 属性

',6),i=[a];function d(_,n,s,p,b,l){return t(),o("div",null,i)}const m=e(r,[["render",d]]);export{f as __pageData,m as default}; diff --git "a/assets/guide_React_\344\273\200\344\271\210\346\230\257Fiber.md.29e5a130.lean.js" "b/assets/guide_React_\344\273\200\344\271\210\346\230\257Fiber.md.9a75b68a.lean.js" similarity index 86% rename from "assets/guide_React_\344\273\200\344\271\210\346\230\257Fiber.md.29e5a130.lean.js" rename to "assets/guide_React_\344\273\200\344\271\210\346\230\257Fiber.md.9a75b68a.lean.js" index 616549b1..d586935f 100644 --- "a/assets/guide_React_\344\273\200\344\271\210\346\230\257Fiber.md.29e5a130.lean.js" +++ "b/assets/guide_React_\344\273\200\344\271\210\346\230\257Fiber.md.9a75b68a.lean.js" @@ -1 +1 @@ -import{_ as e,o as t,c as o,Q as c}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"对 Fiber 的理解","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/什么是Fiber.md","filePath":"guide/React/什么是Fiber.md","lastUpdated":1736065650000}'),r={name:"guide/React/什么是Fiber.md"},a=c("",6),i=[a];function d(_,n,s,p,b,l){return t(),o("div",null,i)}const m=e(r,[["render",d]]);export{f as __pageData,m as default}; +import{_ as e,o as t,c as o,Q as c}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"对 Fiber 的理解","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/什么是Fiber.md","filePath":"guide/React/什么是Fiber.md","lastUpdated":1736839070000}'),r={name:"guide/React/什么是Fiber.md"},a=c("",6),i=[a];function d(_,n,s,p,b,l){return t(),o("div",null,i)}const m=e(r,[["render",d]]);export{f as __pageData,m as default}; diff --git "a/assets/guide_React_\345\246\202\344\275\225\350\256\276\350\256\241React\347\273\204\344\273\266.md.7bfc2acc.js" "b/assets/guide_React_\345\246\202\344\275\225\350\256\276\350\256\241React\347\273\204\344\273\266.md.b6076bcd.js" similarity index 95% rename from "assets/guide_React_\345\246\202\344\275\225\350\256\276\350\256\241React\347\273\204\344\273\266.md.7bfc2acc.js" rename to "assets/guide_React_\345\246\202\344\275\225\350\256\276\350\256\241React\347\273\204\344\273\266.md.b6076bcd.js" index 20d214be..4aca9e65 100644 --- "a/assets/guide_React_\345\246\202\344\275\225\350\256\276\350\256\241React\347\273\204\344\273\266.md.7bfc2acc.js" +++ "b/assets/guide_React_\345\246\202\344\275\225\350\256\276\350\256\241React\347\273\204\344\273\266.md.b6076bcd.js" @@ -1 +1 @@ -import{_ as a,o as e,c as t,Q as r}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/1.df8894ba.png",f=JSON.parse('{"title":"如何设计 React 组件","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/如何设计React组件.md","filePath":"guide/React/如何设计React组件.md","lastUpdated":1736065650000}'),c={name:"guide/React/如何设计React组件.md"},s=r('

如何设计 React 组件

设计分类

  • 展示组件: 只负责 UI 展示,不涉及业务逻辑

  • 业务组件: 复用负责业务逻辑

展示组件

展示组件一般受制于 props, 具有通用性、复用性

业务组件

业务组件一般受制于 state, 具有复用性

组件分类

',8),i=[s];function _(l,n,d,h,p,u){return e(),t("div",null,i)}const b=a(c,[["render",_]]);export{f as __pageData,b as default}; +import{_ as a,o as e,c as t,Q as r}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/1.df8894ba.png",f=JSON.parse('{"title":"如何设计 React 组件","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/如何设计React组件.md","filePath":"guide/React/如何设计React组件.md","lastUpdated":1736839070000}'),c={name:"guide/React/如何设计React组件.md"},s=r('

如何设计 React 组件

设计分类

  • 展示组件: 只负责 UI 展示,不涉及业务逻辑

  • 业务组件: 复用负责业务逻辑

展示组件

展示组件一般受制于 props, 具有通用性、复用性

业务组件

业务组件一般受制于 state, 具有复用性

组件分类

',8),i=[s];function _(l,n,d,h,p,u){return e(),t("div",null,i)}const b=a(c,[["render",_]]);export{f as __pageData,b as default}; diff --git "a/assets/guide_React_\345\246\202\344\275\225\350\256\276\350\256\241React\347\273\204\344\273\266.md.7bfc2acc.lean.js" "b/assets/guide_React_\345\246\202\344\275\225\350\256\276\350\256\241React\347\273\204\344\273\266.md.b6076bcd.lean.js" similarity index 88% rename from "assets/guide_React_\345\246\202\344\275\225\350\256\276\350\256\241React\347\273\204\344\273\266.md.7bfc2acc.lean.js" rename to "assets/guide_React_\345\246\202\344\275\225\350\256\276\350\256\241React\347\273\204\344\273\266.md.b6076bcd.lean.js" index cf7d7826..4a7eaf5f 100644 --- "a/assets/guide_React_\345\246\202\344\275\225\350\256\276\350\256\241React\347\273\204\344\273\266.md.7bfc2acc.lean.js" +++ "b/assets/guide_React_\345\246\202\344\275\225\350\256\276\350\256\241React\347\273\204\344\273\266.md.b6076bcd.lean.js" @@ -1 +1 @@ -import{_ as a,o as e,c as t,Q as r}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/1.df8894ba.png",f=JSON.parse('{"title":"如何设计 React 组件","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/如何设计React组件.md","filePath":"guide/React/如何设计React组件.md","lastUpdated":1736065650000}'),c={name:"guide/React/如何设计React组件.md"},s=r("",8),i=[s];function _(l,n,d,h,p,u){return e(),t("div",null,i)}const b=a(c,[["render",_]]);export{f as __pageData,b as default}; +import{_ as a,o as e,c as t,Q as r}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/1.df8894ba.png",f=JSON.parse('{"title":"如何设计 React 组件","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/如何设计React组件.md","filePath":"guide/React/如何设计React组件.md","lastUpdated":1736839070000}'),c={name:"guide/React/如何设计React组件.md"},s=r("",8),i=[s];function _(l,n,d,h,p,u){return e(),t("div",null,i)}const b=a(c,[["render",_]]);export{f as __pageData,b as default}; diff --git "a/assets/guide_React_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.681d5e24.js" "b/assets/guide_React_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.5afef26f.js" similarity index 91% rename from "assets/guide_React_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.681d5e24.js" rename to "assets/guide_React_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.5afef26f.js" index 7f2e0f99..0afa4bb3 100644 --- "a/assets/guide_React_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.681d5e24.js" +++ "b/assets/guide_React_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.5afef26f.js" @@ -1 +1 @@ -import{_ as a,o as t,c as r,k as e,a as s}from"./chunks/framework.b6910bb2.js";const x=JSON.parse('{"title":"常见的问题","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/常见的问题.md","filePath":"guide/React/常见的问题.md","lastUpdated":1736065650000}'),o={name:"guide/React/常见的问题.md"},c=e("h1",{id:"常见的问题",tabindex:"-1"},[s("常见的问题 "),e("a",{class:"header-anchor",href:"#常见的问题","aria-label":'Permalink to "常见的问题"'},"​")],-1),d=e("h2",{id:"",tabindex:"-1"},[e("a",{class:"header-anchor",href:"#","aria-label":'Permalink to ""'},"​")],-1),n=[c,d];function i(_,l,h,p,m,f){return t(),r("div",null,n)}const k=a(o,[["render",i]]);export{x as __pageData,k as default}; +import{_ as a,o as t,c as r,k as e,a as s}from"./chunks/framework.b6910bb2.js";const x=JSON.parse('{"title":"常见的问题","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/常见的问题.md","filePath":"guide/React/常见的问题.md","lastUpdated":1736839070000}'),o={name:"guide/React/常见的问题.md"},c=e("h1",{id:"常见的问题",tabindex:"-1"},[s("常见的问题 "),e("a",{class:"header-anchor",href:"#常见的问题","aria-label":'Permalink to "常见的问题"'},"​")],-1),d=e("h2",{id:"",tabindex:"-1"},[e("a",{class:"header-anchor",href:"#","aria-label":'Permalink to ""'},"​")],-1),n=[c,d];function i(_,l,h,p,m,f){return t(),r("div",null,n)}const k=a(o,[["render",i]]);export{x as __pageData,k as default}; diff --git "a/assets/guide_React_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.681d5e24.lean.js" "b/assets/guide_React_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.5afef26f.lean.js" similarity index 91% rename from "assets/guide_React_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.681d5e24.lean.js" rename to "assets/guide_React_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.5afef26f.lean.js" index 7f2e0f99..0afa4bb3 100644 --- "a/assets/guide_React_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.681d5e24.lean.js" +++ "b/assets/guide_React_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.5afef26f.lean.js" @@ -1 +1 @@ -import{_ as a,o as t,c as r,k as e,a as s}from"./chunks/framework.b6910bb2.js";const x=JSON.parse('{"title":"常见的问题","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/常见的问题.md","filePath":"guide/React/常见的问题.md","lastUpdated":1736065650000}'),o={name:"guide/React/常见的问题.md"},c=e("h1",{id:"常见的问题",tabindex:"-1"},[s("常见的问题 "),e("a",{class:"header-anchor",href:"#常见的问题","aria-label":'Permalink to "常见的问题"'},"​")],-1),d=e("h2",{id:"",tabindex:"-1"},[e("a",{class:"header-anchor",href:"#","aria-label":'Permalink to ""'},"​")],-1),n=[c,d];function i(_,l,h,p,m,f){return t(),r("div",null,n)}const k=a(o,[["render",i]]);export{x as __pageData,k as default}; +import{_ as a,o as t,c as r,k as e,a as s}from"./chunks/framework.b6910bb2.js";const x=JSON.parse('{"title":"常见的问题","description":"","frontmatter":{},"headers":[],"relativePath":"guide/React/常见的问题.md","filePath":"guide/React/常见的问题.md","lastUpdated":1736839070000}'),o={name:"guide/React/常见的问题.md"},c=e("h1",{id:"常见的问题",tabindex:"-1"},[s("常见的问题 "),e("a",{class:"header-anchor",href:"#常见的问题","aria-label":'Permalink to "常见的问题"'},"​")],-1),d=e("h2",{id:"",tabindex:"-1"},[e("a",{class:"header-anchor",href:"#","aria-label":'Permalink to ""'},"​")],-1),n=[c,d];function i(_,l,h,p,m,f){return t(),r("div",null,n)}const k=a(o,[["render",i]]);export{x as __pageData,k as default}; diff --git "a/assets/guide_Redux_\346\272\220\347\240\201\350\247\243\350\257\273.md.a19e3f7c.js" "b/assets/guide_Redux_\346\272\220\347\240\201\350\247\243\350\257\273.md.e0e871fd.js" similarity index 89% rename from "assets/guide_Redux_\346\272\220\347\240\201\350\247\243\350\257\273.md.a19e3f7c.js" rename to "assets/guide_Redux_\346\272\220\347\240\201\350\247\243\350\257\273.md.e0e871fd.js" index ca720584..4905cdde 100644 --- "a/assets/guide_Redux_\346\272\220\347\240\201\350\247\243\350\257\273.md.a19e3f7c.js" +++ "b/assets/guide_Redux_\346\272\220\347\240\201\350\247\243\350\257\273.md.e0e871fd.js" @@ -1 +1 @@ -import{_ as a,o as t,c as s,k as e,a as d}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"源码解读","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Redux/源码解读.md","filePath":"guide/Redux/源码解读.md","lastUpdated":1736065650000}'),o={name:"guide/Redux/源码解读.md"},r=e("h1",{id:"源码解读",tabindex:"-1"},[d("源码解读 "),e("a",{class:"header-anchor",href:"#源码解读","aria-label":'Permalink to "源码解读"'},"​")],-1),c=[r];function n(i,_,l,p,u,h){return t(),s("div",null,c)}const x=a(o,[["render",n]]);export{f as __pageData,x as default}; +import{_ as a,o as t,c as s,k as e,a as d}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"源码解读","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Redux/源码解读.md","filePath":"guide/Redux/源码解读.md","lastUpdated":1736839070000}'),o={name:"guide/Redux/源码解读.md"},r=e("h1",{id:"源码解读",tabindex:"-1"},[d("源码解读 "),e("a",{class:"header-anchor",href:"#源码解读","aria-label":'Permalink to "源码解读"'},"​")],-1),c=[r];function n(i,_,l,p,u,h){return t(),s("div",null,c)}const x=a(o,[["render",n]]);export{f as __pageData,x as default}; diff --git "a/assets/guide_Redux_\346\272\220\347\240\201\350\247\243\350\257\273.md.a19e3f7c.lean.js" "b/assets/guide_Redux_\346\272\220\347\240\201\350\247\243\350\257\273.md.e0e871fd.lean.js" similarity index 89% rename from "assets/guide_Redux_\346\272\220\347\240\201\350\247\243\350\257\273.md.a19e3f7c.lean.js" rename to "assets/guide_Redux_\346\272\220\347\240\201\350\247\243\350\257\273.md.e0e871fd.lean.js" index ca720584..4905cdde 100644 --- "a/assets/guide_Redux_\346\272\220\347\240\201\350\247\243\350\257\273.md.a19e3f7c.lean.js" +++ "b/assets/guide_Redux_\346\272\220\347\240\201\350\247\243\350\257\273.md.e0e871fd.lean.js" @@ -1 +1 @@ -import{_ as a,o as t,c as s,k as e,a as d}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"源码解读","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Redux/源码解读.md","filePath":"guide/Redux/源码解读.md","lastUpdated":1736065650000}'),o={name:"guide/Redux/源码解读.md"},r=e("h1",{id:"源码解读",tabindex:"-1"},[d("源码解读 "),e("a",{class:"header-anchor",href:"#源码解读","aria-label":'Permalink to "源码解读"'},"​")],-1),c=[r];function n(i,_,l,p,u,h){return t(),s("div",null,c)}const x=a(o,[["render",n]]);export{f as __pageData,x as default}; +import{_ as a,o as t,c as s,k as e,a as d}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"源码解读","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Redux/源码解读.md","filePath":"guide/Redux/源码解读.md","lastUpdated":1736839070000}'),o={name:"guide/Redux/源码解读.md"},r=e("h1",{id:"源码解读",tabindex:"-1"},[d("源码解读 "),e("a",{class:"header-anchor",href:"#源码解读","aria-label":'Permalink to "源码解读"'},"​")],-1),c=[r];function n(i,_,l,p,u,h){return t(),s("div",null,c)}const x=a(o,[["render",n]]);export{f as __pageData,x as default}; diff --git "a/assets/guide_Redux_\350\256\276\350\256\241\347\220\206\345\277\265.md.c290b375.lean.js" "b/assets/guide_Redux_\350\256\276\350\256\241\347\220\206\345\277\265.md.00f4264a.js" similarity index 89% rename from "assets/guide_Redux_\350\256\276\350\256\241\347\220\206\345\277\265.md.c290b375.lean.js" rename to "assets/guide_Redux_\350\256\276\350\256\241\347\220\206\345\277\265.md.00f4264a.js" index e8c4a054..dc91627c 100644 --- "a/assets/guide_Redux_\350\256\276\350\256\241\347\220\206\345\277\265.md.c290b375.lean.js" +++ "b/assets/guide_Redux_\350\256\276\350\256\241\347\220\206\345\277\265.md.00f4264a.js" @@ -1 +1 @@ -import{_ as a,o as t,c as s,k as e,a as d}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"设计理念","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Redux/设计理念.md","filePath":"guide/Redux/设计理念.md","lastUpdated":1736065650000}'),o={name:"guide/Redux/设计理念.md"},r=e("h1",{id:"设计理念",tabindex:"-1"},[d("设计理念 "),e("a",{class:"header-anchor",href:"#设计理念","aria-label":'Permalink to "设计理念"'},"​")],-1),c=[r];function n(i,_,l,p,u,h){return t(),s("div",null,c)}const x=a(o,[["render",n]]);export{f as __pageData,x as default}; +import{_ as a,o as t,c as s,k as e,a as d}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"设计理念","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Redux/设计理念.md","filePath":"guide/Redux/设计理念.md","lastUpdated":1736839070000}'),o={name:"guide/Redux/设计理念.md"},r=e("h1",{id:"设计理念",tabindex:"-1"},[d("设计理念 "),e("a",{class:"header-anchor",href:"#设计理念","aria-label":'Permalink to "设计理念"'},"​")],-1),c=[r];function n(i,_,l,p,u,h){return t(),s("div",null,c)}const x=a(o,[["render",n]]);export{f as __pageData,x as default}; diff --git "a/assets/guide_Redux_\350\256\276\350\256\241\347\220\206\345\277\265.md.c290b375.js" "b/assets/guide_Redux_\350\256\276\350\256\241\347\220\206\345\277\265.md.00f4264a.lean.js" similarity index 89% rename from "assets/guide_Redux_\350\256\276\350\256\241\347\220\206\345\277\265.md.c290b375.js" rename to "assets/guide_Redux_\350\256\276\350\256\241\347\220\206\345\277\265.md.00f4264a.lean.js" index e8c4a054..dc91627c 100644 --- "a/assets/guide_Redux_\350\256\276\350\256\241\347\220\206\345\277\265.md.c290b375.js" +++ "b/assets/guide_Redux_\350\256\276\350\256\241\347\220\206\345\277\265.md.00f4264a.lean.js" @@ -1 +1 @@ -import{_ as a,o as t,c as s,k as e,a as d}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"设计理念","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Redux/设计理念.md","filePath":"guide/Redux/设计理念.md","lastUpdated":1736065650000}'),o={name:"guide/Redux/设计理念.md"},r=e("h1",{id:"设计理念",tabindex:"-1"},[d("设计理念 "),e("a",{class:"header-anchor",href:"#设计理念","aria-label":'Permalink to "设计理念"'},"​")],-1),c=[r];function n(i,_,l,p,u,h){return t(),s("div",null,c)}const x=a(o,[["render",n]]);export{f as __pageData,x as default}; +import{_ as a,o as t,c as s,k as e,a as d}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"设计理念","description":"","frontmatter":{},"headers":[],"relativePath":"guide/Redux/设计理念.md","filePath":"guide/Redux/设计理念.md","lastUpdated":1736839070000}'),o={name:"guide/Redux/设计理念.md"},r=e("h1",{id:"设计理念",tabindex:"-1"},[d("设计理念 "),e("a",{class:"header-anchor",href:"#设计理念","aria-label":'Permalink to "设计理念"'},"​")],-1),c=[r];function n(i,_,l,p,u,h){return t(),s("div",null,c)}const x=a(o,[["render",n]]);export{f as __pageData,x as default}; diff --git "a/assets/guide_ai_Embedding\344\271\213\345\212\240\350\275\275\346\225\260\346\215\256.md.a848af48.js" "b/assets/guide_ai_Embedding\344\271\213\345\212\240\350\275\275\346\225\260\346\215\256.md.6565d23d.js" similarity index 99% rename from "assets/guide_ai_Embedding\344\271\213\345\212\240\350\275\275\346\225\260\346\215\256.md.a848af48.js" rename to "assets/guide_ai_Embedding\344\271\213\345\212\240\350\275\275\346\225\260\346\215\256.md.6565d23d.js" index 32cb1772..e978b7fd 100644 --- "a/assets/guide_ai_Embedding\344\271\213\345\212\240\350\275\275\346\225\260\346\215\256.md.a848af48.js" +++ "b/assets/guide_ai_Embedding\344\271\213\345\212\240\350\275\275\346\225\260\346\215\256.md.6565d23d.js" @@ -1,4 +1,4 @@ -import{_ as s,o as a,c as n,Q as o}from"./chunks/framework.b6910bb2.js";const l="/vitePress-blob/assets/6.fdcb22fa.png",p="/vitePress-blob/assets/7.522a2c1b.png",e="/vitePress-blob/assets/8.adec2266.png",m=JSON.parse('{"title":"Embedding 之加载数据","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/Embedding之加载数据.md","filePath":"guide/ai/Embedding之加载数据.md","lastUpdated":1736065650000}'),t={name:"guide/ai/Embedding之加载数据.md"},c=o(`

Embedding 之加载数据

因为 RAG 本质是给 Chat Bot 额外挂在数据源,而数据源存在的形式是多种多样的,有可能是文件/网页/数据库/代码等等其情况。

针对这个情况,langchain 提供了一系列开箱即用的数据加载器 (loader),方便用户快速加载数据。

Document 对象

Document 对象你可以理解成 langchain 对所有类型的数据的一个统一抽象,其中包含

pageContent 文本内容,即文档对象对应的文本数据 metadata 元数据,文本数据对应的元数据,例如 原始文档的标题、页数等信息,可以用于后面 Retriever 基于此进行筛选。

TypeScript
// Document 对象的定义
+import{_ as s,o as a,c as n,Q as o}from"./chunks/framework.b6910bb2.js";const l="/vitePress-blob/assets/6.fdcb22fa.png",p="/vitePress-blob/assets/7.522a2c1b.png",e="/vitePress-blob/assets/8.adec2266.png",m=JSON.parse('{"title":"Embedding 之加载数据","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/Embedding之加载数据.md","filePath":"guide/ai/Embedding之加载数据.md","lastUpdated":1736839070000}'),t={name:"guide/ai/Embedding之加载数据.md"},c=o(`

Embedding 之加载数据

因为 RAG 本质是给 Chat Bot 额外挂在数据源,而数据源存在的形式是多种多样的,有可能是文件/网页/数据库/代码等等其情况。

针对这个情况,langchain 提供了一系列开箱即用的数据加载器 (loader),方便用户快速加载数据。

Document 对象

Document 对象你可以理解成 langchain 对所有类型的数据的一个统一抽象,其中包含

pageContent 文本内容,即文档对象对应的文本数据 metadata 元数据,文本数据对应的元数据,例如 原始文档的标题、页数等信息,可以用于后面 Retriever 基于此进行筛选。

TypeScript
// Document 对象的定义
 interface Document {
   pageContent: string;
   metadata: Record<string, any>;
diff --git "a/assets/guide_ai_Embedding\344\271\213\345\212\240\350\275\275\346\225\260\346\215\256.md.a848af48.lean.js" "b/assets/guide_ai_Embedding\344\271\213\345\212\240\350\275\275\346\225\260\346\215\256.md.6565d23d.lean.js"
similarity index 89%
rename from "assets/guide_ai_Embedding\344\271\213\345\212\240\350\275\275\346\225\260\346\215\256.md.a848af48.lean.js"
rename to "assets/guide_ai_Embedding\344\271\213\345\212\240\350\275\275\346\225\260\346\215\256.md.6565d23d.lean.js"
index 5b53e86f..1f989432 100644
--- "a/assets/guide_ai_Embedding\344\271\213\345\212\240\350\275\275\346\225\260\346\215\256.md.a848af48.lean.js"
+++ "b/assets/guide_ai_Embedding\344\271\213\345\212\240\350\275\275\346\225\260\346\215\256.md.6565d23d.lean.js"
@@ -1 +1 @@
-import{_ as s,o as a,c as n,Q as o}from"./chunks/framework.b6910bb2.js";const l="/vitePress-blob/assets/6.fdcb22fa.png",p="/vitePress-blob/assets/7.522a2c1b.png",e="/vitePress-blob/assets/8.adec2266.png",m=JSON.parse('{"title":"Embedding 之加载数据","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/Embedding之加载数据.md","filePath":"guide/ai/Embedding之加载数据.md","lastUpdated":1736065650000}'),t={name:"guide/ai/Embedding之加载数据.md"},c=o("",27),r=[c];function E(y,i,d,h,F,u){return a(),n("div",null,r)}const b=s(t,[["render",E]]);export{m as __pageData,b as default};
+import{_ as s,o as a,c as n,Q as o}from"./chunks/framework.b6910bb2.js";const l="/vitePress-blob/assets/6.fdcb22fa.png",p="/vitePress-blob/assets/7.522a2c1b.png",e="/vitePress-blob/assets/8.adec2266.png",m=JSON.parse('{"title":"Embedding 之加载数据","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/Embedding之加载数据.md","filePath":"guide/ai/Embedding之加载数据.md","lastUpdated":1736839070000}'),t={name:"guide/ai/Embedding之加载数据.md"},c=o("",27),r=[c];function E(y,i,d,h,F,u){return a(),n("div",null,r)}const b=s(t,[["render",E]]);export{m as __pageData,b as default};
diff --git "a/assets/guide_ai_Embedding\344\271\213\345\244\247\350\247\204\346\250\241\346\225\260\346\215\256\346\213\206\345\210\206.md.8b24253c.js" "b/assets/guide_ai_Embedding\344\271\213\345\244\247\350\247\204\346\250\241\346\225\260\346\215\256\346\213\206\345\210\206.md.64da80d7.js"
similarity index 99%
rename from "assets/guide_ai_Embedding\344\271\213\345\244\247\350\247\204\346\250\241\346\225\260\346\215\256\346\213\206\345\210\206.md.8b24253c.js"
rename to "assets/guide_ai_Embedding\344\271\213\345\244\247\350\247\204\346\250\241\346\225\260\346\215\256\346\213\206\345\210\206.md.64da80d7.js"
index 3b2ae809..dbe5708d 100644
--- "a/assets/guide_ai_Embedding\344\271\213\345\244\247\350\247\204\346\250\241\346\225\260\346\215\256\346\213\206\345\210\206.md.8b24253c.js"
+++ "b/assets/guide_ai_Embedding\344\271\213\345\244\247\350\247\204\346\250\241\346\225\260\346\215\256\346\213\206\345\210\206.md.64da80d7.js"
@@ -1,4 +1,4 @@
-import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/9.a690b18c.png",F=JSON.parse('{"title":"Embedding 之大规模数据拆分","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/Embedding之大规模数据拆分.md","filePath":"guide/ai/Embedding之大规模数据拆分.md","lastUpdated":1736065650000}'),o={name:"guide/ai/Embedding之大规模数据拆分.md"},e=l(`

Embedding 之大规模数据拆分

受限于常见 LLM 的上下文大小,例如 gpt3.5t 是 16k、gpt4t 是 128k,我们并不能把完整的数据整个塞到对话的上下文中。

即使数据源接近于 LLM 的上下文窗口大小,llm 在读取数据时很容易出现分神,或者忽略其中部分细节的问题。

因此,我们需要对数据进行拆分,然后将最关联的内容输入给 LLM,以便 LLM 能够更好的理解数据。

分割数据

对于分割来说,将数据分割成具有独立逻辑的段落是比较好的选择,段落分割的质量越高,意味着 RAG 的效果越好,LLM 返回数据的质量就越好。

langchain 目前提供的切分工具有:

  • Recursive: 根据给定的切分字符(例如 \\n\\n、\\n等),递归的切分
  • HTML: 根据 html 特定字符进行切分
  • Markdown: 根据 md 的特定字符进行切分
  • Code: 根据不同编程语言的特定字符进行切分
  • Token: 根据文本块的 token 数据进行切分
  • Character: 根据用户给定的字符进行切割

RecursiveCharacterTextSplitter

RecursiveCharacterTextSplitter 默认的分隔符列表是 ["\\n\\n", "\\n", " ", ""], 你可以理解为它将文本分割后,在根据设置的 chunk 大小进行组装。

最影响切分质量的就是两个参数:

  • chunkSize: 定义切分块的大小
  • chunkOverlap: 定义块与块之间重叠的大小。较大的重叠可能会导致分割的内容重复,较小的重叠可能会导致分割的内容不完整。
js
// RecursiveCharacterTextSplitter 对文本进行分割
+import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/9.a690b18c.png",F=JSON.parse('{"title":"Embedding 之大规模数据拆分","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/Embedding之大规模数据拆分.md","filePath":"guide/ai/Embedding之大规模数据拆分.md","lastUpdated":1736839070000}'),o={name:"guide/ai/Embedding之大规模数据拆分.md"},e=l(`

Embedding 之大规模数据拆分

受限于常见 LLM 的上下文大小,例如 gpt3.5t 是 16k、gpt4t 是 128k,我们并不能把完整的数据整个塞到对话的上下文中。

即使数据源接近于 LLM 的上下文窗口大小,llm 在读取数据时很容易出现分神,或者忽略其中部分细节的问题。

因此,我们需要对数据进行拆分,然后将最关联的内容输入给 LLM,以便 LLM 能够更好的理解数据。

分割数据

对于分割来说,将数据分割成具有独立逻辑的段落是比较好的选择,段落分割的质量越高,意味着 RAG 的效果越好,LLM 返回数据的质量就越好。

langchain 目前提供的切分工具有:

  • Recursive: 根据给定的切分字符(例如 \\n\\n、\\n等),递归的切分
  • HTML: 根据 html 特定字符进行切分
  • Markdown: 根据 md 的特定字符进行切分
  • Code: 根据不同编程语言的特定字符进行切分
  • Token: 根据文本块的 token 数据进行切分
  • Character: 根据用户给定的字符进行切割

RecursiveCharacterTextSplitter

RecursiveCharacterTextSplitter 默认的分隔符列表是 ["\\n\\n", "\\n", " ", ""], 你可以理解为它将文本分割后,在根据设置的 chunk 大小进行组装。

最影响切分质量的就是两个参数:

  • chunkSize: 定义切分块的大小
  • chunkOverlap: 定义块与块之间重叠的大小。较大的重叠可能会导致分割的内容重复,较小的重叠可能会导致分割的内容不完整。
js
// RecursiveCharacterTextSplitter 对文本进行分割
 // https://chunkviz.up.railway.app/
 import { RecursiveCharacterTextSplitter } from 'langchain/text_splitter';
 import { TextLoader } from 'langchain/document_loaders/fs/text';
diff --git "a/assets/guide_ai_Embedding\344\271\213\345\244\247\350\247\204\346\250\241\346\225\260\346\215\256\346\213\206\345\210\206.md.8b24253c.lean.js" "b/assets/guide_ai_Embedding\344\271\213\345\244\247\350\247\204\346\250\241\346\225\260\346\215\256\346\213\206\345\210\206.md.64da80d7.lean.js"
similarity index 88%
rename from "assets/guide_ai_Embedding\344\271\213\345\244\247\350\247\204\346\250\241\346\225\260\346\215\256\346\213\206\345\210\206.md.8b24253c.lean.js"
rename to "assets/guide_ai_Embedding\344\271\213\345\244\247\350\247\204\346\250\241\346\225\260\346\215\256\346\213\206\345\210\206.md.64da80d7.lean.js"
index 08c11af7..e66382c0 100644
--- "a/assets/guide_ai_Embedding\344\271\213\345\244\247\350\247\204\346\250\241\346\225\260\346\215\256\346\213\206\345\210\206.md.8b24253c.lean.js"
+++ "b/assets/guide_ai_Embedding\344\271\213\345\244\247\350\247\204\346\250\241\346\225\260\346\215\256\346\213\206\345\210\206.md.64da80d7.lean.js"
@@ -1 +1 @@
-import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/9.a690b18c.png",F=JSON.parse('{"title":"Embedding 之大规模数据拆分","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/Embedding之大规模数据拆分.md","filePath":"guide/ai/Embedding之大规模数据拆分.md","lastUpdated":1736065650000}'),o={name:"guide/ai/Embedding之大规模数据拆分.md"},e=l("",14),t=[e];function c(r,i,E,y,d,u){return a(),n("div",null,t)}const _=s(o,[["render",c]]);export{F as __pageData,_ as default};
+import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/9.a690b18c.png",F=JSON.parse('{"title":"Embedding 之大规模数据拆分","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/Embedding之大规模数据拆分.md","filePath":"guide/ai/Embedding之大规模数据拆分.md","lastUpdated":1736839070000}'),o={name:"guide/ai/Embedding之大规模数据拆分.md"},e=l("",14),t=[e];function c(r,i,E,y,d,u){return a(),n("div",null,t)}const _=s(o,[["render",c]]);export{F as __pageData,_ as default};
diff --git "a/assets/guide_ai_OutputParser\346\236\204\345\273\272\346\240\274\345\274\217\345\214\226\350\276\223\345\207\272.md.d681019e.js" "b/assets/guide_ai_OutputParser\346\236\204\345\273\272\346\240\274\345\274\217\345\214\226\350\276\223\345\207\272.md.1a2532c7.js"
similarity index 99%
rename from "assets/guide_ai_OutputParser\346\236\204\345\273\272\346\240\274\345\274\217\345\214\226\350\276\223\345\207\272.md.d681019e.js"
rename to "assets/guide_ai_OutputParser\346\236\204\345\273\272\346\240\274\345\274\217\345\214\226\350\276\223\345\207\272.md.1a2532c7.js"
index d9a046ad..809d8208 100644
--- "a/assets/guide_ai_OutputParser\346\236\204\345\273\272\346\240\274\345\274\217\345\214\226\350\276\223\345\207\272.md.d681019e.js"
+++ "b/assets/guide_ai_OutputParser\346\236\204\345\273\272\346\240\274\345\274\217\345\214\226\350\276\223\345\207\272.md.1a2532c7.js"
@@ -1,4 +1,4 @@
-import{_ as s,o as n,c as a,Q as o}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/4.43d3c47e.png",m=JSON.parse('{"title":"OutputParser","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/OutputParser构建格式化输出.md","filePath":"guide/ai/OutputParser构建格式化输出.md","lastUpdated":1736065650000}'),l={name:"guide/ai/OutputParser构建格式化输出.md"},t=o(`

OutputParser

OutputParser是一个用于解析模型输出的工具类,它可以帮助用户将模型输出解析为用户需要的格式。

String Output Parser

StringOutputParser是OutputParser的一个实现类,它可以将模型输出解析为字符串。

前面的例子中,我们使用了StringOutputParser来解析模型输出,这里就不过多赘述了。

StructuredOutputParser (结构化的输出)

StructuredOutputParser是OutputParser的另一个实现类,它可以将模型输出解析为结构化的数据。

例如,我们可以将模型输出解析为JSON格式的数据,这样我们就可以更方便地处理模型输出。

下面是一个使用StructuredOutputParser的例子:

js
import { StructuredOutputParser } from "langchain/output_parsers";
+import{_ as s,o as n,c as a,Q as o}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/4.43d3c47e.png",m=JSON.parse('{"title":"OutputParser","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/OutputParser构建格式化输出.md","filePath":"guide/ai/OutputParser构建格式化输出.md","lastUpdated":1736839070000}'),l={name:"guide/ai/OutputParser构建格式化输出.md"},t=o(`

OutputParser

OutputParser是一个用于解析模型输出的工具类,它可以帮助用户将模型输出解析为用户需要的格式。

String Output Parser

StringOutputParser是OutputParser的一个实现类,它可以将模型输出解析为字符串。

前面的例子中,我们使用了StringOutputParser来解析模型输出,这里就不过多赘述了。

StructuredOutputParser (结构化的输出)

StructuredOutputParser是OutputParser的另一个实现类,它可以将模型输出解析为结构化的数据。

例如,我们可以将模型输出解析为JSON格式的数据,这样我们就可以更方便地处理模型输出。

下面是一个使用StructuredOutputParser的例子:

js
import { StructuredOutputParser } from "langchain/output_parsers";
 import { PromptTemplate } from "@langchain/core/prompts";
 
 const parser = StructuredOutputParser.fromNamesAndDescriptions({
diff --git "a/assets/guide_ai_OutputParser\346\236\204\345\273\272\346\240\274\345\274\217\345\214\226\350\276\223\345\207\272.md.d681019e.lean.js" "b/assets/guide_ai_OutputParser\346\236\204\345\273\272\346\240\274\345\274\217\345\214\226\350\276\223\345\207\272.md.1a2532c7.lean.js"
similarity index 88%
rename from "assets/guide_ai_OutputParser\346\236\204\345\273\272\346\240\274\345\274\217\345\214\226\350\276\223\345\207\272.md.d681019e.lean.js"
rename to "assets/guide_ai_OutputParser\346\236\204\345\273\272\346\240\274\345\274\217\345\214\226\350\276\223\345\207\272.md.1a2532c7.lean.js"
index 664916b9..3b30cc2f 100644
--- "a/assets/guide_ai_OutputParser\346\236\204\345\273\272\346\240\274\345\274\217\345\214\226\350\276\223\345\207\272.md.d681019e.lean.js"
+++ "b/assets/guide_ai_OutputParser\346\236\204\345\273\272\346\240\274\345\274\217\345\214\226\350\276\223\345\207\272.md.1a2532c7.lean.js"
@@ -1 +1 @@
-import{_ as s,o as n,c as a,Q as o}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/4.43d3c47e.png",m=JSON.parse('{"title":"OutputParser","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/OutputParser构建格式化输出.md","filePath":"guide/ai/OutputParser构建格式化输出.md","lastUpdated":1736065650000}'),l={name:"guide/ai/OutputParser构建格式化输出.md"},t=o("",31),e=[t];function r(c,E,y,u,i,F){return n(),a("div",null,e)}const d=s(l,[["render",r]]);export{m as __pageData,d as default};
+import{_ as s,o as n,c as a,Q as o}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/4.43d3c47e.png",m=JSON.parse('{"title":"OutputParser","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/OutputParser构建格式化输出.md","filePath":"guide/ai/OutputParser构建格式化输出.md","lastUpdated":1736839070000}'),l={name:"guide/ai/OutputParser构建格式化输出.md"},t=o("",31),e=[t];function r(c,E,y,u,i,F){return n(),a("div",null,e)}const d=s(l,[["render",r]]);export{m as __pageData,d as default};
diff --git "a/assets/guide_ai_RAG \346\243\200\347\264\242\345\242\236\345\274\272\347\224\237\346\210\220\347\232\204\346\265\201\347\250\213.md.c97a9cbd.js" "b/assets/guide_ai_RAG \346\243\200\347\264\242\345\242\236\345\274\272\347\224\237\346\210\220\347\232\204\346\265\201\347\250\213.md.58fac56a.js"
similarity index 98%
rename from "assets/guide_ai_RAG \346\243\200\347\264\242\345\242\236\345\274\272\347\224\237\346\210\220\347\232\204\346\265\201\347\250\213.md.c97a9cbd.js"
rename to "assets/guide_ai_RAG \346\243\200\347\264\242\345\242\236\345\274\272\347\224\237\346\210\220\347\232\204\346\265\201\347\250\213.md.58fac56a.js"
index fb2b57d4..562bc905 100644
--- "a/assets/guide_ai_RAG \346\243\200\347\264\242\345\242\236\345\274\272\347\224\237\346\210\220\347\232\204\346\265\201\347\250\213.md.c97a9cbd.js"	
+++ "b/assets/guide_ai_RAG \346\243\200\347\264\242\345\242\236\345\274\272\347\224\237\346\210\220\347\232\204\346\265\201\347\250\213.md.58fac56a.js"	
@@ -1 +1 @@
-import{_ as a,o as e,c as t,Q as r}from"./chunks/framework.b6910bb2.js";const i="/vitePress-blob/assets/5.ca3137a4.png",m=JSON.parse('{"title":"RAG 检索增强生成的流程","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/RAG 检索增强生成的流程.md","filePath":"guide/ai/RAG 检索增强生成的流程.md","lastUpdated":1736065650000}'),o={name:"guide/ai/RAG 检索增强生成的流程.md"},l=r('

RAG 检索增强生成的流程

RAG 其全称是 Retrieval Augmented Generation,可以被翻译成 检索增强生成技术,从标题上也能了解其核心的流程 检索 => 增强 => 生成。

LLM 的局限

在介绍 RAG 之前,我们先来了解一下 LLM 的局限性。

首先是幻觉问题(hallucination),LLM 底层还不具备真正的逻辑推理能力,是根据大量的数据进行概率性预测,所以在某些情况下,LLM 会生成一些不合理的答案。

其次是对领域知识的欠缺,造成这个问题主要是俩个原因,第一个是对知识的更新慢,另一个是对专业领域的知识训练样本不足导致。

RAG 的优势

当我们了解了 LLM 的局限性后,RAG 会尽可能提供与答案相关的上下文,来增强它正确输出的可能性。

RAG 的优势主要体现在以下几个方面:

  1. 用户输入提问
  2. 检索:根据用户提问对 向量数据库 进行相似性检测,查找与回答用户问题最相关的内容
  3. 增强:根据检索的结果,生成 prompt。 一般都会涉及 “仅依赖下述信息源来回答问题” 这种限制 LLM 参考信息源的语句,来减少幻想,让回答更加聚焦
  4. 生成:将增强后的 prompt 传递给 LLM,返回数据给用户

所以 RAG 就是哪里有问题解决哪里,既然大模型无法获得最新和内部的数据集,那我们就使用外挂的向量数据库为 LLM 提供最新和内部的数据库。既然大模型有幻想问题,我们就将回答问题所需要的信息和知识编码到上下文中,强制大模型只参考这些内容进行回答。

注意:LLM 是逻辑推理引擎,而不是信息引擎。所以,由外挂的向量数据库提供最有效的知识,然后由 LLM 根据知识进行推理,提供有价值的回复。

RAG 流程

输出结果

后续章节,在下将针对 RAG 的各个流程做详细的介绍。

',15),_=[l];function p(s,n,c,d,h,L){return e(),t("div",null,_)}const R=a(o,[["render",p]]);export{m as __pageData,R as default}; +import{_ as a,o as e,c as t,Q as r}from"./chunks/framework.b6910bb2.js";const i="/vitePress-blob/assets/5.ca3137a4.png",m=JSON.parse('{"title":"RAG 检索增强生成的流程","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/RAG 检索增强生成的流程.md","filePath":"guide/ai/RAG 检索增强生成的流程.md","lastUpdated":1736839070000}'),o={name:"guide/ai/RAG 检索增强生成的流程.md"},l=r('

RAG 检索增强生成的流程

RAG 其全称是 Retrieval Augmented Generation,可以被翻译成 检索增强生成技术,从标题上也能了解其核心的流程 检索 => 增强 => 生成。

LLM 的局限

在介绍 RAG 之前,我们先来了解一下 LLM 的局限性。

首先是幻觉问题(hallucination),LLM 底层还不具备真正的逻辑推理能力,是根据大量的数据进行概率性预测,所以在某些情况下,LLM 会生成一些不合理的答案。

其次是对领域知识的欠缺,造成这个问题主要是俩个原因,第一个是对知识的更新慢,另一个是对专业领域的知识训练样本不足导致。

RAG 的优势

当我们了解了 LLM 的局限性后,RAG 会尽可能提供与答案相关的上下文,来增强它正确输出的可能性。

RAG 的优势主要体现在以下几个方面:

  1. 用户输入提问
  2. 检索:根据用户提问对 向量数据库 进行相似性检测,查找与回答用户问题最相关的内容
  3. 增强:根据检索的结果,生成 prompt。 一般都会涉及 “仅依赖下述信息源来回答问题” 这种限制 LLM 参考信息源的语句,来减少幻想,让回答更加聚焦
  4. 生成:将增强后的 prompt 传递给 LLM,返回数据给用户

所以 RAG 就是哪里有问题解决哪里,既然大模型无法获得最新和内部的数据集,那我们就使用外挂的向量数据库为 LLM 提供最新和内部的数据库。既然大模型有幻想问题,我们就将回答问题所需要的信息和知识编码到上下文中,强制大模型只参考这些内容进行回答。

注意:LLM 是逻辑推理引擎,而不是信息引擎。所以,由外挂的向量数据库提供最有效的知识,然后由 LLM 根据知识进行推理,提供有价值的回复。

RAG 流程

输出结果

后续章节,在下将针对 RAG 的各个流程做详细的介绍。

',15),_=[l];function p(s,n,c,d,h,L){return e(),t("div",null,_)}const R=a(o,[["render",p]]);export{m as __pageData,R as default}; diff --git "a/assets/guide_ai_RAG \346\243\200\347\264\242\345\242\236\345\274\272\347\224\237\346\210\220\347\232\204\346\265\201\347\250\213.md.c97a9cbd.lean.js" "b/assets/guide_ai_RAG \346\243\200\347\264\242\345\242\236\345\274\272\347\224\237\346\210\220\347\232\204\346\265\201\347\250\213.md.58fac56a.lean.js" similarity index 88% rename from "assets/guide_ai_RAG \346\243\200\347\264\242\345\242\236\345\274\272\347\224\237\346\210\220\347\232\204\346\265\201\347\250\213.md.c97a9cbd.lean.js" rename to "assets/guide_ai_RAG \346\243\200\347\264\242\345\242\236\345\274\272\347\224\237\346\210\220\347\232\204\346\265\201\347\250\213.md.58fac56a.lean.js" index 9165048e..14168f6c 100644 --- "a/assets/guide_ai_RAG \346\243\200\347\264\242\345\242\236\345\274\272\347\224\237\346\210\220\347\232\204\346\265\201\347\250\213.md.c97a9cbd.lean.js" +++ "b/assets/guide_ai_RAG \346\243\200\347\264\242\345\242\236\345\274\272\347\224\237\346\210\220\347\232\204\346\265\201\347\250\213.md.58fac56a.lean.js" @@ -1 +1 @@ -import{_ as a,o as e,c as t,Q as r}from"./chunks/framework.b6910bb2.js";const i="/vitePress-blob/assets/5.ca3137a4.png",m=JSON.parse('{"title":"RAG 检索增强生成的流程","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/RAG 检索增强生成的流程.md","filePath":"guide/ai/RAG 检索增强生成的流程.md","lastUpdated":1736065650000}'),o={name:"guide/ai/RAG 检索增强生成的流程.md"},l=r("",15),_=[l];function p(s,n,c,d,h,L){return e(),t("div",null,_)}const R=a(o,[["render",p]]);export{m as __pageData,R as default}; +import{_ as a,o as e,c as t,Q as r}from"./chunks/framework.b6910bb2.js";const i="/vitePress-blob/assets/5.ca3137a4.png",m=JSON.parse('{"title":"RAG 检索增强生成的流程","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/RAG 检索增强生成的流程.md","filePath":"guide/ai/RAG 检索增强生成的流程.md","lastUpdated":1736839070000}'),o={name:"guide/ai/RAG 检索增强生成的流程.md"},l=r("",15),_=[l];function p(s,n,c,d,h,L){return e(),t("div",null,_)}const R=a(o,[["render",p]]);export{m as __pageData,R as default}; diff --git "a/assets/guide_ai_Retriever \345\270\270\350\247\201\347\232\204\344\274\230\345\214\226\346\226\271\345\274\217.md.86e88704.js" "b/assets/guide_ai_Retriever \345\270\270\350\247\201\347\232\204\344\274\230\345\214\226\346\226\271\345\274\217.md.a17d2dce.js" similarity index 99% rename from "assets/guide_ai_Retriever \345\270\270\350\247\201\347\232\204\344\274\230\345\214\226\346\226\271\345\274\217.md.86e88704.js" rename to "assets/guide_ai_Retriever \345\270\270\350\247\201\347\232\204\344\274\230\345\214\226\346\226\271\345\274\217.md.a17d2dce.js" index c8082327..4e3a4935 100644 --- "a/assets/guide_ai_Retriever \345\270\270\350\247\201\347\232\204\344\274\230\345\214\226\346\226\271\345\274\217.md.86e88704.js" +++ "b/assets/guide_ai_Retriever \345\270\270\350\247\201\347\232\204\344\274\230\345\214\226\346\226\271\345\274\217.md.a17d2dce.js" @@ -1,4 +1,4 @@ -import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/12.f148ecd9.png",p="/vitePress-blob/assets/13.24cbed74.png",v=JSON.parse('{"title":"Retriever 常见的优化方式","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/Retriever 常见的优化方式.md","filePath":"guide/ai/Retriever 常见的优化方式.md","lastUpdated":1736065650000}'),e={name:"guide/ai/Retriever 常见的优化方式.md"},r=l(`

Retriever 常见的优化方式

上一节我们通过厂商提供的 Embedding 算法去制作匹配向量,通过 Facebook 提供的 Faiss 作为向量数据库,将数据存储到向量数据库中。

接下来我们将优化 Retriever 检索数据的方式

MultiQueryRetriever

MultiQueryRetriever 思路,或者说其他解决 llm 缺陷的思路基本都是一致的:加入更多 llm。

而 MultiQueryRetriever 是其中比较简单的一种解决方案,它使用 LLM 去将用户的输入改写成多个不同写法,从不同的角度来表达同一个意思,来克服因为关键词或者细微措词导致检索效果差的问题。

js
// MultiQueryRetriever: 
+import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/12.f148ecd9.png",p="/vitePress-blob/assets/13.24cbed74.png",v=JSON.parse('{"title":"Retriever 常见的优化方式","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/Retriever 常见的优化方式.md","filePath":"guide/ai/Retriever 常见的优化方式.md","lastUpdated":1736839070000}'),e={name:"guide/ai/Retriever 常见的优化方式.md"},r=l(`

Retriever 常见的优化方式

上一节我们通过厂商提供的 Embedding 算法去制作匹配向量,通过 Facebook 提供的 Faiss 作为向量数据库,将数据存储到向量数据库中。

接下来我们将优化 Retriever 检索数据的方式

MultiQueryRetriever

MultiQueryRetriever 思路,或者说其他解决 llm 缺陷的思路基本都是一致的:加入更多 llm。

而 MultiQueryRetriever 是其中比较简单的一种解决方案,它使用 LLM 去将用户的输入改写成多个不同写法,从不同的角度来表达同一个意思,来克服因为关键词或者细微措词导致检索效果差的问题。

js
// MultiQueryRetriever: 
 import "dotenv/config";
 import { FaissStore } from "@langchain/community/vectorstores/faiss";
 import { BaiduQianfanEmbeddings } from "@langchain/community/embeddings/baidu_qianfan";
diff --git "a/assets/guide_ai_Retriever \345\270\270\350\247\201\347\232\204\344\274\230\345\214\226\346\226\271\345\274\217.md.86e88704.lean.js" "b/assets/guide_ai_Retriever \345\270\270\350\247\201\347\232\204\344\274\230\345\214\226\346\226\271\345\274\217.md.a17d2dce.lean.js"
similarity index 89%
rename from "assets/guide_ai_Retriever \345\270\270\350\247\201\347\232\204\344\274\230\345\214\226\346\226\271\345\274\217.md.86e88704.lean.js"
rename to "assets/guide_ai_Retriever \345\270\270\350\247\201\347\232\204\344\274\230\345\214\226\346\226\271\345\274\217.md.a17d2dce.lean.js"
index 23a63cf3..6f48bbcd 100644
--- "a/assets/guide_ai_Retriever \345\270\270\350\247\201\347\232\204\344\274\230\345\214\226\346\226\271\345\274\217.md.86e88704.lean.js"	
+++ "b/assets/guide_ai_Retriever \345\270\270\350\247\201\347\232\204\344\274\230\345\214\226\346\226\271\345\274\217.md.a17d2dce.lean.js"	
@@ -1 +1 @@
-import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/12.f148ecd9.png",p="/vitePress-blob/assets/13.24cbed74.png",v=JSON.parse('{"title":"Retriever 常见的优化方式","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/Retriever 常见的优化方式.md","filePath":"guide/ai/Retriever 常见的优化方式.md","lastUpdated":1736065650000}'),e={name:"guide/ai/Retriever 常见的优化方式.md"},r=l("",24),t=[r];function c(E,y,i,d,F,u){return n(),a("div",null,t)}const h=s(e,[["render",c]]);export{v as __pageData,h as default};
+import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/12.f148ecd9.png",p="/vitePress-blob/assets/13.24cbed74.png",v=JSON.parse('{"title":"Retriever 常见的优化方式","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/Retriever 常见的优化方式.md","filePath":"guide/ai/Retriever 常见的优化方式.md","lastUpdated":1736839070000}'),e={name:"guide/ai/Retriever 常见的优化方式.md"},r=l("",24),t=[r];function c(E,y,i,d,F,u){return n(),a("div",null,t)}const h=s(e,[["render",c]]);export{v as __pageData,h as default};
diff --git "a/assets/guide_ai_Retriever\344\271\213\345\220\221\351\207\217\346\225\260\346\215\256\345\272\223.md.6b2ddb25.js" "b/assets/guide_ai_Retriever\344\271\213\345\220\221\351\207\217\346\225\260\346\215\256\345\272\223.md.0af240f5.js"
similarity index 99%
rename from "assets/guide_ai_Retriever\344\271\213\345\220\221\351\207\217\346\225\260\346\215\256\345\272\223.md.6b2ddb25.js"
rename to "assets/guide_ai_Retriever\344\271\213\345\220\221\351\207\217\346\225\260\346\215\256\345\272\223.md.0af240f5.js"
index d760bdc4..30e56ea5 100644
--- "a/assets/guide_ai_Retriever\344\271\213\345\220\221\351\207\217\346\225\260\346\215\256\345\272\223.md.6b2ddb25.js"
+++ "b/assets/guide_ai_Retriever\344\271\213\345\220\221\351\207\217\346\225\260\346\215\256\345\272\223.md.0af240f5.js"
@@ -1,4 +1,4 @@
-import{_ as s,o as n,c as a,Q as p}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/10.fdf08991.png",m=JSON.parse('{"title":"Retriever 之向量数据库","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/Retriever之向量数据库.md","filePath":"guide/ai/Retriever之向量数据库.md","lastUpdated":1736065650000}'),l={name:"guide/ai/Retriever之向量数据库.md"},e=p(`

Retriever 之向量数据库

经过前面的学习,我们知道了如何对 数据进行加载和切割,接下来我们就要学习如何将数据通过 Embedding 算法转化为向量加载到向量数据库中。

Embedding

这里我们用最简单的词袋(words bag)模型来描述一下最简单的 embedding 过程,让大家更具象化的理解这个。

简单地说,词袋模型首先将一篇文章拆分成一个个单词,然后将其放入袋子里面。

例如我们有十篇文章,我们可以将文章拆分成一个个单词,然后统计单词出现的次数

js
第一篇文章:
+import{_ as s,o as n,c as a,Q as p}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/10.fdf08991.png",m=JSON.parse('{"title":"Retriever 之向量数据库","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/Retriever之向量数据库.md","filePath":"guide/ai/Retriever之向量数据库.md","lastUpdated":1736839070000}'),l={name:"guide/ai/Retriever之向量数据库.md"},e=p(`

Retriever 之向量数据库

经过前面的学习,我们知道了如何对 数据进行加载和切割,接下来我们就要学习如何将数据通过 Embedding 算法转化为向量加载到向量数据库中。

Embedding

这里我们用最简单的词袋(words bag)模型来描述一下最简单的 embedding 过程,让大家更具象化的理解这个。

简单地说,词袋模型首先将一篇文章拆分成一个个单词,然后将其放入袋子里面。

例如我们有十篇文章,我们可以将文章拆分成一个个单词,然后统计单词出现的次数

js
第一篇文章:
 enson: 10  cool: 5  handsome: 8
 
 第二篇文章:
diff --git "a/assets/guide_ai_Retriever\344\271\213\345\220\221\351\207\217\346\225\260\346\215\256\345\272\223.md.6b2ddb25.lean.js" "b/assets/guide_ai_Retriever\344\271\213\345\220\221\351\207\217\346\225\260\346\215\256\345\272\223.md.0af240f5.lean.js"
similarity index 88%
rename from "assets/guide_ai_Retriever\344\271\213\345\220\221\351\207\217\346\225\260\346\215\256\345\272\223.md.6b2ddb25.lean.js"
rename to "assets/guide_ai_Retriever\344\271\213\345\220\221\351\207\217\346\225\260\346\215\256\345\272\223.md.0af240f5.lean.js"
index 8565d10a..174de5ef 100644
--- "a/assets/guide_ai_Retriever\344\271\213\345\220\221\351\207\217\346\225\260\346\215\256\345\272\223.md.6b2ddb25.lean.js"
+++ "b/assets/guide_ai_Retriever\344\271\213\345\220\221\351\207\217\346\225\260\346\215\256\345\272\223.md.0af240f5.lean.js"
@@ -1 +1 @@
-import{_ as s,o as n,c as a,Q as p}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/10.fdf08991.png",m=JSON.parse('{"title":"Retriever 之向量数据库","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/Retriever之向量数据库.md","filePath":"guide/ai/Retriever之向量数据库.md","lastUpdated":1736065650000}'),l={name:"guide/ai/Retriever之向量数据库.md"},e=p("",21),t=[e];function c(r,y,E,i,d,F){return n(),a("div",null,t)}const u=s(l,[["render",c]]);export{m as __pageData,u as default};
+import{_ as s,o as n,c as a,Q as p}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/10.fdf08991.png",m=JSON.parse('{"title":"Retriever 之向量数据库","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/Retriever之向量数据库.md","filePath":"guide/ai/Retriever之向量数据库.md","lastUpdated":1736839070000}'),l={name:"guide/ai/Retriever之向量数据库.md"},e=p("",21),t=[e];function c(r,y,E,i,d,F){return n(),a("div",null,t)}const u=s(l,[["render",c]]);export{m as __pageData,u as default};
diff --git "a/assets/guide_ai_langchain\345\277\253\351\200\237\345\205\245\351\227\250.md.ab114f06.js" "b/assets/guide_ai_langchain\345\277\253\351\200\237\345\205\245\351\227\250.md.e65bc2d1.js"
similarity index 99%
rename from "assets/guide_ai_langchain\345\277\253\351\200\237\345\205\245\351\227\250.md.ab114f06.js"
rename to "assets/guide_ai_langchain\345\277\253\351\200\237\345\205\245\351\227\250.md.e65bc2d1.js"
index f7fb588f..a6a73b2d 100644
--- "a/assets/guide_ai_langchain\345\277\253\351\200\237\345\205\245\351\227\250.md.ab114f06.js"
+++ "b/assets/guide_ai_langchain\345\277\253\351\200\237\345\205\245\351\227\250.md.e65bc2d1.js"
@@ -1,4 +1,4 @@
-import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/1.111604b1.png",p="/vitePress-blob/assets/2.3cc0a64c.png",e="/vitePress-blob/assets/3.6ba488f9.png",F=JSON.parse('{"title":"LangChain 快速入门","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/langchain快速入门.md","filePath":"guide/ai/langchain快速入门.md","lastUpdated":1736065650000}'),t={name:"guide/ai/langchain快速入门.md"},c=l(`

LangChain 快速入门

安装

要安装这个包,你可以使用 npm 或 yarn:

bash
yarn add langchain
yarn add langchain

安装环境

node >= 18.x

什么是 LCEL (LangChain Expression Language)

LangChain Expression Language is a way to create arbitrary custom chains. It is built on the Runnable protocol.

LCEL 无论是 python 还是 js 版本都在主推的新设计,能创建自定义的链,它是基于 Runnable 协议构建的。

通过 LangChain 加载大模型

本地大模型

在 mac 平台下,推荐用 ollama,使用简单,下载好模型后,点击这个 app,就会自动在 http://localhost:11434 起一个 llm 的服务。

安装 llama3 模型

可以通过 ollama list 查看所有的模型,然后通过 ollama pull 安装模型。

bash
# 模型列表参考:https://ollama.com/library
+import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/1.111604b1.png",p="/vitePress-blob/assets/2.3cc0a64c.png",e="/vitePress-blob/assets/3.6ba488f9.png",F=JSON.parse('{"title":"LangChain 快速入门","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/langchain快速入门.md","filePath":"guide/ai/langchain快速入门.md","lastUpdated":1736839070000}'),t={name:"guide/ai/langchain快速入门.md"},c=l(`

LangChain 快速入门

安装

要安装这个包,你可以使用 npm 或 yarn:

bash
yarn add langchain
yarn add langchain

安装环境

node >= 18.x

什么是 LCEL (LangChain Expression Language)

LangChain Expression Language is a way to create arbitrary custom chains. It is built on the Runnable protocol.

LCEL 无论是 python 还是 js 版本都在主推的新设计,能创建自定义的链,它是基于 Runnable 协议构建的。

通过 LangChain 加载大模型

本地大模型

在 mac 平台下,推荐用 ollama,使用简单,下载好模型后,点击这个 app,就会自动在 http://localhost:11434 起一个 llm 的服务。

安装 llama3 模型

可以通过 ollama list 查看所有的模型,然后通过 ollama pull 安装模型。

bash
# 模型列表参考:https://ollama.com/library
 ollama pull [model]
# 模型列表参考:https://ollama.com/library
 ollama pull [model]

目前下载了一个 ollama3 的模型

初始状态图

当我们在本地启动了一个模型后,我们就可以通过 LangChain 来调用这个模型了。

js
import { Ollama } from '@langchain/community/llms/ollama';
 
diff --git "a/assets/guide_ai_langchain\345\277\253\351\200\237\345\205\245\351\227\250.md.ab114f06.lean.js" "b/assets/guide_ai_langchain\345\277\253\351\200\237\345\205\245\351\227\250.md.e65bc2d1.lean.js"
similarity index 89%
rename from "assets/guide_ai_langchain\345\277\253\351\200\237\345\205\245\351\227\250.md.ab114f06.lean.js"
rename to "assets/guide_ai_langchain\345\277\253\351\200\237\345\205\245\351\227\250.md.e65bc2d1.lean.js"
index f6a5ab91..18e451bb 100644
--- "a/assets/guide_ai_langchain\345\277\253\351\200\237\345\205\245\351\227\250.md.ab114f06.lean.js"
+++ "b/assets/guide_ai_langchain\345\277\253\351\200\237\345\205\245\351\227\250.md.e65bc2d1.lean.js"
@@ -1 +1 @@
-import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/1.111604b1.png",p="/vitePress-blob/assets/2.3cc0a64c.png",e="/vitePress-blob/assets/3.6ba488f9.png",F=JSON.parse('{"title":"LangChain 快速入门","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/langchain快速入门.md","filePath":"guide/ai/langchain快速入门.md","lastUpdated":1736065650000}'),t={name:"guide/ai/langchain快速入门.md"},c=l("",44),r=[c];function i(E,y,h,d,u,g){return a(),n("div",null,r)}const b=s(t,[["render",i]]);export{F as __pageData,b as default};
+import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/1.111604b1.png",p="/vitePress-blob/assets/2.3cc0a64c.png",e="/vitePress-blob/assets/3.6ba488f9.png",F=JSON.parse('{"title":"LangChain 快速入门","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/langchain快速入门.md","filePath":"guide/ai/langchain快速入门.md","lastUpdated":1736839070000}'),t={name:"guide/ai/langchain快速入门.md"},c=l("",44),r=[c];function i(E,y,h,d,u,g){return a(),n("div",null,r)}const b=s(t,[["render",i]]);export{F as __pageData,b as default};
diff --git "a/assets/guide_ai_\344\273\200\344\271\210\346\230\257langchain.md.a8c31c8d.js" "b/assets/guide_ai_\344\273\200\344\271\210\346\230\257langchain.md.487089c2.js"
similarity index 98%
rename from "assets/guide_ai_\344\273\200\344\271\210\346\230\257langchain.md.a8c31c8d.js"
rename to "assets/guide_ai_\344\273\200\344\271\210\346\230\257langchain.md.487089c2.js"
index ae24f17c..01fa88b4 100644
--- "a/assets/guide_ai_\344\273\200\344\271\210\346\230\257langchain.md.a8c31c8d.js"
+++ "b/assets/guide_ai_\344\273\200\344\271\210\346\230\257langchain.md.487089c2.js"
@@ -1 +1 @@
-import{_ as a,o as n,c as i,Q as e}from"./chunks/framework.b6910bb2.js";const _=JSON.parse('{"title":"什么是 LangChain","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/什么是langchain.md","filePath":"guide/ai/什么是langchain.md","lastUpdated":1736065650000}'),t={name:"guide/ai/什么是langchain.md"},r=e('

什么是 LangChain

LangChain 是一个用于开发由大型语言模型 (LLMs) 驱动的应用程序的框架。

为什么是 LangChain

因为 LLM 的 API 只是提供了一个非常基础的调用方式,当我们需要构建一个复杂的 Chat Bot 时,就需要考虑如何保存聊天的上下文、网络搜索、加载 PDF 等工程问题, 而LangChain 提供了一种解决方案,让开发者可以专注于业务逻辑的开发。

足够的流行度和认可度,目前已经在 Github 获得 83k star,并且其上升速度非常恐怖:

Star History Chart

而 LangChain.js 并不是 Python 版本的套壳,而是一个完整的团队从 0 开始构建的生态,足以看出官方对 JavaScript 生态的重视:

Star History Chart

基于此,在下将以 LangChain.js 为例,去使用大模型领域最流行的框架去构建应用,感受大模型的魅力。

支持的环境

  • Node.js (ESM and CommonJS) - 18.x, 19.x, 20.x
  • Cloudflare Workers Cloudflare
  • Vercel / Next.js (Browser, Serverless and Edge functions)
  • Supabase Edge Functions Supabase
  • Browser
  • Deno

相关文档

',13),l=[r];function h(s,o,c,g,p,d){return n(),i("div",null,l)}const u=a(t,[["render",h]]);export{_ as __pageData,u as default}; +import{_ as a,o as n,c as i,Q as e}from"./chunks/framework.b6910bb2.js";const _=JSON.parse('{"title":"什么是 LangChain","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/什么是langchain.md","filePath":"guide/ai/什么是langchain.md","lastUpdated":1736839070000}'),t={name:"guide/ai/什么是langchain.md"},r=e('

什么是 LangChain

LangChain 是一个用于开发由大型语言模型 (LLMs) 驱动的应用程序的框架。

为什么是 LangChain

因为 LLM 的 API 只是提供了一个非常基础的调用方式,当我们需要构建一个复杂的 Chat Bot 时,就需要考虑如何保存聊天的上下文、网络搜索、加载 PDF 等工程问题, 而LangChain 提供了一种解决方案,让开发者可以专注于业务逻辑的开发。

足够的流行度和认可度,目前已经在 Github 获得 83k star,并且其上升速度非常恐怖:

Star History Chart

而 LangChain.js 并不是 Python 版本的套壳,而是一个完整的团队从 0 开始构建的生态,足以看出官方对 JavaScript 生态的重视:

Star History Chart

基于此,在下将以 LangChain.js 为例,去使用大模型领域最流行的框架去构建应用,感受大模型的魅力。

支持的环境

  • Node.js (ESM and CommonJS) - 18.x, 19.x, 20.x
  • Cloudflare Workers Cloudflare
  • Vercel / Next.js (Browser, Serverless and Edge functions)
  • Supabase Edge Functions Supabase
  • Browser
  • Deno

相关文档

',13),l=[r];function h(s,o,c,g,p,d){return n(),i("div",null,l)}const u=a(t,[["render",h]]);export{_ as __pageData,u as default}; diff --git "a/assets/guide_ai_\344\273\200\344\271\210\346\230\257langchain.md.a8c31c8d.lean.js" "b/assets/guide_ai_\344\273\200\344\271\210\346\230\257langchain.md.487089c2.lean.js" similarity index 86% rename from "assets/guide_ai_\344\273\200\344\271\210\346\230\257langchain.md.a8c31c8d.lean.js" rename to "assets/guide_ai_\344\273\200\344\271\210\346\230\257langchain.md.487089c2.lean.js" index 7ae0ca24..1b34ae74 100644 --- "a/assets/guide_ai_\344\273\200\344\271\210\346\230\257langchain.md.a8c31c8d.lean.js" +++ "b/assets/guide_ai_\344\273\200\344\271\210\346\230\257langchain.md.487089c2.lean.js" @@ -1 +1 @@ -import{_ as a,o as n,c as i,Q as e}from"./chunks/framework.b6910bb2.js";const _=JSON.parse('{"title":"什么是 LangChain","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/什么是langchain.md","filePath":"guide/ai/什么是langchain.md","lastUpdated":1736065650000}'),t={name:"guide/ai/什么是langchain.md"},r=e("",13),l=[r];function h(s,o,c,g,p,d){return n(),i("div",null,l)}const u=a(t,[["render",h]]);export{_ as __pageData,u as default}; +import{_ as a,o as n,c as i,Q as e}from"./chunks/framework.b6910bb2.js";const _=JSON.parse('{"title":"什么是 LangChain","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/什么是langchain.md","filePath":"guide/ai/什么是langchain.md","lastUpdated":1736839070000}'),t={name:"guide/ai/什么是langchain.md"},r=e("",13),l=[r];function h(s,o,c,g,p,d){return n(),i("div",null,l)}const u=a(t,[["render",h]]);export{_ as __pageData,u as default}; diff --git "a/assets/guide_ai_\346\236\204\345\273\272\345\217\257\345\244\215\347\224\250\347\232\204PromptTemplate.md.25d3a453.js" "b/assets/guide_ai_\346\236\204\345\273\272\345\217\257\345\244\215\347\224\250\347\232\204PromptTemplate.md.2d5ec969.js" similarity index 99% rename from "assets/guide_ai_\346\236\204\345\273\272\345\217\257\345\244\215\347\224\250\347\232\204PromptTemplate.md.25d3a453.js" rename to "assets/guide_ai_\346\236\204\345\273\272\345\217\257\345\244\215\347\224\250\347\232\204PromptTemplate.md.2d5ec969.js" index 3264244d..39c4e775 100644 --- "a/assets/guide_ai_\346\236\204\345\273\272\345\217\257\345\244\215\347\224\250\347\232\204PromptTemplate.md.25d3a453.js" +++ "b/assets/guide_ai_\346\236\204\345\273\272\345\217\257\345\244\215\347\224\250\347\232\204PromptTemplate.md.2d5ec969.js" @@ -1,4 +1,4 @@ -import{_ as s,o as a,c as n,Q as p}from"./chunks/framework.b6910bb2.js";const u=JSON.parse('{"title":"构建可复用的 PromptTemplate","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/构建可复用的PromptTemplate.md","filePath":"guide/ai/构建可复用的PromptTemplate.md","lastUpdated":1736065650000}'),l={name:"guide/ai/构建可复用的PromptTemplate.md"},o=p(`

构建可复用的 PromptTemplate

Prompt 是大模型的核心,传统的方式一般是通过字符串或者字符串模版来构建 Prompt,但是这种方式不够灵活,也不够易用。为了解决这个问题,LangChain 引入了 PromptTemplate,它是一个可以复用的 Prompt 模版,可以通过参数化的方式来构建 Prompt。

基础的 Prompt 模版使用

js
// 基础的 Prompt 模版使用
+import{_ as s,o as a,c as n,Q as p}from"./chunks/framework.b6910bb2.js";const u=JSON.parse('{"title":"构建可复用的 PromptTemplate","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/构建可复用的PromptTemplate.md","filePath":"guide/ai/构建可复用的PromptTemplate.md","lastUpdated":1736839070000}'),l={name:"guide/ai/构建可复用的PromptTemplate.md"},o=p(`

构建可复用的 PromptTemplate

Prompt 是大模型的核心,传统的方式一般是通过字符串或者字符串模版来构建 Prompt,但是这种方式不够灵活,也不够易用。为了解决这个问题,LangChain 引入了 PromptTemplate,它是一个可以复用的 Prompt 模版,可以通过参数化的方式来构建 Prompt。

基础的 Prompt 模版使用

js
// 基础的 Prompt 模版使用
 import { PromptTemplate } from "@langchain/core/prompts";
 
 const greetingPrompt = new PromptTemplate({
diff --git "a/assets/guide_ai_\346\236\204\345\273\272\345\217\257\345\244\215\347\224\250\347\232\204PromptTemplate.md.25d3a453.lean.js" "b/assets/guide_ai_\346\236\204\345\273\272\345\217\257\345\244\215\347\224\250\347\232\204PromptTemplate.md.2d5ec969.lean.js"
similarity index 87%
rename from "assets/guide_ai_\346\236\204\345\273\272\345\217\257\345\244\215\347\224\250\347\232\204PromptTemplate.md.25d3a453.lean.js"
rename to "assets/guide_ai_\346\236\204\345\273\272\345\217\257\345\244\215\347\224\250\347\232\204PromptTemplate.md.2d5ec969.lean.js"
index b6e0fb18..c532973f 100644
--- "a/assets/guide_ai_\346\236\204\345\273\272\345\217\257\345\244\215\347\224\250\347\232\204PromptTemplate.md.25d3a453.lean.js"
+++ "b/assets/guide_ai_\346\236\204\345\273\272\345\217\257\345\244\215\347\224\250\347\232\204PromptTemplate.md.2d5ec969.lean.js"
@@ -1 +1 @@
-import{_ as s,o as a,c as n,Q as p}from"./chunks/framework.b6910bb2.js";const u=JSON.parse('{"title":"构建可复用的 PromptTemplate","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/构建可复用的PromptTemplate.md","filePath":"guide/ai/构建可复用的PromptTemplate.md","lastUpdated":1736065650000}'),l={name:"guide/ai/构建可复用的PromptTemplate.md"},o=p("",18),e=[o];function t(c,r,E,y,i,m){return a(),n("div",null,e)}const d=s(l,[["render",t]]);export{u as __pageData,d as default};
+import{_ as s,o as a,c as n,Q as p}from"./chunks/framework.b6910bb2.js";const u=JSON.parse('{"title":"构建可复用的 PromptTemplate","description":"","frontmatter":{},"headers":[],"relativePath":"guide/ai/构建可复用的PromptTemplate.md","filePath":"guide/ai/构建可复用的PromptTemplate.md","lastUpdated":1736839070000}'),l={name:"guide/ai/构建可复用的PromptTemplate.md"},o=p("",18),e=[o];function t(c,r,E,y,i,m){return a(),n("div",null,e)}const d=s(l,[["render",t]]);export{u as __pageData,d as default};
diff --git "a/assets/guide_canvas_Canvas\345\260\272\345\257\270\345\217\212\345\210\206\350\276\250\347\216\207\347\237\253\346\255\243.md.e573b0a5.js" "b/assets/guide_canvas_Canvas\345\260\272\345\257\270\345\217\212\345\210\206\350\276\250\347\216\207\347\237\253\346\255\243.md.6671b487.js"
similarity index 99%
rename from "assets/guide_canvas_Canvas\345\260\272\345\257\270\345\217\212\345\210\206\350\276\250\347\216\207\347\237\253\346\255\243.md.e573b0a5.js"
rename to "assets/guide_canvas_Canvas\345\260\272\345\257\270\345\217\212\345\210\206\350\276\250\347\216\207\347\237\253\346\255\243.md.6671b487.js"
index f10abe49..b5b6f7cf 100644
--- "a/assets/guide_canvas_Canvas\345\260\272\345\257\270\345\217\212\345\210\206\350\276\250\347\216\207\347\237\253\346\255\243.md.e573b0a5.js"
+++ "b/assets/guide_canvas_Canvas\345\260\272\345\257\270\345\217\212\345\210\206\350\276\250\347\216\207\347\237\253\346\255\243.md.6671b487.js"
@@ -1,4 +1,4 @@
-import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/1.15edf814.png",o="/vitePress-blob/assets/2.19007fd9.png",t="/vitePress-blob/assets/3.59bb2b90.png",v=JSON.parse('{"title":"Canvas 尺寸与分辨率矫正","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/Canvas尺寸及分辨率矫正.md","filePath":"guide/canvas/Canvas尺寸及分辨率矫正.md","lastUpdated":1736065650000}'),e={name:"guide/canvas/Canvas尺寸及分辨率矫正.md"},c=l(`

Canvas 尺寸与分辨率矫正

前言

Canvas 的默认大小为 300 像素 ×150 像素(宽 × 高,像素的单位是 px)

html
<canvas id="tutorial" width="150" height="150"></canvas>
<canvas id="tutorial" width="150" height="150"></canvas>

canvas 看起来和 img 元素很相像,唯一的不同就是它并没有 src 和 alt 属性。实际上,canvas 标签只有两个属性: width 和 height。

正文

设置画布的 css 大小

canvas 可以使用CSS来定义大小,但在绘制时图像会伸缩以适应它的框架尺寸:如果 CSS 的尺寸与初始画布的比例不一致,它会出现扭曲。

假设一个场景,当我们给 canvas 设置了 css 的宽高 500px * 500px, 但是我们在 canvas 中绘制的图形是 100px * 100px, 那么最终的图形会被拉伸,如下图所示:

html
<canvas id="test-canvas"></canvas>
+import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/1.15edf814.png",o="/vitePress-blob/assets/2.19007fd9.png",t="/vitePress-blob/assets/3.59bb2b90.png",v=JSON.parse('{"title":"Canvas 尺寸与分辨率矫正","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/Canvas尺寸及分辨率矫正.md","filePath":"guide/canvas/Canvas尺寸及分辨率矫正.md","lastUpdated":1736839070000}'),e={name:"guide/canvas/Canvas尺寸及分辨率矫正.md"},c=l(`

Canvas 尺寸与分辨率矫正

前言

Canvas 的默认大小为 300 像素 ×150 像素(宽 × 高,像素的单位是 px)

html
<canvas id="tutorial" width="150" height="150"></canvas>
<canvas id="tutorial" width="150" height="150"></canvas>

canvas 看起来和 img 元素很相像,唯一的不同就是它并没有 src 和 alt 属性。实际上,canvas 标签只有两个属性: width 和 height。

正文

设置画布的 css 大小

canvas 可以使用CSS来定义大小,但在绘制时图像会伸缩以适应它的框架尺寸:如果 CSS 的尺寸与初始画布的比例不一致,它会出现扭曲。

假设一个场景,当我们给 canvas 设置了 css 的宽高 500px * 500px, 但是我们在 canvas 中绘制的图形是 100px * 100px, 那么最终的图形会被拉伸,如下图所示:

html
<canvas id="test-canvas"></canvas>
 <script>
     const canvas = document.getElementById('test-canvas');
     const ctx = canvas.getContext('2d');
diff --git "a/assets/guide_canvas_Canvas\345\260\272\345\257\270\345\217\212\345\210\206\350\276\250\347\216\207\347\237\253\346\255\243.md.e573b0a5.lean.js" "b/assets/guide_canvas_Canvas\345\260\272\345\257\270\345\217\212\345\210\206\350\276\250\347\216\207\347\237\253\346\255\243.md.6671b487.lean.js"
similarity index 90%
rename from "assets/guide_canvas_Canvas\345\260\272\345\257\270\345\217\212\345\210\206\350\276\250\347\216\207\347\237\253\346\255\243.md.e573b0a5.lean.js"
rename to "assets/guide_canvas_Canvas\345\260\272\345\257\270\345\217\212\345\210\206\350\276\250\347\216\207\347\237\253\346\255\243.md.6671b487.lean.js"
index e30f813f..8c65b04c 100644
--- "a/assets/guide_canvas_Canvas\345\260\272\345\257\270\345\217\212\345\210\206\350\276\250\347\216\207\347\237\253\346\255\243.md.e573b0a5.lean.js"
+++ "b/assets/guide_canvas_Canvas\345\260\272\345\257\270\345\217\212\345\210\206\350\276\250\347\216\207\347\237\253\346\255\243.md.6671b487.lean.js"
@@ -1 +1 @@
-import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/1.15edf814.png",o="/vitePress-blob/assets/2.19007fd9.png",t="/vitePress-blob/assets/3.59bb2b90.png",v=JSON.parse('{"title":"Canvas 尺寸与分辨率矫正","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/Canvas尺寸及分辨率矫正.md","filePath":"guide/canvas/Canvas尺寸及分辨率矫正.md","lastUpdated":1736065650000}'),e={name:"guide/canvas/Canvas尺寸及分辨率矫正.md"},c=l("",29),r=[c];function E(y,i,F,d,h,C){return a(),n("div",null,r)}const u=s(e,[["render",E]]);export{v as __pageData,u as default};
+import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/1.15edf814.png",o="/vitePress-blob/assets/2.19007fd9.png",t="/vitePress-blob/assets/3.59bb2b90.png",v=JSON.parse('{"title":"Canvas 尺寸与分辨率矫正","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/Canvas尺寸及分辨率矫正.md","filePath":"guide/canvas/Canvas尺寸及分辨率矫正.md","lastUpdated":1736839070000}'),e={name:"guide/canvas/Canvas尺寸及分辨率矫正.md"},c=l("",29),r=[c];function E(y,i,F,d,h,C){return a(),n("div",null,r)}const u=s(e,[["render",E]]);export{v as __pageData,u as default};
diff --git "a/assets/guide_canvas_\345\217\257\350\247\206\345\214\272\345\237\237\345\206\205\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.e953eb67.js" "b/assets/guide_canvas_\345\217\257\350\247\206\345\214\272\345\237\237\345\206\205\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.1828bbfc.js"
similarity index 99%
rename from "assets/guide_canvas_\345\217\257\350\247\206\345\214\272\345\237\237\345\206\205\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.e953eb67.js"
rename to "assets/guide_canvas_\345\217\257\350\247\206\345\214\272\345\237\237\345\206\205\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.1828bbfc.js"
index 1a544626..401f9fc4 100644
--- "a/assets/guide_canvas_\345\217\257\350\247\206\345\214\272\345\237\237\345\206\205\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.e953eb67.js"
+++ "b/assets/guide_canvas_\345\217\257\350\247\206\345\214\272\345\237\237\345\206\205\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.1828bbfc.js"
@@ -1,4 +1,4 @@
-import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/13.ebfa1a54.png",o="/vitePress-blob/assets/14.92ddec45.png",e="/vitePress-blob/assets/15.3e4e2641.gif",t="/vitePress-blob/assets/17.75314470.png",c="/vitePress-blob/assets/18.34b0479f.gif",D=JSON.parse('{"title":"可视区域内渲染提高 Canvas 的书写性能","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/可视区域内渲染提高Canvas书写性能.md","filePath":"guide/canvas/可视区域内渲染提高Canvas书写性能.md","lastUpdated":1736065650000}'),r={name:"guide/canvas/可视区域内渲染提高Canvas书写性能.md"},E=l('

可视区域内渲染提高 Canvas 的书写性能

前言

上一节我们通过离屏渲染提高了 Canvas 的渲染性能,但是离屏渲染也有一些缺点,比如会增加内存的使用,而且在某些场景下,离屏渲染的性能并不会比直接在 Canvas 上绘制要高。本节我们将介绍如何通过可视区域内渲染提高 Canvas 的书写性能。

前提

一般我们说的可视区域内渲染,是指在 Canvas 上只绘制可视区域内的内容,而不是绘制整个 Canvas 的内容。这样做的好处是可以减少 Canvas 的绘制区域,从而提高 Canvas 的渲染性能。

对于 Canvas 而言,无法无限制地扩大 Canvas 的面积,因此浏览器对 Canvas 的大小也有一定的限制。从 MDN 文档 可知, 在 Chrome 浏览器中,Canvas 的大小限制为 32767px * 32767px。由于 Canvas 的大小限制,因此我们在实现 无限画布 的功能时,不能无限拓展 Canvas 的大小,可以通过坐标的切换,来实现无限画布的功能。

实现无限画布

实现思路

记初始坐标A (x, y), 横向滚动距离为 scrollX, 纵向滚动距离为 scrollY

在初始状态下, scrollX、scrollY 均为 0

初始状态图

假设现在,我们在水平方向向右滚动了scrollX,垂直方向向下滚动scrollY。那么滚动后的坐标就是

x1 = x - scrollX

y1 = y - scrollY

这里大家可能会有疑惑,为什么是减法呢?因为向下滚动后,绘制的图形应该是往上移动的,因此我们需要减去滚动的距离。

滚动后的状态图

在代码中,我们可以通过监听 Canvas 的 WheelEvent 事件,来获取滚动的距离,然后根据上面的公式计算出滚动后的坐标,最后重新绘制 Canvas。

具体实现如下

jsx
ctx.clearRect(0, 0, window.innerWidth, window.innerHeight);
+import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/13.ebfa1a54.png",o="/vitePress-blob/assets/14.92ddec45.png",e="/vitePress-blob/assets/15.3e4e2641.gif",t="/vitePress-blob/assets/17.75314470.png",c="/vitePress-blob/assets/18.34b0479f.gif",D=JSON.parse('{"title":"可视区域内渲染提高 Canvas 的书写性能","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/可视区域内渲染提高Canvas书写性能.md","filePath":"guide/canvas/可视区域内渲染提高Canvas书写性能.md","lastUpdated":1736839070000}'),r={name:"guide/canvas/可视区域内渲染提高Canvas书写性能.md"},E=l('

可视区域内渲染提高 Canvas 的书写性能

前言

上一节我们通过离屏渲染提高了 Canvas 的渲染性能,但是离屏渲染也有一些缺点,比如会增加内存的使用,而且在某些场景下,离屏渲染的性能并不会比直接在 Canvas 上绘制要高。本节我们将介绍如何通过可视区域内渲染提高 Canvas 的书写性能。

前提

一般我们说的可视区域内渲染,是指在 Canvas 上只绘制可视区域内的内容,而不是绘制整个 Canvas 的内容。这样做的好处是可以减少 Canvas 的绘制区域,从而提高 Canvas 的渲染性能。

对于 Canvas 而言,无法无限制地扩大 Canvas 的面积,因此浏览器对 Canvas 的大小也有一定的限制。从 MDN 文档 可知, 在 Chrome 浏览器中,Canvas 的大小限制为 32767px * 32767px。由于 Canvas 的大小限制,因此我们在实现 无限画布 的功能时,不能无限拓展 Canvas 的大小,可以通过坐标的切换,来实现无限画布的功能。

实现无限画布

实现思路

记初始坐标A (x, y), 横向滚动距离为 scrollX, 纵向滚动距离为 scrollY

在初始状态下, scrollX、scrollY 均为 0

初始状态图

假设现在,我们在水平方向向右滚动了scrollX,垂直方向向下滚动scrollY。那么滚动后的坐标就是

x1 = x - scrollX

y1 = y - scrollY

这里大家可能会有疑惑,为什么是减法呢?因为向下滚动后,绘制的图形应该是往上移动的,因此我们需要减去滚动的距离。

滚动后的状态图

在代码中,我们可以通过监听 Canvas 的 WheelEvent 事件,来获取滚动的距离,然后根据上面的公式计算出滚动后的坐标,最后重新绘制 Canvas。

具体实现如下

jsx
ctx.clearRect(0, 0, window.innerWidth, window.innerHeight);
 ctx.save();
 ctx.translate(scrollX, scrollY);
 // 绘制相关的逻辑
diff --git "a/assets/guide_canvas_\345\217\257\350\247\206\345\214\272\345\237\237\345\206\205\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.e953eb67.lean.js" "b/assets/guide_canvas_\345\217\257\350\247\206\345\214\272\345\237\237\345\206\205\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.1828bbfc.lean.js"
similarity index 92%
rename from "assets/guide_canvas_\345\217\257\350\247\206\345\214\272\345\237\237\345\206\205\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.e953eb67.lean.js"
rename to "assets/guide_canvas_\345\217\257\350\247\206\345\214\272\345\237\237\345\206\205\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.1828bbfc.lean.js"
index 20ffe740..829955c6 100644
--- "a/assets/guide_canvas_\345\217\257\350\247\206\345\214\272\345\237\237\345\206\205\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.e953eb67.lean.js"
+++ "b/assets/guide_canvas_\345\217\257\350\247\206\345\214\272\345\237\237\345\206\205\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.1828bbfc.lean.js"
@@ -1 +1 @@
-import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/13.ebfa1a54.png",o="/vitePress-blob/assets/14.92ddec45.png",e="/vitePress-blob/assets/15.3e4e2641.gif",t="/vitePress-blob/assets/17.75314470.png",c="/vitePress-blob/assets/18.34b0479f.gif",D=JSON.parse('{"title":"可视区域内渲染提高 Canvas 的书写性能","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/可视区域内渲染提高Canvas书写性能.md","filePath":"guide/canvas/可视区域内渲染提高Canvas书写性能.md","lastUpdated":1736065650000}'),r={name:"guide/canvas/可视区域内渲染提高Canvas书写性能.md"},E=l("",46),y=[E];function i(F,d,m,h,C,A){return n(),a("div",null,y)}const x=s(r,[["render",i]]);export{D as __pageData,x as default};
+import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/13.ebfa1a54.png",o="/vitePress-blob/assets/14.92ddec45.png",e="/vitePress-blob/assets/15.3e4e2641.gif",t="/vitePress-blob/assets/17.75314470.png",c="/vitePress-blob/assets/18.34b0479f.gif",D=JSON.parse('{"title":"可视区域内渲染提高 Canvas 的书写性能","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/可视区域内渲染提高Canvas书写性能.md","filePath":"guide/canvas/可视区域内渲染提高Canvas书写性能.md","lastUpdated":1736839070000}'),r={name:"guide/canvas/可视区域内渲染提高Canvas书写性能.md"},E=l("",46),y=[E];function i(F,d,m,h,C,A){return n(),a("div",null,y)}const x=s(r,[["render",i]]);export{D as __pageData,x as default};
diff --git "a/assets/guide_canvas_\345\246\202\344\275\225\345\234\250Canvas\347\224\273\346\235\277\344\270\212\350\207\252\347\224\261\344\271\246\345\206\231.md.a3f6d57a.js" "b/assets/guide_canvas_\345\246\202\344\275\225\345\234\250Canvas\347\224\273\346\235\277\344\270\212\350\207\252\347\224\261\344\271\246\345\206\231.md.a9d3a092.js"
similarity index 99%
rename from "assets/guide_canvas_\345\246\202\344\275\225\345\234\250Canvas\347\224\273\346\235\277\344\270\212\350\207\252\347\224\261\344\271\246\345\206\231.md.a3f6d57a.js"
rename to "assets/guide_canvas_\345\246\202\344\275\225\345\234\250Canvas\347\224\273\346\235\277\344\270\212\350\207\252\347\224\261\344\271\246\345\206\231.md.a9d3a092.js"
index af5e465b..e4e07339 100644
--- "a/assets/guide_canvas_\345\246\202\344\275\225\345\234\250Canvas\347\224\273\346\235\277\344\270\212\350\207\252\347\224\261\344\271\246\345\206\231.md.a3f6d57a.js"
+++ "b/assets/guide_canvas_\345\246\202\344\275\225\345\234\250Canvas\347\224\273\346\235\277\344\270\212\350\207\252\347\224\261\344\271\246\345\206\231.md.a9d3a092.js"
@@ -1,4 +1,4 @@
-import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/4.6c4ce0eb.gif",o="/vitePress-blob/assets/6.bb9fb817.png",e="/vitePress-blob/assets/5.016fbc65.png",t="/vitePress-blob/assets/7.2b4ce5bb.gif",h=JSON.parse('{"title":"如何实现一个自由绘制的 Canvas 画板","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/如何在Canvas画板上自由书写.md","filePath":"guide/canvas/如何在Canvas画板上自由书写.md","lastUpdated":1736065650000}'),c={name:"guide/canvas/如何在Canvas画板上自由书写.md"},r=l(`

如何实现一个自由绘制的 Canvas 画板

前言

承接上文,当我们了解了如何去设置一个 Canvas 尺寸,并通过分辨率对 Canvas 进行矫正后,我们就可以开始实现一个自由绘制的 Canvas 画板了。

正文

实现思路

自由画笔的实现可以通过:

    1. 监听鼠标事件
    1. 将鼠标移动的点记录下来
    1. 将这些点连成线,就可以实现自由画笔了。

实现代码

html
<!DOCTYPE html>
+import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/4.6c4ce0eb.gif",o="/vitePress-blob/assets/6.bb9fb817.png",e="/vitePress-blob/assets/5.016fbc65.png",t="/vitePress-blob/assets/7.2b4ce5bb.gif",h=JSON.parse('{"title":"如何实现一个自由绘制的 Canvas 画板","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/如何在Canvas画板上自由书写.md","filePath":"guide/canvas/如何在Canvas画板上自由书写.md","lastUpdated":1736839070000}'),c={name:"guide/canvas/如何在Canvas画板上自由书写.md"},r=l(`

如何实现一个自由绘制的 Canvas 画板

前言

承接上文,当我们了解了如何去设置一个 Canvas 尺寸,并通过分辨率对 Canvas 进行矫正后,我们就可以开始实现一个自由绘制的 Canvas 画板了。

正文

实现思路

自由画笔的实现可以通过:

    1. 监听鼠标事件
    1. 将鼠标移动的点记录下来
    1. 将这些点连成线,就可以实现自由画笔了。

实现代码

html
<!DOCTYPE html>
 <html lang="en">
 
 <head>
diff --git "a/assets/guide_canvas_\345\246\202\344\275\225\345\234\250Canvas\347\224\273\346\235\277\344\270\212\350\207\252\347\224\261\344\271\246\345\206\231.md.a3f6d57a.lean.js" "b/assets/guide_canvas_\345\246\202\344\275\225\345\234\250Canvas\347\224\273\346\235\277\344\270\212\350\207\252\347\224\261\344\271\246\345\206\231.md.a9d3a092.lean.js"
similarity index 91%
rename from "assets/guide_canvas_\345\246\202\344\275\225\345\234\250Canvas\347\224\273\346\235\277\344\270\212\350\207\252\347\224\261\344\271\246\345\206\231.md.a3f6d57a.lean.js"
rename to "assets/guide_canvas_\345\246\202\344\275\225\345\234\250Canvas\347\224\273\346\235\277\344\270\212\350\207\252\347\224\261\344\271\246\345\206\231.md.a9d3a092.lean.js"
index 9933124c..e0a6eae9 100644
--- "a/assets/guide_canvas_\345\246\202\344\275\225\345\234\250Canvas\347\224\273\346\235\277\344\270\212\350\207\252\347\224\261\344\271\246\345\206\231.md.a3f6d57a.lean.js"
+++ "b/assets/guide_canvas_\345\246\202\344\275\225\345\234\250Canvas\347\224\273\346\235\277\344\270\212\350\207\252\347\224\261\344\271\246\345\206\231.md.a9d3a092.lean.js"
@@ -1 +1 @@
-import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/4.6c4ce0eb.gif",o="/vitePress-blob/assets/6.bb9fb817.png",e="/vitePress-blob/assets/5.016fbc65.png",t="/vitePress-blob/assets/7.2b4ce5bb.gif",h=JSON.parse('{"title":"如何实现一个自由绘制的 Canvas 画板","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/如何在Canvas画板上自由书写.md","filePath":"guide/canvas/如何在Canvas画板上自由书写.md","lastUpdated":1736065650000}'),c={name:"guide/canvas/如何在Canvas画板上自由书写.md"},r=l("",23),E=[r];function y(i,F,C,A,D,d){return n(),a("div",null,E)}const B=s(c,[["render",y]]);export{h as __pageData,B as default};
+import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/4.6c4ce0eb.gif",o="/vitePress-blob/assets/6.bb9fb817.png",e="/vitePress-blob/assets/5.016fbc65.png",t="/vitePress-blob/assets/7.2b4ce5bb.gif",h=JSON.parse('{"title":"如何实现一个自由绘制的 Canvas 画板","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/如何在Canvas画板上自由书写.md","filePath":"guide/canvas/如何在Canvas画板上自由书写.md","lastUpdated":1736839070000}'),c={name:"guide/canvas/如何在Canvas画板上自由书写.md"},r=l("",23),E=[r];function y(i,F,C,A,D,d){return n(),a("div",null,E)}const B=s(c,[["render",y]]);export{h as __pageData,B as default};
diff --git "a/assets/guide_canvas_\351\200\232\350\277\207 OffscreenCanvas _ Worker \346\217\220\351\253\230\344\271\246\345\206\231\346\200\247\350\203\275.md.09ce6c22.js" "b/assets/guide_canvas_\351\200\232\350\277\207 OffscreenCanvas _ Worker \346\217\220\351\253\230\344\271\246\345\206\231\346\200\247\350\203\275.md.74068538.js"
similarity index 99%
rename from "assets/guide_canvas_\351\200\232\350\277\207 OffscreenCanvas _ Worker \346\217\220\351\253\230\344\271\246\345\206\231\346\200\247\350\203\275.md.09ce6c22.js"
rename to "assets/guide_canvas_\351\200\232\350\277\207 OffscreenCanvas _ Worker \346\217\220\351\253\230\344\271\246\345\206\231\346\200\247\350\203\275.md.74068538.js"
index be1fe5c4..8ded7bfe 100644
--- "a/assets/guide_canvas_\351\200\232\350\277\207 OffscreenCanvas _ Worker \346\217\220\351\253\230\344\271\246\345\206\231\346\200\247\350\203\275.md.09ce6c22.js"	
+++ "b/assets/guide_canvas_\351\200\232\350\277\207 OffscreenCanvas _ Worker \346\217\220\351\253\230\344\271\246\345\206\231\346\200\247\350\203\275.md.74068538.js"	
@@ -1,4 +1,4 @@
-import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/19.ad6d5c6e.gif",h=JSON.parse('{"title":"通过 OffscreenCanvas + Worker 提高书写性能","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/通过 OffscreenCanvas + Worker 提高书写性能.md","filePath":"guide/canvas/通过 OffscreenCanvas + Worker 提高书写性能.md","lastUpdated":1736065650000}'),o={name:"guide/canvas/通过 OffscreenCanvas + Worker 提高书写性能.md"},e=l(`

通过 OffscreenCanvas + Worker 提高书写性能

前言

上一节我们通过离屏渲染提高了 Canvas 的渲染性能,但是在绘制的过程中,我们会发现,当绘制的图形越来越多时,Canvas 的渲染性能会越来越差,这是因为我们在绘制图形会阻塞主线程,如果主线程中还有其他的任务也会表现出卡顿的效果,Canvas 的渲染性能越来越差。

这节我们将通过 OffscreenCanvas + Worker 将绘制图形的任务放到 Worker 中进行,避免阻塞主线程,从而提高 Canvas 的渲染性能。

实现思路

在 worker 线程中是无法操作 DOM 的,但 OffscreenCanvas 可以在 worker 线程中进行操作,因此我们可以通过 OffscreenCanvas 将绘制图形的任务放到 worker 线程中进行。这样可以减少主线程的任务,从而提高书写性能。

具体实现

创建 OffscreenCanvas

js
const canvas = document.getElementById('draw');
+import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/19.ad6d5c6e.gif",h=JSON.parse('{"title":"通过 OffscreenCanvas + Worker 提高书写性能","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/通过 OffscreenCanvas + Worker 提高书写性能.md","filePath":"guide/canvas/通过 OffscreenCanvas + Worker 提高书写性能.md","lastUpdated":1736839070000}'),o={name:"guide/canvas/通过 OffscreenCanvas + Worker 提高书写性能.md"},e=l(`

通过 OffscreenCanvas + Worker 提高书写性能

前言

上一节我们通过离屏渲染提高了 Canvas 的渲染性能,但是在绘制的过程中,我们会发现,当绘制的图形越来越多时,Canvas 的渲染性能会越来越差,这是因为我们在绘制图形会阻塞主线程,如果主线程中还有其他的任务也会表现出卡顿的效果,Canvas 的渲染性能越来越差。

这节我们将通过 OffscreenCanvas + Worker 将绘制图形的任务放到 Worker 中进行,避免阻塞主线程,从而提高 Canvas 的渲染性能。

实现思路

在 worker 线程中是无法操作 DOM 的,但 OffscreenCanvas 可以在 worker 线程中进行操作,因此我们可以通过 OffscreenCanvas 将绘制图形的任务放到 worker 线程中进行。这样可以减少主线程的任务,从而提高书写性能。

具体实现

创建 OffscreenCanvas

js
const canvas = document.getElementById('draw');
 const offScreenCanvas = canvas.transferControlToOffscreen(); // 将 canvas 转换为 offScreenCanvas
const canvas = document.getElementById('draw');
 const offScreenCanvas = canvas.transferControlToOffscreen(); // 将 canvas 转换为 offScreenCanvas

将 offScreenCanvas 传递给 worker 线程

js
const worker = new Worker('./worker.js'); // 创建一个 webWorker
 const offScreenCanvas = canvas.transferControlToOffscreen(); // 将 canvas 转换为 offScreenCanvas
diff --git "a/assets/guide_canvas_\351\200\232\350\277\207 OffscreenCanvas _ Worker \346\217\220\351\253\230\344\271\246\345\206\231\346\200\247\350\203\275.md.09ce6c22.lean.js" "b/assets/guide_canvas_\351\200\232\350\277\207 OffscreenCanvas _ Worker \346\217\220\351\253\230\344\271\246\345\206\231\346\200\247\350\203\275.md.74068538.lean.js"
similarity index 90%
rename from "assets/guide_canvas_\351\200\232\350\277\207 OffscreenCanvas _ Worker \346\217\220\351\253\230\344\271\246\345\206\231\346\200\247\350\203\275.md.09ce6c22.lean.js"
rename to "assets/guide_canvas_\351\200\232\350\277\207 OffscreenCanvas _ Worker \346\217\220\351\253\230\344\271\246\345\206\231\346\200\247\350\203\275.md.74068538.lean.js"
index bdd78413..790245a5 100644
--- "a/assets/guide_canvas_\351\200\232\350\277\207 OffscreenCanvas _ Worker \346\217\220\351\253\230\344\271\246\345\206\231\346\200\247\350\203\275.md.09ce6c22.lean.js"	
+++ "b/assets/guide_canvas_\351\200\232\350\277\207 OffscreenCanvas _ Worker \346\217\220\351\253\230\344\271\246\345\206\231\346\200\247\350\203\275.md.74068538.lean.js"	
@@ -1 +1 @@
-import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/19.ad6d5c6e.gif",h=JSON.parse('{"title":"通过 OffscreenCanvas + Worker 提高书写性能","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/通过 OffscreenCanvas + Worker 提高书写性能.md","filePath":"guide/canvas/通过 OffscreenCanvas + Worker 提高书写性能.md","lastUpdated":1736065650000}'),o={name:"guide/canvas/通过 OffscreenCanvas + Worker 提高书写性能.md"},e=l("",18),r=[e];function c(t,E,y,i,d,f){return n(),a("div",null,r)}const v=s(o,[["render",c]]);export{h as __pageData,v as default};
+import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/19.ad6d5c6e.gif",h=JSON.parse('{"title":"通过 OffscreenCanvas + Worker 提高书写性能","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/通过 OffscreenCanvas + Worker 提高书写性能.md","filePath":"guide/canvas/通过 OffscreenCanvas + Worker 提高书写性能.md","lastUpdated":1736839070000}'),o={name:"guide/canvas/通过 OffscreenCanvas + Worker 提高书写性能.md"},e=l("",18),r=[e];function c(t,E,y,i,d,f){return n(),a("div",null,r)}const v=s(o,[["render",c]]);export{h as __pageData,v as default};
diff --git "a/assets/guide_canvas_\351\200\232\350\277\207\344\270\212\344\270\213\345\210\206\345\261\202\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.81bf21d6.js" "b/assets/guide_canvas_\351\200\232\350\277\207\344\270\212\344\270\213\345\210\206\345\261\202\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.fc9344af.js"
similarity index 99%
rename from "assets/guide_canvas_\351\200\232\350\277\207\344\270\212\344\270\213\345\210\206\345\261\202\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.81bf21d6.js"
rename to "assets/guide_canvas_\351\200\232\350\277\207\344\270\212\344\270\213\345\210\206\345\261\202\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.fc9344af.js"
index 03ab1277..264ea607 100644
--- "a/assets/guide_canvas_\351\200\232\350\277\207\344\270\212\344\270\213\345\210\206\345\261\202\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.81bf21d6.js"
+++ "b/assets/guide_canvas_\351\200\232\350\277\207\344\270\212\344\270\213\345\210\206\345\261\202\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.fc9344af.js"
@@ -1,4 +1,4 @@
-import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/10.57bb4969.gif",o="/vitePress-blob/assets/11.298d6f19.gif",h=JSON.parse('{"title":"通过上下分层优化 Canvas 书写性能","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/通过上下分层优化Canvas书写性能.md","filePath":"guide/canvas/通过上下分层优化Canvas书写性能.md","lastUpdated":1736065650000}'),e={name:"guide/canvas/通过上下分层优化Canvas书写性能.md"},t=l(`

通过上下分层优化 Canvas 书写性能

前言

上一节,我们通过点稀疏的方式,优化了 Canvas 的绘制性能。但在书写的过程中,因为点的数量减少导致书写的效果不够理想,因此通过贝塞尔曲线的方式,将点连接起来,形成平滑的曲线,从而达到书写的效果。 这节,我们将通过上下分层的方式,优化 Canvas 的书写性能。

基本思路

这里我将 Canvas 分为上下两层,上层称之为动态层,下层称之为静态层,动态层用于书写,静态层用于显示书写的效果。当书写的时候,将书写的点绘制到上层,当书写完成后,将上层的内容绘制到下层,然后清空上层的内容,这样就可以达到书写的效果。

因为 Canvas 每次渲染都会将整个 Canvas 清空,所以我们需要将静态层的内容保存下来,然后在每次书写的时候,只需要渲染当前动态层的内容即可。

实现

创建 2 个 Canvas

首先创建 2 个 Canvas, 上层 Canvas 用于书写, 书写完成后将上层 Canvas 的内容绘制到下层 Canvas 中, 然后清空上层 Canvas 的内容, 这样就可以达到书写的效果。

这样的好处在于, 每次书写的时候, 只需要渲染上层 Canvas 的书写内容即可, 不需要每次都渲染整个 Canvas, 从而达到优化性能的目的。

html
<style>
+import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/10.57bb4969.gif",o="/vitePress-blob/assets/11.298d6f19.gif",h=JSON.parse('{"title":"通过上下分层优化 Canvas 书写性能","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/通过上下分层优化Canvas书写性能.md","filePath":"guide/canvas/通过上下分层优化Canvas书写性能.md","lastUpdated":1736839070000}'),e={name:"guide/canvas/通过上下分层优化Canvas书写性能.md"},t=l(`

通过上下分层优化 Canvas 书写性能

前言

上一节,我们通过点稀疏的方式,优化了 Canvas 的绘制性能。但在书写的过程中,因为点的数量减少导致书写的效果不够理想,因此通过贝塞尔曲线的方式,将点连接起来,形成平滑的曲线,从而达到书写的效果。 这节,我们将通过上下分层的方式,优化 Canvas 的书写性能。

基本思路

这里我将 Canvas 分为上下两层,上层称之为动态层,下层称之为静态层,动态层用于书写,静态层用于显示书写的效果。当书写的时候,将书写的点绘制到上层,当书写完成后,将上层的内容绘制到下层,然后清空上层的内容,这样就可以达到书写的效果。

因为 Canvas 每次渲染都会将整个 Canvas 清空,所以我们需要将静态层的内容保存下来,然后在每次书写的时候,只需要渲染当前动态层的内容即可。

实现

创建 2 个 Canvas

首先创建 2 个 Canvas, 上层 Canvas 用于书写, 书写完成后将上层 Canvas 的内容绘制到下层 Canvas 中, 然后清空上层 Canvas 的内容, 这样就可以达到书写的效果。

这样的好处在于, 每次书写的时候, 只需要渲染上层 Canvas 的书写内容即可, 不需要每次都渲染整个 Canvas, 从而达到优化性能的目的。

html
<style>
     #draw {
         border: 1px solid black;
         position: absolute;
diff --git "a/assets/guide_canvas_\351\200\232\350\277\207\344\270\212\344\270\213\345\210\206\345\261\202\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.81bf21d6.lean.js" "b/assets/guide_canvas_\351\200\232\350\277\207\344\270\212\344\270\213\345\210\206\345\261\202\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.fc9344af.lean.js"
similarity index 80%
rename from "assets/guide_canvas_\351\200\232\350\277\207\344\270\212\344\270\213\345\210\206\345\261\202\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.81bf21d6.lean.js"
rename to "assets/guide_canvas_\351\200\232\350\277\207\344\270\212\344\270\213\345\210\206\345\261\202\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.fc9344af.lean.js"
index 5eca1c93..5cb10e00 100644
--- "a/assets/guide_canvas_\351\200\232\350\277\207\344\270\212\344\270\213\345\210\206\345\261\202\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.81bf21d6.lean.js"
+++ "b/assets/guide_canvas_\351\200\232\350\277\207\344\270\212\344\270\213\345\210\206\345\261\202\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.fc9344af.lean.js"
@@ -1 +1 @@
-import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/10.57bb4969.gif",o="/vitePress-blob/assets/11.298d6f19.gif",h=JSON.parse('{"title":"通过上下分层优化 Canvas 书写性能","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/通过上下分层优化Canvas书写性能.md","filePath":"guide/canvas/通过上下分层优化Canvas书写性能.md","lastUpdated":1736065650000}'),e={name:"guide/canvas/通过上下分层优化Canvas书写性能.md"},t=l("",22),c=[t];function r(E,y,i,d,C,v){return a(),n("div",null,c)}const u=s(e,[["render",r]]);export{h as __pageData,u as default};
+import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/10.57bb4969.gif",o="/vitePress-blob/assets/11.298d6f19.gif",h=JSON.parse('{"title":"通过上下分层优化 Canvas 书写性能","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/通过上下分层优化Canvas书写性能.md","filePath":"guide/canvas/通过上下分层优化Canvas书写性能.md","lastUpdated":1736839070000}'),e={name:"guide/canvas/通过上下分层优化Canvas书写性能.md"},t=l("",22),c=[t];function r(E,y,i,d,C,v){return a(),n("div",null,c)}const u=s(e,[["render",r]]);export{h as __pageData,u as default};
diff --git "a/assets/guide_canvas_\351\200\232\350\277\207\347\246\273\345\261\217\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.5cb7017a.js" "b/assets/guide_canvas_\351\200\232\350\277\207\347\246\273\345\261\217\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.86c812db.js"
similarity index 99%
rename from "assets/guide_canvas_\351\200\232\350\277\207\347\246\273\345\261\217\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.5cb7017a.js"
rename to "assets/guide_canvas_\351\200\232\350\277\207\347\246\273\345\261\217\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.86c812db.js"
index 8ceb2a1e..1678d073 100644
--- "a/assets/guide_canvas_\351\200\232\350\277\207\347\246\273\345\261\217\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.5cb7017a.js"
+++ "b/assets/guide_canvas_\351\200\232\350\277\207\347\246\273\345\261\217\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.86c812db.js"
@@ -1,4 +1,4 @@
-import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/12.31e4f330.gif",D=JSON.parse('{"title":"通过离屏渲染提高 Canvas 书写性能","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/通过离屏渲染提高Canvas书写性能.md","filePath":"guide/canvas/通过离屏渲染提高Canvas书写性能.md","lastUpdated":1736065650000}'),o={name:"guide/canvas/通过离屏渲染提高Canvas书写性能.md"},e=l(`

通过离屏渲染提高 Canvas 书写性能

前言

前面我们通过上下分层的方式,优化了 Canvas 的书写性能,接下来我们通过离屏渲染的方式,进一步优化 Canvas 的书写性能。

基本思路

在书写的过程中,每绘制一笔都需要不断地调用 Canvas 的 API,重新渲染整个 Canvas,这样就会导致性能的浪费。 而离屏渲染则是将 绘制内容存储到离屏的 Canvas 中,相当于一个缓冲区,然后将需要绘制的画面在离屏的 Canvas 缓冲好,最后将离屏的 Canvas 转化成图片,渲染到屏幕上,这样就可以达到优化性能的目的。

实现

创建离屏 Canvas

思路如下: 基于上一节的基础,我们改写 render 函数,如果是离屏渲染的话,将绘制的内容存储到离屏的 Canvas 中,然后将离屏的 Canvas 缓存起来,下次绘制的时候,如果命中缓存的话,就直接使用缓存的 Canvas,从而达到优化性能的目的。

操作如下:

  • 1 在执行 render 函数之前,先判断是否存在缓存的 Canvas,如果存在的话,就直接使用缓存的 Canvas
  • 2 如果命中缓存,使用离屏 Canvas 转化成图片进行绘制
  • 3 如果不存在缓存的 Canvas,就创建一个离屏的 Canvas,然后将绘制的内容存储到离屏的 Canvas 中,最后将离屏的 Canvas 缓存起来
html
<script>
+import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/12.31e4f330.gif",D=JSON.parse('{"title":"通过离屏渲染提高 Canvas 书写性能","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/通过离屏渲染提高Canvas书写性能.md","filePath":"guide/canvas/通过离屏渲染提高Canvas书写性能.md","lastUpdated":1736839070000}'),o={name:"guide/canvas/通过离屏渲染提高Canvas书写性能.md"},e=l(`

通过离屏渲染提高 Canvas 书写性能

前言

前面我们通过上下分层的方式,优化了 Canvas 的书写性能,接下来我们通过离屏渲染的方式,进一步优化 Canvas 的书写性能。

基本思路

在书写的过程中,每绘制一笔都需要不断地调用 Canvas 的 API,重新渲染整个 Canvas,这样就会导致性能的浪费。 而离屏渲染则是将 绘制内容存储到离屏的 Canvas 中,相当于一个缓冲区,然后将需要绘制的画面在离屏的 Canvas 缓冲好,最后将离屏的 Canvas 转化成图片,渲染到屏幕上,这样就可以达到优化性能的目的。

实现

创建离屏 Canvas

思路如下: 基于上一节的基础,我们改写 render 函数,如果是离屏渲染的话,将绘制的内容存储到离屏的 Canvas 中,然后将离屏的 Canvas 缓存起来,下次绘制的时候,如果命中缓存的话,就直接使用缓存的 Canvas,从而达到优化性能的目的。

操作如下:

  • 1 在执行 render 函数之前,先判断是否存在缓存的 Canvas,如果存在的话,就直接使用缓存的 Canvas
  • 2 如果命中缓存,使用离屏 Canvas 转化成图片进行绘制
  • 3 如果不存在缓存的 Canvas,就创建一个离屏的 Canvas,然后将绘制的内容存储到离屏的 Canvas 中,最后将离屏的 Canvas 缓存起来
html
<script>
         const elementWithCanvasCache = new WeakMap(); // 用于存储离屏 Canvas 的缓存
         const generateOffScreenCanvas = (points) => {
             const padding = 20; // 避免笔记被 Canvas 
diff --git "a/assets/guide_canvas_\351\200\232\350\277\207\347\246\273\345\261\217\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.5cb7017a.lean.js" "b/assets/guide_canvas_\351\200\232\350\277\207\347\246\273\345\261\217\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.86c812db.lean.js"
similarity index 89%
rename from "assets/guide_canvas_\351\200\232\350\277\207\347\246\273\345\261\217\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.5cb7017a.lean.js"
rename to "assets/guide_canvas_\351\200\232\350\277\207\347\246\273\345\261\217\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.86c812db.lean.js"
index 72b58673..c9305c58 100644
--- "a/assets/guide_canvas_\351\200\232\350\277\207\347\246\273\345\261\217\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.5cb7017a.lean.js"
+++ "b/assets/guide_canvas_\351\200\232\350\277\207\347\246\273\345\261\217\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.86c812db.lean.js"
@@ -1 +1 @@
-import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/12.31e4f330.gif",D=JSON.parse('{"title":"通过离屏渲染提高 Canvas 书写性能","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/通过离屏渲染提高Canvas书写性能.md","filePath":"guide/canvas/通过离屏渲染提高Canvas书写性能.md","lastUpdated":1736065650000}'),o={name:"guide/canvas/通过离屏渲染提高Canvas书写性能.md"},e=l("",19),t=[e];function c(r,E,y,i,F,C){return n(),a("div",null,t)}const h=s(o,[["render",c]]);export{D as __pageData,h as default};
+import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/12.31e4f330.gif",D=JSON.parse('{"title":"通过离屏渲染提高 Canvas 书写性能","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/通过离屏渲染提高Canvas书写性能.md","filePath":"guide/canvas/通过离屏渲染提高Canvas书写性能.md","lastUpdated":1736839070000}'),o={name:"guide/canvas/通过离屏渲染提高Canvas书写性能.md"},e=l("",19),t=[e];function c(r,E,y,i,F,C){return n(),a("div",null,t)}const h=s(o,[["render",c]]);export{D as __pageData,h as default};
diff --git "a/assets/guide_canvas_\351\200\232\350\277\207\350\264\235\345\241\236\345\260\224\346\233\262\347\272\277\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.61a93128.js" "b/assets/guide_canvas_\351\200\232\350\277\207\350\264\235\345\241\236\345\260\224\346\233\262\347\272\277\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.1acbc131.js"
similarity index 99%
rename from "assets/guide_canvas_\351\200\232\350\277\207\350\264\235\345\241\236\345\260\224\346\233\262\347\272\277\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.61a93128.js"
rename to "assets/guide_canvas_\351\200\232\350\277\207\350\264\235\345\241\236\345\260\224\346\233\262\347\272\277\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.1acbc131.js"
index 4ee0a704..bf358eec 100644
--- "a/assets/guide_canvas_\351\200\232\350\277\207\350\264\235\345\241\236\345\260\224\346\233\262\347\272\277\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.61a93128.js"
+++ "b/assets/guide_canvas_\351\200\232\350\277\207\350\264\235\345\241\236\345\260\224\346\233\262\347\272\277\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.1acbc131.js"
@@ -1,4 +1,4 @@
-import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/8.4bb53b64.gif",o="/vitePress-blob/assets/9.4148f2f2.gif",d=JSON.parse('{"title":"通过贝塞尔曲线解决 Canvas 书写的圆滑问题","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/通过贝塞尔曲线优化Canvas书写性能.md","filePath":"guide/canvas/通过贝塞尔曲线优化Canvas书写性能.md","lastUpdated":1736065650000}'),e={name:"guide/canvas/通过贝塞尔曲线优化Canvas书写性能.md"},t=l('

通过贝塞尔曲线解决 Canvas 书写的圆滑问题

前言

书接上回,当我们实现了一个自由书写的 Canvas 画板后,我们发现采集的点越少,画出来的圈圈越不圆润,这时候我们可以使用贝塞尔曲线来优化书写。

canvas

正文

贝塞尔曲线

贝塞尔曲线是一种数学曲线,常见的有: 线性贝塞尔曲线、二次·贝塞尔曲线、三次贝塞尔曲线等, 它可以用来实现平滑曲线,我们可以通过贝塞尔曲线以相对圆滑的方式来优化我们的 Canvas 书写。

优化思路

我们可以通过贝塞尔曲线来优化我们的 Canvas 书写性能,将笔直的线变的更加圆滑弯曲。

在 Canvas2D API 中有一个方法叫做 quadraticCurveTo 用于创建二次贝塞尔曲线。它需要两个点:一个控制点和一个终点。曲线从当前的绘图位置开始绘制,并以控制点为参考,向终点绘制曲线。

这个方法的语法是:context.quadraticCurveTo(cp1x, cp1y, x, y);

其中:

cp1x 和 cp1y 是控制点坐标。
+import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/8.4bb53b64.gif",o="/vitePress-blob/assets/9.4148f2f2.gif",d=JSON.parse('{"title":"通过贝塞尔曲线解决 Canvas 书写的圆滑问题","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/通过贝塞尔曲线优化Canvas书写性能.md","filePath":"guide/canvas/通过贝塞尔曲线优化Canvas书写性能.md","lastUpdated":1736839070000}'),e={name:"guide/canvas/通过贝塞尔曲线优化Canvas书写性能.md"},t=l('

通过贝塞尔曲线解决 Canvas 书写的圆滑问题

前言

书接上回,当我们实现了一个自由书写的 Canvas 画板后,我们发现采集的点越少,画出来的圈圈越不圆润,这时候我们可以使用贝塞尔曲线来优化书写。

canvas

正文

贝塞尔曲线

贝塞尔曲线是一种数学曲线,常见的有: 线性贝塞尔曲线、二次·贝塞尔曲线、三次贝塞尔曲线等, 它可以用来实现平滑曲线,我们可以通过贝塞尔曲线以相对圆滑的方式来优化我们的 Canvas 书写。

优化思路

我们可以通过贝塞尔曲线来优化我们的 Canvas 书写性能,将笔直的线变的更加圆滑弯曲。

在 Canvas2D API 中有一个方法叫做 quadraticCurveTo 用于创建二次贝塞尔曲线。它需要两个点:一个控制点和一个终点。曲线从当前的绘图位置开始绘制,并以控制点为参考,向终点绘制曲线。

这个方法的语法是:context.quadraticCurveTo(cp1x, cp1y, x, y);

其中:

cp1x 和 cp1y 是控制点坐标。
 x 和 y 是终点坐标。
 

这个方法不会直接绘制曲线,而是将曲线添加到当前的路径中。要在画布上实际绘制曲线,你需要使用 stroke 或 fill 方法

优化实现

1 采集到所有的点集 2 前一个点作为控制点,当前点作为终点,绘制二次贝塞尔曲线

html
<!DOCTYPE html>
 <html lang="en">
diff --git "a/assets/guide_canvas_\351\200\232\350\277\207\350\264\235\345\241\236\345\260\224\346\233\262\347\272\277\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.61a93128.lean.js" "b/assets/guide_canvas_\351\200\232\350\277\207\350\264\235\345\241\236\345\260\224\346\233\262\347\272\277\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.1acbc131.lean.js"
similarity index 90%
rename from "assets/guide_canvas_\351\200\232\350\277\207\350\264\235\345\241\236\345\260\224\346\233\262\347\272\277\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.61a93128.lean.js"
rename to "assets/guide_canvas_\351\200\232\350\277\207\350\264\235\345\241\236\345\260\224\346\233\262\347\272\277\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.1acbc131.lean.js"
index 83c818d3..2144019f 100644
--- "a/assets/guide_canvas_\351\200\232\350\277\207\350\264\235\345\241\236\345\260\224\346\233\262\347\272\277\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.61a93128.lean.js"
+++ "b/assets/guide_canvas_\351\200\232\350\277\207\350\264\235\345\241\236\345\260\224\346\233\262\347\272\277\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.md.1acbc131.lean.js"
@@ -1 +1 @@
-import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/8.4bb53b64.gif",o="/vitePress-blob/assets/9.4148f2f2.gif",d=JSON.parse('{"title":"通过贝塞尔曲线解决 Canvas 书写的圆滑问题","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/通过贝塞尔曲线优化Canvas书写性能.md","filePath":"guide/canvas/通过贝塞尔曲线优化Canvas书写性能.md","lastUpdated":1736065650000}'),e={name:"guide/canvas/通过贝塞尔曲线优化Canvas书写性能.md"},t=l("",21),c=[t];function r(E,y,i,F,A,C){return n(),a("div",null,c)}const u=s(e,[["render",r]]);export{d as __pageData,u as default};
+import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/8.4bb53b64.gif",o="/vitePress-blob/assets/9.4148f2f2.gif",d=JSON.parse('{"title":"通过贝塞尔曲线解决 Canvas 书写的圆滑问题","description":"","frontmatter":{},"headers":[],"relativePath":"guide/canvas/通过贝塞尔曲线优化Canvas书写性能.md","filePath":"guide/canvas/通过贝塞尔曲线优化Canvas书写性能.md","lastUpdated":1736839070000}'),e={name:"guide/canvas/通过贝塞尔曲线优化Canvas书写性能.md"},t=l("",21),c=[t];function r(E,y,i,F,A,C){return n(),a("div",null,c)}const u=s(e,[["render",r]]);export{d as __pageData,u as default};
diff --git "a/assets/guide_css\347\233\270\345\205\263_BFC.md.97daca69.js" "b/assets/guide_css\347\233\270\345\205\263_BFC.md.fbec5d1d.js"
similarity index 96%
rename from "assets/guide_css\347\233\270\345\205\263_BFC.md.97daca69.js"
rename to "assets/guide_css\347\233\270\345\205\263_BFC.md.fbec5d1d.js"
index 16470257..1a630396 100644
--- "a/assets/guide_css\347\233\270\345\205\263_BFC.md.97daca69.js"
+++ "b/assets/guide_css\347\233\270\345\205\263_BFC.md.fbec5d1d.js"
@@ -1 +1 @@
-import{_ as i,o as a,c as l,Q as e}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"BFC","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/BFC.md","filePath":"guide/css相关/BFC.md","lastUpdated":1736065650000}'),o={name:"guide/css相关/BFC.md"},t=e('

BFC

块级格式化上下文

条件

  1. 根元素 (html)
  2. 浮动元素
  3. position 为绝对定位元素 (absolute/fixed)
  4. display: inline-block、inline-flex等
  5. overflow: auto/scroll/hidden (overflow 值不为 visible 或 clip 的块级元素)

特点

  1. BFC 内部元素不影响外部元素
  2. 计算高度需要计算浮动元素
  3. BFC 区域不会与浮动容器发生重叠
  4. 在 BFC 中上下相邻的俩个容器的 margin 会重叠
  5. 每个元素的左 margin 值和容器的左 border 相接触

作用

  1. 清除浮动带来的高度塌陷问题
  2. 解决俩个元素的 margin 重叠问题
  3. 创建自适应的俩栏布局
',8),r=[t];function s(n,c,d,h,_,f){return a(),l("div",null,r)}const p=i(o,[["render",s]]);export{m as __pageData,p as default}; +import{_ as i,o as a,c as l,Q as e}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"BFC","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/BFC.md","filePath":"guide/css相关/BFC.md","lastUpdated":1736839070000}'),o={name:"guide/css相关/BFC.md"},t=e('

BFC

块级格式化上下文

条件

  1. 根元素 (html)
  2. 浮动元素
  3. position 为绝对定位元素 (absolute/fixed)
  4. display: inline-block、inline-flex等
  5. overflow: auto/scroll/hidden (overflow 值不为 visible 或 clip 的块级元素)

特点

  1. BFC 内部元素不影响外部元素
  2. 计算高度需要计算浮动元素
  3. BFC 区域不会与浮动容器发生重叠
  4. 在 BFC 中上下相邻的俩个容器的 margin 会重叠
  5. 每个元素的左 margin 值和容器的左 border 相接触

作用

  1. 清除浮动带来的高度塌陷问题
  2. 解决俩个元素的 margin 重叠问题
  3. 创建自适应的俩栏布局
',8),r=[t];function s(n,c,d,h,_,f){return a(),l("div",null,r)}const p=i(o,[["render",s]]);export{m as __pageData,p as default}; diff --git "a/assets/guide_css\347\233\270\345\205\263_BFC.md.97daca69.lean.js" "b/assets/guide_css\347\233\270\345\205\263_BFC.md.fbec5d1d.lean.js" similarity index 85% rename from "assets/guide_css\347\233\270\345\205\263_BFC.md.97daca69.lean.js" rename to "assets/guide_css\347\233\270\345\205\263_BFC.md.fbec5d1d.lean.js" index aae2f31f..2998d188 100644 --- "a/assets/guide_css\347\233\270\345\205\263_BFC.md.97daca69.lean.js" +++ "b/assets/guide_css\347\233\270\345\205\263_BFC.md.fbec5d1d.lean.js" @@ -1 +1 @@ -import{_ as i,o as a,c as l,Q as e}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"BFC","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/BFC.md","filePath":"guide/css相关/BFC.md","lastUpdated":1736065650000}'),o={name:"guide/css相关/BFC.md"},t=e("",8),r=[t];function s(n,c,d,h,_,f){return a(),l("div",null,r)}const p=i(o,[["render",s]]);export{m as __pageData,p as default}; +import{_ as i,o as a,c as l,Q as e}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"BFC","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/BFC.md","filePath":"guide/css相关/BFC.md","lastUpdated":1736839070000}'),o={name:"guide/css相关/BFC.md"},t=e("",8),r=[t];function s(n,c,d,h,_,f){return a(),l("div",null,r)}const p=i(o,[["render",s]]);export{m as __pageData,p as default}; diff --git "a/assets/guide_css\347\233\270\345\205\263_display.md.3c174711.js" "b/assets/guide_css\347\233\270\345\205\263_display.md.6d34245c.js" similarity index 99% rename from "assets/guide_css\347\233\270\345\205\263_display.md.3c174711.js" rename to "assets/guide_css\347\233\270\345\205\263_display.md.6d34245c.js" index b79c09f5..57fa9487 100644 --- "a/assets/guide_css\347\233\270\345\205\263_display.md.3c174711.js" +++ "b/assets/guide_css\347\233\270\345\205\263_display.md.6d34245c.js" @@ -1,4 +1,4 @@ -import{_ as s,o as n,c as l,Q as a}from"./chunks/framework.b6910bb2.js";const d=JSON.parse('{"title":"display","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/display.md","filePath":"guide/css相关/display.md","lastUpdated":1736065650000}'),p={name:"guide/css相关/display.md"},o=a(`

display

none

不显示元素,脱离文档流

inline 行内元素

不能设置宽高、不换行

水平 padding、margin 有效

垂直方向 padding、margin 无效

line-block

可以设置宽高、不换行

block

块级元素,可以设置宽高,独占一行

flex

弹性布局

table

元素作为块级表格元素使用

inherit

继承父元素 display 值

常见的问题

line-block、inline 都会有空格问题

因为浏览器会将换行符当空格字符处理

解决方案:

  1. 使用 font-size 时,可通过设置 font-size: 0、letter-spacing、word-spacing 解决
  2. 使用弹性布局
  3. 使用 margin 负值

浏览器对于空格的默认表现

  1. 元素的头尾的空白符会直接忽略
  2. 内容中间有多个空格,会被合并成一个空格
js
<!DOCTYPE html>
+import{_ as s,o as n,c as l,Q as a}from"./chunks/framework.b6910bb2.js";const d=JSON.parse('{"title":"display","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/display.md","filePath":"guide/css相关/display.md","lastUpdated":1736839070000}'),p={name:"guide/css相关/display.md"},o=a(`

display

none

不显示元素,脱离文档流

inline 行内元素

不能设置宽高、不换行

水平 padding、margin 有效

垂直方向 padding、margin 无效

line-block

可以设置宽高、不换行

block

块级元素,可以设置宽高,独占一行

flex

弹性布局

table

元素作为块级表格元素使用

inherit

继承父元素 display 值

常见的问题

line-block、inline 都会有空格问题

因为浏览器会将换行符当空格字符处理

解决方案:

  1. 使用 font-size 时,可通过设置 font-size: 0、letter-spacing、word-spacing 解决
  2. 使用弹性布局
  3. 使用 margin 负值

浏览器对于空格的默认表现

  1. 元素的头尾的空白符会直接忽略
  2. 内容中间有多个空格,会被合并成一个空格
js
<!DOCTYPE html>
 <html lang="en">
 
 <head>
diff --git "a/assets/guide_css\347\233\270\345\205\263_display.md.3c174711.lean.js" "b/assets/guide_css\347\233\270\345\205\263_display.md.6d34245c.lean.js"
similarity index 85%
rename from "assets/guide_css\347\233\270\345\205\263_display.md.3c174711.lean.js"
rename to "assets/guide_css\347\233\270\345\205\263_display.md.6d34245c.lean.js"
index 6bf268ec..38183622 100644
--- "a/assets/guide_css\347\233\270\345\205\263_display.md.3c174711.lean.js"
+++ "b/assets/guide_css\347\233\270\345\205\263_display.md.6d34245c.lean.js"
@@ -1 +1 @@
-import{_ as s,o as n,c as l,Q as a}from"./chunks/framework.b6910bb2.js";const d=JSON.parse('{"title":"display","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/display.md","filePath":"guide/css相关/display.md","lastUpdated":1736065650000}'),p={name:"guide/css相关/display.md"},o=a("",25),t=[o];function e(c,E,r,y,i,g){return n(),l("div",null,t)}const u=s(p,[["render",e]]);export{d as __pageData,u as default};
+import{_ as s,o as n,c as l,Q as a}from"./chunks/framework.b6910bb2.js";const d=JSON.parse('{"title":"display","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/display.md","filePath":"guide/css相关/display.md","lastUpdated":1736839070000}'),p={name:"guide/css相关/display.md"},o=a("",25),t=[o];function e(c,E,r,y,i,g){return n(),l("div",null,t)}const u=s(p,[["render",e]]);export{d as __pageData,u as default};
diff --git "a/assets/guide_css\347\233\270\345\205\263_display\343\200\201float\343\200\201position\347\232\204\345\205\263\347\263\273.md.0e4b5524.js" "b/assets/guide_css\347\233\270\345\205\263_display\343\200\201float\343\200\201position\347\232\204\345\205\263\347\263\273.md.f469d683.js"
similarity index 99%
rename from "assets/guide_css\347\233\270\345\205\263_display\343\200\201float\343\200\201position\347\232\204\345\205\263\347\263\273.md.0e4b5524.js"
rename to "assets/guide_css\347\233\270\345\205\263_display\343\200\201float\343\200\201position\347\232\204\345\205\263\347\263\273.md.f469d683.js"
index ff375fbd..ba3ecef7 100644
--- "a/assets/guide_css\347\233\270\345\205\263_display\343\200\201float\343\200\201position\347\232\204\345\205\263\347\263\273.md.0e4b5524.js"
+++ "b/assets/guide_css\347\233\270\345\205\263_display\343\200\201float\343\200\201position\347\232\204\345\205\263\347\263\273.md.f469d683.js"
@@ -1,4 +1,4 @@
-import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const C=JSON.parse('{"title":"display、float、position的关系","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/display、float、position的关系.md","filePath":"guide/css相关/display、float、position的关系.md","lastUpdated":1736065650000}'),p={name:"guide/css相关/display、float、position的关系.md"},o=l(`

display、float、position的关系

  • 如果元素的 display 为 none 直接隐藏
  • 如果元素的 position 为 absolute 或者 fixed 时,float 失效,display 会提升块级元素
  • 如果元素的 position 为 relative, 会在浮动后的元素进行定位,display 会被提升成块级元素
  • 如果是根元素,display 会被提升为块级元素
  • 浮动元素会将 display 提升为块级元素
html
<!DOCTYPE html>
+import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const C=JSON.parse('{"title":"display、float、position的关系","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/display、float、position的关系.md","filePath":"guide/css相关/display、float、position的关系.md","lastUpdated":1736839070000}'),p={name:"guide/css相关/display、float、position的关系.md"},o=l(`

display、float、position的关系

  • 如果元素的 display 为 none 直接隐藏
  • 如果元素的 position 为 absolute 或者 fixed 时,float 失效,display 会提升块级元素
  • 如果元素的 position 为 relative, 会在浮动后的元素进行定位,display 会被提升成块级元素
  • 如果是根元素,display 会被提升为块级元素
  • 浮动元素会将 display 提升为块级元素
html
<!DOCTYPE html>
 <html lang="en">
 
 <head>
diff --git "a/assets/guide_css\347\233\270\345\205\263_display\343\200\201float\343\200\201position\347\232\204\345\205\263\347\263\273.md.0e4b5524.lean.js" "b/assets/guide_css\347\233\270\345\205\263_display\343\200\201float\343\200\201position\347\232\204\345\205\263\347\263\273.md.f469d683.lean.js"
similarity index 88%
rename from "assets/guide_css\347\233\270\345\205\263_display\343\200\201float\343\200\201position\347\232\204\345\205\263\347\263\273.md.0e4b5524.lean.js"
rename to "assets/guide_css\347\233\270\345\205\263_display\343\200\201float\343\200\201position\347\232\204\345\205\263\347\263\273.md.f469d683.lean.js"
index ea71a8bb..8facf0ba 100644
--- "a/assets/guide_css\347\233\270\345\205\263_display\343\200\201float\343\200\201position\347\232\204\345\205\263\347\263\273.md.0e4b5524.lean.js"
+++ "b/assets/guide_css\347\233\270\345\205\263_display\343\200\201float\343\200\201position\347\232\204\345\205\263\347\263\273.md.f469d683.lean.js"
@@ -1 +1 @@
-import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const C=JSON.parse('{"title":"display、float、position的关系","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/display、float、position的关系.md","filePath":"guide/css相关/display、float、position的关系.md","lastUpdated":1736065650000}'),p={name:"guide/css相关/display、float、position的关系.md"},o=l("",3),t=[o];function e(c,E,r,y,i,d){return n(),a("div",null,t)}const u=s(p,[["render",e]]);export{C as __pageData,u as default};
+import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const C=JSON.parse('{"title":"display、float、position的关系","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/display、float、position的关系.md","filePath":"guide/css相关/display、float、position的关系.md","lastUpdated":1736839070000}'),p={name:"guide/css相关/display、float、position的关系.md"},o=l("",3),t=[o];function e(c,E,r,y,i,d){return n(),a("div",null,t)}const u=s(p,[["render",e]]);export{C as __pageData,u as default};
diff --git "a/assets/guide_css\347\233\270\345\205\263_display\343\200\201visibility\343\200\201opacity\345\214\272\345\210\253.md.66ff32b5.js" "b/assets/guide_css\347\233\270\345\205\263_display\343\200\201visibility\343\200\201opacity\345\214\272\345\210\253.md.2678884b.js"
similarity index 96%
rename from "assets/guide_css\347\233\270\345\205\263_display\343\200\201visibility\343\200\201opacity\345\214\272\345\210\253.md.66ff32b5.js"
rename to "assets/guide_css\347\233\270\345\205\263_display\343\200\201visibility\343\200\201opacity\345\214\272\345\210\253.md.2678884b.js"
index 4bbea699..303a8336 100644
--- "a/assets/guide_css\347\233\270\345\205\263_display\343\200\201visibility\343\200\201opacity\345\214\272\345\210\253.md.66ff32b5.js"
+++ "b/assets/guide_css\347\233\270\345\205\263_display\343\200\201visibility\343\200\201opacity\345\214\272\345\210\253.md.2678884b.js"
@@ -1 +1 @@
-import{_ as i,o as l,c as a,Q as t}from"./chunks/framework.b6910bb2.js";const b=JSON.parse('{"title":"display、visibility、opacity区别","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/display、visibility、opacity区别.md","filePath":"guide/css相关/display、visibility、opacity区别.md","lastUpdated":1736065650000}'),s={name:"guide/css相关/display、visibility、opacity区别.md"},e=t('

display、visibility、opacity区别

  • 占据空间

    • opacity、visibility 占据空间,不会引起回流,但是会重绘
    • display 不占据空间,但会引起页面的回流和重绘
  • 绑定事件

    • display、visibility 不会触发绑定事件
    • opacity 会触发绑定事件

display: none 和 visibility: hidden 的区别

  • 从渲染树上看
    • display: none 不存在渲染树中
    • visibility: hidden 的元素存在渲染树中,还会占据空间
  • 从继承上看
    • display 不会被继承
    • visibility 会被继承
  • 从渲染上看
    • display 会影响回流重绘
    • visibility 只会引起重绘
',4),d=[e];function y(o,p,n,c,_,r){return l(),a("div",null,d)}const h=i(s,[["render",y]]);export{b as __pageData,h as default}; +import{_ as i,o as l,c as a,Q as t}from"./chunks/framework.b6910bb2.js";const b=JSON.parse('{"title":"display、visibility、opacity区别","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/display、visibility、opacity区别.md","filePath":"guide/css相关/display、visibility、opacity区别.md","lastUpdated":1736839070000}'),s={name:"guide/css相关/display、visibility、opacity区别.md"},e=t('

display、visibility、opacity区别

  • 占据空间

    • opacity、visibility 占据空间,不会引起回流,但是会重绘
    • display 不占据空间,但会引起页面的回流和重绘
  • 绑定事件

    • display、visibility 不会触发绑定事件
    • opacity 会触发绑定事件

display: none 和 visibility: hidden 的区别

  • 从渲染树上看
    • display: none 不存在渲染树中
    • visibility: hidden 的元素存在渲染树中,还会占据空间
  • 从继承上看
    • display 不会被继承
    • visibility 会被继承
  • 从渲染上看
    • display 会影响回流重绘
    • visibility 只会引起重绘
',4),d=[e];function y(o,p,n,c,_,r){return l(),a("div",null,d)}const h=i(s,[["render",y]]);export{b as __pageData,h as default}; diff --git "a/assets/guide_css\347\233\270\345\205\263_display\343\200\201visibility\343\200\201opacity\345\214\272\345\210\253.md.66ff32b5.lean.js" "b/assets/guide_css\347\233\270\345\205\263_display\343\200\201visibility\343\200\201opacity\345\214\272\345\210\253.md.2678884b.lean.js" similarity index 88% rename from "assets/guide_css\347\233\270\345\205\263_display\343\200\201visibility\343\200\201opacity\345\214\272\345\210\253.md.66ff32b5.lean.js" rename to "assets/guide_css\347\233\270\345\205\263_display\343\200\201visibility\343\200\201opacity\345\214\272\345\210\253.md.2678884b.lean.js" index 430df202..83b75a76 100644 --- "a/assets/guide_css\347\233\270\345\205\263_display\343\200\201visibility\343\200\201opacity\345\214\272\345\210\253.md.66ff32b5.lean.js" +++ "b/assets/guide_css\347\233\270\345\205\263_display\343\200\201visibility\343\200\201opacity\345\214\272\345\210\253.md.2678884b.lean.js" @@ -1 +1 @@ -import{_ as i,o as l,c as a,Q as t}from"./chunks/framework.b6910bb2.js";const b=JSON.parse('{"title":"display、visibility、opacity区别","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/display、visibility、opacity区别.md","filePath":"guide/css相关/display、visibility、opacity区别.md","lastUpdated":1736065650000}'),s={name:"guide/css相关/display、visibility、opacity区别.md"},e=t("",4),d=[e];function y(o,p,n,c,_,r){return l(),a("div",null,d)}const h=i(s,[["render",y]]);export{b as __pageData,h as default}; +import{_ as i,o as l,c as a,Q as t}from"./chunks/framework.b6910bb2.js";const b=JSON.parse('{"title":"display、visibility、opacity区别","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/display、visibility、opacity区别.md","filePath":"guide/css相关/display、visibility、opacity区别.md","lastUpdated":1736839070000}'),s={name:"guide/css相关/display、visibility、opacity区别.md"},e=t("",4),d=[e];function y(o,p,n,c,_,r){return l(),a("div",null,d)}const h=i(s,[["render",y]]);export{b as __pageData,h as default}; diff --git "a/assets/guide_css\347\233\270\345\205\263_position.md.839d8549.js" "b/assets/guide_css\347\233\270\345\205\263_position.md.b4f4db42.js" similarity index 99% rename from "assets/guide_css\347\233\270\345\205\263_position.md.839d8549.js" rename to "assets/guide_css\347\233\270\345\205\263_position.md.b4f4db42.js" index bb9c4d86..8e3535ee 100644 --- "a/assets/guide_css\347\233\270\345\205\263_position.md.839d8549.js" +++ "b/assets/guide_css\347\233\270\345\205\263_position.md.b4f4db42.js" @@ -1,4 +1,4 @@ -import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="",o="/vitePress-blob/assets/2.bf4501cf.jpg",t="",u=JSON.parse('{"title":"Position","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/position.md","filePath":"guide/css相关/position.md","lastUpdated":1736065650000}'),e={name:"guide/css相关/position.md"},c=l('

Position

static [ˈstætɪk]

默认值,没有定位,元素出现在正常的文档流中,会忽略 top,bottom,left,right 或者 z-index 声明

relative

生成相对定位元素,相对于其原来位置进行定位,元素的位置通过 left、top、right、bottom 属性规定

relative

元素定位永远相对于元素自身位置,和其他元素没有关系,也不会影响其他元素。

absolute

生成绝对定位的元素,相对于 static 定位以外的一个父元素进行定位

元素位置通过 left、top、right、bottom 属性规定

relative

浏览器会递归查找该元素的所有父元素,如果找到一个设置了非 static 的定位元素,就以该元素为基准定位,如果没有找到,就以浏览器边界定位

但是它具有破坏性,会导致其他元素位置的变化

fixed

生成绝对定位的元素,指定元素相对于屏幕视口的位置来制定元素位置。

relative

但是它具有破坏性,会导致其他元素位置的变化

sticky [ˈstɪki]

相对它的最近滚动祖先(nearest scrolling ancestor)和 containing block (最近块级祖先 nearest block-level ancestor),包括table-related元素,基于top, right, bottom, 和 left的值进行偏移。偏移值不会影响任何其他元素的位置。

inherit

规定从父元素继承 position 属性的值

js
<!DOCTYPE html>
+import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="",o="/vitePress-blob/assets/2.bf4501cf.jpg",t="",u=JSON.parse('{"title":"Position","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/position.md","filePath":"guide/css相关/position.md","lastUpdated":1736839070000}'),e={name:"guide/css相关/position.md"},c=l('

Position

static [ˈstætɪk]

默认值,没有定位,元素出现在正常的文档流中,会忽略 top,bottom,left,right 或者 z-index 声明

relative

生成相对定位元素,相对于其原来位置进行定位,元素的位置通过 left、top、right、bottom 属性规定

relative

元素定位永远相对于元素自身位置,和其他元素没有关系,也不会影响其他元素。

absolute

生成绝对定位的元素,相对于 static 定位以外的一个父元素进行定位

元素位置通过 left、top、right、bottom 属性规定

relative

浏览器会递归查找该元素的所有父元素,如果找到一个设置了非 static 的定位元素,就以该元素为基准定位,如果没有找到,就以浏览器边界定位

但是它具有破坏性,会导致其他元素位置的变化

fixed

生成绝对定位的元素,指定元素相对于屏幕视口的位置来制定元素位置。

relative

但是它具有破坏性,会导致其他元素位置的变化

sticky [ˈstɪki]

相对它的最近滚动祖先(nearest scrolling ancestor)和 containing block (最近块级祖先 nearest block-level ancestor),包括table-related元素,基于top, right, bottom, 和 left的值进行偏移。偏移值不会影响任何其他元素的位置。

inherit

规定从父元素继承 position 属性的值

js
<!DOCTYPE html>
 <html lang="en">
 
 <head>
diff --git "a/assets/guide_css\347\233\270\345\205\263_position.md.839d8549.lean.js" "b/assets/guide_css\347\233\270\345\205\263_position.md.b4f4db42.lean.js"
similarity index 99%
rename from "assets/guide_css\347\233\270\345\205\263_position.md.839d8549.lean.js"
rename to "assets/guide_css\347\233\270\345\205\263_position.md.b4f4db42.lean.js"
index bb5e423c..733a4e10 100644
--- "a/assets/guide_css\347\233\270\345\205\263_position.md.839d8549.lean.js"
+++ "b/assets/guide_css\347\233\270\345\205\263_position.md.b4f4db42.lean.js"
@@ -1 +1 @@
-import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="",o="/vitePress-blob/assets/2.bf4501cf.jpg",t="",u=JSON.parse('{"title":"Position","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/position.md","filePath":"guide/css相关/position.md","lastUpdated":1736065650000}'),e={name:"guide/css相关/position.md"},c=l("",22),r=[c];function E(i,A,y,g,d,h){return n(),a("div",null,r)}const Q=s(e,[["render",E]]);export{u as __pageData,Q as default};
+import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="",o="/vitePress-blob/assets/2.bf4501cf.jpg",t="",u=JSON.parse('{"title":"Position","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/position.md","filePath":"guide/css相关/position.md","lastUpdated":1736839070000}'),e={name:"guide/css相关/position.md"},c=l("",22),r=[c];function E(i,A,y,g,d,h){return n(),a("div",null,r)}const Q=s(e,[["render",E]]);export{u as __pageData,Q as default};
diff --git "a/assets/guide_css\347\233\270\345\205\263_\345\257\271 line-height \347\232\204\347\220\206\350\247\243\345\217\212\345\205\266\350\265\213\345\200\274\346\226\271\345\274\217.md.838b7cb9.js" "b/assets/guide_css\347\233\270\345\205\263_\345\257\271 line-height \347\232\204\347\220\206\350\247\243\345\217\212\345\205\266\350\265\213\345\200\274\346\226\271\345\274\217.md.7f0ca022.js"
similarity index 97%
rename from "assets/guide_css\347\233\270\345\205\263_\345\257\271 line-height \347\232\204\347\220\206\350\247\243\345\217\212\345\205\266\350\265\213\345\200\274\346\226\271\345\274\217.md.838b7cb9.js"
rename to "assets/guide_css\347\233\270\345\205\263_\345\257\271 line-height \347\232\204\347\220\206\350\247\243\345\217\212\345\205\266\350\265\213\345\200\274\346\226\271\345\274\217.md.7f0ca022.js"
index ccdf7a15..81ece1d3 100644
--- "a/assets/guide_css\347\233\270\345\205\263_\345\257\271 line-height \347\232\204\347\220\206\350\247\243\345\217\212\345\205\266\350\265\213\345\200\274\346\226\271\345\274\217.md.838b7cb9.js"	
+++ "b/assets/guide_css\347\233\270\345\205\263_\345\257\271 line-height \347\232\204\347\220\206\350\247\243\345\217\212\345\205\266\350\265\213\345\200\274\346\226\271\345\274\217.md.7f0ca022.js"	
@@ -1 +1 @@
-import{_ as e,o as a,c as t,Q as i}from"./chunks/framework.b6910bb2.js";const u=JSON.parse('{"title":"对 line-height 的理解及其赋值方式","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/对 line-height 的理解及其赋值方式.md","filePath":"guide/css相关/对 line-height 的理解及其赋值方式.md","lastUpdated":1736065650000}'),h={name:"guide/css相关/对 line-height 的理解及其赋值方式.md"},n=i('

对 line-height 的理解及其赋值方式

概念

line-height 指一行文本的高度,包含了字间距,实际上是下一行基线到上一行基线的距离

line-height 和 height 都能撑开元素高度 (如果 line-height 和 height 一致时可以实现单行文字的垂直居中)

计算方式

带单位的px

line-height 为 固定值

纯数字

直接 line-height 会把比例传递给后代

例如,父级行高为 1.5,子元素字体为 18px,则子元素行高为 1.5 * 18 = 27px

百分比

父元素将计算后的值传递给后代

例如,父元素行高为 200%,父元素字体为 18px,那么子元素行高为 18 * 200% = 36px

',13),r=[n];function l(o,s,_,d,c,p){return a(),t("div",null,r)}const x=e(h,[["render",l]]);export{u as __pageData,x as default}; +import{_ as e,o as a,c as t,Q as i}from"./chunks/framework.b6910bb2.js";const u=JSON.parse('{"title":"对 line-height 的理解及其赋值方式","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/对 line-height 的理解及其赋值方式.md","filePath":"guide/css相关/对 line-height 的理解及其赋值方式.md","lastUpdated":1736839070000}'),h={name:"guide/css相关/对 line-height 的理解及其赋值方式.md"},n=i('

对 line-height 的理解及其赋值方式

概念

line-height 指一行文本的高度,包含了字间距,实际上是下一行基线到上一行基线的距离

line-height 和 height 都能撑开元素高度 (如果 line-height 和 height 一致时可以实现单行文字的垂直居中)

计算方式

带单位的px

line-height 为 固定值

纯数字

直接 line-height 会把比例传递给后代

例如,父级行高为 1.5,子元素字体为 18px,则子元素行高为 1.5 * 18 = 27px

百分比

父元素将计算后的值传递给后代

例如,父元素行高为 200%,父元素字体为 18px,那么子元素行高为 18 * 200% = 36px

',13),r=[n];function l(o,s,_,d,c,p){return a(),t("div",null,r)}const x=e(h,[["render",l]]);export{u as __pageData,x as default}; diff --git "a/assets/guide_css\347\233\270\345\205\263_\345\257\271 line-height \347\232\204\347\220\206\350\247\243\345\217\212\345\205\266\350\265\213\345\200\274\346\226\271\345\274\217.md.838b7cb9.lean.js" "b/assets/guide_css\347\233\270\345\205\263_\345\257\271 line-height \347\232\204\347\220\206\350\247\243\345\217\212\345\205\266\350\265\213\345\200\274\346\226\271\345\274\217.md.7f0ca022.lean.js" similarity index 89% rename from "assets/guide_css\347\233\270\345\205\263_\345\257\271 line-height \347\232\204\347\220\206\350\247\243\345\217\212\345\205\266\350\265\213\345\200\274\346\226\271\345\274\217.md.838b7cb9.lean.js" rename to "assets/guide_css\347\233\270\345\205\263_\345\257\271 line-height \347\232\204\347\220\206\350\247\243\345\217\212\345\205\266\350\265\213\345\200\274\346\226\271\345\274\217.md.7f0ca022.lean.js" index ea66af65..363f62ed 100644 --- "a/assets/guide_css\347\233\270\345\205\263_\345\257\271 line-height \347\232\204\347\220\206\350\247\243\345\217\212\345\205\266\350\265\213\345\200\274\346\226\271\345\274\217.md.838b7cb9.lean.js" +++ "b/assets/guide_css\347\233\270\345\205\263_\345\257\271 line-height \347\232\204\347\220\206\350\247\243\345\217\212\345\205\266\350\265\213\345\200\274\346\226\271\345\274\217.md.7f0ca022.lean.js" @@ -1 +1 @@ -import{_ as e,o as a,c as t,Q as i}from"./chunks/framework.b6910bb2.js";const u=JSON.parse('{"title":"对 line-height 的理解及其赋值方式","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/对 line-height 的理解及其赋值方式.md","filePath":"guide/css相关/对 line-height 的理解及其赋值方式.md","lastUpdated":1736065650000}'),h={name:"guide/css相关/对 line-height 的理解及其赋值方式.md"},n=i("",13),r=[n];function l(o,s,_,d,c,p){return a(),t("div",null,r)}const x=e(h,[["render",l]]);export{u as __pageData,x as default}; +import{_ as e,o as a,c as t,Q as i}from"./chunks/framework.b6910bb2.js";const u=JSON.parse('{"title":"对 line-height 的理解及其赋值方式","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/对 line-height 的理解及其赋值方式.md","filePath":"guide/css相关/对 line-height 的理解及其赋值方式.md","lastUpdated":1736839070000}'),h={name:"guide/css相关/对 line-height 的理解及其赋值方式.md"},n=i("",13),r=[n];function l(o,s,_,d,c,p){return a(),t("div",null,r)}const x=e(h,[["render",l]]);export{u as __pageData,x as default}; diff --git "a/assets/guide_css\347\233\270\345\205\263_\346\246\202\350\246\201.md.6367bf5b.js" "b/assets/guide_css\347\233\270\345\205\263_\346\246\202\350\246\201.md.9e3e6a15.js" similarity index 87% rename from "assets/guide_css\347\233\270\345\205\263_\346\246\202\350\246\201.md.6367bf5b.js" rename to "assets/guide_css\347\233\270\345\205\263_\346\246\202\350\246\201.md.9e3e6a15.js" index 318b8838..47843992 100644 --- "a/assets/guide_css\347\233\270\345\205\263_\346\246\202\350\246\201.md.6367bf5b.js" +++ "b/assets/guide_css\347\233\270\345\205\263_\346\246\202\350\246\201.md.9e3e6a15.js" @@ -1 +1 @@ -import{_ as e}from"./chunks/container.fad5294e.js";import{o as t,c as a,H as s}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{"aside":false,"outline":false},"headers":[],"relativePath":"guide/css相关/概要.md","filePath":"guide/css相关/概要.md","lastUpdated":1736065650000}'),c={name:"guide/css相关/概要.md"},n=Object.assign(c,{setup(o){return(r,d)=>(t(),a("div",null,[s(e,{url:"https://www.mubu.com/doc/1bKlzv2cCSK#m"})]))}});export{m as __pageData,n as default}; +import{_ as e}from"./chunks/container.fad5294e.js";import{o as t,c as a,H as s}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{"aside":false,"outline":false},"headers":[],"relativePath":"guide/css相关/概要.md","filePath":"guide/css相关/概要.md","lastUpdated":1736839070000}'),c={name:"guide/css相关/概要.md"},n=Object.assign(c,{setup(o){return(r,d)=>(t(),a("div",null,[s(e,{url:"https://www.mubu.com/doc/1bKlzv2cCSK#m"})]))}});export{m as __pageData,n as default}; diff --git "a/assets/guide_css\347\233\270\345\205\263_\346\246\202\350\246\201.md.6367bf5b.lean.js" "b/assets/guide_css\347\233\270\345\205\263_\346\246\202\350\246\201.md.9e3e6a15.lean.js" similarity index 87% rename from "assets/guide_css\347\233\270\345\205\263_\346\246\202\350\246\201.md.6367bf5b.lean.js" rename to "assets/guide_css\347\233\270\345\205\263_\346\246\202\350\246\201.md.9e3e6a15.lean.js" index 318b8838..47843992 100644 --- "a/assets/guide_css\347\233\270\345\205\263_\346\246\202\350\246\201.md.6367bf5b.lean.js" +++ "b/assets/guide_css\347\233\270\345\205\263_\346\246\202\350\246\201.md.9e3e6a15.lean.js" @@ -1 +1 @@ -import{_ as e}from"./chunks/container.fad5294e.js";import{o as t,c as a,H as s}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{"aside":false,"outline":false},"headers":[],"relativePath":"guide/css相关/概要.md","filePath":"guide/css相关/概要.md","lastUpdated":1736065650000}'),c={name:"guide/css相关/概要.md"},n=Object.assign(c,{setup(o){return(r,d)=>(t(),a("div",null,[s(e,{url:"https://www.mubu.com/doc/1bKlzv2cCSK#m"})]))}});export{m as __pageData,n as default}; +import{_ as e}from"./chunks/container.fad5294e.js";import{o as t,c as a,H as s}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{"aside":false,"outline":false},"headers":[],"relativePath":"guide/css相关/概要.md","filePath":"guide/css相关/概要.md","lastUpdated":1736839070000}'),c={name:"guide/css相关/概要.md"},n=Object.assign(c,{setup(o){return(r,d)=>(t(),a("div",null,[s(e,{url:"https://www.mubu.com/doc/1bKlzv2cCSK#m"})]))}});export{m as __pageData,n as default}; diff --git "a/assets/guide_css\347\233\270\345\205\263_\351\232\220\350\227\217\345\205\203\347\264\240\347\232\204\346\226\271\346\263\225.md.6677bfcf.js" "b/assets/guide_css\347\233\270\345\205\263_\351\232\220\350\227\217\345\205\203\347\264\240\347\232\204\346\226\271\346\263\225.md.3f129790.js" similarity index 94% rename from "assets/guide_css\347\233\270\345\205\263_\351\232\220\350\227\217\345\205\203\347\264\240\347\232\204\346\226\271\346\263\225.md.6677bfcf.js" rename to "assets/guide_css\347\233\270\345\205\263_\351\232\220\350\227\217\345\205\203\347\264\240\347\232\204\346\226\271\346\263\225.md.3f129790.js" index 945e3a92..345e8fca 100644 --- "a/assets/guide_css\347\233\270\345\205\263_\351\232\220\350\227\217\345\205\203\347\264\240\347\232\204\346\226\271\346\263\225.md.6677bfcf.js" +++ "b/assets/guide_css\347\233\270\345\205\263_\351\232\220\350\227\217\345\205\203\347\264\240\347\232\204\346\226\271\346\263\225.md.3f129790.js" @@ -1 +1 @@ -import{_ as l,o as i,c as t,Q as e}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"隐藏元素的方法","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/隐藏元素的方法.md","filePath":"guide/css相关/隐藏元素的方法.md","lastUpdated":1736065650000}'),a={name:"guide/css相关/隐藏元素的方法.md"},o=e('

隐藏元素的方法

    1. display: none - 不会显示在渲染树上,不占据空间,无法监听事件
    1. visibility: hidden - 占据空间,无法监听事件
    1. opacity: 0 - 占据空间,可以监听事件
    1. transform: scale(0, 0)
    1. 绝对定位
    1. z-index 为负数
',2),s=[o];function _(r,c,n,d,p,h){return i(),t("div",null,s)}const f=l(a,[["render",_]]);export{m as __pageData,f as default}; +import{_ as l,o as i,c as t,Q as e}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"隐藏元素的方法","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/隐藏元素的方法.md","filePath":"guide/css相关/隐藏元素的方法.md","lastUpdated":1736839070000}'),a={name:"guide/css相关/隐藏元素的方法.md"},o=e('

隐藏元素的方法

    1. display: none - 不会显示在渲染树上,不占据空间,无法监听事件
    1. visibility: hidden - 占据空间,无法监听事件
    1. opacity: 0 - 占据空间,可以监听事件
    1. transform: scale(0, 0)
    1. 绝对定位
    1. z-index 为负数
',2),s=[o];function _(r,c,n,d,p,h){return i(),t("div",null,s)}const f=l(a,[["render",_]]);export{m as __pageData,f as default}; diff --git "a/assets/guide_css\347\233\270\345\205\263_\351\232\220\350\227\217\345\205\203\347\264\240\347\232\204\346\226\271\346\263\225.md.6677bfcf.lean.js" "b/assets/guide_css\347\233\270\345\205\263_\351\232\220\350\227\217\345\205\203\347\264\240\347\232\204\346\226\271\346\263\225.md.3f129790.lean.js" similarity index 87% rename from "assets/guide_css\347\233\270\345\205\263_\351\232\220\350\227\217\345\205\203\347\264\240\347\232\204\346\226\271\346\263\225.md.6677bfcf.lean.js" rename to "assets/guide_css\347\233\270\345\205\263_\351\232\220\350\227\217\345\205\203\347\264\240\347\232\204\346\226\271\346\263\225.md.3f129790.lean.js" index 4832692f..a1e25e74 100644 --- "a/assets/guide_css\347\233\270\345\205\263_\351\232\220\350\227\217\345\205\203\347\264\240\347\232\204\346\226\271\346\263\225.md.6677bfcf.lean.js" +++ "b/assets/guide_css\347\233\270\345\205\263_\351\232\220\350\227\217\345\205\203\347\264\240\347\232\204\346\226\271\346\263\225.md.3f129790.lean.js" @@ -1 +1 @@ -import{_ as l,o as i,c as t,Q as e}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"隐藏元素的方法","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/隐藏元素的方法.md","filePath":"guide/css相关/隐藏元素的方法.md","lastUpdated":1736065650000}'),a={name:"guide/css相关/隐藏元素的方法.md"},o=e("",2),s=[o];function _(r,c,n,d,p,h){return i(),t("div",null,s)}const f=l(a,[["render",_]]);export{m as __pageData,f as default}; +import{_ as l,o as i,c as t,Q as e}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"隐藏元素的方法","description":"","frontmatter":{},"headers":[],"relativePath":"guide/css相关/隐藏元素的方法.md","filePath":"guide/css相关/隐藏元素的方法.md","lastUpdated":1736839070000}'),a={name:"guide/css相关/隐藏元素的方法.md"},o=e("",2),s=[o];function _(r,c,n,d,p,h){return i(),t("div",null,s)}const f=l(a,[["render",_]]);export{m as __pageData,f as default}; diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_DOM\347\233\270\345\205\263.md.7690a9dd.js" "b/assets/guide_javaScript\347\233\270\345\205\263_DOM\347\233\270\345\205\263.md.b5647b3c.js" similarity index 99% rename from "assets/guide_javaScript\347\233\270\345\205\263_DOM\347\233\270\345\205\263.md.7690a9dd.js" rename to "assets/guide_javaScript\347\233\270\345\205\263_DOM\347\233\270\345\205\263.md.b5647b3c.js" index be8d1022..87e3e4da 100644 --- "a/assets/guide_javaScript\347\233\270\345\205\263_DOM\347\233\270\345\205\263.md.7690a9dd.js" +++ "b/assets/guide_javaScript\347\233\270\345\205\263_DOM\347\233\270\345\205\263.md.b5647b3c.js" @@ -1,4 +1,4 @@ -import{_ as s,o as a,c as l,Q as n}from"./chunks/framework.b6910bb2.js";const h=JSON.parse('{"title":"DOM 相关","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/DOM相关.md","filePath":"guide/javaScript相关/DOM相关.md","lastUpdated":1736065650000}'),o={name:"guide/javaScript相关/DOM相关.md"},e=n(`

DOM 相关

如何阻止事件的冒泡和默认事件

js
event.stopPropagation(); // 阻止事件冒泡
+import{_ as s,o as a,c as l,Q as n}from"./chunks/framework.b6910bb2.js";const h=JSON.parse('{"title":"DOM 相关","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/DOM相关.md","filePath":"guide/javaScript相关/DOM相关.md","lastUpdated":1736839070000}'),o={name:"guide/javaScript相关/DOM相关.md"},e=n(`

DOM 相关

如何阻止事件的冒泡和默认事件

js
event.stopPropagation(); // 阻止事件冒泡
 event.preventDefalut(); // 阻止默认事件
 event.stopImmediatePropagation(); // 阻止监听同一事件的其他事件监听器被调用
event.stopPropagation(); // 阻止事件冒泡
 event.preventDefalut(); // 阻止默认事件
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_DOM\347\233\270\345\205\263.md.7690a9dd.lean.js" "b/assets/guide_javaScript\347\233\270\345\205\263_DOM\347\233\270\345\205\263.md.b5647b3c.lean.js"
similarity index 86%
rename from "assets/guide_javaScript\347\233\270\345\205\263_DOM\347\233\270\345\205\263.md.7690a9dd.lean.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_DOM\347\233\270\345\205\263.md.b5647b3c.lean.js"
index 10519f36..35223367 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_DOM\347\233\270\345\205\263.md.7690a9dd.lean.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_DOM\347\233\270\345\205\263.md.b5647b3c.lean.js"
@@ -1 +1 @@
-import{_ as s,o as a,c as l,Q as n}from"./chunks/framework.b6910bb2.js";const h=JSON.parse('{"title":"DOM 相关","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/DOM相关.md","filePath":"guide/javaScript相关/DOM相关.md","lastUpdated":1736065650000}'),o={name:"guide/javaScript相关/DOM相关.md"},e=n("",16),p=[e];function t(c,r,i,y,E,d){return a(),l("div",null,p)}const u=s(o,[["render",t]]);export{h as __pageData,u as default};
+import{_ as s,o as a,c as l,Q as n}from"./chunks/framework.b6910bb2.js";const h=JSON.parse('{"title":"DOM 相关","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/DOM相关.md","filePath":"guide/javaScript相关/DOM相关.md","lastUpdated":1736839070000}'),o={name:"guide/javaScript相关/DOM相关.md"},e=n("",16),p=[e];function t(c,r,i,y,E,d){return a(),l("div",null,p)}const u=s(o,[["render",t]]);export{h as __pageData,u as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_ES6\347\233\270\345\205\263.md.4615c5ea.js" "b/assets/guide_javaScript\347\233\270\345\205\263_ES6\347\233\270\345\205\263.md.a46321f6.js"
similarity index 99%
rename from "assets/guide_javaScript\347\233\270\345\205\263_ES6\347\233\270\345\205\263.md.4615c5ea.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_ES6\347\233\270\345\205\263.md.a46321f6.js"
index 8164e400..bbcc7a10 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_ES6\347\233\270\345\205\263.md.4615c5ea.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_ES6\347\233\270\345\205\263.md.a46321f6.js"
@@ -1,4 +1,4 @@
-import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/6.9b165bbb.png",b=JSON.parse('{"title":"ES6相关","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/ES6相关.md","filePath":"guide/javaScript相关/ES6相关.md","lastUpdated":1736065650000}'),o={name:"guide/javaScript相关/ES6相关.md"},e=l('

ES6相关

var、let、const 的区别

1 作用域 let、const 存在块级作用域 var 不存在块级作用域

2 const 常量 定义时赋值

3 变量提升 let/const 声明的变量需要声明后使用 var 声明的变量可以声明前使用

4 重复声明 let/const 不能重复声明 var 可以重复声明

5 暂时性死区 let/const 声明的变量不能在声明前使用

暂时性死区

箭头函数和普通函数的区别

1 箭头函数没有 arguments

2 箭头函数没有自己的 this this 由定义的所在父级上下文决定

3 箭头函数继承来的 this 不会改变 call、apply、bind 方法不能改变箭头函数的 this 指向

js
var id = 'global';
+import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/6.9b165bbb.png",b=JSON.parse('{"title":"ES6相关","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/ES6相关.md","filePath":"guide/javaScript相关/ES6相关.md","lastUpdated":1736839070000}'),o={name:"guide/javaScript相关/ES6相关.md"},e=l('

ES6相关

var、let、const 的区别

1 作用域 let、const 存在块级作用域 var 不存在块级作用域

2 const 常量 定义时赋值

3 变量提升 let/const 声明的变量需要声明后使用 var 声明的变量可以声明前使用

4 重复声明 let/const 不能重复声明 var 可以重复声明

5 暂时性死区 let/const 声明的变量不能在声明前使用

暂时性死区

箭头函数和普通函数的区别

1 箭头函数没有 arguments

2 箭头函数没有自己的 this this 由定义的所在父级上下文决定

3 箭头函数继承来的 this 不会改变 call、apply、bind 方法不能改变箭头函数的 this 指向

js
var id = 'global';
 var obj = {
     id: 'obj',
     a: function () {
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_ES6\347\233\270\345\205\263.md.4615c5ea.lean.js" "b/assets/guide_javaScript\347\233\270\345\205\263_ES6\347\233\270\345\205\263.md.a46321f6.lean.js"
similarity index 87%
rename from "assets/guide_javaScript\347\233\270\345\205\263_ES6\347\233\270\345\205\263.md.4615c5ea.lean.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_ES6\347\233\270\345\205\263.md.a46321f6.lean.js"
index 020590e8..7a2011a5 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_ES6\347\233\270\345\205\263.md.4615c5ea.lean.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_ES6\347\233\270\345\205\263.md.a46321f6.lean.js"
@@ -1 +1 @@
-import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/6.9b165bbb.png",b=JSON.parse('{"title":"ES6相关","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/ES6相关.md","filePath":"guide/javaScript相关/ES6相关.md","lastUpdated":1736065650000}'),o={name:"guide/javaScript相关/ES6相关.md"},e=l("",19),t=[e];function c(r,E,y,i,F,d){return n(),a("div",null,t)}const A=s(o,[["render",c]]);export{b as __pageData,A as default};
+import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/6.9b165bbb.png",b=JSON.parse('{"title":"ES6相关","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/ES6相关.md","filePath":"guide/javaScript相关/ES6相关.md","lastUpdated":1736839070000}'),o={name:"guide/javaScript相关/ES6相关.md"},e=l("",19),t=[e];function c(r,E,y,i,F,d){return n(),a("div",null,t)}const A=s(o,[["render",c]]);export{b as __pageData,A as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_JS\345\274\202\345\270\270\346\215\225\350\216\267\346\234\272\345\210\266.md.b6816c6f.js" "b/assets/guide_javaScript\347\233\270\345\205\263_JS\345\274\202\345\270\270\346\215\225\350\216\267\346\234\272\345\210\266.md.bd59bbe3.js"
similarity index 86%
rename from "assets/guide_javaScript\347\233\270\345\205\263_JS\345\274\202\345\270\270\346\215\225\350\216\267\346\234\272\345\210\266.md.b6816c6f.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_JS\345\274\202\345\270\270\346\215\225\350\216\267\346\234\272\345\210\266.md.bd59bbe3.js"
index 2679e95c..696cc5d4 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_JS\345\274\202\345\270\270\346\215\225\350\216\267\346\234\272\345\210\266.md.b6816c6f.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_JS\345\274\202\345\270\270\346\215\225\350\216\267\346\234\272\345\210\266.md.bd59bbe3.js"
@@ -1 +1 @@
-import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/JS异常捕获机制.md","filePath":"guide/javaScript相关/JS异常捕获机制.md","lastUpdated":1736065650000}'),r={name:"guide/javaScript相关/JS异常捕获机制.md"};function c(o,s,_,i,d,n){return t(),a("div")}const f=e(r,[["render",c]]);export{m as __pageData,f as default};
+import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/JS异常捕获机制.md","filePath":"guide/javaScript相关/JS异常捕获机制.md","lastUpdated":1736839070000}'),r={name:"guide/javaScript相关/JS异常捕获机制.md"};function c(o,s,_,i,d,n){return t(),a("div")}const f=e(r,[["render",c]]);export{m as __pageData,f as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_JS\345\274\202\345\270\270\346\215\225\350\216\267\346\234\272\345\210\266.md.b6816c6f.lean.js" "b/assets/guide_javaScript\347\233\270\345\205\263_JS\345\274\202\345\270\270\346\215\225\350\216\267\346\234\272\345\210\266.md.bd59bbe3.lean.js"
similarity index 86%
rename from "assets/guide_javaScript\347\233\270\345\205\263_JS\345\274\202\345\270\270\346\215\225\350\216\267\346\234\272\345\210\266.md.b6816c6f.lean.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_JS\345\274\202\345\270\270\346\215\225\350\216\267\346\234\272\345\210\266.md.bd59bbe3.lean.js"
index 2679e95c..696cc5d4 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_JS\345\274\202\345\270\270\346\215\225\350\216\267\346\234\272\345\210\266.md.b6816c6f.lean.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_JS\345\274\202\345\270\270\346\215\225\350\216\267\346\234\272\345\210\266.md.bd59bbe3.lean.js"
@@ -1 +1 @@
-import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/JS异常捕获机制.md","filePath":"guide/javaScript相关/JS异常捕获机制.md","lastUpdated":1736065650000}'),r={name:"guide/javaScript相关/JS异常捕获机制.md"};function c(o,s,_,i,d,n){return t(),a("div")}const f=e(r,[["render",c]]);export{m as __pageData,f as default};
+import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/JS异常捕获机制.md","filePath":"guide/javaScript相关/JS异常捕获机制.md","lastUpdated":1736839070000}'),r={name:"guide/javaScript相关/JS异常捕获机制.md"};function c(o,s,_,i,d,n){return t(),a("div")}const f=e(r,[["render",c]]);export{m as __pageData,f as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_JavaScript\346\211\247\350\241\214\346\234\272\345\210\266.md.f9a311e1.js" "b/assets/guide_javaScript\347\233\270\345\205\263_JavaScript\346\211\247\350\241\214\346\234\272\345\210\266.md.b95eb7c2.js"
similarity index 92%
rename from "assets/guide_javaScript\347\233\270\345\205\263_JavaScript\346\211\247\350\241\214\346\234\272\345\210\266.md.f9a311e1.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_JavaScript\346\211\247\350\241\214\346\234\272\345\210\266.md.b95eb7c2.js"
index 8f1a8cbe..61fc8b87 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_JavaScript\346\211\247\350\241\214\346\234\272\345\210\266.md.f9a311e1.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_JavaScript\346\211\247\350\241\214\346\234\272\345\210\266.md.b95eb7c2.js"
@@ -1 +1 @@
-import{_ as t,o as e,c as r,k as a,a as c}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"JavaScript 执行机制","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/JavaScript执行机制.md","filePath":"guide/javaScript相关/JavaScript执行机制.md","lastUpdated":1736065650000}'),i={name:"guide/javaScript相关/JavaScript执行机制.md"},s=a("h1",{id:"javascript-执行机制",tabindex:"-1"},[c("JavaScript 执行机制 "),a("a",{class:"header-anchor",href:"#javascript-执行机制","aria-label":'Permalink to "JavaScript 执行机制"'},"​")],-1),o=a("p",null,"先编译后执行",-1),p=a("p",null,"编译 -> 创建上下文、创建变量环境、创建词法环境 -> 将上下文压入执行调用栈执行代码",-1),n=[s,o,p];function d(_,l,v,h,S,m){return e(),r("div",null,n)}const J=t(i,[["render",d]]);export{f as __pageData,J as default};
+import{_ as t,o as e,c as r,k as a,a as c}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"JavaScript 执行机制","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/JavaScript执行机制.md","filePath":"guide/javaScript相关/JavaScript执行机制.md","lastUpdated":1736839070000}'),i={name:"guide/javaScript相关/JavaScript执行机制.md"},s=a("h1",{id:"javascript-执行机制",tabindex:"-1"},[c("JavaScript 执行机制 "),a("a",{class:"header-anchor",href:"#javascript-执行机制","aria-label":'Permalink to "JavaScript 执行机制"'},"​")],-1),o=a("p",null,"先编译后执行",-1),p=a("p",null,"编译 -> 创建上下文、创建变量环境、创建词法环境 -> 将上下文压入执行调用栈执行代码",-1),n=[s,o,p];function d(_,l,v,h,S,m){return e(),r("div",null,n)}const J=t(i,[["render",d]]);export{f as __pageData,J as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_JavaScript\346\211\247\350\241\214\346\234\272\345\210\266.md.f9a311e1.lean.js" "b/assets/guide_javaScript\347\233\270\345\205\263_JavaScript\346\211\247\350\241\214\346\234\272\345\210\266.md.b95eb7c2.lean.js"
similarity index 92%
rename from "assets/guide_javaScript\347\233\270\345\205\263_JavaScript\346\211\247\350\241\214\346\234\272\345\210\266.md.f9a311e1.lean.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_JavaScript\346\211\247\350\241\214\346\234\272\345\210\266.md.b95eb7c2.lean.js"
index 8f1a8cbe..61fc8b87 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_JavaScript\346\211\247\350\241\214\346\234\272\345\210\266.md.f9a311e1.lean.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_JavaScript\346\211\247\350\241\214\346\234\272\345\210\266.md.b95eb7c2.lean.js"
@@ -1 +1 @@
-import{_ as t,o as e,c as r,k as a,a as c}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"JavaScript 执行机制","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/JavaScript执行机制.md","filePath":"guide/javaScript相关/JavaScript执行机制.md","lastUpdated":1736065650000}'),i={name:"guide/javaScript相关/JavaScript执行机制.md"},s=a("h1",{id:"javascript-执行机制",tabindex:"-1"},[c("JavaScript 执行机制 "),a("a",{class:"header-anchor",href:"#javascript-执行机制","aria-label":'Permalink to "JavaScript 执行机制"'},"​")],-1),o=a("p",null,"先编译后执行",-1),p=a("p",null,"编译 -> 创建上下文、创建变量环境、创建词法环境 -> 将上下文压入执行调用栈执行代码",-1),n=[s,o,p];function d(_,l,v,h,S,m){return e(),r("div",null,n)}const J=t(i,[["render",d]]);export{f as __pageData,J as default};
+import{_ as t,o as e,c as r,k as a,a as c}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"JavaScript 执行机制","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/JavaScript执行机制.md","filePath":"guide/javaScript相关/JavaScript执行机制.md","lastUpdated":1736839070000}'),i={name:"guide/javaScript相关/JavaScript执行机制.md"},s=a("h1",{id:"javascript-执行机制",tabindex:"-1"},[c("JavaScript 执行机制 "),a("a",{class:"header-anchor",href:"#javascript-执行机制","aria-label":'Permalink to "JavaScript 执行机制"'},"​")],-1),o=a("p",null,"先编译后执行",-1),p=a("p",null,"编译 -> 创建上下文、创建变量环境、创建词法环境 -> 将上下文压入执行调用栈执行代码",-1),n=[s,o,p];function d(_,l,v,h,S,m){return e(),r("div",null,n)}const J=t(i,[["render",d]]);export{f as __pageData,J as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_JavaScript\347\274\226\350\257\221\346\234\272\345\210\266.md.4fefe971.lean.js" "b/assets/guide_javaScript\347\233\270\345\205\263_JavaScript\347\274\226\350\257\221\346\234\272\345\210\266.md.4e21f748.js"
similarity index 92%
rename from "assets/guide_javaScript\347\233\270\345\205\263_JavaScript\347\274\226\350\257\221\346\234\272\345\210\266.md.4fefe971.lean.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_JavaScript\347\274\226\350\257\221\346\234\272\345\210\266.md.4e21f748.js"
index 52ce66d9..b22800b8 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_JavaScript\347\274\226\350\257\221\346\234\272\345\210\266.md.4fefe971.lean.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_JavaScript\347\274\226\350\257\221\346\234\272\345\210\266.md.4e21f748.js"
@@ -1 +1 @@
-import{_ as t,o as e,c as r,k as a,a as c}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"JavaScript 编译机制","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/JavaScript编译机制.md","filePath":"guide/javaScript相关/JavaScript编译机制.md","lastUpdated":1736065650000}'),i={name:"guide/javaScript相关/JavaScript编译机制.md"},s=a("h1",{id:"javascript-编译机制",tabindex:"-1"},[c("JavaScript 编译机制 "),a("a",{class:"header-anchor",href:"#javascript-编译机制","aria-label":'Permalink to "JavaScript 编译机制"'},"​")],-1),o=a("p",null,"code -> 词法分析、语法分析 -> AST -> 生成字节码 -> 解释执行",-1),p=[s,o];function d(n,_,l,v,S,h){return e(),r("div",null,p)}const u=t(i,[["render",d]]);export{f as __pageData,u as default};
+import{_ as t,o as e,c as r,k as a,a as c}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"JavaScript 编译机制","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/JavaScript编译机制.md","filePath":"guide/javaScript相关/JavaScript编译机制.md","lastUpdated":1736839070000}'),i={name:"guide/javaScript相关/JavaScript编译机制.md"},s=a("h1",{id:"javascript-编译机制",tabindex:"-1"},[c("JavaScript 编译机制 "),a("a",{class:"header-anchor",href:"#javascript-编译机制","aria-label":'Permalink to "JavaScript 编译机制"'},"​")],-1),o=a("p",null,"code -> 词法分析、语法分析 -> AST -> 生成字节码 -> 解释执行",-1),p=[s,o];function d(n,_,l,v,S,h){return e(),r("div",null,p)}const u=t(i,[["render",d]]);export{f as __pageData,u as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_JavaScript\347\274\226\350\257\221\346\234\272\345\210\266.md.4fefe971.js" "b/assets/guide_javaScript\347\233\270\345\205\263_JavaScript\347\274\226\350\257\221\346\234\272\345\210\266.md.4e21f748.lean.js"
similarity index 92%
rename from "assets/guide_javaScript\347\233\270\345\205\263_JavaScript\347\274\226\350\257\221\346\234\272\345\210\266.md.4fefe971.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_JavaScript\347\274\226\350\257\221\346\234\272\345\210\266.md.4e21f748.lean.js"
index 52ce66d9..b22800b8 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_JavaScript\347\274\226\350\257\221\346\234\272\345\210\266.md.4fefe971.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_JavaScript\347\274\226\350\257\221\346\234\272\345\210\266.md.4e21f748.lean.js"
@@ -1 +1 @@
-import{_ as t,o as e,c as r,k as a,a as c}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"JavaScript 编译机制","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/JavaScript编译机制.md","filePath":"guide/javaScript相关/JavaScript编译机制.md","lastUpdated":1736065650000}'),i={name:"guide/javaScript相关/JavaScript编译机制.md"},s=a("h1",{id:"javascript-编译机制",tabindex:"-1"},[c("JavaScript 编译机制 "),a("a",{class:"header-anchor",href:"#javascript-编译机制","aria-label":'Permalink to "JavaScript 编译机制"'},"​")],-1),o=a("p",null,"code -> 词法分析、语法分析 -> AST -> 生成字节码 -> 解释执行",-1),p=[s,o];function d(n,_,l,v,S,h){return e(),r("div",null,p)}const u=t(i,[["render",d]]);export{f as __pageData,u as default};
+import{_ as t,o as e,c as r,k as a,a as c}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"JavaScript 编译机制","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/JavaScript编译机制.md","filePath":"guide/javaScript相关/JavaScript编译机制.md","lastUpdated":1736839070000}'),i={name:"guide/javaScript相关/JavaScript编译机制.md"},s=a("h1",{id:"javascript-编译机制",tabindex:"-1"},[c("JavaScript 编译机制 "),a("a",{class:"header-anchor",href:"#javascript-编译机制","aria-label":'Permalink to "JavaScript 编译机制"'},"​")],-1),o=a("p",null,"code -> 词法分析、语法分析 -> AST -> 生成字节码 -> 解释执行",-1),p=[s,o];function d(n,_,l,v,S,h){return e(),r("div",null,p)}const u=t(i,[["render",d]]);export{f as __pageData,u as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_Promise.md.54f111a9.js" "b/assets/guide_javaScript\347\233\270\345\205\263_Promise.md.9dcd3d04.js"
similarity index 99%
rename from "assets/guide_javaScript\347\233\270\345\205\263_Promise.md.54f111a9.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_Promise.md.9dcd3d04.js"
index d6a5cc2e..2c02efc3 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_Promise.md.54f111a9.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_Promise.md.9dcd3d04.js"
@@ -1,4 +1,4 @@
-import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/10.5cd5f050.png",o="/vitePress-blob/assets/7.6a576ff8.png",e="/vitePress-blob/assets/8.56c35364.png",r="/vitePress-blob/assets/9.bf057657.png",v=JSON.parse('{"title":"Promise 相关","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/Promise.md","filePath":"guide/javaScript相关/Promise.md","lastUpdated":1736065650000}'),t={name:"guide/javaScript相关/Promise.md"},c=l(`

Promise 相关

定义

Promise 有 3 种状态:

  • 等待状态 pending
  • 成功状态 resolved
  • 失败状态 rejected

当 Promise 状态是 Pending 时,可以转换为 Resolved 或者 Rejected 状态,一旦状态转换,就不会再改变。

resolved、rejected 是异步函数

then 默认返回一个 Promise 对象,第一个参数是 resolved 的回调函数,第二个参数是 rejected 的回调函数 catch 会返回一个 Promise 对象,catch 正常返回 resolved 状态,里面报错返回 rejected 状态

Promise.resolve

Promise.resolve(value) 返回一个成功状态的 promise 对象

js
/*
+import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/10.5cd5f050.png",o="/vitePress-blob/assets/7.6a576ff8.png",e="/vitePress-blob/assets/8.56c35364.png",r="/vitePress-blob/assets/9.bf057657.png",v=JSON.parse('{"title":"Promise 相关","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/Promise.md","filePath":"guide/javaScript相关/Promise.md","lastUpdated":1736839070000}'),t={name:"guide/javaScript相关/Promise.md"},c=l(`

Promise 相关

定义

Promise 有 3 种状态:

  • 等待状态 pending
  • 成功状态 resolved
  • 失败状态 rejected

当 Promise 状态是 Pending 时,可以转换为 Resolved 或者 Rejected 状态,一旦状态转换,就不会再改变。

resolved、rejected 是异步函数

then 默认返回一个 Promise 对象,第一个参数是 resolved 的回调函数,第二个参数是 rejected 的回调函数 catch 会返回一个 Promise 对象,catch 正常返回 resolved 状态,里面报错返回 rejected 状态

Promise.resolve

Promise.resolve(value) 返回一个成功状态的 promise 对象

js
/*
   如果 .then 函数参数不是一个函数,那么会将其封装成 v => v 函数, 这里的 v 是上一个 resolve 的返回值
 */
 
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_Promise.md.54f111a9.lean.js" "b/assets/guide_javaScript\347\233\270\345\205\263_Promise.md.9dcd3d04.lean.js"
similarity index 90%
rename from "assets/guide_javaScript\347\233\270\345\205\263_Promise.md.54f111a9.lean.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_Promise.md.9dcd3d04.lean.js"
index 7fcc7840..64cdee4b 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_Promise.md.54f111a9.lean.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_Promise.md.9dcd3d04.lean.js"
@@ -1 +1 @@
-import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/10.5cd5f050.png",o="/vitePress-blob/assets/7.6a576ff8.png",e="/vitePress-blob/assets/8.56c35364.png",r="/vitePress-blob/assets/9.bf057657.png",v=JSON.parse('{"title":"Promise 相关","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/Promise.md","filePath":"guide/javaScript相关/Promise.md","lastUpdated":1736065650000}'),t={name:"guide/javaScript相关/Promise.md"},c=l("",34),E=[c];function y(i,d,F,h,m,P){return a(),n("div",null,E)}const A=s(t,[["render",y]]);export{v as __pageData,A as default};
+import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/10.5cd5f050.png",o="/vitePress-blob/assets/7.6a576ff8.png",e="/vitePress-blob/assets/8.56c35364.png",r="/vitePress-blob/assets/9.bf057657.png",v=JSON.parse('{"title":"Promise 相关","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/Promise.md","filePath":"guide/javaScript相关/Promise.md","lastUpdated":1736839070000}'),t={name:"guide/javaScript相关/Promise.md"},c=l("",34),E=[c];function y(i,d,F,h,m,P){return a(),n("div",null,E)}const A=s(t,[["render",y]]);export{v as __pageData,A as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_index.md.41467648.js" "b/assets/guide_javaScript\347\233\270\345\205\263_index.md.6fa036d3.js"
similarity index 88%
rename from "assets/guide_javaScript\347\233\270\345\205\263_index.md.41467648.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_index.md.6fa036d3.js"
index 1e5a76c1..701ea43e 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_index.md.41467648.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_index.md.6fa036d3.js"
@@ -1 +1 @@
-import{_ as e}from"./chunks/container.fad5294e.js";import{o as a,c as t,H as i}from"./chunks/framework.b6910bb2.js";const l=JSON.parse('{"title":"","description":"","frontmatter":{"aside":false,"outline":false},"headers":[],"relativePath":"guide/javaScript相关/index.md","filePath":"guide/javaScript相关/index.md","lastUpdated":1736065650000}'),r={name:"guide/javaScript相关/index.md"},m=Object.assign(r,{setup(s){return(c,d)=>(a(),t("div",null,[i(e,{url:"https://www.mubu.com/doc/2GzwQWqBS-K#m"})]))}});export{l as __pageData,m as default};
+import{_ as e}from"./chunks/container.fad5294e.js";import{o as a,c as t,H as i}from"./chunks/framework.b6910bb2.js";const l=JSON.parse('{"title":"","description":"","frontmatter":{"aside":false,"outline":false},"headers":[],"relativePath":"guide/javaScript相关/index.md","filePath":"guide/javaScript相关/index.md","lastUpdated":1736839070000}'),r={name:"guide/javaScript相关/index.md"},m=Object.assign(r,{setup(s){return(c,d)=>(a(),t("div",null,[i(e,{url:"https://www.mubu.com/doc/2GzwQWqBS-K#m"})]))}});export{l as __pageData,m as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_index.md.41467648.lean.js" "b/assets/guide_javaScript\347\233\270\345\205\263_index.md.6fa036d3.lean.js"
similarity index 88%
rename from "assets/guide_javaScript\347\233\270\345\205\263_index.md.41467648.lean.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_index.md.6fa036d3.lean.js"
index 1e5a76c1..701ea43e 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_index.md.41467648.lean.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_index.md.6fa036d3.lean.js"
@@ -1 +1 @@
-import{_ as e}from"./chunks/container.fad5294e.js";import{o as a,c as t,H as i}from"./chunks/framework.b6910bb2.js";const l=JSON.parse('{"title":"","description":"","frontmatter":{"aside":false,"outline":false},"headers":[],"relativePath":"guide/javaScript相关/index.md","filePath":"guide/javaScript相关/index.md","lastUpdated":1736065650000}'),r={name:"guide/javaScript相关/index.md"},m=Object.assign(r,{setup(s){return(c,d)=>(a(),t("div",null,[i(e,{url:"https://www.mubu.com/doc/2GzwQWqBS-K#m"})]))}});export{l as __pageData,m as default};
+import{_ as e}from"./chunks/container.fad5294e.js";import{o as a,c as t,H as i}from"./chunks/framework.b6910bb2.js";const l=JSON.parse('{"title":"","description":"","frontmatter":{"aside":false,"outline":false},"headers":[],"relativePath":"guide/javaScript相关/index.md","filePath":"guide/javaScript相关/index.md","lastUpdated":1736839070000}'),r={name:"guide/javaScript相关/index.md"},m=Object.assign(r,{setup(s){return(c,d)=>(a(),t("div",null,[i(e,{url:"https://www.mubu.com/doc/2GzwQWqBS-K#m"})]))}});export{l as __pageData,m as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_\344\272\213\344\273\266\345\276\252\347\216\257\346\234\272\345\210\266.md.0b3536a3.js" "b/assets/guide_javaScript\347\233\270\345\205\263_\344\272\213\344\273\266\345\276\252\347\216\257\346\234\272\345\210\266.md.d507e357.js"
similarity index 93%
rename from "assets/guide_javaScript\347\233\270\345\205\263_\344\272\213\344\273\266\345\276\252\347\216\257\346\234\272\345\210\266.md.0b3536a3.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_\344\272\213\344\273\266\345\276\252\347\216\257\346\234\272\345\210\266.md.d507e357.js"
index 62adf990..097b3515 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_\344\272\213\344\273\266\345\276\252\347\216\257\346\234\272\345\210\266.md.0b3536a3.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_\344\272\213\344\273\266\345\276\252\347\216\257\346\234\272\345\210\266.md.d507e357.js"
@@ -1 +1 @@
-import{_ as a,o as t,c as s,k as e,a as r}from"./chunks/framework.b6910bb2.js";const v=JSON.parse('{"title":"事件循环机制","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/事件循环机制.md","filePath":"guide/javaScript相关/事件循环机制.md","lastUpdated":1736065650000}'),o={name:"guide/javaScript相关/事件循环机制.md"},c=e("h1",{id:"事件循环机制",tabindex:"-1"},[r("事件循环机制 "),e("a",{class:"header-anchor",href:"#事件循环机制","aria-label":'Permalink to "事件循环机制"'},"​")],-1),d=e("p",null,"执行 js 代码时,会将代码分成同步代码和异步代码, 同步任务会进入主线程执行,异步任务又分成宏任务和微任务 宏任务会进入宏任务队列,微任务会进入微任务队列 当主线程的代码执行完后,会查看是否有微任务队列,有则执行,无则进入下一个宏任务",-1),i=[c,d];function n(_,l,p,h,m,f){return t(),s("div",null,i)}const x=a(o,[["render",n]]);export{v as __pageData,x as default};
+import{_ as a,o as t,c as s,k as e,a as r}from"./chunks/framework.b6910bb2.js";const v=JSON.parse('{"title":"事件循环机制","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/事件循环机制.md","filePath":"guide/javaScript相关/事件循环机制.md","lastUpdated":1736839070000}'),o={name:"guide/javaScript相关/事件循环机制.md"},c=e("h1",{id:"事件循环机制",tabindex:"-1"},[r("事件循环机制 "),e("a",{class:"header-anchor",href:"#事件循环机制","aria-label":'Permalink to "事件循环机制"'},"​")],-1),d=e("p",null,"执行 js 代码时,会将代码分成同步代码和异步代码, 同步任务会进入主线程执行,异步任务又分成宏任务和微任务 宏任务会进入宏任务队列,微任务会进入微任务队列 当主线程的代码执行完后,会查看是否有微任务队列,有则执行,无则进入下一个宏任务",-1),i=[c,d];function n(_,l,p,h,m,f){return t(),s("div",null,i)}const x=a(o,[["render",n]]);export{v as __pageData,x as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_\344\272\213\344\273\266\345\276\252\347\216\257\346\234\272\345\210\266.md.0b3536a3.lean.js" "b/assets/guide_javaScript\347\233\270\345\205\263_\344\272\213\344\273\266\345\276\252\347\216\257\346\234\272\345\210\266.md.d507e357.lean.js"
similarity index 93%
rename from "assets/guide_javaScript\347\233\270\345\205\263_\344\272\213\344\273\266\345\276\252\347\216\257\346\234\272\345\210\266.md.0b3536a3.lean.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_\344\272\213\344\273\266\345\276\252\347\216\257\346\234\272\345\210\266.md.d507e357.lean.js"
index 62adf990..097b3515 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_\344\272\213\344\273\266\345\276\252\347\216\257\346\234\272\345\210\266.md.0b3536a3.lean.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_\344\272\213\344\273\266\345\276\252\347\216\257\346\234\272\345\210\266.md.d507e357.lean.js"
@@ -1 +1 @@
-import{_ as a,o as t,c as s,k as e,a as r}from"./chunks/framework.b6910bb2.js";const v=JSON.parse('{"title":"事件循环机制","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/事件循环机制.md","filePath":"guide/javaScript相关/事件循环机制.md","lastUpdated":1736065650000}'),o={name:"guide/javaScript相关/事件循环机制.md"},c=e("h1",{id:"事件循环机制",tabindex:"-1"},[r("事件循环机制 "),e("a",{class:"header-anchor",href:"#事件循环机制","aria-label":'Permalink to "事件循环机制"'},"​")],-1),d=e("p",null,"执行 js 代码时,会将代码分成同步代码和异步代码, 同步任务会进入主线程执行,异步任务又分成宏任务和微任务 宏任务会进入宏任务队列,微任务会进入微任务队列 当主线程的代码执行完后,会查看是否有微任务队列,有则执行,无则进入下一个宏任务",-1),i=[c,d];function n(_,l,p,h,m,f){return t(),s("div",null,i)}const x=a(o,[["render",n]]);export{v as __pageData,x as default};
+import{_ as a,o as t,c as s,k as e,a as r}from"./chunks/framework.b6910bb2.js";const v=JSON.parse('{"title":"事件循环机制","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/事件循环机制.md","filePath":"guide/javaScript相关/事件循环机制.md","lastUpdated":1736839070000}'),o={name:"guide/javaScript相关/事件循环机制.md"},c=e("h1",{id:"事件循环机制",tabindex:"-1"},[r("事件循环机制 "),e("a",{class:"header-anchor",href:"#事件循环机制","aria-label":'Permalink to "事件循环机制"'},"​")],-1),d=e("p",null,"执行 js 代码时,会将代码分成同步代码和异步代码, 同步任务会进入主线程执行,异步任务又分成宏任务和微任务 宏任务会进入宏任务队列,微任务会进入微任务队列 当主线程的代码执行完后,会查看是否有微任务队列,有则执行,无则进入下一个宏任务",-1),i=[c,d];function n(_,l,p,h,m,f){return t(),s("div",null,i)}const x=a(o,[["render",n]]);export{v as __pageData,x as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_\345\236\203\345\234\276\345\233\236\346\224\266\346\234\272\345\210\266.md.7fe4a5e0.js" "b/assets/guide_javaScript\347\233\270\345\205\263_\345\236\203\345\234\276\345\233\236\346\224\266\346\234\272\345\210\266.md.41875062.js"
similarity index 95%
rename from "assets/guide_javaScript\347\233\270\345\205\263_\345\236\203\345\234\276\345\233\236\346\224\266\346\234\272\345\210\266.md.7fe4a5e0.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_\345\236\203\345\234\276\345\233\236\346\224\266\346\234\272\345\210\266.md.41875062.js"
index 5a209322..1ef894c4 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_\345\236\203\345\234\276\345\233\236\346\224\266\346\234\272\345\210\266.md.7fe4a5e0.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_\345\236\203\345\234\276\345\233\236\346\224\266\346\234\272\345\210\266.md.41875062.js"
@@ -1 +1 @@
-import{_ as t,o as a,c as s,k as e,a as o}from"./chunks/framework.b6910bb2.js";const S=JSON.parse('{"title":"垃圾回收机制","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/垃圾回收机制.md","filePath":"guide/javaScript相关/垃圾回收机制.md","lastUpdated":1736065650000}'),r={name:"guide/javaScript相关/垃圾回收机制.md"},c=e("h1",{id:"垃圾回收机制",tabindex:"-1"},[o("垃圾回收机制 "),e("a",{class:"header-anchor",href:"#垃圾回收机制","aria-label":'Permalink to "垃圾回收机制"'},"​")],-1),n=e("p",null,"在 JS 的垃圾回收机制中,分成新生代和老生代 新生代采用的是 GC 算法,将新生代划分成活动对象区域和空闲区域,例如声明的对象会被放入活动对象区域,当活动对象区域填满时, 会进行一次垃圾回收,将存活的对象移入空闲区域并进行碎片空间整理,完成后再将空闲区域和活动对象区域进行翻转,依次循环回收。",-1),_=e("p",null,"当进行 2次 垃圾回收后,存活的对象会被放入老生代中,",-1),d=e("p",null,"老生代采用的是标记清除和标记整理算法,递归遍历堆内存的变量,将不需要使用的变量标记为垃圾数据,将需要使用的变量标记成活动变量,清除掉垃圾,然后进行标记整理碎片空间。",-1),i=[c,n,_,d];function l(p,h,u,m,f,v){return a(),s("div",null,i)}const g=t(r,[["render",l]]);export{S as __pageData,g as default};
+import{_ as t,o as a,c as s,k as e,a as o}from"./chunks/framework.b6910bb2.js";const S=JSON.parse('{"title":"垃圾回收机制","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/垃圾回收机制.md","filePath":"guide/javaScript相关/垃圾回收机制.md","lastUpdated":1736839070000}'),r={name:"guide/javaScript相关/垃圾回收机制.md"},c=e("h1",{id:"垃圾回收机制",tabindex:"-1"},[o("垃圾回收机制 "),e("a",{class:"header-anchor",href:"#垃圾回收机制","aria-label":'Permalink to "垃圾回收机制"'},"​")],-1),n=e("p",null,"在 JS 的垃圾回收机制中,分成新生代和老生代 新生代采用的是 GC 算法,将新生代划分成活动对象区域和空闲区域,例如声明的对象会被放入活动对象区域,当活动对象区域填满时, 会进行一次垃圾回收,将存活的对象移入空闲区域并进行碎片空间整理,完成后再将空闲区域和活动对象区域进行翻转,依次循环回收。",-1),_=e("p",null,"当进行 2次 垃圾回收后,存活的对象会被放入老生代中,",-1),d=e("p",null,"老生代采用的是标记清除和标记整理算法,递归遍历堆内存的变量,将不需要使用的变量标记为垃圾数据,将需要使用的变量标记成活动变量,清除掉垃圾,然后进行标记整理碎片空间。",-1),i=[c,n,_,d];function l(p,h,u,m,f,v){return a(),s("div",null,i)}const g=t(r,[["render",l]]);export{S as __pageData,g as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_\345\236\203\345\234\276\345\233\236\346\224\266\346\234\272\345\210\266.md.7fe4a5e0.lean.js" "b/assets/guide_javaScript\347\233\270\345\205\263_\345\236\203\345\234\276\345\233\236\346\224\266\346\234\272\345\210\266.md.41875062.lean.js"
similarity index 95%
rename from "assets/guide_javaScript\347\233\270\345\205\263_\345\236\203\345\234\276\345\233\236\346\224\266\346\234\272\345\210\266.md.7fe4a5e0.lean.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_\345\236\203\345\234\276\345\233\236\346\224\266\346\234\272\345\210\266.md.41875062.lean.js"
index 5a209322..1ef894c4 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_\345\236\203\345\234\276\345\233\236\346\224\266\346\234\272\345\210\266.md.7fe4a5e0.lean.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_\345\236\203\345\234\276\345\233\236\346\224\266\346\234\272\345\210\266.md.41875062.lean.js"
@@ -1 +1 @@
-import{_ as t,o as a,c as s,k as e,a as o}from"./chunks/framework.b6910bb2.js";const S=JSON.parse('{"title":"垃圾回收机制","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/垃圾回收机制.md","filePath":"guide/javaScript相关/垃圾回收机制.md","lastUpdated":1736065650000}'),r={name:"guide/javaScript相关/垃圾回收机制.md"},c=e("h1",{id:"垃圾回收机制",tabindex:"-1"},[o("垃圾回收机制 "),e("a",{class:"header-anchor",href:"#垃圾回收机制","aria-label":'Permalink to "垃圾回收机制"'},"​")],-1),n=e("p",null,"在 JS 的垃圾回收机制中,分成新生代和老生代 新生代采用的是 GC 算法,将新生代划分成活动对象区域和空闲区域,例如声明的对象会被放入活动对象区域,当活动对象区域填满时, 会进行一次垃圾回收,将存活的对象移入空闲区域并进行碎片空间整理,完成后再将空闲区域和活动对象区域进行翻转,依次循环回收。",-1),_=e("p",null,"当进行 2次 垃圾回收后,存活的对象会被放入老生代中,",-1),d=e("p",null,"老生代采用的是标记清除和标记整理算法,递归遍历堆内存的变量,将不需要使用的变量标记为垃圾数据,将需要使用的变量标记成活动变量,清除掉垃圾,然后进行标记整理碎片空间。",-1),i=[c,n,_,d];function l(p,h,u,m,f,v){return a(),s("div",null,i)}const g=t(r,[["render",l]]);export{S as __pageData,g as default};
+import{_ as t,o as a,c as s,k as e,a as o}from"./chunks/framework.b6910bb2.js";const S=JSON.parse('{"title":"垃圾回收机制","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/垃圾回收机制.md","filePath":"guide/javaScript相关/垃圾回收机制.md","lastUpdated":1736839070000}'),r={name:"guide/javaScript相关/垃圾回收机制.md"},c=e("h1",{id:"垃圾回收机制",tabindex:"-1"},[o("垃圾回收机制 "),e("a",{class:"header-anchor",href:"#垃圾回收机制","aria-label":'Permalink to "垃圾回收机制"'},"​")],-1),n=e("p",null,"在 JS 的垃圾回收机制中,分成新生代和老生代 新生代采用的是 GC 算法,将新生代划分成活动对象区域和空闲区域,例如声明的对象会被放入活动对象区域,当活动对象区域填满时, 会进行一次垃圾回收,将存活的对象移入空闲区域并进行碎片空间整理,完成后再将空闲区域和活动对象区域进行翻转,依次循环回收。",-1),_=e("p",null,"当进行 2次 垃圾回收后,存活的对象会被放入老生代中,",-1),d=e("p",null,"老生代采用的是标记清除和标记整理算法,递归遍历堆内存的变量,将不需要使用的变量标记为垃圾数据,将需要使用的变量标记成活动变量,清除掉垃圾,然后进行标记整理碎片空间。",-1),i=[c,n,_,d];function l(p,h,u,m,f,v){return a(),s("div",null,i)}const g=t(r,[["render",l]]);export{S as __pageData,g as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_\345\237\272\347\241\200\346\246\202\345\277\265.md.3cedefb2.js" "b/assets/guide_javaScript\347\233\270\345\205\263_\345\237\272\347\241\200\346\246\202\345\277\265.md.53cfc405.js"
similarity index 99%
rename from "assets/guide_javaScript\347\233\270\345\205\263_\345\237\272\347\241\200\346\246\202\345\277\265.md.3cedefb2.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_\345\237\272\347\241\200\346\246\202\345\277\265.md.53cfc405.js"
index 4f8f807b..35228237 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_\345\237\272\347\241\200\346\246\202\345\277\265.md.3cedefb2.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_\345\237\272\347\241\200\346\246\202\345\277\265.md.53cfc405.js"
@@ -1,4 +1,4 @@
-import{_ as a,o as s,c as l,Q as n}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/4.8663f597.png",o="/vitePress-blob/assets/5.83f69d4f.png",_=JSON.parse('{"title":"3 基础概念","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/基础概念.md","filePath":"guide/javaScript相关/基础概念.md","lastUpdated":1736065650000}'),e={name:"guide/javaScript相关/基础概念.md"},t=n('

3 基础概念

3.1 闭包

执行外部函数返回内部函数后,虽然外部函数已经弹出调用栈了,但是内部函数对外部函数变量的引用依然保存在内存中,那么内部函数和这些变量的集合就叫做闭包。

闭包的好处: 1. 私有变量在内存中持久化

闭包的坏处: 1. 使用不当会造成内存泄漏

3.2 原型

在 js 中对象是由构造函数创建的,构造函数中会有一个 prototype 属性,指向一个对象,这个对象包含了由该构造函数创建的实例所共享的属性和方法, 由该构造函数创建的实例可以通过 proto 属性指向这个对象,这个对象就是我们所说的原型。

当想要访问对象的某个属性时,如果在当前对象查找不到,就会往该对象的原型查找,对象的原型也会有属于他的原型对象,如此循环,直到 null 停止,这就是我们所说的原型链

隐式原型: proto 显示原型: prototype

相关方法: 1. hasOwnProperty() 判断属性是否是实例自身的属性 2. Object.getPrototypeOf() 获取实例的原型

3.3 作用域

作用域分成全局作用域函数作用域块级作用域,它标识着一个变量是否合法 (编译过程中就已经确认了)

当查询一个变量时,如果当前作用域查询不到,会往上一级作用域查找,如此循环,直到全局作用域,这就是我们所说的作用域链

3.4 执行上下文

从类型上看

  1. 全局执行上下文
  2. 函数执行上下文
  3. eval 执行上下文

从生命周期上看

  1. 创建阶段
    • this 绑定
    • 创建词法环境 (let、const会被提升到词法环境)
    • 创建变量环境 (var 声明的变量会被提升到变量环境)
  2. 执行阶段
    • 对变量进行赋值,执行代码
  3. 回收阶段
    • 当执行上下文弹出调用站后,会对上下文进行回收

执行上下文栈: 当 JS 执行代码时,首先遇到全局代码,会创建一个全局执行上下文并压入执行栈中,当遇到函数调用时,就会为该函数创建一个新的函数执行上下文压入栈中, 引擎会执行位于执行上下文栈顶的函数,当函数执行完后,执行上下文会从栈中弹出,继续执行下一个上下文,当所有代码都执行完毕后,从栈中弹出全局执行上下文

执行上下文

3.5 类数组对象

一个拥有 length 属性和若干索引属性的对象就是类数组对象。

将类数组对象转化成数组的方法:

1 Array.from(arrayLike)
+import{_ as a,o as s,c as l,Q as n}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/4.8663f597.png",o="/vitePress-blob/assets/5.83f69d4f.png",_=JSON.parse('{"title":"3 基础概念","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/基础概念.md","filePath":"guide/javaScript相关/基础概念.md","lastUpdated":1736839070000}'),e={name:"guide/javaScript相关/基础概念.md"},t=n('

3 基础概念

3.1 闭包

执行外部函数返回内部函数后,虽然外部函数已经弹出调用栈了,但是内部函数对外部函数变量的引用依然保存在内存中,那么内部函数和这些变量的集合就叫做闭包。

闭包的好处: 1. 私有变量在内存中持久化

闭包的坏处: 1. 使用不当会造成内存泄漏

3.2 原型

在 js 中对象是由构造函数创建的,构造函数中会有一个 prototype 属性,指向一个对象,这个对象包含了由该构造函数创建的实例所共享的属性和方法, 由该构造函数创建的实例可以通过 proto 属性指向这个对象,这个对象就是我们所说的原型。

当想要访问对象的某个属性时,如果在当前对象查找不到,就会往该对象的原型查找,对象的原型也会有属于他的原型对象,如此循环,直到 null 停止,这就是我们所说的原型链

隐式原型: proto 显示原型: prototype

相关方法: 1. hasOwnProperty() 判断属性是否是实例自身的属性 2. Object.getPrototypeOf() 获取实例的原型

3.3 作用域

作用域分成全局作用域函数作用域块级作用域,它标识着一个变量是否合法 (编译过程中就已经确认了)

当查询一个变量时,如果当前作用域查询不到,会往上一级作用域查找,如此循环,直到全局作用域,这就是我们所说的作用域链

3.4 执行上下文

从类型上看

  1. 全局执行上下文
  2. 函数执行上下文
  3. eval 执行上下文

从生命周期上看

  1. 创建阶段
    • this 绑定
    • 创建词法环境 (let、const会被提升到词法环境)
    • 创建变量环境 (var 声明的变量会被提升到变量环境)
  2. 执行阶段
    • 对变量进行赋值,执行代码
  3. 回收阶段
    • 当执行上下文弹出调用站后,会对上下文进行回收

执行上下文栈: 当 JS 执行代码时,首先遇到全局代码,会创建一个全局执行上下文并压入执行栈中,当遇到函数调用时,就会为该函数创建一个新的函数执行上下文压入栈中, 引擎会执行位于执行上下文栈顶的函数,当函数执行完后,执行上下文会从栈中弹出,继续执行下一个上下文,当所有代码都执行完毕后,从栈中弹出全局执行上下文

执行上下文

3.5 类数组对象

一个拥有 length 属性和若干索引属性的对象就是类数组对象。

将类数组对象转化成数组的方法:

1 Array.from(arrayLike)
 
 2 Array.prototype.slice.call(arrayLike)
 
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_\345\237\272\347\241\200\346\246\202\345\277\265.md.3cedefb2.lean.js" "b/assets/guide_javaScript\347\233\270\345\205\263_\345\237\272\347\241\200\346\246\202\345\277\265.md.53cfc405.lean.js"
similarity index 88%
rename from "assets/guide_javaScript\347\233\270\345\205\263_\345\237\272\347\241\200\346\246\202\345\277\265.md.3cedefb2.lean.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_\345\237\272\347\241\200\346\246\202\345\277\265.md.53cfc405.lean.js"
index ccbc8155..9bf46233 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_\345\237\272\347\241\200\346\246\202\345\277\265.md.3cedefb2.lean.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_\345\237\272\347\241\200\346\246\202\345\277\265.md.53cfc405.lean.js"
@@ -1 +1 @@
-import{_ as a,o as s,c as l,Q as n}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/4.8663f597.png",o="/vitePress-blob/assets/5.83f69d4f.png",_=JSON.parse('{"title":"3 基础概念","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/基础概念.md","filePath":"guide/javaScript相关/基础概念.md","lastUpdated":1736065650000}'),e={name:"guide/javaScript相关/基础概念.md"},t=n("",63),r=[t];function c(i,y,E,h,d,u){return s(),l("div",null,r)}const b=a(e,[["render",c]]);export{_ as __pageData,b as default};
+import{_ as a,o as s,c as l,Q as n}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/4.8663f597.png",o="/vitePress-blob/assets/5.83f69d4f.png",_=JSON.parse('{"title":"3 基础概念","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/基础概念.md","filePath":"guide/javaScript相关/基础概念.md","lastUpdated":1736839070000}'),e={name:"guide/javaScript相关/基础概念.md"},t=n("",63),r=[t];function c(i,y,E,h,d,u){return s(),l("div",null,r)}const b=a(e,[["render",c]]);export{_ as __pageData,b as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_\345\255\227\347\254\246\344\270\262\345\270\270\350\247\201\347\232\204API.md.c4f598fd.js" "b/assets/guide_javaScript\347\233\270\345\205\263_\345\255\227\347\254\246\344\270\262\345\270\270\350\247\201\347\232\204API.md.699074e1.js"
similarity index 86%
rename from "assets/guide_javaScript\347\233\270\345\205\263_\345\255\227\347\254\246\344\270\262\345\270\270\350\247\201\347\232\204API.md.c4f598fd.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_\345\255\227\347\254\246\344\270\262\345\270\270\350\247\201\347\232\204API.md.699074e1.js"
index 2f173713..9f13fd21 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_\345\255\227\347\254\246\344\270\262\345\270\270\350\247\201\347\232\204API.md.c4f598fd.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_\345\255\227\347\254\246\344\270\262\345\270\270\350\247\201\347\232\204API.md.699074e1.js"
@@ -1 +1 @@
-import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/字符串常见的API.md","filePath":"guide/javaScript相关/字符串常见的API.md","lastUpdated":1736065650000}'),r={name:"guide/javaScript相关/字符串常见的API.md"};function c(o,s,_,i,d,n){return t(),a("div")}const f=e(r,[["render",c]]);export{m as __pageData,f as default};
+import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/字符串常见的API.md","filePath":"guide/javaScript相关/字符串常见的API.md","lastUpdated":1736839070000}'),r={name:"guide/javaScript相关/字符串常见的API.md"};function c(o,s,_,i,d,n){return t(),a("div")}const f=e(r,[["render",c]]);export{m as __pageData,f as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_\345\255\227\347\254\246\344\270\262\345\270\270\350\247\201\347\232\204API.md.c4f598fd.lean.js" "b/assets/guide_javaScript\347\233\270\345\205\263_\345\255\227\347\254\246\344\270\262\345\270\270\350\247\201\347\232\204API.md.699074e1.lean.js"
similarity index 86%
rename from "assets/guide_javaScript\347\233\270\345\205\263_\345\255\227\347\254\246\344\270\262\345\270\270\350\247\201\347\232\204API.md.c4f598fd.lean.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_\345\255\227\347\254\246\344\270\262\345\270\270\350\247\201\347\232\204API.md.699074e1.lean.js"
index 2f173713..9f13fd21 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_\345\255\227\347\254\246\344\270\262\345\270\270\350\247\201\347\232\204API.md.c4f598fd.lean.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_\345\255\227\347\254\246\344\270\262\345\270\270\350\247\201\347\232\204API.md.699074e1.lean.js"
@@ -1 +1 @@
-import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/字符串常见的API.md","filePath":"guide/javaScript相关/字符串常见的API.md","lastUpdated":1736065650000}'),r={name:"guide/javaScript相关/字符串常见的API.md"};function c(o,s,_,i,d,n){return t(),a("div")}const f=e(r,[["render",c]]);export{m as __pageData,f as default};
+import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/字符串常见的API.md","filePath":"guide/javaScript相关/字符串常见的API.md","lastUpdated":1736839070000}'),r={name:"guide/javaScript相关/字符串常见的API.md"};function c(o,s,_,i,d,n){return t(),a("div")}const f=e(r,[["render",c]]);export{m as __pageData,f as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_\345\256\232\344\271\211.md.05754f7e.js" "b/assets/guide_javaScript\347\233\270\345\205\263_\345\256\232\344\271\211.md.ea2dc8ec.js"
similarity index 94%
rename from "assets/guide_javaScript\347\233\270\345\205\263_\345\256\232\344\271\211.md.05754f7e.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_\345\256\232\344\271\211.md.ea2dc8ec.js"
index 6e73be61..46d3b5a4 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_\345\256\232\344\271\211.md.05754f7e.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_\345\256\232\344\271\211.md.ea2dc8ec.js"
@@ -1 +1 @@
-import{_ as a,o as s,c as o,k as t,a as e}from"./chunks/framework.b6910bb2.js";const c="/vitePress-blob/assets/2.192658da.png",k=JSON.parse('{"title":"1 定义","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/定义.md","filePath":"guide/javaScript相关/定义.md","lastUpdated":1736065650000}'),r={name:"guide/javaScript相关/定义.md"},i=t("h1",{id:"_1-定义",tabindex:"-1"},[e("1 定义 "),t("a",{class:"header-anchor",href:"#_1-定义","aria-label":'Permalink to "1 定义"'},"​")],-1),n=t("p",null,"JavaScript 是一门动态语言,可以不需要定义变量类型",-1),_=t("p",null,"JavaScript 是一门弱类型语言,存在隐式类型转化",-1),d=t("p",null,"JavaScript 是一门解释型语言,在运行程序时动态编译",-1),l=t("p",null,"解释型语言特点: 1 每次运行都需要将源代码转化成机器码并执行,效率低 2 只要平台提供相应的解析器,就可以运行源代码,跨平台",-1),p=t("p",null,[e("V8 执行一段代码的流程图 "),t("img",{src:c,alt:"流程"})],-1),h=[i,n,_,d,l,p];function u(m,f,v,S,g,x){return s(),o("div",null,h)}const J=a(r,[["render",u]]);export{k as __pageData,J as default};
+import{_ as a,o as s,c as o,k as t,a as e}from"./chunks/framework.b6910bb2.js";const c="/vitePress-blob/assets/2.192658da.png",k=JSON.parse('{"title":"1 定义","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/定义.md","filePath":"guide/javaScript相关/定义.md","lastUpdated":1736839070000}'),r={name:"guide/javaScript相关/定义.md"},i=t("h1",{id:"_1-定义",tabindex:"-1"},[e("1 定义 "),t("a",{class:"header-anchor",href:"#_1-定义","aria-label":'Permalink to "1 定义"'},"​")],-1),n=t("p",null,"JavaScript 是一门动态语言,可以不需要定义变量类型",-1),_=t("p",null,"JavaScript 是一门弱类型语言,存在隐式类型转化",-1),d=t("p",null,"JavaScript 是一门解释型语言,在运行程序时动态编译",-1),l=t("p",null,"解释型语言特点: 1 每次运行都需要将源代码转化成机器码并执行,效率低 2 只要平台提供相应的解析器,就可以运行源代码,跨平台",-1),p=t("p",null,[e("V8 执行一段代码的流程图 "),t("img",{src:c,alt:"流程"})],-1),h=[i,n,_,d,l,p];function u(m,f,v,S,g,x){return s(),o("div",null,h)}const J=a(r,[["render",u]]);export{k as __pageData,J as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_\345\256\232\344\271\211.md.05754f7e.lean.js" "b/assets/guide_javaScript\347\233\270\345\205\263_\345\256\232\344\271\211.md.ea2dc8ec.lean.js"
similarity index 94%
rename from "assets/guide_javaScript\347\233\270\345\205\263_\345\256\232\344\271\211.md.05754f7e.lean.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_\345\256\232\344\271\211.md.ea2dc8ec.lean.js"
index 6e73be61..46d3b5a4 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_\345\256\232\344\271\211.md.05754f7e.lean.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_\345\256\232\344\271\211.md.ea2dc8ec.lean.js"
@@ -1 +1 @@
-import{_ as a,o as s,c as o,k as t,a as e}from"./chunks/framework.b6910bb2.js";const c="/vitePress-blob/assets/2.192658da.png",k=JSON.parse('{"title":"1 定义","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/定义.md","filePath":"guide/javaScript相关/定义.md","lastUpdated":1736065650000}'),r={name:"guide/javaScript相关/定义.md"},i=t("h1",{id:"_1-定义",tabindex:"-1"},[e("1 定义 "),t("a",{class:"header-anchor",href:"#_1-定义","aria-label":'Permalink to "1 定义"'},"​")],-1),n=t("p",null,"JavaScript 是一门动态语言,可以不需要定义变量类型",-1),_=t("p",null,"JavaScript 是一门弱类型语言,存在隐式类型转化",-1),d=t("p",null,"JavaScript 是一门解释型语言,在运行程序时动态编译",-1),l=t("p",null,"解释型语言特点: 1 每次运行都需要将源代码转化成机器码并执行,效率低 2 只要平台提供相应的解析器,就可以运行源代码,跨平台",-1),p=t("p",null,[e("V8 执行一段代码的流程图 "),t("img",{src:c,alt:"流程"})],-1),h=[i,n,_,d,l,p];function u(m,f,v,S,g,x){return s(),o("div",null,h)}const J=a(r,[["render",u]]);export{k as __pageData,J as default};
+import{_ as a,o as s,c as o,k as t,a as e}from"./chunks/framework.b6910bb2.js";const c="/vitePress-blob/assets/2.192658da.png",k=JSON.parse('{"title":"1 定义","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/定义.md","filePath":"guide/javaScript相关/定义.md","lastUpdated":1736839070000}'),r={name:"guide/javaScript相关/定义.md"},i=t("h1",{id:"_1-定义",tabindex:"-1"},[e("1 定义 "),t("a",{class:"header-anchor",href:"#_1-定义","aria-label":'Permalink to "1 定义"'},"​")],-1),n=t("p",null,"JavaScript 是一门动态语言,可以不需要定义变量类型",-1),_=t("p",null,"JavaScript 是一门弱类型语言,存在隐式类型转化",-1),d=t("p",null,"JavaScript 是一门解释型语言,在运行程序时动态编译",-1),l=t("p",null,"解释型语言特点: 1 每次运行都需要将源代码转化成机器码并执行,效率低 2 只要平台提供相应的解析器,就可以运行源代码,跨平台",-1),p=t("p",null,[e("V8 执行一段代码的流程图 "),t("img",{src:c,alt:"流程"})],-1),h=[i,n,_,d,l,p];function u(m,f,v,S,g,x){return s(),o("div",null,h)}const J=a(r,[["render",u]]);export{k as __pageData,J as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_\345\257\271\350\261\241\345\270\270\350\247\201\347\232\204API.md.c8d02937.js" "b/assets/guide_javaScript\347\233\270\345\205\263_\345\257\271\350\261\241\345\270\270\350\247\201\347\232\204API.md.6057b79b.js"
similarity index 86%
rename from "assets/guide_javaScript\347\233\270\345\205\263_\345\257\271\350\261\241\345\270\270\350\247\201\347\232\204API.md.c8d02937.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_\345\257\271\350\261\241\345\270\270\350\247\201\347\232\204API.md.6057b79b.js"
index 488bd180..37b805fc 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_\345\257\271\350\261\241\345\270\270\350\247\201\347\232\204API.md.c8d02937.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_\345\257\271\350\261\241\345\270\270\350\247\201\347\232\204API.md.6057b79b.js"
@@ -1 +1 @@
-import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/对象常见的API.md","filePath":"guide/javaScript相关/对象常见的API.md","lastUpdated":1736065650000}'),r={name:"guide/javaScript相关/对象常见的API.md"};function c(o,s,i,_,d,n){return t(),a("div")}const f=e(r,[["render",c]]);export{m as __pageData,f as default};
+import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/对象常见的API.md","filePath":"guide/javaScript相关/对象常见的API.md","lastUpdated":1736839070000}'),r={name:"guide/javaScript相关/对象常见的API.md"};function c(o,s,i,_,d,n){return t(),a("div")}const f=e(r,[["render",c]]);export{m as __pageData,f as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_\345\257\271\350\261\241\345\270\270\350\247\201\347\232\204API.md.c8d02937.lean.js" "b/assets/guide_javaScript\347\233\270\345\205\263_\345\257\271\350\261\241\345\270\270\350\247\201\347\232\204API.md.6057b79b.lean.js"
similarity index 86%
rename from "assets/guide_javaScript\347\233\270\345\205\263_\345\257\271\350\261\241\345\270\270\350\247\201\347\232\204API.md.c8d02937.lean.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_\345\257\271\350\261\241\345\270\270\350\247\201\347\232\204API.md.6057b79b.lean.js"
index 488bd180..37b805fc 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_\345\257\271\350\261\241\345\270\270\350\247\201\347\232\204API.md.c8d02937.lean.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_\345\257\271\350\261\241\345\270\270\350\247\201\347\232\204API.md.6057b79b.lean.js"
@@ -1 +1 @@
-import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/对象常见的API.md","filePath":"guide/javaScript相关/对象常见的API.md","lastUpdated":1736065650000}'),r={name:"guide/javaScript相关/对象常见的API.md"};function c(o,s,i,_,d,n){return t(),a("div")}const f=e(r,[["render",c]]);export{m as __pageData,f as default};
+import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/对象常见的API.md","filePath":"guide/javaScript相关/对象常见的API.md","lastUpdated":1736839070000}'),r={name:"guide/javaScript相关/对象常见的API.md"};function c(o,s,i,_,d,n){return t(),a("div")}const f=e(r,[["render",c]]);export{m as __pageData,f as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\346\215\256\347\261\273\345\236\213.md.b73ffe80.js" "b/assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\346\215\256\347\261\273\345\236\213.md.77b639a5.js"
similarity index 99%
rename from "assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\346\215\256\347\261\273\345\236\213.md.b73ffe80.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\346\215\256\347\261\273\345\236\213.md.77b639a5.js"
index df62c491..38933f87 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\346\215\256\347\261\273\345\236\213.md.b73ffe80.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\346\215\256\347\261\273\345\236\213.md.77b639a5.js"
@@ -1,4 +1,4 @@
-import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/3.d5809f49.png",b=JSON.parse('{"title":"2 数据类型","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/数据类型.md","filePath":"guide/javaScript相关/数据类型.md","lastUpdated":1736065650000}'),o={name:"guide/javaScript相关/数据类型.md"},e=l(`

2 数据类型

2.1 数据类型

一共有 7 种基本数据类型

  • 1 Number - 基于 IEEE 754 标准实现,采用双精度 64 位二进制格式, -(2 ^ 63 - 1) ~ 2 ^ 63 - 1
  • 2 Boolean - 只有 true 和 false
  • 3 Undefined - 没有定义的值
  • 4 Null - 空值
  • 5 String - 字符串
  • 6 Symbol - 唯一不可修改的值
  • 7 BigInt - 大整数类型

一种引用类型

  • Object - 引用类型,存储在堆中

2.2 类型检测

2.2.1 typeof

判断基础数据类型,除了 null 也可以判断 function

js
typeof (function() {}) // function
+import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/3.d5809f49.png",b=JSON.parse('{"title":"2 数据类型","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/数据类型.md","filePath":"guide/javaScript相关/数据类型.md","lastUpdated":1736839070000}'),o={name:"guide/javaScript相关/数据类型.md"},e=l(`

2 数据类型

2.1 数据类型

一共有 7 种基本数据类型

  • 1 Number - 基于 IEEE 754 标准实现,采用双精度 64 位二进制格式, -(2 ^ 63 - 1) ~ 2 ^ 63 - 1
  • 2 Boolean - 只有 true 和 false
  • 3 Undefined - 没有定义的值
  • 4 Null - 空值
  • 5 String - 字符串
  • 6 Symbol - 唯一不可修改的值
  • 7 BigInt - 大整数类型

一种引用类型

  • Object - 引用类型,存储在堆中

2.2 类型检测

2.2.1 typeof

判断基础数据类型,除了 null 也可以判断 function

js
typeof (function() {}) // function
 typeof (() => {}) // function
typeof (function() {}) // function
 typeof (() => {}) // function

但数组、对象、null 都会返回 object

js
typeof [] // object
 typeof {} // object
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\346\215\256\347\261\273\345\236\213.md.b73ffe80.lean.js" "b/assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\346\215\256\347\261\273\345\236\213.md.77b639a5.lean.js"
similarity index 87%
rename from "assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\346\215\256\347\261\273\345\236\213.md.b73ffe80.lean.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\346\215\256\347\261\273\345\236\213.md.77b639a5.lean.js"
index 4928bb87..051a26fa 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\346\215\256\347\261\273\345\236\213.md.b73ffe80.lean.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\346\215\256\347\261\273\345\236\213.md.77b639a5.lean.js"
@@ -1 +1 @@
-import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/3.d5809f49.png",b=JSON.parse('{"title":"2 数据类型","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/数据类型.md","filePath":"guide/javaScript相关/数据类型.md","lastUpdated":1736065650000}'),o={name:"guide/javaScript相关/数据类型.md"},e=l("",59),t=[e];function c(r,y,E,i,d,u){return a(),n("div",null,t)}const g=s(o,[["render",c]]);export{b as __pageData,g as default};
+import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.b6910bb2.js";const p="/vitePress-blob/assets/3.d5809f49.png",b=JSON.parse('{"title":"2 数据类型","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/数据类型.md","filePath":"guide/javaScript相关/数据类型.md","lastUpdated":1736839070000}'),o={name:"guide/javaScript相关/数据类型.md"},e=l("",59),t=[e];function c(r,y,E,i,d,u){return a(),n("div",null,t)}const g=s(o,[["render",c]]);export{b as __pageData,g as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\347\273\204\345\270\270\350\247\201\347\232\204API.md.9011f496.js" "b/assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\347\273\204\345\270\270\350\247\201\347\232\204API.md.452ffdf7.js"
similarity index 86%
rename from "assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\347\273\204\345\270\270\350\247\201\347\232\204API.md.9011f496.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\347\273\204\345\270\270\350\247\201\347\232\204API.md.452ffdf7.js"
index d770b7f2..a776183a 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\347\273\204\345\270\270\350\247\201\347\232\204API.md.9011f496.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\347\273\204\345\270\270\350\247\201\347\232\204API.md.452ffdf7.js"
@@ -1 +1 @@
-import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/数组常见的API.md","filePath":"guide/javaScript相关/数组常见的API.md","lastUpdated":1736065650000}'),r={name:"guide/javaScript相关/数组常见的API.md"};function c(o,s,i,_,d,n){return t(),a("div")}const f=e(r,[["render",c]]);export{m as __pageData,f as default};
+import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/数组常见的API.md","filePath":"guide/javaScript相关/数组常见的API.md","lastUpdated":1736839070000}'),r={name:"guide/javaScript相关/数组常见的API.md"};function c(o,s,i,_,d,n){return t(),a("div")}const f=e(r,[["render",c]]);export{m as __pageData,f as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\347\273\204\345\270\270\350\247\201\347\232\204API.md.9011f496.lean.js" "b/assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\347\273\204\345\270\270\350\247\201\347\232\204API.md.452ffdf7.lean.js"
similarity index 86%
rename from "assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\347\273\204\345\270\270\350\247\201\347\232\204API.md.9011f496.lean.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\347\273\204\345\270\270\350\247\201\347\232\204API.md.452ffdf7.lean.js"
index d770b7f2..a776183a 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\347\273\204\345\270\270\350\247\201\347\232\204API.md.9011f496.lean.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_\346\225\260\347\273\204\345\270\270\350\247\201\347\232\204API.md.452ffdf7.lean.js"
@@ -1 +1 @@
-import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/数组常见的API.md","filePath":"guide/javaScript相关/数组常见的API.md","lastUpdated":1736065650000}'),r={name:"guide/javaScript相关/数组常见的API.md"};function c(o,s,i,_,d,n){return t(),a("div")}const f=e(r,[["render",c]]);export{m as __pageData,f as default};
+import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/数组常见的API.md","filePath":"guide/javaScript相关/数组常见的API.md","lastUpdated":1736839070000}'),r={name:"guide/javaScript相关/数组常见的API.md"};function c(o,s,i,_,d,n){return t(),a("div")}const f=e(r,[["render",c]]);export{m as __pageData,f as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_\346\226\260\347\232\204\350\277\220\347\256\227\347\254\246.md.c349c9ff.js" "b/assets/guide_javaScript\347\233\270\345\205\263_\346\226\260\347\232\204\350\277\220\347\256\227\347\254\246.md.c8bd4cf5.js"
similarity index 85%
rename from "assets/guide_javaScript\347\233\270\345\205\263_\346\226\260\347\232\204\350\277\220\347\256\227\347\254\246.md.c349c9ff.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_\346\226\260\347\232\204\350\277\220\347\256\227\347\254\246.md.c8bd4cf5.js"
index 3441fd8f..643c18b5 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_\346\226\260\347\232\204\350\277\220\347\256\227\347\254\246.md.c349c9ff.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_\346\226\260\347\232\204\350\277\220\347\256\227\347\254\246.md.c8bd4cf5.js"
@@ -1 +1 @@
-import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/新的运算符.md","filePath":"guide/javaScript相关/新的运算符.md","lastUpdated":1736065650000}'),r={name:"guide/javaScript相关/新的运算符.md"};function c(o,s,i,_,d,n){return t(),a("div")}const f=e(r,[["render",c]]);export{m as __pageData,f as default};
+import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/新的运算符.md","filePath":"guide/javaScript相关/新的运算符.md","lastUpdated":1736839070000}'),r={name:"guide/javaScript相关/新的运算符.md"};function c(o,s,i,_,d,n){return t(),a("div")}const f=e(r,[["render",c]]);export{m as __pageData,f as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_\346\226\260\347\232\204\350\277\220\347\256\227\347\254\246.md.c349c9ff.lean.js" "b/assets/guide_javaScript\347\233\270\345\205\263_\346\226\260\347\232\204\350\277\220\347\256\227\347\254\246.md.c8bd4cf5.lean.js"
similarity index 85%
rename from "assets/guide_javaScript\347\233\270\345\205\263_\346\226\260\347\232\204\350\277\220\347\256\227\347\254\246.md.c349c9ff.lean.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_\346\226\260\347\232\204\350\277\220\347\256\227\347\254\246.md.c8bd4cf5.lean.js"
index 3441fd8f..643c18b5 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_\346\226\260\347\232\204\350\277\220\347\256\227\347\254\246.md.c349c9ff.lean.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_\346\226\260\347\232\204\350\277\220\347\256\227\347\254\246.md.c8bd4cf5.lean.js"
@@ -1 +1 @@
-import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/新的运算符.md","filePath":"guide/javaScript相关/新的运算符.md","lastUpdated":1736065650000}'),r={name:"guide/javaScript相关/新的运算符.md"};function c(o,s,i,_,d,n){return t(),a("div")}const f=e(r,[["render",c]]);export{m as __pageData,f as default};
+import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/新的运算符.md","filePath":"guide/javaScript相关/新的运算符.md","lastUpdated":1736839070000}'),r={name:"guide/javaScript相关/新的运算符.md"};function c(o,s,i,_,d,n){return t(),a("div")}const f=e(r,[["render",c]]);export{m as __pageData,f as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_\346\255\243\345\210\231\350\241\250\350\276\276\345\274\217.md.e5d049d6.js" "b/assets/guide_javaScript\347\233\270\345\205\263_\346\255\243\345\210\231\350\241\250\350\276\276\345\274\217.md.df2c1874.js"
similarity index 85%
rename from "assets/guide_javaScript\347\233\270\345\205\263_\346\255\243\345\210\231\350\241\250\350\276\276\345\274\217.md.e5d049d6.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_\346\255\243\345\210\231\350\241\250\350\276\276\345\274\217.md.df2c1874.js"
index 313bd2be..81bad2a7 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_\346\255\243\345\210\231\350\241\250\350\276\276\345\274\217.md.e5d049d6.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_\346\255\243\345\210\231\350\241\250\350\276\276\345\274\217.md.df2c1874.js"
@@ -1 +1 @@
-import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/正则表达式.md","filePath":"guide/javaScript相关/正则表达式.md","lastUpdated":1736065650000}'),r={name:"guide/javaScript相关/正则表达式.md"};function c(o,s,i,_,d,n){return t(),a("div")}const f=e(r,[["render",c]]);export{m as __pageData,f as default};
+import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/正则表达式.md","filePath":"guide/javaScript相关/正则表达式.md","lastUpdated":1736839070000}'),r={name:"guide/javaScript相关/正则表达式.md"};function c(o,s,i,_,d,n){return t(),a("div")}const f=e(r,[["render",c]]);export{m as __pageData,f as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_\346\255\243\345\210\231\350\241\250\350\276\276\345\274\217.md.e5d049d6.lean.js" "b/assets/guide_javaScript\347\233\270\345\205\263_\346\255\243\345\210\231\350\241\250\350\276\276\345\274\217.md.df2c1874.lean.js"
similarity index 85%
rename from "assets/guide_javaScript\347\233\270\345\205\263_\346\255\243\345\210\231\350\241\250\350\276\276\345\274\217.md.e5d049d6.lean.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_\346\255\243\345\210\231\350\241\250\350\276\276\345\274\217.md.df2c1874.lean.js"
index 313bd2be..81bad2a7 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_\346\255\243\345\210\231\350\241\250\350\276\276\345\274\217.md.e5d049d6.lean.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_\346\255\243\345\210\231\350\241\250\350\276\276\345\274\217.md.df2c1874.lean.js"
@@ -1 +1 @@
-import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/正则表达式.md","filePath":"guide/javaScript相关/正则表达式.md","lastUpdated":1736065650000}'),r={name:"guide/javaScript相关/正则表达式.md"};function c(o,s,i,_,d,n){return t(),a("div")}const f=e(r,[["render",c]]);export{m as __pageData,f as default};
+import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/正则表达式.md","filePath":"guide/javaScript相关/正则表达式.md","lastUpdated":1736839070000}'),r={name:"guide/javaScript相关/正则表达式.md"};function c(o,s,i,_,d,n){return t(),a("div")}const f=e(r,[["render",c]]);export{m as __pageData,f as default};
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_\347\273\247\346\211\277\347\233\270\345\205\263.md.7058e93b.js" "b/assets/guide_javaScript\347\233\270\345\205\263_\347\273\247\346\211\277\347\233\270\345\205\263.md.86520b75.js"
similarity index 99%
rename from "assets/guide_javaScript\347\233\270\345\205\263_\347\273\247\346\211\277\347\233\270\345\205\263.md.7058e93b.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_\347\273\247\346\211\277\347\233\270\345\205\263.md.86520b75.js"
index cb396fc7..206e8181 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_\347\273\247\346\211\277\347\233\270\345\205\263.md.7058e93b.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_\347\273\247\346\211\277\347\233\270\345\205\263.md.86520b75.js"
@@ -1,4 +1,4 @@
-import{_ as s,o as n,c as a,Q as p}from"./chunks/framework.b6910bb2.js";const d=JSON.parse('{"title":"继承","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/继承相关.md","filePath":"guide/javaScript相关/继承相关.md","lastUpdated":1736065650000}'),l={name:"guide/javaScript相关/继承相关.md"},o=p(`

继承

原型链继承

子类的原型对象是父类的实例对象

优点:

  • 简单易懂 缺点:
  • 1 实例化子类时,没办法向父类的构造函数传值
  • 2 所有子类实例共享一个父类,当某个子类修改了父类的属性时,其他子类也会受到影响
js
function Parent() {
+import{_ as s,o as n,c as a,Q as p}from"./chunks/framework.b6910bb2.js";const d=JSON.parse('{"title":"继承","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/继承相关.md","filePath":"guide/javaScript相关/继承相关.md","lastUpdated":1736839070000}'),l={name:"guide/javaScript相关/继承相关.md"},o=p(`

继承

原型链继承

子类的原型对象是父类的实例对象

优点:

  • 简单易懂 缺点:
  • 1 实例化子类时,没办法向父类的构造函数传值
  • 2 所有子类实例共享一个父类,当某个子类修改了父类的属性时,其他子类也会受到影响
js
function Parent() {
     this.name = 'parent';
 }
 
diff --git "a/assets/guide_javaScript\347\233\270\345\205\263_\347\273\247\346\211\277\347\233\270\345\205\263.md.7058e93b.lean.js" "b/assets/guide_javaScript\347\233\270\345\205\263_\347\273\247\346\211\277\347\233\270\345\205\263.md.86520b75.lean.js"
similarity index 86%
rename from "assets/guide_javaScript\347\233\270\345\205\263_\347\273\247\346\211\277\347\233\270\345\205\263.md.7058e93b.lean.js"
rename to "assets/guide_javaScript\347\233\270\345\205\263_\347\273\247\346\211\277\347\233\270\345\205\263.md.86520b75.lean.js"
index dc48d74f..10c37f63 100644
--- "a/assets/guide_javaScript\347\233\270\345\205\263_\347\273\247\346\211\277\347\233\270\345\205\263.md.7058e93b.lean.js"
+++ "b/assets/guide_javaScript\347\233\270\345\205\263_\347\273\247\346\211\277\347\233\270\345\205\263.md.86520b75.lean.js"
@@ -1 +1 @@
-import{_ as s,o as n,c as a,Q as p}from"./chunks/framework.b6910bb2.js";const d=JSON.parse('{"title":"继承","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/继承相关.md","filePath":"guide/javaScript相关/继承相关.md","lastUpdated":1736065650000}'),l={name:"guide/javaScript相关/继承相关.md"},o=p("",31),e=[o];function t(c,r,E,y,i,F){return n(),a("div",null,e)}const h=s(l,[["render",t]]);export{d as __pageData,h as default};
+import{_ as s,o as n,c as a,Q as p}from"./chunks/framework.b6910bb2.js";const d=JSON.parse('{"title":"继承","description":"","frontmatter":{},"headers":[],"relativePath":"guide/javaScript相关/继承相关.md","filePath":"guide/javaScript相关/继承相关.md","lastUpdated":1736839070000}'),l={name:"guide/javaScript相关/继承相关.md"},o=p("",31),e=[o];function t(c,r,E,y,i,F){return n(),a("div",null,e)}const h=s(l,[["render",t]]);export{d as __pageData,h as default};
diff --git a/assets/guide_webpack_Loader.md.daa9d3ad.js b/assets/guide_webpack_Loader.md.8ee4eae1.js
similarity index 99%
rename from assets/guide_webpack_Loader.md.daa9d3ad.js
rename to assets/guide_webpack_Loader.md.8ee4eae1.js
index 5f5f0262..af352cc1 100644
--- a/assets/guide_webpack_Loader.md.daa9d3ad.js
+++ b/assets/guide_webpack_Loader.md.8ee4eae1.js
@@ -1,4 +1,4 @@
-import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/1.f8d8fb4d.png",p="/vitePress-blob/assets/2.ca1be426.png",g=JSON.parse('{"title":"Loader","description":"","frontmatter":{},"headers":[],"relativePath":"guide/webpack/Loader.md","filePath":"guide/webpack/Loader.md","lastUpdated":1736065650000}'),e={name:"guide/webpack/Loader.md"},c=l(`

Loader

loader 是一个加载器,让 webpack 拥有加载和解析非 JavaScript 文件的能力

类型

  1. pre - 前置
  2. normal - 普通
  3. inline - 行内
  4. post - 后置
js
// 定义在 require 请求内部的叫做行内 loader
+import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/1.f8d8fb4d.png",p="/vitePress-blob/assets/2.ca1be426.png",g=JSON.parse('{"title":"Loader","description":"","frontmatter":{},"headers":[],"relativePath":"guide/webpack/Loader.md","filePath":"guide/webpack/Loader.md","lastUpdated":1736839070000}'),e={name:"guide/webpack/Loader.md"},c=l(`

Loader

loader 是一个加载器,让 webpack 拥有加载和解析非 JavaScript 文件的能力

类型

  1. pre - 前置
  2. normal - 普通
  3. inline - 行内
  4. post - 后置
js
// 定义在 require 请求内部的叫做行内 loader
 const a = require('inline1-loader!inline2-loader!a.js');
// 定义在 require 请求内部的叫做行内 loader
 const a = require('inline1-loader!inline2-loader!a.js');
  • 如果是 ! 作为前缀,将禁用 normal loader,例如 !inline1-loader!inline2-loader!a.js
  • 如果是 !! 作为前缀,将禁用所有 loader,例如 !!inline1-loader!inline2-loader!a.js
  • 如果是 -! 作为前缀,将禁用所有 loader 但不包含 post loader,例如 -!inline1-loader!inline2-loader!a.js

pitch

一个 loader 中存在 normal (必须) 和 pitch (可选)

先从左往右执行 loader 的 pitch 方法,在从右往左执行 loader 的默认方法

js
function preLoader(source) {
   console.log("pre1");
diff --git a/assets/guide_webpack_Loader.md.daa9d3ad.lean.js b/assets/guide_webpack_Loader.md.8ee4eae1.lean.js
similarity index 87%
rename from assets/guide_webpack_Loader.md.daa9d3ad.lean.js
rename to assets/guide_webpack_Loader.md.8ee4eae1.lean.js
index 63f28589..7fc4256b 100644
--- a/assets/guide_webpack_Loader.md.daa9d3ad.lean.js
+++ b/assets/guide_webpack_Loader.md.8ee4eae1.lean.js
@@ -1 +1 @@
-import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/1.f8d8fb4d.png",p="/vitePress-blob/assets/2.ca1be426.png",g=JSON.parse('{"title":"Loader","description":"","frontmatter":{},"headers":[],"relativePath":"guide/webpack/Loader.md","filePath":"guide/webpack/Loader.md","lastUpdated":1736065650000}'),e={name:"guide/webpack/Loader.md"},c=l("",13),t=[c];function r(E,y,i,d,F,u){return n(),a("div",null,t)}const q=s(e,[["render",r]]);export{g as __pageData,q as default};
+import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/1.f8d8fb4d.png",p="/vitePress-blob/assets/2.ca1be426.png",g=JSON.parse('{"title":"Loader","description":"","frontmatter":{},"headers":[],"relativePath":"guide/webpack/Loader.md","filePath":"guide/webpack/Loader.md","lastUpdated":1736839070000}'),e={name:"guide/webpack/Loader.md"},c=l("",13),t=[c];function r(E,y,i,d,F,u){return n(),a("div",null,t)}const q=s(e,[["render",r]]);export{g as __pageData,q as default};
diff --git a/assets/guide_webpack_index.md.fcc0bdf4.js b/assets/guide_webpack_index.md.8c12b279.js
similarity index 94%
rename from assets/guide_webpack_index.md.fcc0bdf4.js
rename to assets/guide_webpack_index.md.8c12b279.js
index 9364d5bb..95d6cfa4 100644
--- a/assets/guide_webpack_index.md.fcc0bdf4.js
+++ b/assets/guide_webpack_index.md.8c12b279.js
@@ -1 +1 @@
-import{_ as t,o as a,c as r,k as e,a as o}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"参考文章","description":"","frontmatter":{},"headers":[],"relativePath":"guide/webpack/index.md","filePath":"guide/webpack/index.md","lastUpdated":1736065650000}'),i={name:"guide/webpack/index.md"},n=e("h1",{id:"参考文章",tabindex:"-1"},[o("参考文章 "),e("a",{class:"header-anchor",href:"#参考文章","aria-label":'Permalink to "参考文章"'},"​")],-1),s=e("ul",null,[e("li",null,[e("a",{href:"https://www.youtube.com/watch?v=Gc9-7PBqOC8&list=LLHK1mTHpwrUeYgF5gu-Kd4g",target:"_blank",rel:"noreferrer"},"https://www.youtube.com/watch?v=Gc9-7PBqOC8&list=LLHK1mTHpwrUeYgF5gu-Kd4g")]),e("li",null,[e("a",{href:"https://tsejx.github.io/webpack-guidebook/infra/implementation-principle/workflow",target:"_blank",rel:"noreferrer"},"https://tsejx.github.io/webpack-guidebook/infra/implementation-principle/workflow")])],-1),c=[n,s];function l(d,p,u,h,w,_){return a(),r("div",null,c)}const g=t(i,[["render",l]]);export{f as __pageData,g as default};
+import{_ as t,o as a,c as r,k as e,a as o}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"参考文章","description":"","frontmatter":{},"headers":[],"relativePath":"guide/webpack/index.md","filePath":"guide/webpack/index.md","lastUpdated":1736839070000}'),i={name:"guide/webpack/index.md"},n=e("h1",{id:"参考文章",tabindex:"-1"},[o("参考文章 "),e("a",{class:"header-anchor",href:"#参考文章","aria-label":'Permalink to "参考文章"'},"​")],-1),s=e("ul",null,[e("li",null,[e("a",{href:"https://www.youtube.com/watch?v=Gc9-7PBqOC8&list=LLHK1mTHpwrUeYgF5gu-Kd4g",target:"_blank",rel:"noreferrer"},"https://www.youtube.com/watch?v=Gc9-7PBqOC8&list=LLHK1mTHpwrUeYgF5gu-Kd4g")]),e("li",null,[e("a",{href:"https://tsejx.github.io/webpack-guidebook/infra/implementation-principle/workflow",target:"_blank",rel:"noreferrer"},"https://tsejx.github.io/webpack-guidebook/infra/implementation-principle/workflow")])],-1),c=[n,s];function l(d,p,u,h,w,_){return a(),r("div",null,c)}const g=t(i,[["render",l]]);export{f as __pageData,g as default};
diff --git a/assets/guide_webpack_index.md.fcc0bdf4.lean.js b/assets/guide_webpack_index.md.8c12b279.lean.js
similarity index 94%
rename from assets/guide_webpack_index.md.fcc0bdf4.lean.js
rename to assets/guide_webpack_index.md.8c12b279.lean.js
index 9364d5bb..95d6cfa4 100644
--- a/assets/guide_webpack_index.md.fcc0bdf4.lean.js
+++ b/assets/guide_webpack_index.md.8c12b279.lean.js
@@ -1 +1 @@
-import{_ as t,o as a,c as r,k as e,a as o}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"参考文章","description":"","frontmatter":{},"headers":[],"relativePath":"guide/webpack/index.md","filePath":"guide/webpack/index.md","lastUpdated":1736065650000}'),i={name:"guide/webpack/index.md"},n=e("h1",{id:"参考文章",tabindex:"-1"},[o("参考文章 "),e("a",{class:"header-anchor",href:"#参考文章","aria-label":'Permalink to "参考文章"'},"​")],-1),s=e("ul",null,[e("li",null,[e("a",{href:"https://www.youtube.com/watch?v=Gc9-7PBqOC8&list=LLHK1mTHpwrUeYgF5gu-Kd4g",target:"_blank",rel:"noreferrer"},"https://www.youtube.com/watch?v=Gc9-7PBqOC8&list=LLHK1mTHpwrUeYgF5gu-Kd4g")]),e("li",null,[e("a",{href:"https://tsejx.github.io/webpack-guidebook/infra/implementation-principle/workflow",target:"_blank",rel:"noreferrer"},"https://tsejx.github.io/webpack-guidebook/infra/implementation-principle/workflow")])],-1),c=[n,s];function l(d,p,u,h,w,_){return a(),r("div",null,c)}const g=t(i,[["render",l]]);export{f as __pageData,g as default};
+import{_ as t,o as a,c as r,k as e,a as o}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"参考文章","description":"","frontmatter":{},"headers":[],"relativePath":"guide/webpack/index.md","filePath":"guide/webpack/index.md","lastUpdated":1736839070000}'),i={name:"guide/webpack/index.md"},n=e("h1",{id:"参考文章",tabindex:"-1"},[o("参考文章 "),e("a",{class:"header-anchor",href:"#参考文章","aria-label":'Permalink to "参考文章"'},"​")],-1),s=e("ul",null,[e("li",null,[e("a",{href:"https://www.youtube.com/watch?v=Gc9-7PBqOC8&list=LLHK1mTHpwrUeYgF5gu-Kd4g",target:"_blank",rel:"noreferrer"},"https://www.youtube.com/watch?v=Gc9-7PBqOC8&list=LLHK1mTHpwrUeYgF5gu-Kd4g")]),e("li",null,[e("a",{href:"https://tsejx.github.io/webpack-guidebook/infra/implementation-principle/workflow",target:"_blank",rel:"noreferrer"},"https://tsejx.github.io/webpack-guidebook/infra/implementation-principle/workflow")])],-1),c=[n,s];function l(d,p,u,h,w,_){return a(),r("div",null,c)}const g=t(i,[["render",l]]);export{f as __pageData,g as default};
diff --git a/assets/guide_webpack_mini-webpack.md.938949bc.js b/assets/guide_webpack_mini-webpack.md.7a56cd34.js
similarity index 91%
rename from assets/guide_webpack_mini-webpack.md.938949bc.js
rename to assets/guide_webpack_mini-webpack.md.7a56cd34.js
index 6a44d95a..a75694f4 100644
--- a/assets/guide_webpack_mini-webpack.md.938949bc.js
+++ b/assets/guide_webpack_mini-webpack.md.7a56cd34.js
@@ -1 +1 @@
-import{_ as a,o as i,c as t,k as e,a as n}from"./chunks/framework.b6910bb2.js";const w=JSON.parse('{"title":"mini-webpack","description":"","frontmatter":{},"headers":[],"relativePath":"guide/webpack/mini-webpack.md","filePath":"guide/webpack/mini-webpack.md","lastUpdated":1736065650000}'),c={name:"guide/webpack/mini-webpack.md"},r=e("h1",{id:"mini-webpack",tabindex:"-1"},[n("mini-webpack "),e("a",{class:"header-anchor",href:"#mini-webpack","aria-label":'Permalink to "mini-webpack"'},"​")],-1),o=e("p",null,[e("a",{href:"https://github.com/enson0131/mini-webpack",target:"_blank",rel:"noreferrer"},"mini-webpack 仓库")],-1),s=[r,o];function p(d,m,k,b,l,_){return i(),t("div",null,s)}const f=a(c,[["render",p]]);export{w as __pageData,f as default};
+import{_ as a,o as i,c as t,k as e,a as n}from"./chunks/framework.b6910bb2.js";const w=JSON.parse('{"title":"mini-webpack","description":"","frontmatter":{},"headers":[],"relativePath":"guide/webpack/mini-webpack.md","filePath":"guide/webpack/mini-webpack.md","lastUpdated":1736839070000}'),c={name:"guide/webpack/mini-webpack.md"},r=e("h1",{id:"mini-webpack",tabindex:"-1"},[n("mini-webpack "),e("a",{class:"header-anchor",href:"#mini-webpack","aria-label":'Permalink to "mini-webpack"'},"​")],-1),o=e("p",null,[e("a",{href:"https://github.com/enson0131/mini-webpack",target:"_blank",rel:"noreferrer"},"mini-webpack 仓库")],-1),s=[r,o];function p(d,m,k,b,l,_){return i(),t("div",null,s)}const f=a(c,[["render",p]]);export{w as __pageData,f as default};
diff --git a/assets/guide_webpack_mini-webpack.md.938949bc.lean.js b/assets/guide_webpack_mini-webpack.md.7a56cd34.lean.js
similarity index 91%
rename from assets/guide_webpack_mini-webpack.md.938949bc.lean.js
rename to assets/guide_webpack_mini-webpack.md.7a56cd34.lean.js
index 6a44d95a..a75694f4 100644
--- a/assets/guide_webpack_mini-webpack.md.938949bc.lean.js
+++ b/assets/guide_webpack_mini-webpack.md.7a56cd34.lean.js
@@ -1 +1 @@
-import{_ as a,o as i,c as t,k as e,a as n}from"./chunks/framework.b6910bb2.js";const w=JSON.parse('{"title":"mini-webpack","description":"","frontmatter":{},"headers":[],"relativePath":"guide/webpack/mini-webpack.md","filePath":"guide/webpack/mini-webpack.md","lastUpdated":1736065650000}'),c={name:"guide/webpack/mini-webpack.md"},r=e("h1",{id:"mini-webpack",tabindex:"-1"},[n("mini-webpack "),e("a",{class:"header-anchor",href:"#mini-webpack","aria-label":'Permalink to "mini-webpack"'},"​")],-1),o=e("p",null,[e("a",{href:"https://github.com/enson0131/mini-webpack",target:"_blank",rel:"noreferrer"},"mini-webpack 仓库")],-1),s=[r,o];function p(d,m,k,b,l,_){return i(),t("div",null,s)}const f=a(c,[["render",p]]);export{w as __pageData,f as default};
+import{_ as a,o as i,c as t,k as e,a as n}from"./chunks/framework.b6910bb2.js";const w=JSON.parse('{"title":"mini-webpack","description":"","frontmatter":{},"headers":[],"relativePath":"guide/webpack/mini-webpack.md","filePath":"guide/webpack/mini-webpack.md","lastUpdated":1736839070000}'),c={name:"guide/webpack/mini-webpack.md"},r=e("h1",{id:"mini-webpack",tabindex:"-1"},[n("mini-webpack "),e("a",{class:"header-anchor",href:"#mini-webpack","aria-label":'Permalink to "mini-webpack"'},"​")],-1),o=e("p",null,[e("a",{href:"https://github.com/enson0131/mini-webpack",target:"_blank",rel:"noreferrer"},"mini-webpack 仓库")],-1),s=[r,o];function p(d,m,k,b,l,_){return i(),t("div",null,s)}const f=a(c,[["render",p]]);export{w as __pageData,f as default};
diff --git "a/assets/guide_webpack_\346\236\204\345\273\272\346\265\201\347\250\213.md.e14ee84d.js" "b/assets/guide_webpack_\346\236\204\345\273\272\346\265\201\347\250\213.md.61545129.js"
similarity index 99%
rename from "assets/guide_webpack_\346\236\204\345\273\272\346\265\201\347\250\213.md.e14ee84d.js"
rename to "assets/guide_webpack_\346\236\204\345\273\272\346\265\201\347\250\213.md.61545129.js"
index 22331206..4c7438d3 100644
--- "a/assets/guide_webpack_\346\236\204\345\273\272\346\265\201\347\250\213.md.e14ee84d.js"
+++ "b/assets/guide_webpack_\346\236\204\345\273\272\346\265\201\347\250\213.md.61545129.js"
@@ -1,4 +1,4 @@
-import{_ as a,o as s,c as l,Q as n}from"./chunks/framework.b6910bb2.js";const y=JSON.parse('{"title":"构建流程","description":"","frontmatter":{},"headers":[],"relativePath":"guide/webpack/构建流程.md","filePath":"guide/webpack/构建流程.md","lastUpdated":1736065650000}'),e={name:"guide/webpack/构建流程.md"},p=n(`

构建流程

工作流程

webpack 的工作流程是一个串行的过程,从启动到结束会依次执行以下流程:

  1. 初始化参数:从配置文件和 Shell 语句中读取与合并参数,得出最终的参数;
  2. 开始编译:用上一步得到的参数初始化 Compiler 对象,加载所有配置的插件,执行对象的 run 方法开始执行编译;
  3. 确定入口:根据配置中的 entry 确定入口文件;
  4. 编译模块:从入口文件出发,调用所有配置的 Loader 对模块进行编译,再找出该模块依赖的模块,得到了每个模块被编译后的 最终内容 以及他们之间的 依赖关系图
  5. 输出资源:根据入口和模块之间的依赖关系,组装成一个个包含多个模块的 Chunk,再把每个 Chunk 转换成一个单独的文件加入到输出列表,这步是可以修改输出内容的最后机会;
  6. 输出完成:在确定好输出内容后,根据配置确定输出的路径和文件名,把文件内容写入到文件系统。

常见的问题

Module、Chunk、Bundle 的区别

module: 模块,一个模块就是一个文件

chunk: 代码块,一个 chunk 可以由多个模块组合而成

bundle: webpack 打包出来的文件

Loader 和 Plugin

loader 是一个加载器,让 webpack 拥有加载和解析非 JavaScript 文件的能力

Plugin 是一个插件,是对 webpack 功能的扩展,让 webpack 具有更高的灵活性

Compiler 和 Compilation 的区别

Compiler 对象在 webpack 启动时候被实例化,可以访问当前运行的 webpack 配置,包括 entry、output、loader 等配置,它是全局唯一的。

Plugin 的 apply 方法会传入一个 Compiler 对象,通过这个 Compiler 对象可以注册各种钩子函数,执行插件任务,也可以通过该对象获取配置信息。

js
Compiler.plugin('emit', function(compilation, callback) {
+import{_ as a,o as s,c as l,Q as n}from"./chunks/framework.b6910bb2.js";const y=JSON.parse('{"title":"构建流程","description":"","frontmatter":{},"headers":[],"relativePath":"guide/webpack/构建流程.md","filePath":"guide/webpack/构建流程.md","lastUpdated":1736839070000}'),e={name:"guide/webpack/构建流程.md"},p=n(`

构建流程

工作流程

webpack 的工作流程是一个串行的过程,从启动到结束会依次执行以下流程:

  1. 初始化参数:从配置文件和 Shell 语句中读取与合并参数,得出最终的参数;
  2. 开始编译:用上一步得到的参数初始化 Compiler 对象,加载所有配置的插件,执行对象的 run 方法开始执行编译;
  3. 确定入口:根据配置中的 entry 确定入口文件;
  4. 编译模块:从入口文件出发,调用所有配置的 Loader 对模块进行编译,再找出该模块依赖的模块,得到了每个模块被编译后的 最终内容 以及他们之间的 依赖关系图
  5. 输出资源:根据入口和模块之间的依赖关系,组装成一个个包含多个模块的 Chunk,再把每个 Chunk 转换成一个单独的文件加入到输出列表,这步是可以修改输出内容的最后机会;
  6. 输出完成:在确定好输出内容后,根据配置确定输出的路径和文件名,把文件内容写入到文件系统。

常见的问题

Module、Chunk、Bundle 的区别

module: 模块,一个模块就是一个文件

chunk: 代码块,一个 chunk 可以由多个模块组合而成

bundle: webpack 打包出来的文件

Loader 和 Plugin

loader 是一个加载器,让 webpack 拥有加载和解析非 JavaScript 文件的能力

Plugin 是一个插件,是对 webpack 功能的扩展,让 webpack 具有更高的灵活性

Compiler 和 Compilation 的区别

Compiler 对象在 webpack 启动时候被实例化,可以访问当前运行的 webpack 配置,包括 entry、output、loader 等配置,它是全局唯一的。

Plugin 的 apply 方法会传入一个 Compiler 对象,通过这个 Compiler 对象可以注册各种钩子函数,执行插件任务,也可以通过该对象获取配置信息。

js
Compiler.plugin('emit', function(compilation, callback) {
     // emit 是异步 hook, 在生成之前输出到目录之前调用
  });
Compiler.plugin('emit', function(compilation, callback) {
     // emit 是异步 hook, 在生成之前输出到目录之前调用
diff --git "a/assets/guide_webpack_\346\236\204\345\273\272\346\265\201\347\250\213.md.e14ee84d.lean.js" "b/assets/guide_webpack_\346\236\204\345\273\272\346\265\201\347\250\213.md.61545129.lean.js"
similarity index 86%
rename from "assets/guide_webpack_\346\236\204\345\273\272\346\265\201\347\250\213.md.e14ee84d.lean.js"
rename to "assets/guide_webpack_\346\236\204\345\273\272\346\265\201\347\250\213.md.61545129.lean.js"
index 78f632d6..c2405e09 100644
--- "a/assets/guide_webpack_\346\236\204\345\273\272\346\265\201\347\250\213.md.e14ee84d.lean.js"
+++ "b/assets/guide_webpack_\346\236\204\345\273\272\346\265\201\347\250\213.md.61545129.lean.js"
@@ -1 +1 @@
-import{_ as a,o as s,c as l,Q as n}from"./chunks/framework.b6910bb2.js";const y=JSON.parse('{"title":"构建流程","description":"","frontmatter":{},"headers":[],"relativePath":"guide/webpack/构建流程.md","filePath":"guide/webpack/构建流程.md","lastUpdated":1736065650000}'),e={name:"guide/webpack/构建流程.md"},p=n("",34),o=[p];function c(t,i,r,d,E,u){return s(),l("div",null,o)}const k=a(e,[["render",c]]);export{y as __pageData,k as default};
+import{_ as a,o as s,c as l,Q as n}from"./chunks/framework.b6910bb2.js";const y=JSON.parse('{"title":"构建流程","description":"","frontmatter":{},"headers":[],"relativePath":"guide/webpack/构建流程.md","filePath":"guide/webpack/构建流程.md","lastUpdated":1736839070000}'),e={name:"guide/webpack/构建流程.md"},p=n("",34),o=[p];function c(t,i,r,d,E,u){return s(),l("div",null,o)}const k=a(e,[["render",c]]);export{y as __pageData,k as default};
diff --git "a/assets/guide_webpack_\346\250\241\345\235\227\350\201\224\351\202\246.md.9176a16d.js" "b/assets/guide_webpack_\346\250\241\345\235\227\350\201\224\351\202\246.md.a4b55cfe.js"
similarity index 99%
rename from "assets/guide_webpack_\346\250\241\345\235\227\350\201\224\351\202\246.md.9176a16d.js"
rename to "assets/guide_webpack_\346\250\241\345\235\227\350\201\224\351\202\246.md.a4b55cfe.js"
index 8fb631ae..3e62bb5e 100644
--- "a/assets/guide_webpack_\346\250\241\345\235\227\350\201\224\351\202\246.md.9176a16d.js"
+++ "b/assets/guide_webpack_\346\250\241\345\235\227\350\201\224\351\202\246.md.a4b55cfe.js"
@@ -1,4 +1,4 @@
-import{_ as a,o as l,c as p,Q as n,k as s}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/3.6834884f.png",e="/vitePress-blob/assets/4.ca1b5ab5.png",t="/vitePress-blob/assets/5.ccaac5d9.png",B=JSON.parse('{"title":"模块联邦","description":"","frontmatter":{},"headers":[],"relativePath":"guide/webpack/模块联邦.md","filePath":"guide/webpack/模块联邦.md","lastUpdated":1736065650000}'),c={name:"guide/webpack/模块联邦.md"},r=n('

模块联邦

目的

更好的复用应用块或者库

每个应用块都是独立构建,被称之为容器

Module Federation

一个被引用的容器被称之为 remote

引用者被称之为 host

配置参数

',8),E=s("table",null,[s("thead",null,[s("tr",null,[s("th",null,"字段"),s("th",null,"类型"),s("th",null,"含义")])]),s("tbody",null,[s("tr",{expose:""},[s("td",null,"name"),s("td",null,"string"),s("td",null,"必传值,即输出的模块名,被远程引用时的路径为 ${name}/$")]),s("tr",null,[s("td",null,"library"),s("td",null,"object"),s("td",null,"声明全局变量的方式,name为 umd 的name")]),s("tr",null,[s("td",null,"filename"),s("td",null,"string"),s("td",null,"构建输出的文件名称")]),s("tr",null,[s("td",null,"remotes"),s("td",null,"object"),s("td",null,"(host 使用)远程引用的应用名及其别名的映射,使用时以 key 值作为 name")]),s("tr",null,[s("td",null,"exposes"),s("td",null,"object"),s("td",null,"(remote 使用)被引用时可暴露的资源路径以其别名")]),s("tr",null,[s("td",null,"shared"),s("td",null,"object"),s("td",null,"与其他应用之间可以共享的第三方依赖,使你的代码中不用重复加载同一份依赖")])])],-1),y=n(`

remote 配置

js
// webpack.config.js
+import{_ as a,o as l,c as p,Q as n,k as s}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/3.6834884f.png",e="/vitePress-blob/assets/4.ca1b5ab5.png",t="/vitePress-blob/assets/5.ccaac5d9.png",B=JSON.parse('{"title":"模块联邦","description":"","frontmatter":{},"headers":[],"relativePath":"guide/webpack/模块联邦.md","filePath":"guide/webpack/模块联邦.md","lastUpdated":1736839070000}'),c={name:"guide/webpack/模块联邦.md"},r=n('

模块联邦

目的

更好的复用应用块或者库

每个应用块都是独立构建,被称之为容器

Module Federation

一个被引用的容器被称之为 remote

引用者被称之为 host

配置参数

',8),E=s("table",null,[s("thead",null,[s("tr",null,[s("th",null,"字段"),s("th",null,"类型"),s("th",null,"含义")])]),s("tbody",null,[s("tr",{expose:""},[s("td",null,"name"),s("td",null,"string"),s("td",null,"必传值,即输出的模块名,被远程引用时的路径为 ${name}/$")]),s("tr",null,[s("td",null,"library"),s("td",null,"object"),s("td",null,"声明全局变量的方式,name为 umd 的name")]),s("tr",null,[s("td",null,"filename"),s("td",null,"string"),s("td",null,"构建输出的文件名称")]),s("tr",null,[s("td",null,"remotes"),s("td",null,"object"),s("td",null,"(host 使用)远程引用的应用名及其别名的映射,使用时以 key 值作为 name")]),s("tr",null,[s("td",null,"exposes"),s("td",null,"object"),s("td",null,"(remote 使用)被引用时可暴露的资源路径以其别名")]),s("tr",null,[s("td",null,"shared"),s("td",null,"object"),s("td",null,"与其他应用之间可以共享的第三方依赖,使你的代码中不用重复加载同一份依赖")])])],-1),y=n(`

remote 配置

js
// webpack.config.js
 const path = require("path");
 const webpack = require("webpack");
 const HtmlWebpackPlugin = require("html-webpack-plugin");
diff --git "a/assets/guide_webpack_\346\250\241\345\235\227\350\201\224\351\202\246.md.9176a16d.lean.js" "b/assets/guide_webpack_\346\250\241\345\235\227\350\201\224\351\202\246.md.a4b55cfe.lean.js"
similarity index 96%
rename from "assets/guide_webpack_\346\250\241\345\235\227\350\201\224\351\202\246.md.9176a16d.lean.js"
rename to "assets/guide_webpack_\346\250\241\345\235\227\350\201\224\351\202\246.md.a4b55cfe.lean.js"
index 3eb7fa27..5d3fa5d7 100644
--- "a/assets/guide_webpack_\346\250\241\345\235\227\350\201\224\351\202\246.md.9176a16d.lean.js"
+++ "b/assets/guide_webpack_\346\250\241\345\235\227\350\201\224\351\202\246.md.a4b55cfe.lean.js"
@@ -1 +1 @@
-import{_ as a,o as l,c as p,Q as n,k as s}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/3.6834884f.png",e="/vitePress-blob/assets/4.ca1b5ab5.png",t="/vitePress-blob/assets/5.ccaac5d9.png",B=JSON.parse('{"title":"模块联邦","description":"","frontmatter":{},"headers":[],"relativePath":"guide/webpack/模块联邦.md","filePath":"guide/webpack/模块联邦.md","lastUpdated":1736065650000}'),c={name:"guide/webpack/模块联邦.md"},r=n("",8),E=s("table",null,[s("thead",null,[s("tr",null,[s("th",null,"字段"),s("th",null,"类型"),s("th",null,"含义")])]),s("tbody",null,[s("tr",{expose:""},[s("td",null,"name"),s("td",null,"string"),s("td",null,"必传值,即输出的模块名,被远程引用时的路径为 ${name}/$")]),s("tr",null,[s("td",null,"library"),s("td",null,"object"),s("td",null,"声明全局变量的方式,name为 umd 的name")]),s("tr",null,[s("td",null,"filename"),s("td",null,"string"),s("td",null,"构建输出的文件名称")]),s("tr",null,[s("td",null,"remotes"),s("td",null,"object"),s("td",null,"(host 使用)远程引用的应用名及其别名的映射,使用时以 key 值作为 name")]),s("tr",null,[s("td",null,"exposes"),s("td",null,"object"),s("td",null,"(remote 使用)被引用时可暴露的资源路径以其别名")]),s("tr",null,[s("td",null,"shared"),s("td",null,"object"),s("td",null,"与其他应用之间可以共享的第三方依赖,使你的代码中不用重复加载同一份依赖")])])],-1),y=n("",15),i=[r,E,y];function u(F,d,q,h,m,b){return l(),p("div",null,i)}const g=a(c,[["render",u]]);export{B as __pageData,g as default};
+import{_ as a,o as l,c as p,Q as n,k as s}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/3.6834884f.png",e="/vitePress-blob/assets/4.ca1b5ab5.png",t="/vitePress-blob/assets/5.ccaac5d9.png",B=JSON.parse('{"title":"模块联邦","description":"","frontmatter":{},"headers":[],"relativePath":"guide/webpack/模块联邦.md","filePath":"guide/webpack/模块联邦.md","lastUpdated":1736839070000}'),c={name:"guide/webpack/模块联邦.md"},r=n("",8),E=s("table",null,[s("thead",null,[s("tr",null,[s("th",null,"字段"),s("th",null,"类型"),s("th",null,"含义")])]),s("tbody",null,[s("tr",{expose:""},[s("td",null,"name"),s("td",null,"string"),s("td",null,"必传值,即输出的模块名,被远程引用时的路径为 ${name}/$")]),s("tr",null,[s("td",null,"library"),s("td",null,"object"),s("td",null,"声明全局变量的方式,name为 umd 的name")]),s("tr",null,[s("td",null,"filename"),s("td",null,"string"),s("td",null,"构建输出的文件名称")]),s("tr",null,[s("td",null,"remotes"),s("td",null,"object"),s("td",null,"(host 使用)远程引用的应用名及其别名的映射,使用时以 key 值作为 name")]),s("tr",null,[s("td",null,"exposes"),s("td",null,"object"),s("td",null,"(remote 使用)被引用时可暴露的资源路径以其别名")]),s("tr",null,[s("td",null,"shared"),s("td",null,"object"),s("td",null,"与其他应用之间可以共享的第三方依赖,使你的代码中不用重复加载同一份依赖")])])],-1),y=n("",15),i=[r,E,y];function u(F,d,q,h,m,b){return l(),p("div",null,i)}const g=a(c,[["render",u]]);export{B as __pageData,g as default};
diff --git "a/assets/guide_webpack_\347\203\255\346\233\264\346\226\260\345\216\237\347\220\206.md.f90caa71.js" "b/assets/guide_webpack_\347\203\255\346\233\264\346\226\260\345\216\237\347\220\206.md.87ba4ad8.js"
similarity index 90%
rename from "assets/guide_webpack_\347\203\255\346\233\264\346\226\260\345\216\237\347\220\206.md.f90caa71.js"
rename to "assets/guide_webpack_\347\203\255\346\233\264\346\226\260\345\216\237\347\220\206.md.87ba4ad8.js"
index e1ef9624..6eea14e0 100644
--- "a/assets/guide_webpack_\347\203\255\346\233\264\346\226\260\345\216\237\347\220\206.md.f90caa71.js"
+++ "b/assets/guide_webpack_\347\203\255\346\233\264\346\226\260\345\216\237\347\220\206.md.87ba4ad8.js"
@@ -1 +1 @@
-import{_ as a,o as t,c as s,k as e,a as o}from"./chunks/framework.b6910bb2.js";const u=JSON.parse('{"title":"热更新原理","description":"","frontmatter":{},"headers":[],"relativePath":"guide/webpack/热更新原理.md","filePath":"guide/webpack/热更新原理.md","lastUpdated":1736065650000}'),r={name:"guide/webpack/热更新原理.md"},c=e("h1",{id:"热更新原理",tabindex:"-1"},[o("热更新原理 "),e("a",{class:"header-anchor",href:"#热更新原理","aria-label":'Permalink to "热更新原理"'},"​")],-1),d=[c];function n(_,i,p,l,h,m){return t(),s("div",null,d)}const k=a(r,[["render",n]]);export{u as __pageData,k as default};
+import{_ as a,o as t,c as s,k as e,a as o}from"./chunks/framework.b6910bb2.js";const u=JSON.parse('{"title":"热更新原理","description":"","frontmatter":{},"headers":[],"relativePath":"guide/webpack/热更新原理.md","filePath":"guide/webpack/热更新原理.md","lastUpdated":1736839070000}'),r={name:"guide/webpack/热更新原理.md"},c=e("h1",{id:"热更新原理",tabindex:"-1"},[o("热更新原理 "),e("a",{class:"header-anchor",href:"#热更新原理","aria-label":'Permalink to "热更新原理"'},"​")],-1),d=[c];function n(_,i,p,l,h,m){return t(),s("div",null,d)}const k=a(r,[["render",n]]);export{u as __pageData,k as default};
diff --git "a/assets/guide_webpack_\347\203\255\346\233\264\346\226\260\345\216\237\347\220\206.md.f90caa71.lean.js" "b/assets/guide_webpack_\347\203\255\346\233\264\346\226\260\345\216\237\347\220\206.md.87ba4ad8.lean.js"
similarity index 90%
rename from "assets/guide_webpack_\347\203\255\346\233\264\346\226\260\345\216\237\347\220\206.md.f90caa71.lean.js"
rename to "assets/guide_webpack_\347\203\255\346\233\264\346\226\260\345\216\237\347\220\206.md.87ba4ad8.lean.js"
index e1ef9624..6eea14e0 100644
--- "a/assets/guide_webpack_\347\203\255\346\233\264\346\226\260\345\216\237\347\220\206.md.f90caa71.lean.js"
+++ "b/assets/guide_webpack_\347\203\255\346\233\264\346\226\260\345\216\237\347\220\206.md.87ba4ad8.lean.js"
@@ -1 +1 @@
-import{_ as a,o as t,c as s,k as e,a as o}from"./chunks/framework.b6910bb2.js";const u=JSON.parse('{"title":"热更新原理","description":"","frontmatter":{},"headers":[],"relativePath":"guide/webpack/热更新原理.md","filePath":"guide/webpack/热更新原理.md","lastUpdated":1736065650000}'),r={name:"guide/webpack/热更新原理.md"},c=e("h1",{id:"热更新原理",tabindex:"-1"},[o("热更新原理 "),e("a",{class:"header-anchor",href:"#热更新原理","aria-label":'Permalink to "热更新原理"'},"​")],-1),d=[c];function n(_,i,p,l,h,m){return t(),s("div",null,d)}const k=a(r,[["render",n]]);export{u as __pageData,k as default};
+import{_ as a,o as t,c as s,k as e,a as o}from"./chunks/framework.b6910bb2.js";const u=JSON.parse('{"title":"热更新原理","description":"","frontmatter":{},"headers":[],"relativePath":"guide/webpack/热更新原理.md","filePath":"guide/webpack/热更新原理.md","lastUpdated":1736839070000}'),r={name:"guide/webpack/热更新原理.md"},c=e("h1",{id:"热更新原理",tabindex:"-1"},[o("热更新原理 "),e("a",{class:"header-anchor",href:"#热更新原理","aria-label":'Permalink to "热更新原理"'},"​")],-1),d=[c];function n(_,i,p,l,h,m){return t(),s("div",null,d)}const k=a(r,[["render",n]]);export{u as __pageData,k as default};
diff --git "a/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\246\202\350\246\201.md.053a8f89.js" "b/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\246\202\350\246\201.md.8b5cc56c.js"
similarity index 88%
rename from "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\246\202\350\246\201.md.053a8f89.js"
rename to "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\246\202\350\246\201.md.8b5cc56c.js"
index 2add6cc8..4bb00ed1 100644
--- "a/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\246\202\350\246\201.md.053a8f89.js"
+++ "b/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\246\202\350\246\201.md.8b5cc56c.js"
@@ -1 +1 @@
-import{_ as e}from"./chunks/container.fad5294e.js";import{o as t,c as a,H as s}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{"aside":false,"outline":false},"headers":[],"relativePath":"guide/浏览器相关/概要.md","filePath":"guide/浏览器相关/概要.md","lastUpdated":1736065650000}'),o={name:"guide/浏览器相关/概要.md"},n=Object.assign(o,{setup(r){return(c,d)=>(t(),a("div",null,[s(e,{url:"https://www.mubu.com/doc/-ip51T7dbe#m"})]))}});export{m as __pageData,n as default};
+import{_ as e}from"./chunks/container.fad5294e.js";import{o as t,c as a,H as s}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{"aside":false,"outline":false},"headers":[],"relativePath":"guide/浏览器相关/概要.md","filePath":"guide/浏览器相关/概要.md","lastUpdated":1736839070000}'),o={name:"guide/浏览器相关/概要.md"},n=Object.assign(o,{setup(r){return(c,d)=>(t(),a("div",null,[s(e,{url:"https://www.mubu.com/doc/-ip51T7dbe#m"})]))}});export{m as __pageData,n as default};
diff --git "a/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\246\202\350\246\201.md.053a8f89.lean.js" "b/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\246\202\350\246\201.md.8b5cc56c.lean.js"
similarity index 88%
rename from "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\246\202\350\246\201.md.053a8f89.lean.js"
rename to "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\246\202\350\246\201.md.8b5cc56c.lean.js"
index 2add6cc8..4bb00ed1 100644
--- "a/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\246\202\350\246\201.md.053a8f89.lean.js"
+++ "b/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\246\202\350\246\201.md.8b5cc56c.lean.js"
@@ -1 +1 @@
-import{_ as e}from"./chunks/container.fad5294e.js";import{o as t,c as a,H as s}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{"aside":false,"outline":false},"headers":[],"relativePath":"guide/浏览器相关/概要.md","filePath":"guide/浏览器相关/概要.md","lastUpdated":1736065650000}'),o={name:"guide/浏览器相关/概要.md"},n=Object.assign(o,{setup(r){return(c,d)=>(t(),a("div",null,[s(e,{url:"https://www.mubu.com/doc/-ip51T7dbe#m"})]))}});export{m as __pageData,n as default};
+import{_ as e}from"./chunks/container.fad5294e.js";import{o as t,c as a,H as s}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{"aside":false,"outline":false},"headers":[],"relativePath":"guide/浏览器相关/概要.md","filePath":"guide/浏览器相关/概要.md","lastUpdated":1736839070000}'),o={name:"guide/浏览器相关/概要.md"},n=Object.assign(o,{setup(r){return(c,d)=>(t(),a("div",null,[s(e,{url:"https://www.mubu.com/doc/-ip51T7dbe#m"})]))}});export{m as __pageData,n as default};
diff --git "a/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\206\205\346\240\270.md.8232bb8e.js" "b/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\206\205\346\240\270.md.6fdfde81.js"
similarity index 92%
rename from "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\206\205\346\240\270.md.8232bb8e.js"
rename to "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\206\205\346\240\270.md.6fdfde81.js"
index 81903fea..ccc5d66f 100644
--- "a/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\206\205\346\240\270.md.8232bb8e.js"
+++ "b/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\206\205\346\240\270.md.6fdfde81.js"
@@ -1 +1 @@
-import{_ as t,o as a,c as i,k as e,a as l}from"./chunks/framework.b6910bb2.js";const k=JSON.parse('{"title":"浏览器内核","description":"","frontmatter":{},"headers":[],"relativePath":"guide/浏览器相关/浏览器内核.md","filePath":"guide/浏览器相关/浏览器内核.md","lastUpdated":1736065650000}'),n={name:"guide/浏览器相关/浏览器内核.md"},o=e("h1",{id:"浏览器内核",tabindex:"-1"},[l("浏览器内核 "),e("a",{class:"header-anchor",href:"#浏览器内核","aria-label":'Permalink to "浏览器内核"'},"​")],-1),r=e("ul",null,[e("li",null,"Trident (IE)"),e("li",null,"Gecko (Firefox)"),e("li",null,"Webkit (Safari)"),e("li",null,"Blink (Chrome、Opera) Blink 是 Webkit 的一个分支")],-1),s=[o,r];function d(c,_,u,p,h,f){return a(),i("div",null,s)}const x=t(n,[["render",d]]);export{k as __pageData,x as default};
+import{_ as t,o as a,c as i,k as e,a as l}from"./chunks/framework.b6910bb2.js";const k=JSON.parse('{"title":"浏览器内核","description":"","frontmatter":{},"headers":[],"relativePath":"guide/浏览器相关/浏览器内核.md","filePath":"guide/浏览器相关/浏览器内核.md","lastUpdated":1736839070000}'),n={name:"guide/浏览器相关/浏览器内核.md"},o=e("h1",{id:"浏览器内核",tabindex:"-1"},[l("浏览器内核 "),e("a",{class:"header-anchor",href:"#浏览器内核","aria-label":'Permalink to "浏览器内核"'},"​")],-1),r=e("ul",null,[e("li",null,"Trident (IE)"),e("li",null,"Gecko (Firefox)"),e("li",null,"Webkit (Safari)"),e("li",null,"Blink (Chrome、Opera) Blink 是 Webkit 的一个分支")],-1),s=[o,r];function d(c,_,u,p,h,f){return a(),i("div",null,s)}const x=t(n,[["render",d]]);export{k as __pageData,x as default};
diff --git "a/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\206\205\346\240\270.md.8232bb8e.lean.js" "b/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\206\205\346\240\270.md.6fdfde81.lean.js"
similarity index 92%
rename from "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\206\205\346\240\270.md.8232bb8e.lean.js"
rename to "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\206\205\346\240\270.md.6fdfde81.lean.js"
index 81903fea..ccc5d66f 100644
--- "a/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\206\205\346\240\270.md.8232bb8e.lean.js"
+++ "b/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\206\205\346\240\270.md.6fdfde81.lean.js"
@@ -1 +1 @@
-import{_ as t,o as a,c as i,k as e,a as l}from"./chunks/framework.b6910bb2.js";const k=JSON.parse('{"title":"浏览器内核","description":"","frontmatter":{},"headers":[],"relativePath":"guide/浏览器相关/浏览器内核.md","filePath":"guide/浏览器相关/浏览器内核.md","lastUpdated":1736065650000}'),n={name:"guide/浏览器相关/浏览器内核.md"},o=e("h1",{id:"浏览器内核",tabindex:"-1"},[l("浏览器内核 "),e("a",{class:"header-anchor",href:"#浏览器内核","aria-label":'Permalink to "浏览器内核"'},"​")],-1),r=e("ul",null,[e("li",null,"Trident (IE)"),e("li",null,"Gecko (Firefox)"),e("li",null,"Webkit (Safari)"),e("li",null,"Blink (Chrome、Opera) Blink 是 Webkit 的一个分支")],-1),s=[o,r];function d(c,_,u,p,h,f){return a(),i("div",null,s)}const x=t(n,[["render",d]]);export{k as __pageData,x as default};
+import{_ as t,o as a,c as i,k as e,a as l}from"./chunks/framework.b6910bb2.js";const k=JSON.parse('{"title":"浏览器内核","description":"","frontmatter":{},"headers":[],"relativePath":"guide/浏览器相关/浏览器内核.md","filePath":"guide/浏览器相关/浏览器内核.md","lastUpdated":1736839070000}'),n={name:"guide/浏览器相关/浏览器内核.md"},o=e("h1",{id:"浏览器内核",tabindex:"-1"},[l("浏览器内核 "),e("a",{class:"header-anchor",href:"#浏览器内核","aria-label":'Permalink to "浏览器内核"'},"​")],-1),r=e("ul",null,[e("li",null,"Trident (IE)"),e("li",null,"Gecko (Firefox)"),e("li",null,"Webkit (Safari)"),e("li",null,"Blink (Chrome、Opera) Blink 是 Webkit 的一个分支")],-1),s=[o,r];function d(c,_,u,p,h,f){return a(),i("div",null,s)}const x=t(n,[["render",d]]);export{k as __pageData,x as default};
diff --git "a/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\256\211\345\205\250.md.73ccefca.js" "b/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\256\211\345\205\250.md.ff5d442d.js"
similarity index 99%
rename from "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\256\211\345\205\250.md.73ccefca.js"
rename to "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\256\211\345\205\250.md.ff5d442d.js"
index 1852ca8f..84cf02d3 100644
--- "a/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\256\211\345\205\250.md.73ccefca.js"
+++ "b/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\256\211\345\205\250.md.ff5d442d.js"
@@ -1 +1 @@
-import{_ as l,o as i,c as e,Q as a}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/6.36673f2f.jpg",t="/vitePress-blob/assets/7.d0aefc74.jpg",r="/vitePress-blob/assets/8.28d1dcd9.jpg",s="/vitePress-blob/assets/9.b32cc031.jpg",P=JSON.parse('{"title":"浏览器安全","description":"","frontmatter":{},"headers":[],"relativePath":"guide/浏览器相关/浏览器安全.md","filePath":"guide/浏览器相关/浏览器安全.md","lastUpdated":1736065650000}'),n={name:"guide/浏览器相关/浏览器安全.md"},c=a('

浏览器安全

页面安全

同源策略

协议、域名、端口号都相同,才是同源。

限制

  1. 无法进行 DOM 操作
  2. 无法获取 Cookie、LocalStorage 等数据
  3. 无法向不同源发请求

开放

  1. 默认页面中可以引用任意第三方资源

    引入 CSP (内容安全策略)

    渲染流程图

  2. 只能请求同源的接口

    引入 CORS (跨域资源共享)

    • 简单请求 (HEAD/POST/GET请求)

      1. 浏览器在请求头上添加 Oirgin 字段,该字段用来说明请求来自那个源,服务器可以根据这个值决定是否同意这次请求
      2. 当服务器收到请求后,根据 Origin 判断是否在许可范围内
      3. 如果不在范围内,服务器会返回一个正常的 HTTP 响应,浏览器发现该响应头没有包含 Access-Control-Allow-Origin 字段,就会抛出一个错误,被 XMLHttpRequest 的 onerror 回调函数捕获 (由于正常响应,其状态码是 200,所以该错误不能通过状态码识别)
      4. 如果 Origin 在指定范围内,服务器返回的响应会多几个头信息字段
        • Access-Control-Allow-Origin(必须) 判断源是否可以跨域,该值要么是请求的 Origin,要么是 *

        • Access-Control-Allow-Credentials 是否允许发送 Cookie,默认为 false 如果 Access-Control-Allow-Origin 的值为 *,则无法发送 Cookie

        • Access-Control-Expose-Headers 制定其他的头信息字段可以暴露给客户端 因为默认情况下,只有 6 个字段可以被暴露,其他的都会被过滤掉

          简单请求

    • 非简单请求 (PUT/DELETE 或者 Content-type字段类型是 application/json)

      1. 浏览器发起 Option 预检请求
      2. 服务器收到预检请求后,检查
        • Origin 发起请求的源信息
        • Access-Control-Request-Method 浏览器会发起请求的方法
        • Access-Control-Request-Headers 请求额外发送的头信息字段
      3. 如果服务器否定预检请求,会返回一个正常的 HTTP 响应,但是没有任何的 CORS 相关的头信息字段,这是浏览器会认定服务器不同意预检,触发错误
      4. 如果服务器同意预检请求,会返回一个正常的 HTTP 响应,但是会多几个 CORS 相关的头信息字段
        • Access-Control-Allow-Origin (必须)
        • Access-Control-Allow-Methods (必须)
        • Access-Control-Allow-Credentials
        • Access-Control-Expose-Headers
        • Access-Control-Max-Age - 预检请求的有效期, 单位秒
        • Access-Control-Allow-Headers

      非简单请求

  3. 只能操作同源的DOM

    引入了跨文档消息机制 postMessage

XSS 攻击

跨域脚本攻击。

类型

  1. 存储型

    脚本存储在服务器,用户访问时,脚本从服务器返回,浏览器执行脚本

  2. 反射型

    在文章链接参数上存放了一段恶意脚本,用户访问时,脚本从参数中取出,浏览器执行脚本

  3. DOM 型

    比如在搜索的时候,添加了一段脚本, 如果网站的搜索功能没有对用户输入进行适当的过滤和转义,搜索结果页面可能会将恶意的脚本代码直接插入到 HTML 中,并在用户浏览器中执行

    web 资源在传输或者用户在使用的过程中修改了 Web 页面的数据

预防

  1. 对输入的内容进行截取或者转义(encode)
  2. 限制输入的长度和类型
  3. 引入内容安全策略 (CSP)
  4. 对 Cookie 设置 HttpOnly 属性

CSRF 攻击

跨站请求伪造。

比如诱导用户点击链接,访问网址后,网址通过用户的登录信息伪造请求

必要条件

  1. 站点有 CSRF 漏洞
  2. 用户登录了
  3. 用户点击了诱导链接

预防

  1. 给 Cookie 新增 SameSite 属性
    • Strict: 第三方站点无法发送 Cookie
    • Lax (默认): 第三方站点如果是 GET 请求可以发送 Cookie
    • None 不做限制 Cookie的书写
  2. 通过请求的 Origin or Referer 字段判断请求合法性
  3. 在请求地址中添加 token 并验证

浏览器安全

将渲染进程放在沙箱中

  • 无法持久存储
  • 无法访问网络
  • 无法监听用户交互
  • 需要通过 IPC 与浏览器主进程通信,由浏览器进程处理后,再将结果传递给渲染进程

网络安全

HTTPS、重放攻击、验签

  • 基本属性:

    • name
    • value
  • 访问性:

    • expire
    • path
    • domain
  • 安全性:

    • secure
    • httpOnly
    • sameSite
',28),p=[c];function h(d,u,_,b,C,m){return i(),e("div",null,p)}const f=l(n,[["render",h]]);export{P as __pageData,f as default}; +import{_ as l,o as i,c as e,Q as a}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/6.36673f2f.jpg",t="/vitePress-blob/assets/7.d0aefc74.jpg",r="/vitePress-blob/assets/8.28d1dcd9.jpg",s="/vitePress-blob/assets/9.b32cc031.jpg",P=JSON.parse('{"title":"浏览器安全","description":"","frontmatter":{},"headers":[],"relativePath":"guide/浏览器相关/浏览器安全.md","filePath":"guide/浏览器相关/浏览器安全.md","lastUpdated":1736839070000}'),n={name:"guide/浏览器相关/浏览器安全.md"},c=a('

浏览器安全

页面安全

同源策略

协议、域名、端口号都相同,才是同源。

限制

  1. 无法进行 DOM 操作
  2. 无法获取 Cookie、LocalStorage 等数据
  3. 无法向不同源发请求

开放

  1. 默认页面中可以引用任意第三方资源

    引入 CSP (内容安全策略)

    渲染流程图

  2. 只能请求同源的接口

    引入 CORS (跨域资源共享)

    • 简单请求 (HEAD/POST/GET请求)

      1. 浏览器在请求头上添加 Oirgin 字段,该字段用来说明请求来自那个源,服务器可以根据这个值决定是否同意这次请求
      2. 当服务器收到请求后,根据 Origin 判断是否在许可范围内
      3. 如果不在范围内,服务器会返回一个正常的 HTTP 响应,浏览器发现该响应头没有包含 Access-Control-Allow-Origin 字段,就会抛出一个错误,被 XMLHttpRequest 的 onerror 回调函数捕获 (由于正常响应,其状态码是 200,所以该错误不能通过状态码识别)
      4. 如果 Origin 在指定范围内,服务器返回的响应会多几个头信息字段
        • Access-Control-Allow-Origin(必须) 判断源是否可以跨域,该值要么是请求的 Origin,要么是 *

        • Access-Control-Allow-Credentials 是否允许发送 Cookie,默认为 false 如果 Access-Control-Allow-Origin 的值为 *,则无法发送 Cookie

        • Access-Control-Expose-Headers 制定其他的头信息字段可以暴露给客户端 因为默认情况下,只有 6 个字段可以被暴露,其他的都会被过滤掉

          简单请求

    • 非简单请求 (PUT/DELETE 或者 Content-type字段类型是 application/json)

      1. 浏览器发起 Option 预检请求
      2. 服务器收到预检请求后,检查
        • Origin 发起请求的源信息
        • Access-Control-Request-Method 浏览器会发起请求的方法
        • Access-Control-Request-Headers 请求额外发送的头信息字段
      3. 如果服务器否定预检请求,会返回一个正常的 HTTP 响应,但是没有任何的 CORS 相关的头信息字段,这是浏览器会认定服务器不同意预检,触发错误
      4. 如果服务器同意预检请求,会返回一个正常的 HTTP 响应,但是会多几个 CORS 相关的头信息字段
        • Access-Control-Allow-Origin (必须)
        • Access-Control-Allow-Methods (必须)
        • Access-Control-Allow-Credentials
        • Access-Control-Expose-Headers
        • Access-Control-Max-Age - 预检请求的有效期, 单位秒
        • Access-Control-Allow-Headers

      非简单请求

  3. 只能操作同源的DOM

    引入了跨文档消息机制 postMessage

XSS 攻击

跨域脚本攻击。

类型

  1. 存储型

    脚本存储在服务器,用户访问时,脚本从服务器返回,浏览器执行脚本

  2. 反射型

    在文章链接参数上存放了一段恶意脚本,用户访问时,脚本从参数中取出,浏览器执行脚本

  3. DOM 型

    比如在搜索的时候,添加了一段脚本, 如果网站的搜索功能没有对用户输入进行适当的过滤和转义,搜索结果页面可能会将恶意的脚本代码直接插入到 HTML 中,并在用户浏览器中执行

    web 资源在传输或者用户在使用的过程中修改了 Web 页面的数据

预防

  1. 对输入的内容进行截取或者转义(encode)
  2. 限制输入的长度和类型
  3. 引入内容安全策略 (CSP)
  4. 对 Cookie 设置 HttpOnly 属性

CSRF 攻击

跨站请求伪造。

比如诱导用户点击链接,访问网址后,网址通过用户的登录信息伪造请求

必要条件

  1. 站点有 CSRF 漏洞
  2. 用户登录了
  3. 用户点击了诱导链接

预防

  1. 给 Cookie 新增 SameSite 属性
    • Strict: 第三方站点无法发送 Cookie
    • Lax (默认): 第三方站点如果是 GET 请求可以发送 Cookie
    • None 不做限制 Cookie的书写
  2. 通过请求的 Origin or Referer 字段判断请求合法性
  3. 在请求地址中添加 token 并验证

浏览器安全

将渲染进程放在沙箱中

  • 无法持久存储
  • 无法访问网络
  • 无法监听用户交互
  • 需要通过 IPC 与浏览器主进程通信,由浏览器进程处理后,再将结果传递给渲染进程

网络安全

HTTPS、重放攻击、验签

  • 基本属性:

    • name
    • value
  • 访问性:

    • expire
    • path
    • domain
  • 安全性:

    • secure
    • httpOnly
    • sameSite
',28),p=[c];function h(d,u,_,b,C,m){return i(),e("div",null,p)}const f=l(n,[["render",h]]);export{P as __pageData,f as default}; diff --git "a/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\256\211\345\205\250.md.73ccefca.lean.js" "b/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\256\211\345\205\250.md.ff5d442d.lean.js" similarity index 90% rename from "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\256\211\345\205\250.md.73ccefca.lean.js" rename to "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\256\211\345\205\250.md.ff5d442d.lean.js" index 66fc9260..1a5751c1 100644 --- "a/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\256\211\345\205\250.md.73ccefca.lean.js" +++ "b/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\345\256\211\345\205\250.md.ff5d442d.lean.js" @@ -1 +1 @@ -import{_ as l,o as i,c as e,Q as a}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/6.36673f2f.jpg",t="/vitePress-blob/assets/7.d0aefc74.jpg",r="/vitePress-blob/assets/8.28d1dcd9.jpg",s="/vitePress-blob/assets/9.b32cc031.jpg",P=JSON.parse('{"title":"浏览器安全","description":"","frontmatter":{},"headers":[],"relativePath":"guide/浏览器相关/浏览器安全.md","filePath":"guide/浏览器相关/浏览器安全.md","lastUpdated":1736065650000}'),n={name:"guide/浏览器相关/浏览器安全.md"},c=a("",28),p=[c];function h(d,u,_,b,C,m){return i(),e("div",null,p)}const f=l(n,[["render",h]]);export{P as __pageData,f as default}; +import{_ as l,o as i,c as e,Q as a}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/6.36673f2f.jpg",t="/vitePress-blob/assets/7.d0aefc74.jpg",r="/vitePress-blob/assets/8.28d1dcd9.jpg",s="/vitePress-blob/assets/9.b32cc031.jpg",P=JSON.parse('{"title":"浏览器安全","description":"","frontmatter":{},"headers":[],"relativePath":"guide/浏览器相关/浏览器安全.md","filePath":"guide/浏览器相关/浏览器安全.md","lastUpdated":1736839070000}'),n={name:"guide/浏览器相关/浏览器安全.md"},c=a("",28),p=[c];function h(d,u,_,b,C,m){return i(),e("div",null,p)}const f=l(n,[["render",h]]);export{P as __pageData,f as default}; diff --git "a/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\346\270\262\346\237\223\346\265\201\347\250\213.md.f3c634f1.js" "b/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\346\270\262\346\237\223\346\265\201\347\250\213.md.08093b9d.js" similarity index 99% rename from "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\346\270\262\346\237\223\346\265\201\347\250\213.md.f3c634f1.js" rename to "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\346\270\262\346\237\223\346\265\201\347\250\213.md.08093b9d.js" index 740666bb..1b33c518 100644 --- "a/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\346\270\262\346\237\223\346\265\201\347\250\213.md.f3c634f1.js" +++ "b/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\346\270\262\346\237\223\346\265\201\347\250\213.md.08093b9d.js" @@ -1 +1 @@ -import{_ as l,o as i,c as e,Q as a}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/3.8bc85ddd.png",t="/vitePress-blob/assets/4.9caf6b36.png",s="/vitePress-blob/assets/5.bbea4579.jpg",m=JSON.parse('{"title":"浏览器渲染流程","description":"","frontmatter":{},"headers":[],"relativePath":"guide/浏览器相关/浏览器渲染流程.md","filePath":"guide/浏览器相关/浏览器渲染流程.md","lastUpdated":1736065650000}'),r={name:"guide/浏览器相关/浏览器渲染流程.md"},p=a('

浏览器渲染流程

浏览器渲染流程

  1. 发出请求到页面首次绘制

    • 第一阶段: 页面提交请求到服务器响应,这时候页面还是之前的页面
    • 第二阶段: 获取到响应数据提交到渲染进程,进行 HTML 解析、CSS 加载、JS 加载、JS 执行、CSSOM 解析、布局树生成、页面绘制
    • 第三阶段: 等首次加载完成后,页面一点点被渲染
  2. HTML 解析

  3. 生成 CSSOM 树

    CSS 不会阻塞 HTML 解析,但是会阻塞页面渲染,因为要生成渲染树

  4. 生成布局树

    去除不显示的节点,计算样式

  5. 分层和合成机制

    • 分层: 分层树在布局树之后,分层树的每一个节点都是图层,如没有,则和父节点同一个图层

    • 绘制阶段: 根据图层在绘制阶段生成绘制指令

    • 光栅化: 根据绘制指令,将每个图层都绘制成一张图片

    • 合成: 合成线程将多张图片合成一张图片,然后显示在屏幕上 (由合成线程完成,不影响主线程)

      • 优化操作1: 合成线程内会维护一个光栅化线程池,将绘制指令列表提交到 GPU 进行光栅化,生成位图,放在内存中
      • 优化操作2: 分块,合成线程将图层分块,优先渲染离屏幕最近的图块
      • 合成线程
  6. 页面显示

常见的问题

1 从输入 URL 到页面渲染完成,发生了什么?

  1. 浏览器会根据用户输入的内容判断是关键字还是URL
  2. 如果是关键字,会将其组成成带有搜索关键字的URL,通过IPC进程通信发送给网络进程
  3. 网络进程发起请求前,会判断是否命中强缓存,如命中直接返回存储资源
  4. 否则发起请求,根据 DNS 解析获取域名对应的 IP 地址,进行 TCP、HTTP 连接
  5. 服务端收到请求后,会判断是否命中协商缓存,如命中则返回304状态码
  6. 如返回的是301、302状态码,浏览器会根据响应头返回的location字段,进行重定向
  7. 如是正常返回资源类型,浏览器会根据 content-type 对资源做相对应的操作,如果是下载类型则进行下载,如果是html类型则会提交到渲染进程进行解析
  8. 解析 HTML,转换成浏览器能识别的 DOM 树
  9. 解析 CSS,转化成浏览器能识别的 CSS 样式树
  10. 根据 DOM 和 CSS 样式树,通过 布局计算生成 布局树
  11. 根据布局树上的分层属性(z-index),生成分层树
  12. 根据分层树,生成绘制指令列表
  13. 渲染主线程会将绘制指令列表提交给合成线程
  14. 合成线程会维护一个光栅化线程池,将绘制指令列表提交到 GPU 进行光栅化,生成位图,放在内存中
  15. 合成线程通信浏览器进程,浏览器进程将内存中的数据输出到显卡的后缓存区,在下一帧绘制之前,显卡的后缓冲区与前缓冲区对换,显示屏读取前缓冲区数据,显示到屏幕上

整体流程

2 回流和重绘有什么区别

回流:元素尺寸、定位改变,可能会影响到其他元素的位置

重绘:元素外观改变,如背景色、颜色,元素尺寸位置不改变且不影响其他元素

减少回流的方法:

  1. 集中修改
    • 修改样式,使用 class
    • 修改前将 DOM 改成 display: none,修改后再显示
    • 使用 DocumentFragment
    • resize、scroll 事件,使用防抖
  2. 使用 BFC
    • 隔离内部元素对外部元素的影响
  3. 脱离文档流
    • position: absolute、fixed
    • float
  4. 提升合成图层
    • CSS3 属性: will-change、transform: translate3d(0);
  5. 不是用 offsetHeight、getBoundingClientRect
    • 使用 intersectionObserver API - 判断元素是否在可视区域内

BFC: 块级格式化上下文 特点: BFC 内部元素不会影响到外部元素 形成条件:

  1. HTML 元素
  2. overflow: hidden、auto、scroll
  3. position: absolute、fixed
  4. float 元素
  5. display: inline-block、flex、inline-flex等

3 渲染流程图

渲染流程图

JS 会阻塞 HTML 的解析和渲染

CSS 不会阻塞 HTML 解析,但会阻塞 DOM 渲染,还会阻塞 JS 的执行

为什么 CSS 会阻塞 JS 的执行?

因为 JS 可能会操作 DOM 节点和 CSS 样式,因此浏览器为了获取到最新的 CSS 样式,样式表会在后面的 JS 执行前先加载执行完毕,所以 CSS 会阻塞 JS 的执行

',20),c=[p];function d(n,_,u,h,S,b){return i(),e("div",null,c)}const C=l(r,[["render",d]]);export{m as __pageData,C as default}; +import{_ as l,o as i,c as e,Q as a}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/3.8bc85ddd.png",t="/vitePress-blob/assets/4.9caf6b36.png",s="/vitePress-blob/assets/5.bbea4579.jpg",m=JSON.parse('{"title":"浏览器渲染流程","description":"","frontmatter":{},"headers":[],"relativePath":"guide/浏览器相关/浏览器渲染流程.md","filePath":"guide/浏览器相关/浏览器渲染流程.md","lastUpdated":1736839070000}'),r={name:"guide/浏览器相关/浏览器渲染流程.md"},p=a('

浏览器渲染流程

浏览器渲染流程

  1. 发出请求到页面首次绘制

    • 第一阶段: 页面提交请求到服务器响应,这时候页面还是之前的页面
    • 第二阶段: 获取到响应数据提交到渲染进程,进行 HTML 解析、CSS 加载、JS 加载、JS 执行、CSSOM 解析、布局树生成、页面绘制
    • 第三阶段: 等首次加载完成后,页面一点点被渲染
  2. HTML 解析

  3. 生成 CSSOM 树

    CSS 不会阻塞 HTML 解析,但是会阻塞页面渲染,因为要生成渲染树

  4. 生成布局树

    去除不显示的节点,计算样式

  5. 分层和合成机制

    • 分层: 分层树在布局树之后,分层树的每一个节点都是图层,如没有,则和父节点同一个图层

    • 绘制阶段: 根据图层在绘制阶段生成绘制指令

    • 光栅化: 根据绘制指令,将每个图层都绘制成一张图片

    • 合成: 合成线程将多张图片合成一张图片,然后显示在屏幕上 (由合成线程完成,不影响主线程)

      • 优化操作1: 合成线程内会维护一个光栅化线程池,将绘制指令列表提交到 GPU 进行光栅化,生成位图,放在内存中
      • 优化操作2: 分块,合成线程将图层分块,优先渲染离屏幕最近的图块
      • 合成线程
  6. 页面显示

常见的问题

1 从输入 URL 到页面渲染完成,发生了什么?

  1. 浏览器会根据用户输入的内容判断是关键字还是URL
  2. 如果是关键字,会将其组成成带有搜索关键字的URL,通过IPC进程通信发送给网络进程
  3. 网络进程发起请求前,会判断是否命中强缓存,如命中直接返回存储资源
  4. 否则发起请求,根据 DNS 解析获取域名对应的 IP 地址,进行 TCP、HTTP 连接
  5. 服务端收到请求后,会判断是否命中协商缓存,如命中则返回304状态码
  6. 如返回的是301、302状态码,浏览器会根据响应头返回的location字段,进行重定向
  7. 如是正常返回资源类型,浏览器会根据 content-type 对资源做相对应的操作,如果是下载类型则进行下载,如果是html类型则会提交到渲染进程进行解析
  8. 解析 HTML,转换成浏览器能识别的 DOM 树
  9. 解析 CSS,转化成浏览器能识别的 CSS 样式树
  10. 根据 DOM 和 CSS 样式树,通过 布局计算生成 布局树
  11. 根据布局树上的分层属性(z-index),生成分层树
  12. 根据分层树,生成绘制指令列表
  13. 渲染主线程会将绘制指令列表提交给合成线程
  14. 合成线程会维护一个光栅化线程池,将绘制指令列表提交到 GPU 进行光栅化,生成位图,放在内存中
  15. 合成线程通信浏览器进程,浏览器进程将内存中的数据输出到显卡的后缓存区,在下一帧绘制之前,显卡的后缓冲区与前缓冲区对换,显示屏读取前缓冲区数据,显示到屏幕上

整体流程

2 回流和重绘有什么区别

回流:元素尺寸、定位改变,可能会影响到其他元素的位置

重绘:元素外观改变,如背景色、颜色,元素尺寸位置不改变且不影响其他元素

减少回流的方法:

  1. 集中修改
    • 修改样式,使用 class
    • 修改前将 DOM 改成 display: none,修改后再显示
    • 使用 DocumentFragment
    • resize、scroll 事件,使用防抖
  2. 使用 BFC
    • 隔离内部元素对外部元素的影响
  3. 脱离文档流
    • position: absolute、fixed
    • float
  4. 提升合成图层
    • CSS3 属性: will-change、transform: translate3d(0);
  5. 不是用 offsetHeight、getBoundingClientRect
    • 使用 intersectionObserver API - 判断元素是否在可视区域内

BFC: 块级格式化上下文 特点: BFC 内部元素不会影响到外部元素 形成条件:

  1. HTML 元素
  2. overflow: hidden、auto、scroll
  3. position: absolute、fixed
  4. float 元素
  5. display: inline-block、flex、inline-flex等

3 渲染流程图

渲染流程图

JS 会阻塞 HTML 的解析和渲染

CSS 不会阻塞 HTML 解析,但会阻塞 DOM 渲染,还会阻塞 JS 的执行

为什么 CSS 会阻塞 JS 的执行?

因为 JS 可能会操作 DOM 节点和 CSS 样式,因此浏览器为了获取到最新的 CSS 样式,样式表会在后面的 JS 执行前先加载执行完毕,所以 CSS 会阻塞 JS 的执行

',20),c=[p];function d(n,_,u,h,S,b){return i(),e("div",null,c)}const C=l(r,[["render",d]]);export{m as __pageData,C as default}; diff --git "a/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\346\270\262\346\237\223\346\265\201\347\250\213.md.f3c634f1.lean.js" "b/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\346\270\262\346\237\223\346\265\201\347\250\213.md.08093b9d.lean.js" similarity index 90% rename from "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\346\270\262\346\237\223\346\265\201\347\250\213.md.f3c634f1.lean.js" rename to "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\346\270\262\346\237\223\346\265\201\347\250\213.md.08093b9d.lean.js" index dd520c1e..f6f75c33 100644 --- "a/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\346\270\262\346\237\223\346\265\201\347\250\213.md.f3c634f1.lean.js" +++ "b/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\346\270\262\346\237\223\346\265\201\347\250\213.md.08093b9d.lean.js" @@ -1 +1 @@ -import{_ as l,o as i,c as e,Q as a}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/3.8bc85ddd.png",t="/vitePress-blob/assets/4.9caf6b36.png",s="/vitePress-blob/assets/5.bbea4579.jpg",m=JSON.parse('{"title":"浏览器渲染流程","description":"","frontmatter":{},"headers":[],"relativePath":"guide/浏览器相关/浏览器渲染流程.md","filePath":"guide/浏览器相关/浏览器渲染流程.md","lastUpdated":1736065650000}'),r={name:"guide/浏览器相关/浏览器渲染流程.md"},p=a("",20),c=[p];function d(n,_,u,h,S,b){return i(),e("div",null,c)}const C=l(r,[["render",d]]);export{m as __pageData,C as default}; +import{_ as l,o as i,c as e,Q as a}from"./chunks/framework.b6910bb2.js";const o="/vitePress-blob/assets/3.8bc85ddd.png",t="/vitePress-blob/assets/4.9caf6b36.png",s="/vitePress-blob/assets/5.bbea4579.jpg",m=JSON.parse('{"title":"浏览器渲染流程","description":"","frontmatter":{},"headers":[],"relativePath":"guide/浏览器相关/浏览器渲染流程.md","filePath":"guide/浏览器相关/浏览器渲染流程.md","lastUpdated":1736839070000}'),r={name:"guide/浏览器相关/浏览器渲染流程.md"},p=a("",20),c=[p];function d(n,_,u,h,S,b){return i(),e("div",null,c)}const C=l(r,[["render",d]]);export{m as __pageData,C as default}; diff --git "a/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\347\274\223\345\255\230.md.4f6289a9.js" "b/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\347\274\223\345\255\230.md.3b8afe38.js" similarity index 99% rename from "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\347\274\223\345\255\230.md.4f6289a9.js" rename to "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\347\274\223\345\255\230.md.3b8afe38.js" index b172a09b..0fd455f3 100644 --- "a/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\347\274\223\345\255\230.md.4f6289a9.js" +++ "b/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\347\274\223\345\255\230.md.3b8afe38.js" @@ -1,4 +1,4 @@ -import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const F=JSON.parse('{"title":"浏览器缓存","description":"","frontmatter":{},"headers":[],"relativePath":"guide/浏览器相关/浏览器缓存.md","filePath":"guide/浏览器相关/浏览器缓存.md","lastUpdated":1736065650000}'),p={name:"guide/浏览器相关/浏览器缓存.md"},o=l(`

浏览器缓存

缓存

从 HTTP 缓存策略来说

  • 强缓存
  • 协商缓存

从浏览器的缓存位置来说

  • Service Worker Cache (必须是 Https)
  • Memory Cache (内存缓存)
  • Disk Cache (硬盘缓存)
  • Push Cache (Http2.0)

本地存储

请求时携带,用于信息持久化

Local Storage 本地存储

用于长期不变化的数据,只能存储字符串

Session Storage 会话存储

用于存储当前会话的数据,只能存储字符串

indexedDB

非关系型数据库

Service Worker

运行在浏览器背后的独立线程(必须 HTTPS 才能使用)

js
// main.js
+import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const F=JSON.parse('{"title":"浏览器缓存","description":"","frontmatter":{},"headers":[],"relativePath":"guide/浏览器相关/浏览器缓存.md","filePath":"guide/浏览器相关/浏览器缓存.md","lastUpdated":1736839070000}'),p={name:"guide/浏览器相关/浏览器缓存.md"},o=l(`

浏览器缓存

缓存

从 HTTP 缓存策略来说

  • 强缓存
  • 协商缓存

从浏览器的缓存位置来说

  • Service Worker Cache (必须是 Https)
  • Memory Cache (内存缓存)
  • Disk Cache (硬盘缓存)
  • Push Cache (Http2.0)

本地存储

请求时携带,用于信息持久化

Local Storage 本地存储

用于长期不变化的数据,只能存储字符串

Session Storage 会话存储

用于存储当前会话的数据,只能存储字符串

indexedDB

非关系型数据库

Service Worker

运行在浏览器背后的独立线程(必须 HTTPS 才能使用)

js
// main.js
 /* 判断当前浏览器是否支持serviceWorker */
 if ('serviceWorker' in navigator) {
   /* 当页面加载完成就创建一个 serviceWorker */
diff --git "a/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\347\274\223\345\255\230.md.4f6289a9.lean.js" "b/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\347\274\223\345\255\230.md.3b8afe38.lean.js"
similarity index 87%
rename from "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\347\274\223\345\255\230.md.4f6289a9.lean.js"
rename to "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\347\274\223\345\255\230.md.3b8afe38.lean.js"
index 15531373..ec0b07c4 100644
--- "a/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\347\274\223\345\255\230.md.4f6289a9.lean.js"
+++ "b/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\347\274\223\345\255\230.md.3b8afe38.lean.js"
@@ -1 +1 @@
-import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const F=JSON.parse('{"title":"浏览器缓存","description":"","frontmatter":{},"headers":[],"relativePath":"guide/浏览器相关/浏览器缓存.md","filePath":"guide/浏览器相关/浏览器缓存.md","lastUpdated":1736065650000}'),p={name:"guide/浏览器相关/浏览器缓存.md"},o=l("",19),e=[o];function c(t,r,E,y,i,h){return n(),a("div",null,e)}const u=s(p,[["render",c]]);export{F as __pageData,u as default};
+import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.b6910bb2.js";const F=JSON.parse('{"title":"浏览器缓存","description":"","frontmatter":{},"headers":[],"relativePath":"guide/浏览器相关/浏览器缓存.md","filePath":"guide/浏览器相关/浏览器缓存.md","lastUpdated":1736839070000}'),p={name:"guide/浏览器相关/浏览器缓存.md"},o=l("",19),e=[o];function c(t,r,E,y,i,h){return n(),a("div",null,e)}const u=s(p,[["render",c]]);export{F as __pageData,u as default};
diff --git "a/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\350\277\233\347\250\213\346\236\266\346\236\204.md.5bd0b06d.js" "b/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\350\277\233\347\250\213\346\236\266\346\236\204.md.4157b63e.js"
similarity index 97%
rename from "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\350\277\233\347\250\213\346\236\266\346\236\204.md.5bd0b06d.js"
rename to "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\350\277\233\347\250\213\346\236\266\346\236\204.md.4157b63e.js"
index 100bb165..ccc70802 100644
--- "a/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\350\277\233\347\250\213\346\236\266\346\236\204.md.5bd0b06d.js"
+++ "b/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\350\277\233\347\250\213\346\236\266\346\236\204.md.4157b63e.js"
@@ -1 +1 @@
-import{_ as a,o as e,c as t,Q as r}from"./chunks/framework.b6910bb2.js";const b=JSON.parse('{"title":"浏览器进程架构","description":"","frontmatter":{},"headers":[],"relativePath":"guide/浏览器相关/浏览器进程架构.md","filePath":"guide/浏览器相关/浏览器进程架构.md","lastUpdated":1736065650000}'),i={name:"guide/浏览器相关/浏览器进程架构.md"},l=r('

浏览器进程架构

进程与线程

进程

进程是一个程序的实例

进程是系统进行资源调度和分配的最小单位

进程之间相互独立,通过 IPC 进行通信

线程

线程是 CPU 调度的最小单位

线程之间共享进程的数据

线程里面会有协程,但只能运行一个

浏览器架构

1 浏览器主进程

用户交互、文件存储、页面显示、子进程管理

2 渲染进程

解析资源、执行 js

渲染进程被浏览器放在安全沙箱里面,因此渲染进程的资源是通过网络获取

  • 渲染线程: 渲染页面
  • JS 线程:解析执行 js
  • 事件触发线程: 事件循环
  • 定时器线程:定时器相关
  • 异步 http 请求线程: 请求相关

3 GPU 进程

UI界面的绘制

4 网络进程

网络资源加载

5 插件进程

管理插件

',23),o=[l];function h(d,n,s,_,c,p){return e(),t("div",null,o)}const q=a(i,[["render",h]]);export{b as __pageData,q as default}; +import{_ as a,o as e,c as t,Q as r}from"./chunks/framework.b6910bb2.js";const b=JSON.parse('{"title":"浏览器进程架构","description":"","frontmatter":{},"headers":[],"relativePath":"guide/浏览器相关/浏览器进程架构.md","filePath":"guide/浏览器相关/浏览器进程架构.md","lastUpdated":1736839070000}'),i={name:"guide/浏览器相关/浏览器进程架构.md"},l=r('

浏览器进程架构

进程与线程

进程

进程是一个程序的实例

进程是系统进行资源调度和分配的最小单位

进程之间相互独立,通过 IPC 进行通信

线程

线程是 CPU 调度的最小单位

线程之间共享进程的数据

线程里面会有协程,但只能运行一个

浏览器架构

1 浏览器主进程

用户交互、文件存储、页面显示、子进程管理

2 渲染进程

解析资源、执行 js

渲染进程被浏览器放在安全沙箱里面,因此渲染进程的资源是通过网络获取

  • 渲染线程: 渲染页面
  • JS 线程:解析执行 js
  • 事件触发线程: 事件循环
  • 定时器线程:定时器相关
  • 异步 http 请求线程: 请求相关

3 GPU 进程

UI界面的绘制

4 网络进程

网络资源加载

5 插件进程

管理插件

',23),o=[l];function h(d,n,s,_,c,p){return e(),t("div",null,o)}const q=a(i,[["render",h]]);export{b as __pageData,q as default}; diff --git "a/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\350\277\233\347\250\213\346\236\266\346\236\204.md.5bd0b06d.lean.js" "b/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\350\277\233\347\250\213\346\236\266\346\236\204.md.4157b63e.lean.js" similarity index 87% rename from "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\350\277\233\347\250\213\346\236\266\346\236\204.md.5bd0b06d.lean.js" rename to "assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\350\277\233\347\250\213\346\236\266\346\236\204.md.4157b63e.lean.js" index 8e457a19..6d432111 100644 --- "a/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\350\277\233\347\250\213\346\236\266\346\236\204.md.5bd0b06d.lean.js" +++ "b/assets/guide_\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263_\346\265\217\350\247\210\345\231\250\350\277\233\347\250\213\346\236\266\346\236\204.md.4157b63e.lean.js" @@ -1 +1 @@ -import{_ as a,o as e,c as t,Q as r}from"./chunks/framework.b6910bb2.js";const b=JSON.parse('{"title":"浏览器进程架构","description":"","frontmatter":{},"headers":[],"relativePath":"guide/浏览器相关/浏览器进程架构.md","filePath":"guide/浏览器相关/浏览器进程架构.md","lastUpdated":1736065650000}'),i={name:"guide/浏览器相关/浏览器进程架构.md"},l=r("",23),o=[l];function h(d,n,s,_,c,p){return e(),t("div",null,o)}const q=a(i,[["render",h]]);export{b as __pageData,q as default}; +import{_ as a,o as e,c as t,Q as r}from"./chunks/framework.b6910bb2.js";const b=JSON.parse('{"title":"浏览器进程架构","description":"","frontmatter":{},"headers":[],"relativePath":"guide/浏览器相关/浏览器进程架构.md","filePath":"guide/浏览器相关/浏览器进程架构.md","lastUpdated":1736839070000}'),i={name:"guide/浏览器相关/浏览器进程架构.md"},l=r("",23),o=[l];function h(d,n,s,_,c,p){return e(),t("div",null,o)}const q=a(i,[["render",h]]);export{b as __pageData,q as default}; diff --git "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_CDN.md.0fd3f4e9.js" "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_CDN.md.426d00a8.js" similarity index 98% rename from "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_CDN.md.0fd3f4e9.js" rename to "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_CDN.md.426d00a8.js" index 1128b375..dffbe09f 100644 --- "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_CDN.md.0fd3f4e9.js" +++ "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_CDN.md.426d00a8.js" @@ -1 +1 @@ -import{_ as l,o as a,c as i,Q as e}from"./chunks/framework.b6910bb2.js";const t="/vitePress-blob/assets/13.1322b23d.jpg",C=JSON.parse('{"title":"CDN","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/CDN.md","filePath":"guide/网络相关/CDN.md","lastUpdated":1736065650000}'),o={name:"guide/网络相关/CDN.md"},r=e('

CDN

定义

内容分发网络,利用最靠近用户的服务器,将资源更快更可靠的给到用户

组成

  1. 分发服务系统

    最小单位是 Cache 设备

    • 响应用户请求
    • 与源站点进行资源同步
  2. 负载均衡系统

    负责对用户的请求进行调度

  3. 运营管理系统

    负责业务层面与外界系统的交互所需要的收集、整理、交付等工作 (为了做闭环)

    运营管理子系统 网络管理子系统

作用

托管 web 资源,加速 web 资源的获取速度

性能方面

  1. 用户从最近的 CDN 站点获取资源,访问速度更快
  2. 突破 TCP 只有同域名下只有 6 个的限制
  3. 减少源服务器的负担

安全方面

  1. 针对 DDos: 通过监控分析异常流量,限制请求频率
  2. 针对 MITM: 从源服务器到 CDN 节点到 ISP (Internet Service Provider, 网络运营提供商),全链路 HTTPS 通信

工作原理

没有使用 CDN 过程

  1. 浏览器通过 DNS 对域名进行解析,得到此域名对应的 IP 地址
  2. 浏览器根据得到的 IP 地址,向域名的服务主机发送数据请求
  3. 服务器向浏览器返回响应数据

使用了 CDN 过程

  1. 获取负载均衡服务器 IP 地址 (自底向上)

      1. 数据的 URL 先进行 DNS 系统解析,发现该 URL 对应的是一个 CDN 专用的 DNS 服务器,DNS 系统就会将域名解析权交给 CNAME 指向的 CDN 专用的 DNS 服务器
      1. CDN 专用的 DNS 服务器将 CDN 全局负载均衡设备 IP 地址返回给用户
      1. 用户向 CDN 的全局负载均衡设备发起数据请求
  2. 获取目标缓存服务器 IP 地址 (自顶向下)

      1. CDN 全局负载均衡设备根据用户 IP 地址以及用户请求的 URL,选择一台用户所属区域的负载均衡设备
      1. 区域负责均衡设备选择一台适合的缓存服务器来提供服务,将该缓存服务器的 IP 地址返回给全局负载均衡设备
      1. 全局负载均衡设备把服务器 IP 地址返回给用户
  3. 用户发起请求获取

    • 用户向该缓存服务器发起请求,缓存服务器响应用户的请求,将用户所需内容发送支用户终端 (如果缓存服务器没有用户想要的内容,那么缓存服务器会穿透 CDN 集群往源服务器获取设备)

流程

使用场景

  1. 使用第三方 CDN 服务 (例如开源框架可以使用第三方 CDN 服务,如 React、Vue、JQuery)
  2. 使用 CDN 进行静态资源的缓存 (将网站的资源放在 CDN 上)
  3. 直播传送 (直播本质也是使用流媒体进行传送,CDN 也支持流媒体传送,所以直播完全可以使用 CDN 来提高访问速度)
',19),h=[r];function n(d,s,c,p,u,D){return a(),i("div",null,h)}const _=l(o,[["render",n]]);export{C as __pageData,_ as default}; +import{_ as l,o as a,c as i,Q as e}from"./chunks/framework.b6910bb2.js";const t="/vitePress-blob/assets/13.1322b23d.jpg",C=JSON.parse('{"title":"CDN","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/CDN.md","filePath":"guide/网络相关/CDN.md","lastUpdated":1736839070000}'),o={name:"guide/网络相关/CDN.md"},r=e('

CDN

定义

内容分发网络,利用最靠近用户的服务器,将资源更快更可靠的给到用户

组成

  1. 分发服务系统

    最小单位是 Cache 设备

    • 响应用户请求
    • 与源站点进行资源同步
  2. 负载均衡系统

    负责对用户的请求进行调度

  3. 运营管理系统

    负责业务层面与外界系统的交互所需要的收集、整理、交付等工作 (为了做闭环)

    运营管理子系统 网络管理子系统

作用

托管 web 资源,加速 web 资源的获取速度

性能方面

  1. 用户从最近的 CDN 站点获取资源,访问速度更快
  2. 突破 TCP 只有同域名下只有 6 个的限制
  3. 减少源服务器的负担

安全方面

  1. 针对 DDos: 通过监控分析异常流量,限制请求频率
  2. 针对 MITM: 从源服务器到 CDN 节点到 ISP (Internet Service Provider, 网络运营提供商),全链路 HTTPS 通信

工作原理

没有使用 CDN 过程

  1. 浏览器通过 DNS 对域名进行解析,得到此域名对应的 IP 地址
  2. 浏览器根据得到的 IP 地址,向域名的服务主机发送数据请求
  3. 服务器向浏览器返回响应数据

使用了 CDN 过程

  1. 获取负载均衡服务器 IP 地址 (自底向上)

      1. 数据的 URL 先进行 DNS 系统解析,发现该 URL 对应的是一个 CDN 专用的 DNS 服务器,DNS 系统就会将域名解析权交给 CNAME 指向的 CDN 专用的 DNS 服务器
      1. CDN 专用的 DNS 服务器将 CDN 全局负载均衡设备 IP 地址返回给用户
      1. 用户向 CDN 的全局负载均衡设备发起数据请求
  2. 获取目标缓存服务器 IP 地址 (自顶向下)

      1. CDN 全局负载均衡设备根据用户 IP 地址以及用户请求的 URL,选择一台用户所属区域的负载均衡设备
      1. 区域负责均衡设备选择一台适合的缓存服务器来提供服务,将该缓存服务器的 IP 地址返回给全局负载均衡设备
      1. 全局负载均衡设备把服务器 IP 地址返回给用户
  3. 用户发起请求获取

    • 用户向该缓存服务器发起请求,缓存服务器响应用户的请求,将用户所需内容发送支用户终端 (如果缓存服务器没有用户想要的内容,那么缓存服务器会穿透 CDN 集群往源服务器获取设备)

流程

使用场景

  1. 使用第三方 CDN 服务 (例如开源框架可以使用第三方 CDN 服务,如 React、Vue、JQuery)
  2. 使用 CDN 进行静态资源的缓存 (将网站的资源放在 CDN 上)
  3. 直播传送 (直播本质也是使用流媒体进行传送,CDN 也支持流媒体传送,所以直播完全可以使用 CDN 来提高访问速度)
',19),h=[r];function n(d,s,c,p,u,D){return a(),i("div",null,h)}const _=l(o,[["render",n]]);export{C as __pageData,_ as default}; diff --git "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_CDN.md.0fd3f4e9.lean.js" "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_CDN.md.426d00a8.lean.js" similarity index 86% rename from "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_CDN.md.0fd3f4e9.lean.js" rename to "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_CDN.md.426d00a8.lean.js" index 235eeb49..cade1bc0 100644 --- "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_CDN.md.0fd3f4e9.lean.js" +++ "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_CDN.md.426d00a8.lean.js" @@ -1 +1 @@ -import{_ as l,o as a,c as i,Q as e}from"./chunks/framework.b6910bb2.js";const t="/vitePress-blob/assets/13.1322b23d.jpg",C=JSON.parse('{"title":"CDN","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/CDN.md","filePath":"guide/网络相关/CDN.md","lastUpdated":1736065650000}'),o={name:"guide/网络相关/CDN.md"},r=e("",19),h=[r];function n(d,s,c,p,u,D){return a(),i("div",null,h)}const _=l(o,[["render",n]]);export{C as __pageData,_ as default}; +import{_ as l,o as a,c as i,Q as e}from"./chunks/framework.b6910bb2.js";const t="/vitePress-blob/assets/13.1322b23d.jpg",C=JSON.parse('{"title":"CDN","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/CDN.md","filePath":"guide/网络相关/CDN.md","lastUpdated":1736839070000}'),o={name:"guide/网络相关/CDN.md"},r=e("",19),h=[r];function n(d,s,c,p,u,D){return a(),i("div",null,h)}const _=l(o,[["render",n]]);export{C as __pageData,_ as default}; diff --git "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_DNS.md.584b66a4.js" "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_DNS.md.dc1c7d42.js" similarity index 93% rename from "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_DNS.md.584b66a4.js" rename to "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_DNS.md.dc1c7d42.js" index 711cf197..e3f44dad 100644 --- "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_DNS.md.584b66a4.js" +++ "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_DNS.md.dc1c7d42.js" @@ -1 +1 @@ -import{_ as l,o as i,c as a,Q as e}from"./chunks/framework.b6910bb2.js";const S=JSON.parse('{"title":"DNS","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/DNS.md","filePath":"guide/网络相关/DNS.md","lastUpdated":1736065650000}'),o={name:"guide/网络相关/DNS.md"},t=e('

DNS

定义

域名解析系统

工作原理 (自下而上)

例如查找 www.baidu.com 的 ip 地址

  1. 查找缓存
      1. 检查浏览器缓存
      1. 检查操作系统缓存 (常见的有 host 文件)
      1. 检查路由器缓存
  2. 如何缓存查找到,会向 ISP (网络服务提供商) 的本地 DNS 服务器查询
  3. 如果本地 DNS 服务器没有找到,会向根域名服务器请求解析,分为以下几步:(自顶向下)
      1. 根服务器返回顶级域名服务器(如.com、.cn、.org等)地址,例如该例子中的 .com 的地址
      1. 接着向顶级瑜域名服务器发送请求,然后会返回次级域名服务器的地址,例如该例子会返回 .baidu 的地址
      1. 接着向次级域名服务器发送请求,会返回通过三级域名地址的 IP,例如本例子返回的 www.baidu.com 的地址
    • 4.Local DNS Server会缓存结果,并返回给用户,缓存在系统中
',6),r=[t];function s(d,c,n,_,h,u){return i(),a("div",null,r)}const m=l(o,[["render",s]]);export{S as __pageData,m as default}; +import{_ as l,o as i,c as a,Q as e}from"./chunks/framework.b6910bb2.js";const S=JSON.parse('{"title":"DNS","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/DNS.md","filePath":"guide/网络相关/DNS.md","lastUpdated":1736839070000}'),o={name:"guide/网络相关/DNS.md"},t=e('

DNS

定义

域名解析系统

工作原理 (自下而上)

例如查找 www.baidu.com 的 ip 地址

  1. 查找缓存
      1. 检查浏览器缓存
      1. 检查操作系统缓存 (常见的有 host 文件)
      1. 检查路由器缓存
  2. 如何缓存查找到,会向 ISP (网络服务提供商) 的本地 DNS 服务器查询
  3. 如果本地 DNS 服务器没有找到,会向根域名服务器请求解析,分为以下几步:(自顶向下)
      1. 根服务器返回顶级域名服务器(如.com、.cn、.org等)地址,例如该例子中的 .com 的地址
      1. 接着向顶级瑜域名服务器发送请求,然后会返回次级域名服务器的地址,例如该例子会返回 .baidu 的地址
      1. 接着向次级域名服务器发送请求,会返回通过三级域名地址的 IP,例如本例子返回的 www.baidu.com 的地址
    • 4.Local DNS Server会缓存结果,并返回给用户,缓存在系统中
',6),r=[t];function s(d,c,n,_,h,u){return i(),a("div",null,r)}const m=l(o,[["render",s]]);export{S as __pageData,m as default}; diff --git "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_DNS.md.584b66a4.lean.js" "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_DNS.md.dc1c7d42.lean.js" similarity index 70% rename from "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_DNS.md.584b66a4.lean.js" rename to "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_DNS.md.dc1c7d42.lean.js" index bf3f6e59..a6f461d7 100644 --- "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_DNS.md.584b66a4.lean.js" +++ "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_DNS.md.dc1c7d42.lean.js" @@ -1 +1 @@ -import{_ as l,o as i,c as a,Q as e}from"./chunks/framework.b6910bb2.js";const S=JSON.parse('{"title":"DNS","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/DNS.md","filePath":"guide/网络相关/DNS.md","lastUpdated":1736065650000}'),o={name:"guide/网络相关/DNS.md"},t=e("",6),r=[t];function s(d,c,n,_,h,u){return i(),a("div",null,r)}const m=l(o,[["render",s]]);export{S as __pageData,m as default}; +import{_ as l,o as i,c as a,Q as e}from"./chunks/framework.b6910bb2.js";const S=JSON.parse('{"title":"DNS","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/DNS.md","filePath":"guide/网络相关/DNS.md","lastUpdated":1736839070000}'),o={name:"guide/网络相关/DNS.md"},t=e("",6),r=[t];function s(d,c,n,_,h,u){return i(),a("div",null,r)}const m=l(o,[["render",s]]);export{S as __pageData,m as default}; diff --git "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_GET\350\257\267\346\261\202\345\222\214POST\350\257\267\346\261\202\347\232\204\345\214\272\345\210\253.md.2f19dc13.js" "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_GET\350\257\267\346\261\202\345\222\214POST\350\257\267\346\261\202\347\232\204\345\214\272\345\210\253.md.675258cb.js" similarity index 95% rename from "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_GET\350\257\267\346\261\202\345\222\214POST\350\257\267\346\261\202\347\232\204\345\214\272\345\210\253.md.2f19dc13.js" rename to "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_GET\350\257\267\346\261\202\345\222\214POST\350\257\267\346\261\202\347\232\204\345\214\272\345\210\253.md.675258cb.js" index 276051b5..56f94ff2 100644 --- "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_GET\350\257\267\346\261\202\345\222\214POST\350\257\267\346\261\202\347\232\204\345\214\272\345\210\253.md.2f19dc13.js" +++ "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_GET\350\257\267\346\261\202\345\222\214POST\350\257\267\346\261\202\347\232\204\345\214\272\345\210\253.md.675258cb.js" @@ -1 +1 @@ -import{_ as t,o as a,c as l,k as e,a as o}from"./chunks/framework.b6910bb2.js";const O=JSON.parse('{"title":"GET 请求和 POST 请求的区别","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/GET请求和POST请求的区别.md","filePath":"guide/网络相关/GET请求和POST请求的区别.md","lastUpdated":1736065650000}'),s={name:"guide/网络相关/GET请求和POST请求的区别.md"},n=e("h1",{id:"get-请求和-post-请求的区别",tabindex:"-1"},[o("GET 请求和 POST 请求的区别 "),e("a",{class:"header-anchor",href:"#get-请求和-post-请求的区别","aria-label":'Permalink to "GET 请求和 POST 请求的区别"'},"​")],-1),i=e("ol",null,[e("li",null,"功能上: GET 请求常用于获取数据,POST 请求常用于提交数据"),e("li",null,"参数上: GET 请求参数放在 URL 上,POST 请求参数放在请求体上"),e("li",null,"参数类型上: POST 请求支持更多的参数类型"),e("li",null,"安全性上: POST 请求更加安全"),e("li",null,"是否有缓存上: GET 请求会被缓存"),e("li",null,"请求长度上: 由于浏览器存在 URL 长度限制,因此会影响 GET 请求的长度")],-1),r=[n,i];function _(d,T,c,u,p,P){return a(),l("div",null,r)}const S=t(s,[["render",_]]);export{O as __pageData,S as default}; +import{_ as t,o as a,c as l,k as e,a as o}from"./chunks/framework.b6910bb2.js";const O=JSON.parse('{"title":"GET 请求和 POST 请求的区别","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/GET请求和POST请求的区别.md","filePath":"guide/网络相关/GET请求和POST请求的区别.md","lastUpdated":1736839070000}'),s={name:"guide/网络相关/GET请求和POST请求的区别.md"},n=e("h1",{id:"get-请求和-post-请求的区别",tabindex:"-1"},[o("GET 请求和 POST 请求的区别 "),e("a",{class:"header-anchor",href:"#get-请求和-post-请求的区别","aria-label":'Permalink to "GET 请求和 POST 请求的区别"'},"​")],-1),i=e("ol",null,[e("li",null,"功能上: GET 请求常用于获取数据,POST 请求常用于提交数据"),e("li",null,"参数上: GET 请求参数放在 URL 上,POST 请求参数放在请求体上"),e("li",null,"参数类型上: POST 请求支持更多的参数类型"),e("li",null,"安全性上: POST 请求更加安全"),e("li",null,"是否有缓存上: GET 请求会被缓存"),e("li",null,"请求长度上: 由于浏览器存在 URL 长度限制,因此会影响 GET 请求的长度")],-1),r=[n,i];function _(d,T,c,u,p,P){return a(),l("div",null,r)}const S=t(s,[["render",_]]);export{O as __pageData,S as default}; diff --git "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_GET\350\257\267\346\261\202\345\222\214POST\350\257\267\346\261\202\347\232\204\345\214\272\345\210\253.md.2f19dc13.lean.js" "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_GET\350\257\267\346\261\202\345\222\214POST\350\257\267\346\261\202\347\232\204\345\214\272\345\210\253.md.675258cb.lean.js" similarity index 95% rename from "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_GET\350\257\267\346\261\202\345\222\214POST\350\257\267\346\261\202\347\232\204\345\214\272\345\210\253.md.2f19dc13.lean.js" rename to "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_GET\350\257\267\346\261\202\345\222\214POST\350\257\267\346\261\202\347\232\204\345\214\272\345\210\253.md.675258cb.lean.js" index 276051b5..56f94ff2 100644 --- "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_GET\350\257\267\346\261\202\345\222\214POST\350\257\267\346\261\202\347\232\204\345\214\272\345\210\253.md.2f19dc13.lean.js" +++ "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_GET\350\257\267\346\261\202\345\222\214POST\350\257\267\346\261\202\347\232\204\345\214\272\345\210\253.md.675258cb.lean.js" @@ -1 +1 @@ -import{_ as t,o as a,c as l,k as e,a as o}from"./chunks/framework.b6910bb2.js";const O=JSON.parse('{"title":"GET 请求和 POST 请求的区别","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/GET请求和POST请求的区别.md","filePath":"guide/网络相关/GET请求和POST请求的区别.md","lastUpdated":1736065650000}'),s={name:"guide/网络相关/GET请求和POST请求的区别.md"},n=e("h1",{id:"get-请求和-post-请求的区别",tabindex:"-1"},[o("GET 请求和 POST 请求的区别 "),e("a",{class:"header-anchor",href:"#get-请求和-post-请求的区别","aria-label":'Permalink to "GET 请求和 POST 请求的区别"'},"​")],-1),i=e("ol",null,[e("li",null,"功能上: GET 请求常用于获取数据,POST 请求常用于提交数据"),e("li",null,"参数上: GET 请求参数放在 URL 上,POST 请求参数放在请求体上"),e("li",null,"参数类型上: POST 请求支持更多的参数类型"),e("li",null,"安全性上: POST 请求更加安全"),e("li",null,"是否有缓存上: GET 请求会被缓存"),e("li",null,"请求长度上: 由于浏览器存在 URL 长度限制,因此会影响 GET 请求的长度")],-1),r=[n,i];function _(d,T,c,u,p,P){return a(),l("div",null,r)}const S=t(s,[["render",_]]);export{O as __pageData,S as default}; +import{_ as t,o as a,c as l,k as e,a as o}from"./chunks/framework.b6910bb2.js";const O=JSON.parse('{"title":"GET 请求和 POST 请求的区别","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/GET请求和POST请求的区别.md","filePath":"guide/网络相关/GET请求和POST请求的区别.md","lastUpdated":1736839070000}'),s={name:"guide/网络相关/GET请求和POST请求的区别.md"},n=e("h1",{id:"get-请求和-post-请求的区别",tabindex:"-1"},[o("GET 请求和 POST 请求的区别 "),e("a",{class:"header-anchor",href:"#get-请求和-post-请求的区别","aria-label":'Permalink to "GET 请求和 POST 请求的区别"'},"​")],-1),i=e("ol",null,[e("li",null,"功能上: GET 请求常用于获取数据,POST 请求常用于提交数据"),e("li",null,"参数上: GET 请求参数放在 URL 上,POST 请求参数放在请求体上"),e("li",null,"参数类型上: POST 请求支持更多的参数类型"),e("li",null,"安全性上: POST 请求更加安全"),e("li",null,"是否有缓存上: GET 请求会被缓存"),e("li",null,"请求长度上: 由于浏览器存在 URL 长度限制,因此会影响 GET 请求的长度")],-1),r=[n,i];function _(d,T,c,u,p,P){return a(),l("div",null,r)}const S=t(s,[["render",_]]);export{O as __pageData,S as default}; diff --git "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTP.md.5112c2aa.js" "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTP.md.2fd00196.js" similarity index 98% rename from "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTP.md.5112c2aa.js" rename to "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTP.md.2fd00196.js" index 37542b4a..4f2a62c4 100644 --- "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTP.md.5112c2aa.js" +++ "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTP.md.2fd00196.js" @@ -1 +1 @@ -import{_ as a,o as t,c as e,Q as l}from"./chunks/framework.b6910bb2.js";const i="/vitePress-blob/assets/5.c8205ece.jpg",o="/vitePress-blob/assets/6.76a1cb50.jpg",r="/vitePress-blob/assets/7.ee9188c9.jpg",H=JSON.parse('{"title":"HTTP","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/HTTP.md","filePath":"guide/网络相关/HTTP.md","lastUpdated":1736065650000}'),h={name:"guide/网络相关/HTTP.md"},T=l('

HTTP

HTTP 0.9

早期做学术研究使用,只是单纯的文本传输(只有请求行、没有请求头和请求体)

HTTP1.0

支持图片、视频传输,但传输效率太低

核心需求:支持多种类型的文件下载(引入了请求头和响应头)

HTTP1.1

持久连接 keep-alive

在此之前一个 HTTP 请求需要建立一个 TCP 连接

目前浏览器中对于 同一个域名,默认允许同时建立 6 个 TCP 连接

管道化连接

将串行请求改成并行请求

支持虚拟主机

HTTP 根据 Host 去区分同一 IP 物理机子上的不同虚拟主机

对动态生成的内容提供完美支持

Chunk transfer 机制

Cookie、完全机制

主要问题

    1. HTTP/1.1 对头阻塞问题 在 HTTP 中,因为前一条请求因某些原因没有及时返回,会导致后面的请求无法发起,容易导致对头阻塞(串行)。虽然 HTTP1.1 引入了管道化技术尝试解决队头阻塞问题,但服务端还是需要按顺序处理请求并返回,因此阻塞问题依然存在
    1. 带宽利用率不理想
    • 原因:
      • TCP 的慢启动: 传输数据的速度从慢到快
      • 同时开启多个 TCP 连接,那么这些连接会竞争固定的带宽(带宽不足时,TCP 传输速度会变慢)

HTTP/1.0 与 HTTP/1.1 的 区别?

1 连接方面

持久化连接: HTTP1.1 支持持久化连接,减少每次 HTTP 请求都需要建立 TCP 连接的耗时

管道化连接:提高请求效率

2 缓存方面

HTTP1.1 新增了协商缓存的字段 ETag 比 if-None-Match 更加紧准

3 新增 Host 字段

根据 Host 区分同一 IP 服务器上的不同网站

之前只能通过 IP 区分网站

4 请求资源方面

添加范围请求 (206 状态码)

5 新增了请求方法

PUT、HEAD、OPTIONS 等

HTTP/2.0

一个域名只有一个 TCP 长连接传输数据 (避规 HTTP 1.1 的 TCP 慢启动和多个 TCP 连接时竞争带宽资源)

多路复用机制

通过引入二进制分帧层,实现多路复用机制

整体流程整体流程

  1. 浏览器发起请求,经过二进制分帧层将请求拆分成一个个带请求id的帧,发送给服务端
  2. 服务端收到这些帧后,根据请求id组装成完整请求信息,服务端处理完请求后,将响应请求发送到二进制分帧层,经过二进制分帧层转化为一个个带有响应id的帧发送给浏览器
  3. 浏览器收到这些帧后,会组合成一个完整的响应请求

实现基础: HTTP2 是一个二进制协议,在 HTTP/1.1 版中,报文的头信息必须是文本(ASCII 编码),数据体可以是文本,也可以是二进制。HTTP/2 则是一个彻底的二进制协议,头信息和数据体都是二进制,并且统称为"帧",可以分为头信息帧和数据帧。 帧的概念是它实现多路复用的基础

头部压缩

HTTP/2 实现了头信息压缩,由于 HTTP 1.1 协议不带状态,每次请求都必须附上所有信息。所以,请求的很多字段都是重复的,比如 Cookie 和 User Agent ,一模一样的内容,每次请求都必须附带,这会浪费很多带宽,也影响速度。HTTP/2 对这一点做了优化,引入了头信息压缩机制。一方面,头信息使用 gzip 或 compress 压缩后再发送;另一方面,客户端和服务器同时维护一张头信息表,所有字段都会存入这个表,生成一个索引号,以后就不发送同样字段了,只发送索引号,这样就能提高速度了。

设置请求优先级

服务器推送

存在的问题

TCP 协议本身存在的问题:

    1. TCP 的队头阻塞 TCP 传输过程中把一份数据分成多个数据包进行传输,当某个数据包没有按顺序返回,接收端会一直保持连接等待数据包返回,这就阻塞了后续数据包的传输 整体流程
    1. TCP 建立连接需要耗时

优点

多路复用技术能有效利用带宽(只有一个TCP连接),缓解 TCP 慢启动带来的问题,解决了 HTTP 队头阻塞问题,同时还支持设置优先级、服务器推送、头部压缩极大的提高了 HTTP 的传输效率

http2 与 http1 的区别?

  1. 为了解决 HTTP1.1 的队头阻塞问题引入了二进制分帧层
  2. 只建立一个 TCP 连接,解决 HTTP1.1中 TCP 慢启动以及多个 TCP 之间相互竞争带宽问题
  3. 头部压缩,HTTP2 请求头部采用头部压缩,减少数据内容的大小,提高效率
  4. 服务端推送: 服务端可以向客户端推送资源

HTTP3.0

采用UDP + QUIC协议

  1. UDP 可以减少连接耗费的 RTT,加快数据传输
  2. 使用 QUIC 协议,可以使用多路复用、TLS、可靠性传输等 TCP 协议的特点

存在的问题:

  1. 规范的制定和落地有着较大的差异(比如官方的QUIC和谷歌的QUIC协议有较大差异)
  2. 是对底层协议的改造,落地成本高。

HTTP1.0/1.1/2.0有什么区别?

这道题应该从HTTP的发展历史的角度出发

  1. http1.0 在 http0.9 时代新增了图片、视频的传输,同时新增了请求、响应头,支持文件下载,但请求的效率低,因为每个http请求需要一次TCP连接
  2. HTTP1.1 引入了缓存策略、支持长连接keep-alive等,支持PUT/DELETE/OPTION方法,提高了HTTP的传输效率,但并发请求时会有一个HTTP的头部阻塞问题
  3. HTTP2 通过引入二进制分帧层、采用多路复用避规了HTTP的头部阻塞问题,还支持头部压缩,提高HTTP的传输效率,服务端推送

RTT

浏览器发送数据包到服务器 + 浏览器接收到服务器确认接收的数据包的往返时间,成为RTT

',60),s=[T];function P(n,p,d,c,u,b){return t(),e("div",null,s)}const m=a(h,[["render",P]]);export{H as __pageData,m as default}; +import{_ as a,o as t,c as e,Q as l}from"./chunks/framework.b6910bb2.js";const i="/vitePress-blob/assets/5.c8205ece.jpg",o="/vitePress-blob/assets/6.76a1cb50.jpg",r="/vitePress-blob/assets/7.ee9188c9.jpg",H=JSON.parse('{"title":"HTTP","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/HTTP.md","filePath":"guide/网络相关/HTTP.md","lastUpdated":1736839070000}'),h={name:"guide/网络相关/HTTP.md"},T=l('

HTTP

HTTP 0.9

早期做学术研究使用,只是单纯的文本传输(只有请求行、没有请求头和请求体)

HTTP1.0

支持图片、视频传输,但传输效率太低

核心需求:支持多种类型的文件下载(引入了请求头和响应头)

HTTP1.1

持久连接 keep-alive

在此之前一个 HTTP 请求需要建立一个 TCP 连接

目前浏览器中对于 同一个域名,默认允许同时建立 6 个 TCP 连接

管道化连接

将串行请求改成并行请求

支持虚拟主机

HTTP 根据 Host 去区分同一 IP 物理机子上的不同虚拟主机

对动态生成的内容提供完美支持

Chunk transfer 机制

Cookie、完全机制

主要问题

    1. HTTP/1.1 对头阻塞问题 在 HTTP 中,因为前一条请求因某些原因没有及时返回,会导致后面的请求无法发起,容易导致对头阻塞(串行)。虽然 HTTP1.1 引入了管道化技术尝试解决队头阻塞问题,但服务端还是需要按顺序处理请求并返回,因此阻塞问题依然存在
    1. 带宽利用率不理想
    • 原因:
      • TCP 的慢启动: 传输数据的速度从慢到快
      • 同时开启多个 TCP 连接,那么这些连接会竞争固定的带宽(带宽不足时,TCP 传输速度会变慢)

HTTP/1.0 与 HTTP/1.1 的 区别?

1 连接方面

持久化连接: HTTP1.1 支持持久化连接,减少每次 HTTP 请求都需要建立 TCP 连接的耗时

管道化连接:提高请求效率

2 缓存方面

HTTP1.1 新增了协商缓存的字段 ETag 比 if-None-Match 更加紧准

3 新增 Host 字段

根据 Host 区分同一 IP 服务器上的不同网站

之前只能通过 IP 区分网站

4 请求资源方面

添加范围请求 (206 状态码)

5 新增了请求方法

PUT、HEAD、OPTIONS 等

HTTP/2.0

一个域名只有一个 TCP 长连接传输数据 (避规 HTTP 1.1 的 TCP 慢启动和多个 TCP 连接时竞争带宽资源)

多路复用机制

通过引入二进制分帧层,实现多路复用机制

整体流程整体流程

  1. 浏览器发起请求,经过二进制分帧层将请求拆分成一个个带请求id的帧,发送给服务端
  2. 服务端收到这些帧后,根据请求id组装成完整请求信息,服务端处理完请求后,将响应请求发送到二进制分帧层,经过二进制分帧层转化为一个个带有响应id的帧发送给浏览器
  3. 浏览器收到这些帧后,会组合成一个完整的响应请求

实现基础: HTTP2 是一个二进制协议,在 HTTP/1.1 版中,报文的头信息必须是文本(ASCII 编码),数据体可以是文本,也可以是二进制。HTTP/2 则是一个彻底的二进制协议,头信息和数据体都是二进制,并且统称为"帧",可以分为头信息帧和数据帧。 帧的概念是它实现多路复用的基础

头部压缩

HTTP/2 实现了头信息压缩,由于 HTTP 1.1 协议不带状态,每次请求都必须附上所有信息。所以,请求的很多字段都是重复的,比如 Cookie 和 User Agent ,一模一样的内容,每次请求都必须附带,这会浪费很多带宽,也影响速度。HTTP/2 对这一点做了优化,引入了头信息压缩机制。一方面,头信息使用 gzip 或 compress 压缩后再发送;另一方面,客户端和服务器同时维护一张头信息表,所有字段都会存入这个表,生成一个索引号,以后就不发送同样字段了,只发送索引号,这样就能提高速度了。

设置请求优先级

服务器推送

存在的问题

TCP 协议本身存在的问题:

    1. TCP 的队头阻塞 TCP 传输过程中把一份数据分成多个数据包进行传输,当某个数据包没有按顺序返回,接收端会一直保持连接等待数据包返回,这就阻塞了后续数据包的传输 整体流程
    1. TCP 建立连接需要耗时

优点

多路复用技术能有效利用带宽(只有一个TCP连接),缓解 TCP 慢启动带来的问题,解决了 HTTP 队头阻塞问题,同时还支持设置优先级、服务器推送、头部压缩极大的提高了 HTTP 的传输效率

http2 与 http1 的区别?

  1. 为了解决 HTTP1.1 的队头阻塞问题引入了二进制分帧层
  2. 只建立一个 TCP 连接,解决 HTTP1.1中 TCP 慢启动以及多个 TCP 之间相互竞争带宽问题
  3. 头部压缩,HTTP2 请求头部采用头部压缩,减少数据内容的大小,提高效率
  4. 服务端推送: 服务端可以向客户端推送资源

HTTP3.0

采用UDP + QUIC协议

  1. UDP 可以减少连接耗费的 RTT,加快数据传输
  2. 使用 QUIC 协议,可以使用多路复用、TLS、可靠性传输等 TCP 协议的特点

存在的问题:

  1. 规范的制定和落地有着较大的差异(比如官方的QUIC和谷歌的QUIC协议有较大差异)
  2. 是对底层协议的改造,落地成本高。

HTTP1.0/1.1/2.0有什么区别?

这道题应该从HTTP的发展历史的角度出发

  1. http1.0 在 http0.9 时代新增了图片、视频的传输,同时新增了请求、响应头,支持文件下载,但请求的效率低,因为每个http请求需要一次TCP连接
  2. HTTP1.1 引入了缓存策略、支持长连接keep-alive等,支持PUT/DELETE/OPTION方法,提高了HTTP的传输效率,但并发请求时会有一个HTTP的头部阻塞问题
  3. HTTP2 通过引入二进制分帧层、采用多路复用避规了HTTP的头部阻塞问题,还支持头部压缩,提高HTTP的传输效率,服务端推送

RTT

浏览器发送数据包到服务器 + 浏览器接收到服务器确认接收的数据包的往返时间,成为RTT

',60),s=[T];function P(n,p,d,c,u,b){return t(),e("div",null,s)}const m=a(h,[["render",P]]);export{H as __pageData,m as default}; diff --git "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTP.md.5112c2aa.lean.js" "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTP.md.2fd00196.lean.js" similarity index 77% rename from "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTP.md.5112c2aa.lean.js" rename to "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTP.md.2fd00196.lean.js" index 7fc2c4a0..455ef0aa 100644 --- "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTP.md.5112c2aa.lean.js" +++ "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTP.md.2fd00196.lean.js" @@ -1 +1 @@ -import{_ as a,o as t,c as e,Q as l}from"./chunks/framework.b6910bb2.js";const i="/vitePress-blob/assets/5.c8205ece.jpg",o="/vitePress-blob/assets/6.76a1cb50.jpg",r="/vitePress-blob/assets/7.ee9188c9.jpg",H=JSON.parse('{"title":"HTTP","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/HTTP.md","filePath":"guide/网络相关/HTTP.md","lastUpdated":1736065650000}'),h={name:"guide/网络相关/HTTP.md"},T=l("",60),s=[T];function P(n,p,d,c,u,b){return t(),e("div",null,s)}const m=a(h,[["render",P]]);export{H as __pageData,m as default}; +import{_ as a,o as t,c as e,Q as l}from"./chunks/framework.b6910bb2.js";const i="/vitePress-blob/assets/5.c8205ece.jpg",o="/vitePress-blob/assets/6.76a1cb50.jpg",r="/vitePress-blob/assets/7.ee9188c9.jpg",H=JSON.parse('{"title":"HTTP","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/HTTP.md","filePath":"guide/网络相关/HTTP.md","lastUpdated":1736839070000}'),h={name:"guide/网络相关/HTTP.md"},T=l("",60),s=[T];function P(n,p,d,c,u,b){return t(),e("div",null,s)}const m=a(h,[["render",P]]);export{H as __pageData,m as default}; diff --git "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTPS.md.efe51aa4.js" "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTPS.md.1a833174.js" similarity index 98% rename from "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTPS.md.efe51aa4.js" rename to "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTPS.md.1a833174.js" index d1cc6a6e..c2c8a79f 100644 --- "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTPS.md.efe51aa4.js" +++ "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTPS.md.1a833174.js" @@ -1 +1 @@ -import{_ as a,o as t,c as e,Q as i}from"./chunks/framework.b6910bb2.js";const l="/vitePress-blob/assets/8.428d7ec3.jpg",s="/vitePress-blob/assets/9.7081f11f.jpg",o="/vitePress-blob/assets/10.80b27028.jpg",r="/vitePress-blob/assets/11.8b44a308.jpg",h="/vitePress-blob/assets/12.c1a5b37b.jpg",S=JSON.parse('{"title":"HTTPS","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/HTTPS.md","filePath":"guide/网络相关/HTTPS.md","lastUpdated":1736065650000}'),T={name:"guide/网络相关/HTTPS.md"},n=i('

HTTPS

HTTP 和 HTTPS 的区别

  1. HTTP 是超文本传输协议,明文传输、简单、无状态,而 HTTPS 是在 HTTP 基础上增加了 SSL 协议,更加安全
  2. HTTP 协议默认端口是 80,HTTPS 协议默认端口是 443
  3. HTTPS 需要 CA 证书、相对于 HTTP 费用更高

HTTPS 演进过程

第一版对称加密 - 不安全

流程

非对称加密 - 传输效率低

流程

对称加密与非对称加密结合 - 存在中间人攻击

流程

通过数字证书校验网站的真实性和获取网站的公钥

之前传输的是密钥,这种方式传输的是装有密钥的保险箱,就算获取到了保险箱,也没有保险箱的钥匙🔑

流程

  1. 浏览器将对称加密方法列表、非对称加密方法列表、随机数 A 传输给服务端
  2. 服务器接受后,将对称加密、非对称加密方法、服务器生成的随机数 B 、数字证书发送给浏览器
  3. 浏览器接受后,验证证书的可靠性,并获取证书内的非对称加密的公钥
  4. 浏览器通过 2 个随机数生成新的随机数 C,并通过非对成加密的公钥对随机数 C 进行加密发送给服务器
  5. 服务器确认后,服务器与浏览器通过 3 个随机数生成对称加密密钥,进行数据传输

如何验证证书的可靠性

  1. 浏览器获取到证书后,通过 CA 相同的 Hash 算法对证书信息进行加密得到摘要 A
  2. 通过 CA 的公钥对证书内的数字签名进行解密,获取到摘要 B
  3. 如果 AB 相同,则证书可靠

什么是 HTTPS 中间人攻击?如何预防?

  1. 先说下 HTTPS 传输过程
  2. 客户端和服务端通信之间,新增一个中间人,伪造 CA 证书和加密数据,获取服务器和客户端的通信信息

流程

如何预防?

使用正规厂商的第三方证书

',21),c=[n];function p(d,P,_,b,m,u){return t(),e("div",null,c)}const f=a(T,[["render",p]]);export{S as __pageData,f as default}; +import{_ as a,o as t,c as e,Q as i}from"./chunks/framework.b6910bb2.js";const l="/vitePress-blob/assets/8.428d7ec3.jpg",s="/vitePress-blob/assets/9.7081f11f.jpg",o="/vitePress-blob/assets/10.80b27028.jpg",r="/vitePress-blob/assets/11.8b44a308.jpg",h="/vitePress-blob/assets/12.c1a5b37b.jpg",S=JSON.parse('{"title":"HTTPS","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/HTTPS.md","filePath":"guide/网络相关/HTTPS.md","lastUpdated":1736839070000}'),T={name:"guide/网络相关/HTTPS.md"},n=i('

HTTPS

HTTP 和 HTTPS 的区别

  1. HTTP 是超文本传输协议,明文传输、简单、无状态,而 HTTPS 是在 HTTP 基础上增加了 SSL 协议,更加安全
  2. HTTP 协议默认端口是 80,HTTPS 协议默认端口是 443
  3. HTTPS 需要 CA 证书、相对于 HTTP 费用更高

HTTPS 演进过程

第一版对称加密 - 不安全

流程

非对称加密 - 传输效率低

流程

对称加密与非对称加密结合 - 存在中间人攻击

流程

通过数字证书校验网站的真实性和获取网站的公钥

之前传输的是密钥,这种方式传输的是装有密钥的保险箱,就算获取到了保险箱,也没有保险箱的钥匙🔑

流程

  1. 浏览器将对称加密方法列表、非对称加密方法列表、随机数 A 传输给服务端
  2. 服务器接受后,将对称加密、非对称加密方法、服务器生成的随机数 B 、数字证书发送给浏览器
  3. 浏览器接受后,验证证书的可靠性,并获取证书内的非对称加密的公钥
  4. 浏览器通过 2 个随机数生成新的随机数 C,并通过非对成加密的公钥对随机数 C 进行加密发送给服务器
  5. 服务器确认后,服务器与浏览器通过 3 个随机数生成对称加密密钥,进行数据传输

如何验证证书的可靠性

  1. 浏览器获取到证书后,通过 CA 相同的 Hash 算法对证书信息进行加密得到摘要 A
  2. 通过 CA 的公钥对证书内的数字签名进行解密,获取到摘要 B
  3. 如果 AB 相同,则证书可靠

什么是 HTTPS 中间人攻击?如何预防?

  1. 先说下 HTTPS 传输过程
  2. 客户端和服务端通信之间,新增一个中间人,伪造 CA 证书和加密数据,获取服务器和客户端的通信信息

流程

如何预防?

使用正规厂商的第三方证书

',21),c=[n];function p(d,P,_,b,m,u){return t(),e("div",null,c)}const f=a(T,[["render",p]]);export{S as __pageData,f as default}; diff --git "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTPS.md.efe51aa4.lean.js" "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTPS.md.1a833174.lean.js" similarity index 90% rename from "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTPS.md.efe51aa4.lean.js" rename to "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTPS.md.1a833174.lean.js" index 43ebdbf0..34f9104d 100644 --- "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTPS.md.efe51aa4.lean.js" +++ "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_HTTPS.md.1a833174.lean.js" @@ -1 +1 @@ -import{_ as a,o as t,c as e,Q as i}from"./chunks/framework.b6910bb2.js";const l="/vitePress-blob/assets/8.428d7ec3.jpg",s="/vitePress-blob/assets/9.7081f11f.jpg",o="/vitePress-blob/assets/10.80b27028.jpg",r="/vitePress-blob/assets/11.8b44a308.jpg",h="/vitePress-blob/assets/12.c1a5b37b.jpg",S=JSON.parse('{"title":"HTTPS","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/HTTPS.md","filePath":"guide/网络相关/HTTPS.md","lastUpdated":1736065650000}'),T={name:"guide/网络相关/HTTPS.md"},n=i("",21),c=[n];function p(d,P,_,b,m,u){return t(),e("div",null,c)}const f=a(T,[["render",p]]);export{S as __pageData,f as default}; +import{_ as a,o as t,c as e,Q as i}from"./chunks/framework.b6910bb2.js";const l="/vitePress-blob/assets/8.428d7ec3.jpg",s="/vitePress-blob/assets/9.7081f11f.jpg",o="/vitePress-blob/assets/10.80b27028.jpg",r="/vitePress-blob/assets/11.8b44a308.jpg",h="/vitePress-blob/assets/12.c1a5b37b.jpg",S=JSON.parse('{"title":"HTTPS","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/HTTPS.md","filePath":"guide/网络相关/HTTPS.md","lastUpdated":1736839070000}'),T={name:"guide/网络相关/HTTPS.md"},n=i("",21),c=[n];function p(d,P,_,b,m,u){return t(),e("div",null,c)}const f=a(T,[["render",p]]);export{S as __pageData,f as default}; diff --git "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\223\215\345\272\224\346\212\245\346\226\207.md.cc269f40.js" "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\223\215\345\272\224\346\212\245\346\226\207.md.b46d175f.js" similarity index 99% rename from "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\223\215\345\272\224\346\212\245\346\226\207.md.cc269f40.js" rename to "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\223\215\345\272\224\346\212\245\346\226\207.md.b46d175f.js" index b29d49b7..d76d2ac9 100644 --- "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\223\215\345\272\224\346\212\245\346\226\207.md.cc269f40.js" +++ "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\223\215\345\272\224\346\212\245\346\226\207.md.b46d175f.js" @@ -1 +1 @@ -import{_ as i,o as l,c as a,Q as e}from"./chunks/framework.b6910bb2.js";const t="",C=JSON.parse('{"title":"响应报文","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/响应报文.md","filePath":"guide/网络相关/响应报文.md","lastUpdated":1736065650000}'),o={name:"guide/网络相关/响应报文.md"},r=e('

响应报文

由响应行、响应头、响应体组成

响应行

由版本、状态码、原因语句组成

状态码

  • 1xx: 请求已被接受,需要继续处理

  • 2xx: 请求已成功处理

    • 200 请求正常处理、命中强缓存
    • 204 请求处理成功,但没有资源返回
    • 206 客户端进行了范围请求,服务端成功执行了这部分 GET 请求
  • 3xx:客户端需要采取进一步操作才能完成

    • 301 永久重定向
    • 302 临时重定向
    • 304 命中协商缓存
  • 4xx: 客户端错误

    • 400 请求存在语法错误
    • 401 用户登录权限不通过
    • 403 用户登录了,但操作权限不通过
    • 404 资源不存在
    • 405 请求行中制定的方法不能被用于请求相应的资源
  • 5xx: 服务端错误

    • 500 服务端报错
    • 504 服务/网关超时

流程

响应头

响应内容相关

  • content-type 内容类型 Content-Type: text/plain;charset=UTF-8
    • 常见的有:
      • application/x-www-form-urlencoded: 浏览器的原生 form 表单,如果不设置 enctype 属性,那么最终就会以 application/x-www-form-urlencoded 方式提交数据。该种方式提交的数据放在 body 里面,数据按照 key1=val1&key2=val2 的方式进行编码,key 和 val 都进行了 URL转码
      • application/json: 服务器消息主体是序列化后的 JSON 字符串
      • multipart/form-data: 该种方式也是一个常见的 POST 提交方式,通常表单上传文件时使用该种方式。
  • content-length 内容长度

缓存相关

  • Cache-Control
  • Last-Modified
  • ETag

客户端相关

  • Set-Cookie: isGray=true;

跨域相关

  • 简单请求

    • Access-Control-Allow-Origin
  • 非简单请求

    • Access-Control-Allow-Methods
    • Access-Control-Allow-Headers
    • Access-Control-Max-Age
  • Cookie 相关

    • Access-Control-Allow-Credentials: true;

时间相关

Date: Mon, 21 Mar 2022 03:36:53 GMT

响应体

',19),d=[r];function n(h,s,p,u,c,A){return l(),a("div",null,d)}const f=i(o,[["render",n]]);export{C as __pageData,f as default}; +import{_ as i,o as l,c as a,Q as e}from"./chunks/framework.b6910bb2.js";const t="",C=JSON.parse('{"title":"响应报文","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/响应报文.md","filePath":"guide/网络相关/响应报文.md","lastUpdated":1736839070000}'),o={name:"guide/网络相关/响应报文.md"},r=e('

响应报文

由响应行、响应头、响应体组成

响应行

由版本、状态码、原因语句组成

状态码

  • 1xx: 请求已被接受,需要继续处理

  • 2xx: 请求已成功处理

    • 200 请求正常处理、命中强缓存
    • 204 请求处理成功,但没有资源返回
    • 206 客户端进行了范围请求,服务端成功执行了这部分 GET 请求
  • 3xx:客户端需要采取进一步操作才能完成

    • 301 永久重定向
    • 302 临时重定向
    • 304 命中协商缓存
  • 4xx: 客户端错误

    • 400 请求存在语法错误
    • 401 用户登录权限不通过
    • 403 用户登录了,但操作权限不通过
    • 404 资源不存在
    • 405 请求行中制定的方法不能被用于请求相应的资源
  • 5xx: 服务端错误

    • 500 服务端报错
    • 504 服务/网关超时

流程

响应头

响应内容相关

  • content-type 内容类型 Content-Type: text/plain;charset=UTF-8
    • 常见的有:
      • application/x-www-form-urlencoded: 浏览器的原生 form 表单,如果不设置 enctype 属性,那么最终就会以 application/x-www-form-urlencoded 方式提交数据。该种方式提交的数据放在 body 里面,数据按照 key1=val1&key2=val2 的方式进行编码,key 和 val 都进行了 URL转码
      • application/json: 服务器消息主体是序列化后的 JSON 字符串
      • multipart/form-data: 该种方式也是一个常见的 POST 提交方式,通常表单上传文件时使用该种方式。
  • content-length 内容长度

缓存相关

  • Cache-Control
  • Last-Modified
  • ETag

客户端相关

  • Set-Cookie: isGray=true;

跨域相关

  • 简单请求

    • Access-Control-Allow-Origin
  • 非简单请求

    • Access-Control-Allow-Methods
    • Access-Control-Allow-Headers
    • Access-Control-Max-Age
  • Cookie 相关

    • Access-Control-Allow-Credentials: true;

时间相关

Date: Mon, 21 Mar 2022 03:36:53 GMT

响应体

',19),d=[r];function n(h,s,p,u,c,A){return l(),a("div",null,d)}const f=i(o,[["render",n]]);export{C as __pageData,f as default}; diff --git "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\223\215\345\272\224\346\212\245\346\226\207.md.cc269f40.lean.js" "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\223\215\345\272\224\346\212\245\346\226\207.md.b46d175f.lean.js" similarity index 98% rename from "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\223\215\345\272\224\346\212\245\346\226\207.md.cc269f40.lean.js" rename to "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\223\215\345\272\224\346\212\245\346\226\207.md.b46d175f.lean.js" index f51d012b..04b9d909 100644 --- "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\223\215\345\272\224\346\212\245\346\226\207.md.cc269f40.lean.js" +++ "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\223\215\345\272\224\346\212\245\346\226\207.md.b46d175f.lean.js" @@ -1 +1 @@ -import{_ as i,o as l,c as a,Q as e}from"./chunks/framework.b6910bb2.js";const t="",C=JSON.parse('{"title":"响应报文","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/响应报文.md","filePath":"guide/网络相关/响应报文.md","lastUpdated":1736065650000}'),o={name:"guide/网络相关/响应报文.md"},r=e("",19),d=[r];function n(h,s,p,u,c,A){return l(),a("div",null,d)}const f=i(o,[["render",n]]);export{C as __pageData,f as default}; +import{_ as i,o as l,c as a,Q as e}from"./chunks/framework.b6910bb2.js";const t="",C=JSON.parse('{"title":"响应报文","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/响应报文.md","filePath":"guide/网络相关/响应报文.md","lastUpdated":1736839070000}'),o={name:"guide/网络相关/响应报文.md"},r=e("",19),d=[r];function n(h,s,p,u,c,A){return l(),a("div",null,d)}const f=i(o,[["render",n]]);export{C as __pageData,f as default}; diff --git "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.0601ae85.js" "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.fee49d4c.js" similarity index 85% rename from "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.0601ae85.js" rename to "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.fee49d4c.js" index f7667b0e..1f2898af 100644 --- "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.0601ae85.js" +++ "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.fee49d4c.js" @@ -1 +1 @@ -import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/常见的问题.md","filePath":"guide/网络相关/常见的问题.md","lastUpdated":1736065650000}'),r={name:"guide/网络相关/常见的问题.md"};function o(s,_,c,d,n,i){return t(),a("div")}const f=e(r,[["render",o]]);export{m as __pageData,f as default}; +import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/常见的问题.md","filePath":"guide/网络相关/常见的问题.md","lastUpdated":1736839070000}'),r={name:"guide/网络相关/常见的问题.md"};function o(s,_,c,d,n,i){return t(),a("div")}const f=e(r,[["render",o]]);export{m as __pageData,f as default}; diff --git "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.0601ae85.lean.js" "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.fee49d4c.lean.js" similarity index 85% rename from "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.0601ae85.lean.js" rename to "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.fee49d4c.lean.js" index f7667b0e..1f2898af 100644 --- "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.0601ae85.lean.js" +++ "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.md.fee49d4c.lean.js" @@ -1 +1 @@ -import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/常见的问题.md","filePath":"guide/网络相关/常见的问题.md","lastUpdated":1736065650000}'),r={name:"guide/网络相关/常见的问题.md"};function o(s,_,c,d,n,i){return t(),a("div")}const f=e(r,[["render",o]]);export{m as __pageData,f as default}; +import{_ as e,o as t,c as a}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/常见的问题.md","filePath":"guide/网络相关/常见的问题.md","lastUpdated":1736839070000}'),r={name:"guide/网络相关/常见的问题.md"};function o(s,_,c,d,n,i){return t(),a("div")}const f=e(r,[["render",o]]);export{m as __pageData,f as default}; diff --git "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\346\246\202\350\246\201.md.a9f4d298.js" "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\346\246\202\350\246\201.md.3fbc149d.js" similarity index 88% rename from "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\346\246\202\350\246\201.md.a9f4d298.js" rename to "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\346\246\202\350\246\201.md.3fbc149d.js" index 7674a4b1..fbe9efe6 100644 --- "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\346\246\202\350\246\201.md.a9f4d298.js" +++ "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\346\246\202\350\246\201.md.3fbc149d.js" @@ -1 +1 @@ -import{_ as e}from"./chunks/container.fad5294e.js";import{o as t,c as a,H as s}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{"aside":false,"outline":false},"headers":[],"relativePath":"guide/网络相关/概要.md","filePath":"guide/网络相关/概要.md","lastUpdated":1736065650000}'),o={name:"guide/网络相关/概要.md"},n=Object.assign(o,{setup(r){return(c,d)=>(t(),a("div",null,[s(e,{url:"https://www.mubu.com/doc/7RYbHADBDCK#m"})]))}});export{m as __pageData,n as default}; +import{_ as e}from"./chunks/container.fad5294e.js";import{o as t,c as a,H as s}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{"aside":false,"outline":false},"headers":[],"relativePath":"guide/网络相关/概要.md","filePath":"guide/网络相关/概要.md","lastUpdated":1736839070000}'),o={name:"guide/网络相关/概要.md"},n=Object.assign(o,{setup(r){return(c,d)=>(t(),a("div",null,[s(e,{url:"https://www.mubu.com/doc/7RYbHADBDCK#m"})]))}});export{m as __pageData,n as default}; diff --git "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\346\246\202\350\246\201.md.a9f4d298.lean.js" "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\346\246\202\350\246\201.md.3fbc149d.lean.js" similarity index 88% rename from "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\346\246\202\350\246\201.md.a9f4d298.lean.js" rename to "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\346\246\202\350\246\201.md.3fbc149d.lean.js" index 7674a4b1..fbe9efe6 100644 --- "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\346\246\202\350\246\201.md.a9f4d298.lean.js" +++ "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\346\246\202\350\246\201.md.3fbc149d.lean.js" @@ -1 +1 @@ -import{_ as e}from"./chunks/container.fad5294e.js";import{o as t,c as a,H as s}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{"aside":false,"outline":false},"headers":[],"relativePath":"guide/网络相关/概要.md","filePath":"guide/网络相关/概要.md","lastUpdated":1736065650000}'),o={name:"guide/网络相关/概要.md"},n=Object.assign(o,{setup(r){return(c,d)=>(t(),a("div",null,[s(e,{url:"https://www.mubu.com/doc/7RYbHADBDCK#m"})]))}});export{m as __pageData,n as default}; +import{_ as e}from"./chunks/container.fad5294e.js";import{o as t,c as a,H as s}from"./chunks/framework.b6910bb2.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{"aside":false,"outline":false},"headers":[],"relativePath":"guide/网络相关/概要.md","filePath":"guide/网络相关/概要.md","lastUpdated":1736839070000}'),o={name:"guide/网络相关/概要.md"},n=Object.assign(o,{setup(r){return(c,d)=>(t(),a("div",null,[s(e,{url:"https://www.mubu.com/doc/7RYbHADBDCK#m"})]))}});export{m as __pageData,n as default}; diff --git "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\257\267\346\261\202\346\212\245\346\226\207.md.c7f4f8d0.js" "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\257\267\346\261\202\346\212\245\346\226\207.md.dd8a47bc.js" similarity index 98% rename from "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\257\267\346\261\202\346\212\245\346\226\207.md.c7f4f8d0.js" rename to "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\257\267\346\261\202\346\212\245\346\226\207.md.dd8a47bc.js" index 3e3ce68d..e6c4d77e 100644 --- "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\257\267\346\261\202\346\212\245\346\226\207.md.c7f4f8d0.js" +++ "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\257\267\346\261\202\346\212\245\346\226\207.md.dd8a47bc.js" @@ -1 +1 @@ -import{_ as a,o as e,c as i,Q as l}from"./chunks/framework.b6910bb2.js";const t="/vitePress-blob/assets/2.8a82dbf4.png",o="/vitePress-blob/assets/3.8de7cbe1.jpg",m=JSON.parse('{"title":"请求报文","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/请求报文.md","filePath":"guide/网络相关/请求报文.md","lastUpdated":1736065650000}'),r={name:"guide/网络相关/请求报文.md"},n=l('

请求报文

由请求行、请求头、请求体组成

请求行

由请求方法、URI、协议版本组成

流程

请求方法

  1. GET 请求获取资源
  2. POST 提交数据
  3. PUT 更新资源
  4. DELETE 删除资源
  5. OPTIONS 查询针对请求 URL 指定的资源支持的方法(常用在非简单请求的预检上)
  6. HEAD 获取响应头
  7. CONNECT 要求用隧道协议链接代理
  8. TRACE 追踪请求经过的路径

URI 字段

协议版本

HTTP0.9、 HTTP1.0、HTTP1.1、HTTP2.0、HTTP3.0

请求头

接受内容相关

  1. Accept 浏览器接受的格式 Accept: application/json, text/plain, /
  2. Accept-Encoding 浏览器接受的压缩格式 Accept-Encoding: gzip, deflate, br
  3. Accept-Language 浏览器接受的语言 Accept-Language: zh-CN
  4. Accept-Charset 浏览器接受的字符集

强缓存相关

  1. Cache-Control

协商缓存相关

  1. If-Modified-Since 上一次访问时文件的更改时间 (存在校验问题,比如文件内容只是添加了空格,但是 Last-Modified 改变了)
  2. If-None-Match 上次访问的 ETag 信息 (如果 If-None-Match 和 ETag 一致,则命中 304 协商缓存)

请求域名相关

  1. HOST HTTP 请求的域名
  2. Origin 页面域名 (常用于防止 CSRF 攻击和跨域请求)
  3. Referer 发出请求的页面 URL

客户端相关

  • UA
  • COOKIE

连接相关

  • Connection Connection: keep-alive

流程

请求体

',25),h=[n];function c(s,d,u,p,_,b){return e(),i("div",null,h)}const P=a(r,[["render",c]]);export{m as __pageData,P as default}; +import{_ as a,o as e,c as i,Q as l}from"./chunks/framework.b6910bb2.js";const t="/vitePress-blob/assets/2.8a82dbf4.png",o="/vitePress-blob/assets/3.8de7cbe1.jpg",m=JSON.parse('{"title":"请求报文","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/请求报文.md","filePath":"guide/网络相关/请求报文.md","lastUpdated":1736839070000}'),r={name:"guide/网络相关/请求报文.md"},n=l('

请求报文

由请求行、请求头、请求体组成

请求行

由请求方法、URI、协议版本组成

流程

请求方法

  1. GET 请求获取资源
  2. POST 提交数据
  3. PUT 更新资源
  4. DELETE 删除资源
  5. OPTIONS 查询针对请求 URL 指定的资源支持的方法(常用在非简单请求的预检上)
  6. HEAD 获取响应头
  7. CONNECT 要求用隧道协议链接代理
  8. TRACE 追踪请求经过的路径

URI 字段

协议版本

HTTP0.9、 HTTP1.0、HTTP1.1、HTTP2.0、HTTP3.0

请求头

接受内容相关

  1. Accept 浏览器接受的格式 Accept: application/json, text/plain, /
  2. Accept-Encoding 浏览器接受的压缩格式 Accept-Encoding: gzip, deflate, br
  3. Accept-Language 浏览器接受的语言 Accept-Language: zh-CN
  4. Accept-Charset 浏览器接受的字符集

强缓存相关

  1. Cache-Control

协商缓存相关

  1. If-Modified-Since 上一次访问时文件的更改时间 (存在校验问题,比如文件内容只是添加了空格,但是 Last-Modified 改变了)
  2. If-None-Match 上次访问的 ETag 信息 (如果 If-None-Match 和 ETag 一致,则命中 304 协商缓存)

请求域名相关

  1. HOST HTTP 请求的域名
  2. Origin 页面域名 (常用于防止 CSRF 攻击和跨域请求)
  3. Referer 发出请求的页面 URL

客户端相关

  • UA
  • COOKIE

连接相关

  • Connection Connection: keep-alive

流程

请求体

',25),h=[n];function c(s,d,u,p,_,b){return e(),i("div",null,h)}const P=a(r,[["render",c]]);export{m as __pageData,P as default}; diff --git "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\257\267\346\261\202\346\212\245\346\226\207.md.c7f4f8d0.lean.js" "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\257\267\346\261\202\346\212\245\346\226\207.md.dd8a47bc.lean.js" similarity index 88% rename from "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\257\267\346\261\202\346\212\245\346\226\207.md.c7f4f8d0.lean.js" rename to "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\257\267\346\261\202\346\212\245\346\226\207.md.dd8a47bc.lean.js" index a7a719a6..3ef595c3 100644 --- "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\257\267\346\261\202\346\212\245\346\226\207.md.c7f4f8d0.lean.js" +++ "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\257\267\346\261\202\346\212\245\346\226\207.md.dd8a47bc.lean.js" @@ -1 +1 @@ -import{_ as a,o as e,c as i,Q as l}from"./chunks/framework.b6910bb2.js";const t="/vitePress-blob/assets/2.8a82dbf4.png",o="/vitePress-blob/assets/3.8de7cbe1.jpg",m=JSON.parse('{"title":"请求报文","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/请求报文.md","filePath":"guide/网络相关/请求报文.md","lastUpdated":1736065650000}'),r={name:"guide/网络相关/请求报文.md"},n=l("",25),h=[n];function c(s,d,u,p,_,b){return e(),i("div",null,h)}const P=a(r,[["render",c]]);export{m as __pageData,P as default}; +import{_ as a,o as e,c as i,Q as l}from"./chunks/framework.b6910bb2.js";const t="/vitePress-blob/assets/2.8a82dbf4.png",o="/vitePress-blob/assets/3.8de7cbe1.jpg",m=JSON.parse('{"title":"请求报文","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/请求报文.md","filePath":"guide/网络相关/请求报文.md","lastUpdated":1736839070000}'),r={name:"guide/网络相关/请求报文.md"},n=l("",25),h=[n];function c(s,d,u,p,_,b){return e(),i("div",null,h)}const P=a(r,[["render",c]]);export{m as __pageData,P as default}; diff --git "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\267\250\345\237\237\350\257\267\346\261\202.md.bcf87cbb.js" "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\267\250\345\237\237\350\257\267\346\261\202.md.848f5d06.js" similarity index 99% rename from "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\267\250\345\237\237\350\257\267\346\261\202.md.bcf87cbb.js" rename to "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\267\250\345\237\237\350\257\267\346\261\202.md.848f5d06.js" index 7576c86a..15275c11 100644 --- "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\267\250\345\237\237\350\257\267\346\261\202.md.bcf87cbb.js" +++ "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\267\250\345\237\237\350\257\267\346\261\202.md.848f5d06.js" @@ -1 +1 @@ -import{_ as i,o,c as l,Q as e}from"./chunks/framework.b6910bb2.js";const a="/vitePress-blob/assets/7.d0aefc74.jpg",t="/vitePress-blob/assets/8.28d1dcd9.jpg",s="/vitePress-blob/assets/16.ecfc65d2.jpg",r="/vitePress-blob/assets/17.8329667f.jpg",n="/vitePress-blob/assets/18.5204f2f6.jpg",q=JSON.parse('{"title":"跨域请求","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/跨域请求.md","filePath":"guide/网络相关/跨域请求.md","lastUpdated":1736065650000}'),c={name:"guide/网络相关/跨域请求.md"},d=e('

跨域请求

简单请求

若请求满足所有下述条件,则该请求可视为简单请求:

  1. HEAD/GET/POST 请求
  2. Content-Type: text/plain、multipart/form-data、application/x-www-form-urlencoded
  3. 除了被用户代理自动设置的标头字段,剩下的请求头是: Accept、Accept-Language、Content-Language、Content-Type、Range

请求过程

  1. 发起请求,请求头会带上 Origin 字段,该字段用来说明请求来自哪个源(协议 + 域名 + 端口),服务器根据这个值决定是否同意这次请求
  2. 当服务器接收到请求后,根据 Origin 判断是否在允许的范围内
  3. 如果不在范围内,服务器会返回一个正常的 HTTP 响应,浏览器发现响应头信息没有 Access-Control-Allow-Origin 字段,就知道出错了,从而抛出错误,被 XML 的 onerror 回调函数捕获。(注意:由于正常响应,其状态码为200,因此该错误不能通过状态码识别)
  4. 如果 Origin 指定的域名在范围内,服务器返回的响应会多出几个头信息字段(Access-Control-Allow-Origin、Access-Control-Allow-Credentials、Access-Control-Expose-Header 等)

流程

非简单请求

请求方法是 PUT、DELETE 或者 Content-Type 是 application/json 类型

请求过程 (源、请求头、方法、缓存、Cookie)

  1. 浏览器发起 Option 预检请求
  2. 服务器收到预检请求以后,检查了 Origin、Access-Control-Request-Method 和 Access-Control-Request-Headers 字段后,确认允许跨域请求,就可以做出回应
    • Origin (必须): 发起请求的源信息
    • Access-Control-Request-Method(必须): 用来列出浏览器的 CORS 请求会用到哪些 HTTP 方法
    • Access-Control-Request-Headers 跨域请求而外的请求头字段
  3. 如果服务器预检请求,会返回一个正常的 HTTP 回应,但没有任何的 CORS 相关的头信息字段,这时浏览器就会认定服务器不同意预检请求,触发错误。
  4. 如果服务器通过了预检请求,以后每次浏览器正常的 CORS 请求就跟简单请求一样
    • Access-Control-Allow-Origin(必须): 运行哪些源跨域,如果是 * 无法携带 Cookie
    • Access-Control-Allow-Method(必须): 服务器支持哪些请求
    • Access-Control-Allow-Header: 服务器支持的头信息字段
    • Access-Control-Max-Age: 预检请求的有效期,单位秒

流程

  1. 响应头设置 Access-Control-Allow-Credentials 等于 true
  2. Access-Control-Allow-Origin 不能设置成 * (Cookie 的 SameSite 属性如果是 Lax 可能也会导致带不上去)
  3. 前端设置 withCredentials: true

常见的问题

cookie 一般用于登录验证,存储用户信息,大小为 4KB,会随着网络请求携带给服务端

session 一般存储在服务端,常用于与 Cookie 配合做登录检验

  • cookie 是 HTTP 的内容,token 是自定义的数据
  • cookie 可以默认存储在浏览器中,token 需要自行存储
  • token 没有跨域限制,cookie 存在跨域限制
  • token 常用于 CSRF 或者 JWT(JSON WEB TOKEN)
  • Cookie 常于 Session 配合,做用户登录鉴权
  • 基本属性:

    • name
    • value
  • 访问性:

    • expire
    • path
    • domain
  • 安全性:

    • secure
    • httpOnly
    • sameSite

Session 和 JWT 哪个更合适?

  • Session

    • 优点:
      1. 易于学习
      2. 用户信息存储在服务器,可以快速封禁某个用户
    • 缺点:
      1. 占用服务器资源,硬件成本高
      2. 多进程、多服务器时,不好同步 (需要第三方缓存, Redis)
      3. Session 需要配合 Cookie 使用,cookie 有域名限制
  • Json Web Token

    • 优点:
      1. 存储在客户端,不占用服务端资源
      2. 易于同步
      3. 没有跨域限制
    • 缺点:
      1. 无法快速封禁用户
      2. Token 不安全,一但秘钥被泄漏,容易窃取用户信息
      3. Token很大,影响请求体积
  • 使用场景:

    • 用户信息安全 -> 使用 Session
    • 没有特殊要求 -> 使用 JWT

如何实现 SSO 单点登录

Cookie 默认跨域不共享

可以通过设置 Cookie 的 domain 为相同的主域名,即可共享 Cookie

比如 www.baidu.com、image.baidu.com 主域名是相同的,设置 cookie domain 为主域名,即可共享 cookie

流程

如果主域名不一致

  1. 使用 SSO 第三方登录, 获取 ticket 返回给 A、B 系统

流程

  1. OAuth2.0

第三方登录(例如微信扫码登录)

流程

',36),h=[d];function p(u,k,m,b,C,_){return o(),l("div",null,h)}const A=i(c,[["render",p]]);export{q as __pageData,A as default}; +import{_ as i,o,c as l,Q as e}from"./chunks/framework.b6910bb2.js";const a="/vitePress-blob/assets/7.d0aefc74.jpg",t="/vitePress-blob/assets/8.28d1dcd9.jpg",s="/vitePress-blob/assets/16.ecfc65d2.jpg",r="/vitePress-blob/assets/17.8329667f.jpg",n="/vitePress-blob/assets/18.5204f2f6.jpg",q=JSON.parse('{"title":"跨域请求","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/跨域请求.md","filePath":"guide/网络相关/跨域请求.md","lastUpdated":1736839070000}'),c={name:"guide/网络相关/跨域请求.md"},d=e('

跨域请求

简单请求

若请求满足所有下述条件,则该请求可视为简单请求:

  1. HEAD/GET/POST 请求
  2. Content-Type: text/plain、multipart/form-data、application/x-www-form-urlencoded
  3. 除了被用户代理自动设置的标头字段,剩下的请求头是: Accept、Accept-Language、Content-Language、Content-Type、Range

请求过程

  1. 发起请求,请求头会带上 Origin 字段,该字段用来说明请求来自哪个源(协议 + 域名 + 端口),服务器根据这个值决定是否同意这次请求
  2. 当服务器接收到请求后,根据 Origin 判断是否在允许的范围内
  3. 如果不在范围内,服务器会返回一个正常的 HTTP 响应,浏览器发现响应头信息没有 Access-Control-Allow-Origin 字段,就知道出错了,从而抛出错误,被 XML 的 onerror 回调函数捕获。(注意:由于正常响应,其状态码为200,因此该错误不能通过状态码识别)
  4. 如果 Origin 指定的域名在范围内,服务器返回的响应会多出几个头信息字段(Access-Control-Allow-Origin、Access-Control-Allow-Credentials、Access-Control-Expose-Header 等)

流程

非简单请求

请求方法是 PUT、DELETE 或者 Content-Type 是 application/json 类型

请求过程 (源、请求头、方法、缓存、Cookie)

  1. 浏览器发起 Option 预检请求
  2. 服务器收到预检请求以后,检查了 Origin、Access-Control-Request-Method 和 Access-Control-Request-Headers 字段后,确认允许跨域请求,就可以做出回应
    • Origin (必须): 发起请求的源信息
    • Access-Control-Request-Method(必须): 用来列出浏览器的 CORS 请求会用到哪些 HTTP 方法
    • Access-Control-Request-Headers 跨域请求而外的请求头字段
  3. 如果服务器预检请求,会返回一个正常的 HTTP 回应,但没有任何的 CORS 相关的头信息字段,这时浏览器就会认定服务器不同意预检请求,触发错误。
  4. 如果服务器通过了预检请求,以后每次浏览器正常的 CORS 请求就跟简单请求一样
    • Access-Control-Allow-Origin(必须): 运行哪些源跨域,如果是 * 无法携带 Cookie
    • Access-Control-Allow-Method(必须): 服务器支持哪些请求
    • Access-Control-Allow-Header: 服务器支持的头信息字段
    • Access-Control-Max-Age: 预检请求的有效期,单位秒

流程

  1. 响应头设置 Access-Control-Allow-Credentials 等于 true
  2. Access-Control-Allow-Origin 不能设置成 * (Cookie 的 SameSite 属性如果是 Lax 可能也会导致带不上去)
  3. 前端设置 withCredentials: true

常见的问题

cookie 一般用于登录验证,存储用户信息,大小为 4KB,会随着网络请求携带给服务端

session 一般存储在服务端,常用于与 Cookie 配合做登录检验

  • cookie 是 HTTP 的内容,token 是自定义的数据
  • cookie 可以默认存储在浏览器中,token 需要自行存储
  • token 没有跨域限制,cookie 存在跨域限制
  • token 常用于 CSRF 或者 JWT(JSON WEB TOKEN)
  • Cookie 常于 Session 配合,做用户登录鉴权
  • 基本属性:

    • name
    • value
  • 访问性:

    • expire
    • path
    • domain
  • 安全性:

    • secure
    • httpOnly
    • sameSite

Session 和 JWT 哪个更合适?

  • Session

    • 优点:
      1. 易于学习
      2. 用户信息存储在服务器,可以快速封禁某个用户
    • 缺点:
      1. 占用服务器资源,硬件成本高
      2. 多进程、多服务器时,不好同步 (需要第三方缓存, Redis)
      3. Session 需要配合 Cookie 使用,cookie 有域名限制
  • Json Web Token

    • 优点:
      1. 存储在客户端,不占用服务端资源
      2. 易于同步
      3. 没有跨域限制
    • 缺点:
      1. 无法快速封禁用户
      2. Token 不安全,一但秘钥被泄漏,容易窃取用户信息
      3. Token很大,影响请求体积
  • 使用场景:

    • 用户信息安全 -> 使用 Session
    • 没有特殊要求 -> 使用 JWT

如何实现 SSO 单点登录

Cookie 默认跨域不共享

可以通过设置 Cookie 的 domain 为相同的主域名,即可共享 Cookie

比如 www.baidu.com、image.baidu.com 主域名是相同的,设置 cookie domain 为主域名,即可共享 cookie

流程

如果主域名不一致

  1. 使用 SSO 第三方登录, 获取 ticket 返回给 A、B 系统

流程

  1. OAuth2.0

第三方登录(例如微信扫码登录)

流程

',36),h=[d];function p(u,k,m,b,C,_){return o(),l("div",null,h)}const A=i(c,[["render",p]]);export{q as __pageData,A as default}; diff --git "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\267\250\345\237\237\350\257\267\346\261\202.md.bcf87cbb.lean.js" "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\267\250\345\237\237\350\257\267\346\261\202.md.848f5d06.lean.js" similarity index 90% rename from "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\267\250\345\237\237\350\257\267\346\261\202.md.bcf87cbb.lean.js" rename to "assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\267\250\345\237\237\350\257\267\346\261\202.md.848f5d06.lean.js" index 1a411a5a..b0f5597d 100644 --- "a/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\267\250\345\237\237\350\257\267\346\261\202.md.bcf87cbb.lean.js" +++ "b/assets/guide_\347\275\221\347\273\234\347\233\270\345\205\263_\350\267\250\345\237\237\350\257\267\346\261\202.md.848f5d06.lean.js" @@ -1 +1 @@ -import{_ as i,o,c as l,Q as e}from"./chunks/framework.b6910bb2.js";const a="/vitePress-blob/assets/7.d0aefc74.jpg",t="/vitePress-blob/assets/8.28d1dcd9.jpg",s="/vitePress-blob/assets/16.ecfc65d2.jpg",r="/vitePress-blob/assets/17.8329667f.jpg",n="/vitePress-blob/assets/18.5204f2f6.jpg",q=JSON.parse('{"title":"跨域请求","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/跨域请求.md","filePath":"guide/网络相关/跨域请求.md","lastUpdated":1736065650000}'),c={name:"guide/网络相关/跨域请求.md"},d=e("",36),h=[d];function p(u,k,m,b,C,_){return o(),l("div",null,h)}const A=i(c,[["render",p]]);export{q as __pageData,A as default}; +import{_ as i,o,c as l,Q as e}from"./chunks/framework.b6910bb2.js";const a="/vitePress-blob/assets/7.d0aefc74.jpg",t="/vitePress-blob/assets/8.28d1dcd9.jpg",s="/vitePress-blob/assets/16.ecfc65d2.jpg",r="/vitePress-blob/assets/17.8329667f.jpg",n="/vitePress-blob/assets/18.5204f2f6.jpg",q=JSON.parse('{"title":"跨域请求","description":"","frontmatter":{},"headers":[],"relativePath":"guide/网络相关/跨域请求.md","filePath":"guide/网络相关/跨域请求.md","lastUpdated":1736839070000}'),c={name:"guide/网络相关/跨域请求.md"},d=e("",36),h=[d];function p(u,k,m,b,C,_){return o(),l("div",null,h)}const A=i(c,[["render",p]]);export{q as __pageData,A as default}; diff --git a/assets/index.md.b3d414ec.js b/assets/index.md.dd3d0939.js similarity index 92% rename from assets/index.md.b3d414ec.js rename to assets/index.md.dd3d0939.js index d83fa337..bf8ed223 100644 --- a/assets/index.md.b3d414ec.js +++ b/assets/index.md.dd3d0939.js @@ -1 +1 @@ -import{_ as t,o as a,c as s,k as e,a as n}from"./chunks/framework.b6910bb2.js";const k=JSON.parse('{"title":"","description":"","frontmatter":{"layout":"doc"},"headers":[],"relativePath":"index.md","filePath":"index.md","lastUpdated":1736065650000}'),o={name:"index.md"},d=e("h3",{id:"个人知识库",tabindex:"-1"},[n("个人知识库 "),e("a",{class:"header-anchor",href:"#个人知识库","aria-label":'Permalink to "个人知识库"'},"​")],-1),r=e("p",null,"👋 欢迎来到我的博客,这里是我记录学习和生活的地方,希望能帮助到你",-1),_=e("p",null,"采用 VitePress 进行构建,为了有更好的阅读体验,请使用新版本浏览器",-1),c=[d,r,_];function i(l,p,h,m,x,f){return a(),s("div",null,c)}const P=t(o,[["render",i]]);export{k as __pageData,P as default}; +import{_ as t,o as a,c as s,k as e,a as n}from"./chunks/framework.b6910bb2.js";const k=JSON.parse('{"title":"","description":"","frontmatter":{"layout":"doc"},"headers":[],"relativePath":"index.md","filePath":"index.md","lastUpdated":1736839070000}'),o={name:"index.md"},d=e("h3",{id:"个人知识库",tabindex:"-1"},[n("个人知识库 "),e("a",{class:"header-anchor",href:"#个人知识库","aria-label":'Permalink to "个人知识库"'},"​")],-1),r=e("p",null,"👋 欢迎来到我的博客,这里是我记录学习和生活的地方,希望能帮助到你",-1),_=e("p",null,"采用 VitePress 进行构建,为了有更好的阅读体验,请使用新版本浏览器",-1),c=[d,r,_];function i(l,p,h,m,x,f){return a(),s("div",null,c)}const P=t(o,[["render",i]]);export{k as __pageData,P as default}; diff --git a/assets/index.md.b3d414ec.lean.js b/assets/index.md.dd3d0939.lean.js similarity index 92% rename from assets/index.md.b3d414ec.lean.js rename to assets/index.md.dd3d0939.lean.js index d83fa337..bf8ed223 100644 --- a/assets/index.md.b3d414ec.lean.js +++ b/assets/index.md.dd3d0939.lean.js @@ -1 +1 @@ -import{_ as t,o as a,c as s,k as e,a as n}from"./chunks/framework.b6910bb2.js";const k=JSON.parse('{"title":"","description":"","frontmatter":{"layout":"doc"},"headers":[],"relativePath":"index.md","filePath":"index.md","lastUpdated":1736065650000}'),o={name:"index.md"},d=e("h3",{id:"个人知识库",tabindex:"-1"},[n("个人知识库 "),e("a",{class:"header-anchor",href:"#个人知识库","aria-label":'Permalink to "个人知识库"'},"​")],-1),r=e("p",null,"👋 欢迎来到我的博客,这里是我记录学习和生活的地方,希望能帮助到你",-1),_=e("p",null,"采用 VitePress 进行构建,为了有更好的阅读体验,请使用新版本浏览器",-1),c=[d,r,_];function i(l,p,h,m,x,f){return a(),s("div",null,c)}const P=t(o,[["render",i]]);export{k as __pageData,P as default}; +import{_ as t,o as a,c as s,k as e,a as n}from"./chunks/framework.b6910bb2.js";const k=JSON.parse('{"title":"","description":"","frontmatter":{"layout":"doc"},"headers":[],"relativePath":"index.md","filePath":"index.md","lastUpdated":1736839070000}'),o={name:"index.md"},d=e("h3",{id:"个人知识库",tabindex:"-1"},[n("个人知识库 "),e("a",{class:"header-anchor",href:"#个人知识库","aria-label":'Permalink to "个人知识库"'},"​")],-1),r=e("p",null,"👋 欢迎来到我的博客,这里是我记录学习和生活的地方,希望能帮助到你",-1),_=e("p",null,"采用 VitePress 进行构建,为了有更好的阅读体验,请使用新版本浏览器",-1),c=[d,r,_];function i(l,p,h,m,x,f){return a(),s("div",null,c)}const P=t(o,[["render",i]]);export{k as __pageData,P as default}; diff --git "a/assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_ISO\346\204\237\345\205\211\345\272\246.md.2f26bf17.js" "b/assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_ISO\346\204\237\345\205\211\345\272\246.md.1329ace6.js" similarity index 91% rename from "assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_ISO\346\204\237\345\205\211\345\272\246.md.2f26bf17.js" rename to "assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_ISO\346\204\237\345\205\211\345\272\246.md.1329ace6.js" index 9a8de07e..85511852 100644 --- "a/assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_ISO\346\204\237\345\205\211\345\272\246.md.2f26bf17.js" +++ "b/assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_ISO\346\204\237\345\205\211\345\272\246.md.1329ace6.js" @@ -1 +1 @@ -import{_ as t,o as a,c as o,k as e,a as s}from"./chunks/framework.b6910bb2.js";const S=JSON.parse('{"title":"ISO 感光度","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/基础概念/ISO感光度.md","filePath":"shoot/基础概念/ISO感光度.md","lastUpdated":1736065650000}'),n={name:"shoot/基础概念/ISO感光度.md"},r=e("h1",{id:"iso-感光度",tabindex:"-1"},[s("ISO 感光度 "),e("a",{class:"header-anchor",href:"#iso-感光度","aria-label":'Permalink to "ISO 感光度"'},"​")],-1),c=e("p",null,"传感器对光线的敏感程度",-1),l=e("ul",null,[e("li",null,"感光度越大、图片越亮,但也容易出现噪点")],-1),_=[r,c,l];function d(i,h,p,m,f,u){return a(),o("div",null,_)}const I=t(n,[["render",d]]);export{S as __pageData,I as default}; +import{_ as t,o as a,c as o,k as e,a as s}from"./chunks/framework.b6910bb2.js";const S=JSON.parse('{"title":"ISO 感光度","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/基础概念/ISO感光度.md","filePath":"shoot/基础概念/ISO感光度.md","lastUpdated":1736839070000}'),n={name:"shoot/基础概念/ISO感光度.md"},r=e("h1",{id:"iso-感光度",tabindex:"-1"},[s("ISO 感光度 "),e("a",{class:"header-anchor",href:"#iso-感光度","aria-label":'Permalink to "ISO 感光度"'},"​")],-1),c=e("p",null,"传感器对光线的敏感程度",-1),l=e("ul",null,[e("li",null,"感光度越大、图片越亮,但也容易出现噪点")],-1),_=[r,c,l];function d(i,h,p,m,f,u){return a(),o("div",null,_)}const I=t(n,[["render",d]]);export{S as __pageData,I as default}; diff --git "a/assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_ISO\346\204\237\345\205\211\345\272\246.md.2f26bf17.lean.js" "b/assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_ISO\346\204\237\345\205\211\345\272\246.md.1329ace6.lean.js" similarity index 91% rename from "assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_ISO\346\204\237\345\205\211\345\272\246.md.2f26bf17.lean.js" rename to "assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_ISO\346\204\237\345\205\211\345\272\246.md.1329ace6.lean.js" index 9a8de07e..85511852 100644 --- "a/assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_ISO\346\204\237\345\205\211\345\272\246.md.2f26bf17.lean.js" +++ "b/assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_ISO\346\204\237\345\205\211\345\272\246.md.1329ace6.lean.js" @@ -1 +1 @@ -import{_ as t,o as a,c as o,k as e,a as s}from"./chunks/framework.b6910bb2.js";const S=JSON.parse('{"title":"ISO 感光度","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/基础概念/ISO感光度.md","filePath":"shoot/基础概念/ISO感光度.md","lastUpdated":1736065650000}'),n={name:"shoot/基础概念/ISO感光度.md"},r=e("h1",{id:"iso-感光度",tabindex:"-1"},[s("ISO 感光度 "),e("a",{class:"header-anchor",href:"#iso-感光度","aria-label":'Permalink to "ISO 感光度"'},"​")],-1),c=e("p",null,"传感器对光线的敏感程度",-1),l=e("ul",null,[e("li",null,"感光度越大、图片越亮,但也容易出现噪点")],-1),_=[r,c,l];function d(i,h,p,m,f,u){return a(),o("div",null,_)}const I=t(n,[["render",d]]);export{S as __pageData,I as default}; +import{_ as t,o as a,c as o,k as e,a as s}from"./chunks/framework.b6910bb2.js";const S=JSON.parse('{"title":"ISO 感光度","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/基础概念/ISO感光度.md","filePath":"shoot/基础概念/ISO感光度.md","lastUpdated":1736839070000}'),n={name:"shoot/基础概念/ISO感光度.md"},r=e("h1",{id:"iso-感光度",tabindex:"-1"},[s("ISO 感光度 "),e("a",{class:"header-anchor",href:"#iso-感光度","aria-label":'Permalink to "ISO 感光度"'},"​")],-1),c=e("p",null,"传感器对光线的敏感程度",-1),l=e("ul",null,[e("li",null,"感光度越大、图片越亮,但也容易出现噪点")],-1),_=[r,c,l];function d(i,h,p,m,f,u){return a(),o("div",null,_)}const I=t(n,[["render",d]]);export{S as __pageData,I as default}; diff --git "a/assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\205\211\345\234\210.md.ed2aed3b.js" "b/assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\205\211\345\234\210.md.5071eb5f.js" similarity index 92% rename from "assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\205\211\345\234\210.md.ed2aed3b.js" rename to "assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\205\211\345\234\210.md.5071eb5f.js" index a92e7b1c..f120e8d9 100644 --- "a/assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\205\211\345\234\210.md.ed2aed3b.js" +++ "b/assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\205\211\345\234\210.md.5071eb5f.js" @@ -1 +1 @@ -import{_ as t,o as a,c as o,k as e,a as s}from"./chunks/framework.b6910bb2.js";const k=JSON.parse('{"title":"光圈","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/基础概念/光圈.md","filePath":"shoot/基础概念/光圈.md","lastUpdated":1736065650000}'),n={name:"shoot/基础概念/光圈.md"},l=e("h1",{id:"光圈",tabindex:"-1"},[s("光圈 "),e("a",{class:"header-anchor",href:"#光圈","aria-label":'Permalink to "光圈"'},"​")],-1),r=e("p",null,"控制光圈大小",-1),c=e("ol",null,[e("li",null,"数字越大、光圈越小就越暗 - 图像越清晰"),e("li",null,"数字越小、光圈越大就越亮 - 光圈过大虚化效果明显")],-1),d=[l,r,c];function _(i,h,p,m,f,u){return a(),o("div",null,d)}const $=t(n,[["render",_]]);export{k as __pageData,$ as default}; +import{_ as t,o as a,c as o,k as e,a as s}from"./chunks/framework.b6910bb2.js";const k=JSON.parse('{"title":"光圈","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/基础概念/光圈.md","filePath":"shoot/基础概念/光圈.md","lastUpdated":1736839070000}'),n={name:"shoot/基础概念/光圈.md"},l=e("h1",{id:"光圈",tabindex:"-1"},[s("光圈 "),e("a",{class:"header-anchor",href:"#光圈","aria-label":'Permalink to "光圈"'},"​")],-1),r=e("p",null,"控制光圈大小",-1),c=e("ol",null,[e("li",null,"数字越大、光圈越小就越暗 - 图像越清晰"),e("li",null,"数字越小、光圈越大就越亮 - 光圈过大虚化效果明显")],-1),d=[l,r,c];function _(i,h,p,m,f,u){return a(),o("div",null,d)}const $=t(n,[["render",_]]);export{k as __pageData,$ as default}; diff --git "a/assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\205\211\345\234\210.md.ed2aed3b.lean.js" "b/assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\205\211\345\234\210.md.5071eb5f.lean.js" similarity index 92% rename from "assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\205\211\345\234\210.md.ed2aed3b.lean.js" rename to "assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\205\211\345\234\210.md.5071eb5f.lean.js" index a92e7b1c..f120e8d9 100644 --- "a/assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\205\211\345\234\210.md.ed2aed3b.lean.js" +++ "b/assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\205\211\345\234\210.md.5071eb5f.lean.js" @@ -1 +1 @@ -import{_ as t,o as a,c as o,k as e,a as s}from"./chunks/framework.b6910bb2.js";const k=JSON.parse('{"title":"光圈","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/基础概念/光圈.md","filePath":"shoot/基础概念/光圈.md","lastUpdated":1736065650000}'),n={name:"shoot/基础概念/光圈.md"},l=e("h1",{id:"光圈",tabindex:"-1"},[s("光圈 "),e("a",{class:"header-anchor",href:"#光圈","aria-label":'Permalink to "光圈"'},"​")],-1),r=e("p",null,"控制光圈大小",-1),c=e("ol",null,[e("li",null,"数字越大、光圈越小就越暗 - 图像越清晰"),e("li",null,"数字越小、光圈越大就越亮 - 光圈过大虚化效果明显")],-1),d=[l,r,c];function _(i,h,p,m,f,u){return a(),o("div",null,d)}const $=t(n,[["render",_]]);export{k as __pageData,$ as default}; +import{_ as t,o as a,c as o,k as e,a as s}from"./chunks/framework.b6910bb2.js";const k=JSON.parse('{"title":"光圈","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/基础概念/光圈.md","filePath":"shoot/基础概念/光圈.md","lastUpdated":1736839070000}'),n={name:"shoot/基础概念/光圈.md"},l=e("h1",{id:"光圈",tabindex:"-1"},[s("光圈 "),e("a",{class:"header-anchor",href:"#光圈","aria-label":'Permalink to "光圈"'},"​")],-1),r=e("p",null,"控制光圈大小",-1),c=e("ol",null,[e("li",null,"数字越大、光圈越小就越暗 - 图像越清晰"),e("li",null,"数字越小、光圈越大就越亮 - 光圈过大虚化效果明显")],-1),d=[l,r,c];function _(i,h,p,m,f,u){return a(),o("div",null,d)}const $=t(n,[["render",_]]);export{k as __pageData,$ as default}; diff --git "a/assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\277\253\351\227\250\351\200\237\345\272\246.md.8bdd3100.js" "b/assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\277\253\351\227\250\351\200\237\345\272\246.md.858c51e6.js" similarity index 93% rename from "assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\277\253\351\227\250\351\200\237\345\272\246.md.8bdd3100.js" rename to "assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\277\253\351\227\250\351\200\237\345\272\246.md.858c51e6.js" index 24119b32..555615ed 100644 --- "a/assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\277\253\351\227\250\351\200\237\345\272\246.md.8bdd3100.js" +++ "b/assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\277\253\351\227\250\351\200\237\345\272\246.md.858c51e6.js" @@ -1 +1 @@ -import{_ as a,o as s,c as o,k as e,a as t}from"./chunks/framework.b6910bb2.js";const l="/vitePress-blob/assets/1.2aa2aedd.png",k=JSON.parse('{"title":"快门速度","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/基础概念/快门速度.md","filePath":"shoot/基础概念/快门速度.md","lastUpdated":1736065650000}'),n={name:"shoot/基础概念/快门速度.md"},r=e("h1",{id:"快门速度",tabindex:"-1"},[t("快门速度 "),e("a",{class:"header-anchor",href:"#快门速度","aria-label":'Permalink to "快门速度"'},"​")],-1),c=e("p",null,"控制开启传感器的时间",-1),i=e("ul",null,[e("li",null,"快门速度越短,越能凝固物体,常用在抓拍"),e("li",null,[t("快门速度越长,容易得到拖影的图 "),e("ul",null,[e("li",null,[e("img",{src:l,alt:"快门速度"})]),e("li",null,"拍静止的水面 (一般需要借助三脚架)")])])],-1),_=[r,c,i];function d(h,p,u,m,f,x){return s(),o("div",null,_)}const P=a(n,[["render",d]]);export{k as __pageData,P as default}; +import{_ as a,o as s,c as o,k as e,a as t}from"./chunks/framework.b6910bb2.js";const l="/vitePress-blob/assets/1.2aa2aedd.png",k=JSON.parse('{"title":"快门速度","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/基础概念/快门速度.md","filePath":"shoot/基础概念/快门速度.md","lastUpdated":1736839070000}'),n={name:"shoot/基础概念/快门速度.md"},r=e("h1",{id:"快门速度",tabindex:"-1"},[t("快门速度 "),e("a",{class:"header-anchor",href:"#快门速度","aria-label":'Permalink to "快门速度"'},"​")],-1),c=e("p",null,"控制开启传感器的时间",-1),i=e("ul",null,[e("li",null,"快门速度越短,越能凝固物体,常用在抓拍"),e("li",null,[t("快门速度越长,容易得到拖影的图 "),e("ul",null,[e("li",null,[e("img",{src:l,alt:"快门速度"})]),e("li",null,"拍静止的水面 (一般需要借助三脚架)")])])],-1),_=[r,c,i];function d(h,p,u,m,f,x){return s(),o("div",null,_)}const P=a(n,[["render",d]]);export{k as __pageData,P as default}; diff --git "a/assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\277\253\351\227\250\351\200\237\345\272\246.md.8bdd3100.lean.js" "b/assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\277\253\351\227\250\351\200\237\345\272\246.md.858c51e6.lean.js" similarity index 93% rename from "assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\277\253\351\227\250\351\200\237\345\272\246.md.8bdd3100.lean.js" rename to "assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\277\253\351\227\250\351\200\237\345\272\246.md.858c51e6.lean.js" index 24119b32..555615ed 100644 --- "a/assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\277\253\351\227\250\351\200\237\345\272\246.md.8bdd3100.lean.js" +++ "b/assets/shoot_\345\237\272\347\241\200\346\246\202\345\277\265_\345\277\253\351\227\250\351\200\237\345\272\246.md.858c51e6.lean.js" @@ -1 +1 @@ -import{_ as a,o as s,c as o,k as e,a as t}from"./chunks/framework.b6910bb2.js";const l="/vitePress-blob/assets/1.2aa2aedd.png",k=JSON.parse('{"title":"快门速度","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/基础概念/快门速度.md","filePath":"shoot/基础概念/快门速度.md","lastUpdated":1736065650000}'),n={name:"shoot/基础概念/快门速度.md"},r=e("h1",{id:"快门速度",tabindex:"-1"},[t("快门速度 "),e("a",{class:"header-anchor",href:"#快门速度","aria-label":'Permalink to "快门速度"'},"​")],-1),c=e("p",null,"控制开启传感器的时间",-1),i=e("ul",null,[e("li",null,"快门速度越短,越能凝固物体,常用在抓拍"),e("li",null,[t("快门速度越长,容易得到拖影的图 "),e("ul",null,[e("li",null,[e("img",{src:l,alt:"快门速度"})]),e("li",null,"拍静止的水面 (一般需要借助三脚架)")])])],-1),_=[r,c,i];function d(h,p,u,m,f,x){return s(),o("div",null,_)}const P=a(n,[["render",d]]);export{k as __pageData,P as default}; +import{_ as a,o as s,c as o,k as e,a as t}from"./chunks/framework.b6910bb2.js";const l="/vitePress-blob/assets/1.2aa2aedd.png",k=JSON.parse('{"title":"快门速度","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/基础概念/快门速度.md","filePath":"shoot/基础概念/快门速度.md","lastUpdated":1736839070000}'),n={name:"shoot/基础概念/快门速度.md"},r=e("h1",{id:"快门速度",tabindex:"-1"},[t("快门速度 "),e("a",{class:"header-anchor",href:"#快门速度","aria-label":'Permalink to "快门速度"'},"​")],-1),c=e("p",null,"控制开启传感器的时间",-1),i=e("ul",null,[e("li",null,"快门速度越短,越能凝固物体,常用在抓拍"),e("li",null,[t("快门速度越长,容易得到拖影的图 "),e("ul",null,[e("li",null,[e("img",{src:l,alt:"快门速度"})]),e("li",null,"拍静止的水面 (一般需要借助三脚架)")])])],-1),_=[r,c,i];function d(h,p,u,m,f,x){return s(),o("div",null,_)}const P=a(n,[["render",d]]);export{k as __pageData,P as default}; diff --git "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_index.md.5e1171b3.js" "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_index.md.e5c3a073.js" similarity index 92% rename from "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_index.md.5e1171b3.js" rename to "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_index.md.e5c3a073.js" index 7ae3c220..cc3de5b2 100644 --- "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_index.md.5e1171b3.js" +++ "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_index.md.e5c3a073.js" @@ -1 +1 @@ -import{_ as a,o as t,c as o,k as e,a as s}from"./chunks/framework.b6910bb2.js";const x=JSON.parse('{"title":"参考文章","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/index.md","filePath":"shoot/实战技巧/index.md","lastUpdated":1736065650000}'),r={name:"shoot/实战技巧/index.md"},i=e("h1",{id:"参考文章",tabindex:"-1"},[s("参考文章 "),e("a",{class:"header-anchor",href:"#参考文章","aria-label":'Permalink to "参考文章"'},"​")],-1),c=e("ol",null,[e("li",null,[e("a",{href:"https://www.bilibili.com/video/BV1x4411F7bs?p=13&vd_source=3ede5fc64ebab3ac5654a5b43523a9b2",target:"_blank",rel:"noreferrer"},"https://www.bilibili.com/video/BV1x4411F7bs?p=13&vd_source=3ede5fc64ebab3ac5654a5b43523a9b2")])],-1),d=[i,c];function n(l,_,b,h,p,f){return t(),o("div",null,d)}const u=a(r,[["render",n]]);export{x as __pageData,u as default}; +import{_ as a,o as t,c as o,k as e,a as s}from"./chunks/framework.b6910bb2.js";const x=JSON.parse('{"title":"参考文章","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/index.md","filePath":"shoot/实战技巧/index.md","lastUpdated":1736839070000}'),r={name:"shoot/实战技巧/index.md"},i=e("h1",{id:"参考文章",tabindex:"-1"},[s("参考文章 "),e("a",{class:"header-anchor",href:"#参考文章","aria-label":'Permalink to "参考文章"'},"​")],-1),c=e("ol",null,[e("li",null,[e("a",{href:"https://www.bilibili.com/video/BV1x4411F7bs?p=13&vd_source=3ede5fc64ebab3ac5654a5b43523a9b2",target:"_blank",rel:"noreferrer"},"https://www.bilibili.com/video/BV1x4411F7bs?p=13&vd_source=3ede5fc64ebab3ac5654a5b43523a9b2")])],-1),d=[i,c];function n(l,_,b,h,p,f){return t(),o("div",null,d)}const u=a(r,[["render",n]]);export{x as __pageData,u as default}; diff --git "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_index.md.5e1171b3.lean.js" "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_index.md.e5c3a073.lean.js" similarity index 92% rename from "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_index.md.5e1171b3.lean.js" rename to "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_index.md.e5c3a073.lean.js" index 7ae3c220..cc3de5b2 100644 --- "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_index.md.5e1171b3.lean.js" +++ "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_index.md.e5c3a073.lean.js" @@ -1 +1 @@ -import{_ as a,o as t,c as o,k as e,a as s}from"./chunks/framework.b6910bb2.js";const x=JSON.parse('{"title":"参考文章","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/index.md","filePath":"shoot/实战技巧/index.md","lastUpdated":1736065650000}'),r={name:"shoot/实战技巧/index.md"},i=e("h1",{id:"参考文章",tabindex:"-1"},[s("参考文章 "),e("a",{class:"header-anchor",href:"#参考文章","aria-label":'Permalink to "参考文章"'},"​")],-1),c=e("ol",null,[e("li",null,[e("a",{href:"https://www.bilibili.com/video/BV1x4411F7bs?p=13&vd_source=3ede5fc64ebab3ac5654a5b43523a9b2",target:"_blank",rel:"noreferrer"},"https://www.bilibili.com/video/BV1x4411F7bs?p=13&vd_source=3ede5fc64ebab3ac5654a5b43523a9b2")])],-1),d=[i,c];function n(l,_,b,h,p,f){return t(),o("div",null,d)}const u=a(r,[["render",n]]);export{x as __pageData,u as default}; +import{_ as a,o as t,c as o,k as e,a as s}from"./chunks/framework.b6910bb2.js";const x=JSON.parse('{"title":"参考文章","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/index.md","filePath":"shoot/实战技巧/index.md","lastUpdated":1736839070000}'),r={name:"shoot/实战技巧/index.md"},i=e("h1",{id:"参考文章",tabindex:"-1"},[s("参考文章 "),e("a",{class:"header-anchor",href:"#参考文章","aria-label":'Permalink to "参考文章"'},"​")],-1),c=e("ol",null,[e("li",null,[e("a",{href:"https://www.bilibili.com/video/BV1x4411F7bs?p=13&vd_source=3ede5fc64ebab3ac5654a5b43523a9b2",target:"_blank",rel:"noreferrer"},"https://www.bilibili.com/video/BV1x4411F7bs?p=13&vd_source=3ede5fc64ebab3ac5654a5b43523a9b2")])],-1),d=[i,c];function n(l,_,b,h,p,f){return t(),o("div",null,d)}const u=a(r,[["render",n]]);export{x as __pageData,u as default}; diff --git "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\344\272\256\350\260\203\344\272\272\345\203\217.md.1d135fd4.js" "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\344\272\256\350\260\203\344\272\272\345\203\217.md.55937770.js" similarity index 94% rename from "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\344\272\256\350\260\203\344\272\272\345\203\217.md.1d135fd4.js" rename to "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\344\272\256\350\260\203\344\272\272\345\203\217.md.55937770.js" index bfbbba83..0269b402 100644 --- "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\344\272\256\350\260\203\344\272\272\345\203\217.md.1d135fd4.js" +++ "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\344\272\256\350\260\203\344\272\272\345\203\217.md.55937770.js" @@ -1 +1 @@ -import{_ as a,o as e,c as t,Q as o}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"亮调人像","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/亮调人像.md","filePath":"shoot/实战技巧/亮调人像.md","lastUpdated":1736065650000}'),l={name:"shoot/实战技巧/亮调人像.md"},r=o('

亮调人像

百分之 70 都是浅色系

前置

  1. 背景是浅色
  2. 人物衣服也是浅色

如何拍摄

  1. 评价测光
  2. 可适当增加曝光补偿
',6),i=[r];function s(_,n,c,d,h,p){return e(),t("div",null,i)}const u=a(l,[["render",s]]);export{f as __pageData,u as default}; +import{_ as a,o as e,c as t,Q as o}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"亮调人像","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/亮调人像.md","filePath":"shoot/实战技巧/亮调人像.md","lastUpdated":1736839070000}'),l={name:"shoot/实战技巧/亮调人像.md"},r=o('

亮调人像

百分之 70 都是浅色系

前置

  1. 背景是浅色
  2. 人物衣服也是浅色

如何拍摄

  1. 评价测光
  2. 可适当增加曝光补偿
',6),i=[r];function s(_,n,c,d,h,p){return e(),t("div",null,i)}const u=a(l,[["render",s]]);export{f as __pageData,u as default}; diff --git "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\344\272\256\350\260\203\344\272\272\345\203\217.md.1d135fd4.lean.js" "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\344\272\256\350\260\203\344\272\272\345\203\217.md.55937770.lean.js" similarity index 86% rename from "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\344\272\256\350\260\203\344\272\272\345\203\217.md.1d135fd4.lean.js" rename to "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\344\272\256\350\260\203\344\272\272\345\203\217.md.55937770.lean.js" index 37e02f47..e1bac05c 100644 --- "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\344\272\256\350\260\203\344\272\272\345\203\217.md.1d135fd4.lean.js" +++ "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\344\272\256\350\260\203\344\272\272\345\203\217.md.55937770.lean.js" @@ -1 +1 @@ -import{_ as a,o as e,c as t,Q as o}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"亮调人像","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/亮调人像.md","filePath":"shoot/实战技巧/亮调人像.md","lastUpdated":1736065650000}'),l={name:"shoot/实战技巧/亮调人像.md"},r=o("",6),i=[r];function s(_,n,c,d,h,p){return e(),t("div",null,i)}const u=a(l,[["render",s]]);export{f as __pageData,u as default}; +import{_ as a,o as e,c as t,Q as o}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"亮调人像","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/亮调人像.md","filePath":"shoot/实战技巧/亮调人像.md","lastUpdated":1736839070000}'),l={name:"shoot/实战技巧/亮调人像.md"},r=o("",6),i=[r];function s(_,n,c,d,h,p){return e(),t("div",null,i)}const u=a(l,[["render",s]]);export{f as __pageData,u as default}; diff --git "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\206\267\346\232\226\345\257\271\346\257\224\344\272\272\345\203\217.md.70050c0c.js" "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\206\267\346\232\226\345\257\271\346\257\224\344\272\272\345\203\217.md.852b078d.js" similarity index 94% rename from "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\206\267\346\232\226\345\257\271\346\257\224\344\272\272\345\203\217.md.70050c0c.js" rename to "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\206\267\346\232\226\345\257\271\346\257\224\344\272\272\345\203\217.md.852b078d.js" index a6f448e1..6758265a 100644 --- "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\206\267\346\232\226\345\257\271\346\257\224\344\272\272\345\203\217.md.70050c0c.js" +++ "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\206\267\346\232\226\345\257\271\346\257\224\344\272\272\345\203\217.md.852b078d.js" @@ -1 +1 @@ -import{_ as a,o as e,c as t,Q as o}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"冷暖对比人像","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/冷暖对比人像.md","filePath":"shoot/实战技巧/冷暖对比人像.md","lastUpdated":1736065650000}'),l={name:"shoot/实战技巧/冷暖对比人像.md"},r=o('

冷暖对比人像

人物和背景冷暖色系不同

场景

  1. 蓝调时刻 - 太阳刚落山

如何拍摄

  1. 暖色 - 可以通过补光灯
  2. 色温 - 3500 - 4000k
  3. 白平衡 - b2m2
',6),i=[r];function _(s,n,c,d,h,p){return e(),t("div",null,i)}const u=a(l,[["render",_]]);export{f as __pageData,u as default}; +import{_ as a,o as e,c as t,Q as o}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"冷暖对比人像","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/冷暖对比人像.md","filePath":"shoot/实战技巧/冷暖对比人像.md","lastUpdated":1736839070000}'),l={name:"shoot/实战技巧/冷暖对比人像.md"},r=o('

冷暖对比人像

人物和背景冷暖色系不同

场景

  1. 蓝调时刻 - 太阳刚落山

如何拍摄

  1. 暖色 - 可以通过补光灯
  2. 色温 - 3500 - 4000k
  3. 白平衡 - b2m2
',6),i=[r];function _(s,n,c,d,h,p){return e(),t("div",null,i)}const u=a(l,[["render",_]]);export{f as __pageData,u as default}; diff --git "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\206\267\346\232\226\345\257\271\346\257\224\344\272\272\345\203\217.md.70050c0c.lean.js" "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\206\267\346\232\226\345\257\271\346\257\224\344\272\272\345\203\217.md.852b078d.lean.js" similarity index 87% rename from "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\206\267\346\232\226\345\257\271\346\257\224\344\272\272\345\203\217.md.70050c0c.lean.js" rename to "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\206\267\346\232\226\345\257\271\346\257\224\344\272\272\345\203\217.md.852b078d.lean.js" index fe6b4a51..9f0d1542 100644 --- "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\206\267\346\232\226\345\257\271\346\257\224\344\272\272\345\203\217.md.70050c0c.lean.js" +++ "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\206\267\346\232\226\345\257\271\346\257\224\344\272\272\345\203\217.md.852b078d.lean.js" @@ -1 +1 @@ -import{_ as a,o as e,c as t,Q as o}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"冷暖对比人像","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/冷暖对比人像.md","filePath":"shoot/实战技巧/冷暖对比人像.md","lastUpdated":1736065650000}'),l={name:"shoot/实战技巧/冷暖对比人像.md"},r=o("",6),i=[r];function _(s,n,c,d,h,p){return e(),t("div",null,i)}const u=a(l,[["render",_]]);export{f as __pageData,u as default}; +import{_ as a,o as e,c as t,Q as o}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"冷暖对比人像","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/冷暖对比人像.md","filePath":"shoot/实战技巧/冷暖对比人像.md","lastUpdated":1736839070000}'),l={name:"shoot/实战技巧/冷暖对比人像.md"},r=o("",6),i=[r];function _(s,n,c,d,h,p){return e(),t("div",null,i)}const u=a(l,[["render",_]]);export{f as __pageData,u as default}; diff --git "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\350\212\261\347\232\204\351\273\221\350\203\214\346\231\257.md.fb63b890.js" "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\350\212\261\347\232\204\351\273\221\350\203\214\346\231\257.md.5e8263dc.js" similarity index 93% rename from "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\350\212\261\347\232\204\351\273\221\350\203\214\346\231\257.md.fb63b890.js" rename to "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\350\212\261\347\232\204\351\273\221\350\203\214\346\231\257.md.5e8263dc.js" index eda8410f..54fd7213 100644 --- "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\350\212\261\347\232\204\351\273\221\350\203\214\346\231\257.md.fb63b890.js" +++ "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\350\212\261\347\232\204\351\273\221\350\203\214\346\231\257.md.5e8263dc.js" @@ -1 +1 @@ -import{_ as t,o,c as s,k as e,a}from"./chunks/framework.b6910bb2.js";const k=JSON.parse('{"title":"如何拍花","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/如何拍花的黑背景.md","filePath":"shoot/实战技巧/如何拍花的黑背景.md","lastUpdated":1736065650000}'),l={name:"shoot/实战技巧/如何拍花的黑背景.md"},n=e("h1",{id:"如何拍花",tabindex:"-1"},[a("如何拍花 "),e("a",{class:"header-anchor",href:"#如何拍花","aria-label":'Permalink to "如何拍花"'},"​")],-1),r=e("h2",{id:"如何拍摄",tabindex:"-1"},[a("如何拍摄 "),e("a",{class:"header-anchor",href:"#如何拍摄","aria-label":'Permalink to "如何拍摄"'},"​")],-1),c=e("ol",null,[e("li",null,[a("采用 "),e("code",null,"点测光"),a(" 突出花的亮度")]),e("li",null,[a("背景采用 "),e("code",null,"深色系"),a(" 突出主体")])],-1),_=[n,r,c];function d(i,h,p,m,f,u){return o(),s("div",null,_)}const b=t(l,[["render",d]]);export{k as __pageData,b as default}; +import{_ as t,o,c as s,k as e,a}from"./chunks/framework.b6910bb2.js";const k=JSON.parse('{"title":"如何拍花","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/如何拍花的黑背景.md","filePath":"shoot/实战技巧/如何拍花的黑背景.md","lastUpdated":1736839070000}'),l={name:"shoot/实战技巧/如何拍花的黑背景.md"},n=e("h1",{id:"如何拍花",tabindex:"-1"},[a("如何拍花 "),e("a",{class:"header-anchor",href:"#如何拍花","aria-label":'Permalink to "如何拍花"'},"​")],-1),r=e("h2",{id:"如何拍摄",tabindex:"-1"},[a("如何拍摄 "),e("a",{class:"header-anchor",href:"#如何拍摄","aria-label":'Permalink to "如何拍摄"'},"​")],-1),c=e("ol",null,[e("li",null,[a("采用 "),e("code",null,"点测光"),a(" 突出花的亮度")]),e("li",null,[a("背景采用 "),e("code",null,"深色系"),a(" 突出主体")])],-1),_=[n,r,c];function d(i,h,p,m,f,u){return o(),s("div",null,_)}const b=t(l,[["render",d]]);export{k as __pageData,b as default}; diff --git "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\350\212\261\347\232\204\351\273\221\350\203\214\346\231\257.md.fb63b890.lean.js" "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\350\212\261\347\232\204\351\273\221\350\203\214\346\231\257.md.5e8263dc.lean.js" similarity index 93% rename from "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\350\212\261\347\232\204\351\273\221\350\203\214\346\231\257.md.fb63b890.lean.js" rename to "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\350\212\261\347\232\204\351\273\221\350\203\214\346\231\257.md.5e8263dc.lean.js" index eda8410f..54fd7213 100644 --- "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\350\212\261\347\232\204\351\273\221\350\203\214\346\231\257.md.fb63b890.lean.js" +++ "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\350\212\261\347\232\204\351\273\221\350\203\214\346\231\257.md.5e8263dc.lean.js" @@ -1 +1 @@ -import{_ as t,o,c as s,k as e,a}from"./chunks/framework.b6910bb2.js";const k=JSON.parse('{"title":"如何拍花","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/如何拍花的黑背景.md","filePath":"shoot/实战技巧/如何拍花的黑背景.md","lastUpdated":1736065650000}'),l={name:"shoot/实战技巧/如何拍花的黑背景.md"},n=e("h1",{id:"如何拍花",tabindex:"-1"},[a("如何拍花 "),e("a",{class:"header-anchor",href:"#如何拍花","aria-label":'Permalink to "如何拍花"'},"​")],-1),r=e("h2",{id:"如何拍摄",tabindex:"-1"},[a("如何拍摄 "),e("a",{class:"header-anchor",href:"#如何拍摄","aria-label":'Permalink to "如何拍摄"'},"​")],-1),c=e("ol",null,[e("li",null,[a("采用 "),e("code",null,"点测光"),a(" 突出花的亮度")]),e("li",null,[a("背景采用 "),e("code",null,"深色系"),a(" 突出主体")])],-1),_=[n,r,c];function d(i,h,p,m,f,u){return o(),s("div",null,_)}const b=t(l,[["render",d]]);export{k as __pageData,b as default}; +import{_ as t,o,c as s,k as e,a}from"./chunks/framework.b6910bb2.js";const k=JSON.parse('{"title":"如何拍花","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/如何拍花的黑背景.md","filePath":"shoot/实战技巧/如何拍花的黑背景.md","lastUpdated":1736839070000}'),l={name:"shoot/实战技巧/如何拍花的黑背景.md"},n=e("h1",{id:"如何拍花",tabindex:"-1"},[a("如何拍花 "),e("a",{class:"header-anchor",href:"#如何拍花","aria-label":'Permalink to "如何拍花"'},"​")],-1),r=e("h2",{id:"如何拍摄",tabindex:"-1"},[a("如何拍摄 "),e("a",{class:"header-anchor",href:"#如何拍摄","aria-label":'Permalink to "如何拍摄"'},"​")],-1),c=e("ol",null,[e("li",null,[a("采用 "),e("code",null,"点测光"),a(" 突出花的亮度")]),e("li",null,[a("背景采用 "),e("code",null,"深色系"),a(" 突出主体")])],-1),_=[n,r,c];function d(i,h,p,m,f,u){return o(),s("div",null,_)}const b=t(l,[["render",d]]);export{k as __pageData,b as default}; diff --git "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\351\233\250\344\270\235.md.e607e8e7.js" "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\351\233\250\344\270\235.md.7e47cf94.js" similarity index 96% rename from "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\351\233\250\344\270\235.md.e607e8e7.js" rename to "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\351\233\250\344\270\235.md.7e47cf94.js" index 1f81d5c3..b9565aa1 100644 --- "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\351\233\250\344\270\235.md.e607e8e7.js" +++ "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\351\233\250\344\270\235.md.7e47cf94.js" @@ -1 +1 @@ -import{_ as a,o as e,c as t,Q as o}from"./chunks/framework.b6910bb2.js";const r="/vitePress-blob/assets/1.b55cef29.png",f=JSON.parse('{"title":"如何拍雨丝","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/如何拍雨丝.md","filePath":"shoot/实战技巧/如何拍雨丝.md","lastUpdated":1736065650000}'),i={name:"shoot/实战技巧/如何拍雨丝.md"},l=o('

如何拍雨丝

如何拍摄

灯位要求

需要俩盏灯

  • 逆光灯: 在拍摄物体后方 (容易拍摄到雨丝,而且物体会有光边效果)
  • 正光灯:在拍摄物体前方(避免物体逆光,变黑)

逆光灯亮度 > 主光灯

灯位要求

白平衡

如果拍摄物体偏黄,可以调低白平衡,使物体变白

快门速度

  • 快门速度越慢,雨水成丝
  • 快门速度越快,雨水成点
',11),s=[l];function n(h,_,c,d,p,u){return e(),t("div",null,s)}const b=a(i,[["render",n]]);export{f as __pageData,b as default}; +import{_ as a,o as e,c as t,Q as o}from"./chunks/framework.b6910bb2.js";const r="/vitePress-blob/assets/1.b55cef29.png",f=JSON.parse('{"title":"如何拍雨丝","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/如何拍雨丝.md","filePath":"shoot/实战技巧/如何拍雨丝.md","lastUpdated":1736839070000}'),i={name:"shoot/实战技巧/如何拍雨丝.md"},l=o('

如何拍雨丝

如何拍摄

灯位要求

需要俩盏灯

  • 逆光灯: 在拍摄物体后方 (容易拍摄到雨丝,而且物体会有光边效果)
  • 正光灯:在拍摄物体前方(避免物体逆光,变黑)

逆光灯亮度 > 主光灯

灯位要求

白平衡

如果拍摄物体偏黄,可以调低白平衡,使物体变白

快门速度

  • 快门速度越慢,雨水成丝
  • 快门速度越快,雨水成点
',11),s=[l];function n(h,_,c,d,p,u){return e(),t("div",null,s)}const b=a(i,[["render",n]]);export{f as __pageData,b as default}; diff --git "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\351\233\250\344\270\235.md.e607e8e7.lean.js" "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\351\233\250\344\270\235.md.7e47cf94.lean.js" similarity index 87% rename from "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\351\233\250\344\270\235.md.e607e8e7.lean.js" rename to "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\351\233\250\344\270\235.md.7e47cf94.lean.js" index 473fee91..d6ee8706 100644 --- "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\351\233\250\344\270\235.md.e607e8e7.lean.js" +++ "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\345\246\202\344\275\225\346\213\215\351\233\250\344\270\235.md.7e47cf94.lean.js" @@ -1 +1 @@ -import{_ as a,o as e,c as t,Q as o}from"./chunks/framework.b6910bb2.js";const r="/vitePress-blob/assets/1.b55cef29.png",f=JSON.parse('{"title":"如何拍雨丝","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/如何拍雨丝.md","filePath":"shoot/实战技巧/如何拍雨丝.md","lastUpdated":1736065650000}'),i={name:"shoot/实战技巧/如何拍雨丝.md"},l=o("",11),s=[l];function n(h,_,c,d,p,u){return e(),t("div",null,s)}const b=a(i,[["render",n]]);export{f as __pageData,b as default}; +import{_ as a,o as e,c as t,Q as o}from"./chunks/framework.b6910bb2.js";const r="/vitePress-blob/assets/1.b55cef29.png",f=JSON.parse('{"title":"如何拍雨丝","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/如何拍雨丝.md","filePath":"shoot/实战技巧/如何拍雨丝.md","lastUpdated":1736839070000}'),i={name:"shoot/实战技巧/如何拍雨丝.md"},l=o("",11),s=[l];function n(h,_,c,d,p,u){return e(),t("div",null,s)}const b=a(i,[["render",n]]);export{f as __pageData,b as default}; diff --git "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\213\215\346\221\204\346\242\246\345\271\273\345\205\211\346\226\221.md.498fd396.js" "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\213\215\346\221\204\346\242\246\345\271\273\345\205\211\346\226\221.md.8ca08bfb.js" similarity index 94% rename from "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\213\215\346\221\204\346\242\246\345\271\273\345\205\211\346\226\221.md.498fd396.js" rename to "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\213\215\346\221\204\346\242\246\345\271\273\345\205\211\346\226\221.md.8ca08bfb.js" index 873d5547..9a888cd7 100644 --- "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\213\215\346\221\204\346\242\246\345\271\273\345\205\211\346\226\221.md.498fd396.js" +++ "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\213\215\346\221\204\346\242\246\345\271\273\345\205\211\346\226\221.md.8ca08bfb.js" @@ -1 +1 @@ -import{_ as t,o,c as l,k as e,a}from"./chunks/framework.b6910bb2.js";const k=JSON.parse('{"title":"拍摄梦幻光斑","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/拍摄梦幻光斑.md","filePath":"shoot/实战技巧/拍摄梦幻光斑.md","lastUpdated":1736065650000}'),s={name:"shoot/实战技巧/拍摄梦幻光斑.md"},n=e("h1",{id:"拍摄梦幻光斑",tabindex:"-1"},[a("拍摄梦幻光斑 "),e("a",{class:"header-anchor",href:"#拍摄梦幻光斑","aria-label":'Permalink to "拍摄梦幻光斑"'},"​")],-1),r=e("h2",{id:"如何拍摄",tabindex:"-1"},[a("如何拍摄 "),e("a",{class:"header-anchor",href:"#如何拍摄","aria-label":'Permalink to "如何拍摄"'},"​")],-1),i=e("ul",null,[e("li",null,"采用大光圈镜头,有很好的背景虚化效果"),e("li",null,[a("需要一个补光灯,减少避免人物逆光导致过黑 "),e("ul",null,[e("li",null,"拍摄物体在光斑前面"),e("li",null,"对焦到物体上(也可以通过精细对焦,即放大后手动对焦)")])])],-1),c=[n,r,i];function _(d,h,u,p,m,f){return o(),l("div",null,c)}const b=t(s,[["render",_]]);export{k as __pageData,b as default}; +import{_ as t,o,c as l,k as e,a}from"./chunks/framework.b6910bb2.js";const k=JSON.parse('{"title":"拍摄梦幻光斑","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/拍摄梦幻光斑.md","filePath":"shoot/实战技巧/拍摄梦幻光斑.md","lastUpdated":1736839070000}'),s={name:"shoot/实战技巧/拍摄梦幻光斑.md"},n=e("h1",{id:"拍摄梦幻光斑",tabindex:"-1"},[a("拍摄梦幻光斑 "),e("a",{class:"header-anchor",href:"#拍摄梦幻光斑","aria-label":'Permalink to "拍摄梦幻光斑"'},"​")],-1),r=e("h2",{id:"如何拍摄",tabindex:"-1"},[a("如何拍摄 "),e("a",{class:"header-anchor",href:"#如何拍摄","aria-label":'Permalink to "如何拍摄"'},"​")],-1),i=e("ul",null,[e("li",null,"采用大光圈镜头,有很好的背景虚化效果"),e("li",null,[a("需要一个补光灯,减少避免人物逆光导致过黑 "),e("ul",null,[e("li",null,"拍摄物体在光斑前面"),e("li",null,"对焦到物体上(也可以通过精细对焦,即放大后手动对焦)")])])],-1),c=[n,r,i];function _(d,h,u,p,m,f){return o(),l("div",null,c)}const b=t(s,[["render",_]]);export{k as __pageData,b as default}; diff --git "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\213\215\346\221\204\346\242\246\345\271\273\345\205\211\346\226\221.md.498fd396.lean.js" "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\213\215\346\221\204\346\242\246\345\271\273\345\205\211\346\226\221.md.8ca08bfb.lean.js" similarity index 94% rename from "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\213\215\346\221\204\346\242\246\345\271\273\345\205\211\346\226\221.md.498fd396.lean.js" rename to "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\213\215\346\221\204\346\242\246\345\271\273\345\205\211\346\226\221.md.8ca08bfb.lean.js" index 873d5547..9a888cd7 100644 --- "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\213\215\346\221\204\346\242\246\345\271\273\345\205\211\346\226\221.md.498fd396.lean.js" +++ "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\213\215\346\221\204\346\242\246\345\271\273\345\205\211\346\226\221.md.8ca08bfb.lean.js" @@ -1 +1 @@ -import{_ as t,o,c as l,k as e,a}from"./chunks/framework.b6910bb2.js";const k=JSON.parse('{"title":"拍摄梦幻光斑","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/拍摄梦幻光斑.md","filePath":"shoot/实战技巧/拍摄梦幻光斑.md","lastUpdated":1736065650000}'),s={name:"shoot/实战技巧/拍摄梦幻光斑.md"},n=e("h1",{id:"拍摄梦幻光斑",tabindex:"-1"},[a("拍摄梦幻光斑 "),e("a",{class:"header-anchor",href:"#拍摄梦幻光斑","aria-label":'Permalink to "拍摄梦幻光斑"'},"​")],-1),r=e("h2",{id:"如何拍摄",tabindex:"-1"},[a("如何拍摄 "),e("a",{class:"header-anchor",href:"#如何拍摄","aria-label":'Permalink to "如何拍摄"'},"​")],-1),i=e("ul",null,[e("li",null,"采用大光圈镜头,有很好的背景虚化效果"),e("li",null,[a("需要一个补光灯,减少避免人物逆光导致过黑 "),e("ul",null,[e("li",null,"拍摄物体在光斑前面"),e("li",null,"对焦到物体上(也可以通过精细对焦,即放大后手动对焦)")])])],-1),c=[n,r,i];function _(d,h,u,p,m,f){return o(),l("div",null,c)}const b=t(s,[["render",_]]);export{k as __pageData,b as default}; +import{_ as t,o,c as l,k as e,a}from"./chunks/framework.b6910bb2.js";const k=JSON.parse('{"title":"拍摄梦幻光斑","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/拍摄梦幻光斑.md","filePath":"shoot/实战技巧/拍摄梦幻光斑.md","lastUpdated":1736839070000}'),s={name:"shoot/实战技巧/拍摄梦幻光斑.md"},n=e("h1",{id:"拍摄梦幻光斑",tabindex:"-1"},[a("拍摄梦幻光斑 "),e("a",{class:"header-anchor",href:"#拍摄梦幻光斑","aria-label":'Permalink to "拍摄梦幻光斑"'},"​")],-1),r=e("h2",{id:"如何拍摄",tabindex:"-1"},[a("如何拍摄 "),e("a",{class:"header-anchor",href:"#如何拍摄","aria-label":'Permalink to "如何拍摄"'},"​")],-1),i=e("ul",null,[e("li",null,"采用大光圈镜头,有很好的背景虚化效果"),e("li",null,[a("需要一个补光灯,减少避免人物逆光导致过黑 "),e("ul",null,[e("li",null,"拍摄物体在光斑前面"),e("li",null,"对焦到物体上(也可以通过精细对焦,即放大后手动对焦)")])])],-1),c=[n,r,i];function _(d,h,u,p,m,f){return o(),l("div",null,c)}const b=t(s,[["render",_]]);export{k as __pageData,b as default}; diff --git "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\232\227\350\260\203\344\272\272\345\203\217.md.09e0e873.js" "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\232\227\350\260\203\344\272\272\345\203\217.md.740aab34.js" similarity index 95% rename from "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\232\227\350\260\203\344\272\272\345\203\217.md.09e0e873.js" rename to "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\232\227\350\260\203\344\272\272\345\203\217.md.740aab34.js" index 811de2bf..d866430e 100644 --- "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\232\227\350\260\203\344\272\272\345\203\217.md.09e0e873.js" +++ "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\232\227\350\260\203\344\272\272\345\203\217.md.740aab34.js" @@ -1 +1 @@ -import{_ as a,o as e,c as t,Q as o}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"暗调人像","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/暗调人像.md","filePath":"shoot/实战技巧/暗调人像.md","lastUpdated":1736065650000}'),l={name:"shoot/实战技巧/暗调人像.md"},i=o('

暗调人像

百分之 70 都是黑色的

前置

  1. 背景是黑色
  2. 人物衣服也是黑色

如何拍摄

  1. 点测光人物面部,保证面部光亮
  2. 为了避免人物和背景重合,可以在人物背面打一束光
  3. 如果周围环境还是比较亮,可以减少一两档曝光补偿

参数参考

  • 光圈:f/2.8、曝光时间 1/125、ISO: 1600
',8),r=[i];function s(_,n,h,c,d,u){return e(),t("div",null,r)}const m=a(l,[["render",s]]);export{f as __pageData,m as default}; +import{_ as a,o as e,c as t,Q as o}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"暗调人像","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/暗调人像.md","filePath":"shoot/实战技巧/暗调人像.md","lastUpdated":1736839070000}'),l={name:"shoot/实战技巧/暗调人像.md"},i=o('

暗调人像

百分之 70 都是黑色的

前置

  1. 背景是黑色
  2. 人物衣服也是黑色

如何拍摄

  1. 点测光人物面部,保证面部光亮
  2. 为了避免人物和背景重合,可以在人物背面打一束光
  3. 如果周围环境还是比较亮,可以减少一两档曝光补偿

参数参考

  • 光圈:f/2.8、曝光时间 1/125、ISO: 1600
',8),r=[i];function s(_,n,h,c,d,u){return e(),t("div",null,r)}const m=a(l,[["render",s]]);export{f as __pageData,m as default}; diff --git "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\232\227\350\260\203\344\272\272\345\203\217.md.09e0e873.lean.js" "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\232\227\350\260\203\344\272\272\345\203\217.md.740aab34.lean.js" similarity index 86% rename from "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\232\227\350\260\203\344\272\272\345\203\217.md.09e0e873.lean.js" rename to "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\232\227\350\260\203\344\272\272\345\203\217.md.740aab34.lean.js" index 60e093a0..334ed81f 100644 --- "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\232\227\350\260\203\344\272\272\345\203\217.md.09e0e873.lean.js" +++ "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\346\232\227\350\260\203\344\272\272\345\203\217.md.740aab34.lean.js" @@ -1 +1 @@ -import{_ as a,o as e,c as t,Q as o}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"暗调人像","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/暗调人像.md","filePath":"shoot/实战技巧/暗调人像.md","lastUpdated":1736065650000}'),l={name:"shoot/实战技巧/暗调人像.md"},i=o("",8),r=[i];function s(_,n,h,c,d,u){return e(),t("div",null,r)}const m=a(l,[["render",s]]);export{f as __pageData,m as default}; +import{_ as a,o as e,c as t,Q as o}from"./chunks/framework.b6910bb2.js";const f=JSON.parse('{"title":"暗调人像","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/暗调人像.md","filePath":"shoot/实战技巧/暗调人像.md","lastUpdated":1736839070000}'),l={name:"shoot/实战技巧/暗调人像.md"},i=o("",8),r=[i];function s(_,n,h,c,d,u){return e(),t("div",null,r)}const m=a(l,[["render",s]]);export{f as __pageData,m as default}; diff --git "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\351\200\206\345\205\211\344\272\272\345\203\217.md.0e2e6697.js" "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\351\200\206\345\205\211\344\272\272\345\203\217.md.57538260.js" similarity index 95% rename from "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\351\200\206\345\205\211\344\272\272\345\203\217.md.0e2e6697.js" rename to "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\351\200\206\345\205\211\344\272\272\345\203\217.md.57538260.js" index 96a58367..4b344daa 100644 --- "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\351\200\206\345\205\211\344\272\272\345\203\217.md.0e2e6697.js" +++ "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\351\200\206\345\205\211\344\272\272\345\203\217.md.57538260.js" @@ -1 +1 @@ -import{_ as a,o as t,c as e,Q as o}from"./chunks/framework.b6910bb2.js";const s="/vitePress-blob/assets/2.cbd8c2b6.png",u=JSON.parse('{"title":"逆光人像","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/逆光人像.md","filePath":"shoot/实战技巧/逆光人像.md","lastUpdated":1736065650000}'),i={name:"shoot/实战技巧/逆光人像.md"},l=o('

逆光人像

前置

  1. 逆光,光线较好(如夕阳)
  2. 背景建议干净,突出人物

如何拍摄

  1. 采用手动对焦,避免对焦有偏差
  2. 使用点测光,对人物面部最亮部分测光
  3. 前景补光,避免人像面部过暗

灯位要求

',6),r=[l];function _(c,n,d,h,p,m){return t(),e("div",null,r)}const b=a(i,[["render",_]]);export{u as __pageData,b as default}; +import{_ as a,o as t,c as e,Q as o}from"./chunks/framework.b6910bb2.js";const s="/vitePress-blob/assets/2.cbd8c2b6.png",u=JSON.parse('{"title":"逆光人像","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/逆光人像.md","filePath":"shoot/实战技巧/逆光人像.md","lastUpdated":1736839070000}'),i={name:"shoot/实战技巧/逆光人像.md"},l=o('

逆光人像

前置

  1. 逆光,光线较好(如夕阳)
  2. 背景建议干净,突出人物

如何拍摄

  1. 采用手动对焦,避免对焦有偏差
  2. 使用点测光,对人物面部最亮部分测光
  3. 前景补光,避免人像面部过暗

灯位要求

',6),r=[l];function _(c,n,d,h,p,m){return t(),e("div",null,r)}const b=a(i,[["render",_]]);export{u as __pageData,b as default}; diff --git "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\351\200\206\345\205\211\344\272\272\345\203\217.md.0e2e6697.lean.js" "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\351\200\206\345\205\211\344\272\272\345\203\217.md.57538260.lean.js" similarity index 87% rename from "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\351\200\206\345\205\211\344\272\272\345\203\217.md.0e2e6697.lean.js" rename to "assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\351\200\206\345\205\211\344\272\272\345\203\217.md.57538260.lean.js" index 178c9404..d759b0f7 100644 --- "a/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\351\200\206\345\205\211\344\272\272\345\203\217.md.0e2e6697.lean.js" +++ "b/assets/shoot_\345\256\236\346\210\230\346\212\200\345\267\247_\351\200\206\345\205\211\344\272\272\345\203\217.md.57538260.lean.js" @@ -1 +1 @@ -import{_ as a,o as t,c as e,Q as o}from"./chunks/framework.b6910bb2.js";const s="/vitePress-blob/assets/2.cbd8c2b6.png",u=JSON.parse('{"title":"逆光人像","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/逆光人像.md","filePath":"shoot/实战技巧/逆光人像.md","lastUpdated":1736065650000}'),i={name:"shoot/实战技巧/逆光人像.md"},l=o("",6),r=[l];function _(c,n,d,h,p,m){return t(),e("div",null,r)}const b=a(i,[["render",_]]);export{u as __pageData,b as default}; +import{_ as a,o as t,c as e,Q as o}from"./chunks/framework.b6910bb2.js";const s="/vitePress-blob/assets/2.cbd8c2b6.png",u=JSON.parse('{"title":"逆光人像","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/实战技巧/逆光人像.md","filePath":"shoot/实战技巧/逆光人像.md","lastUpdated":1736839070000}'),i={name:"shoot/实战技巧/逆光人像.md"},l=o("",6),r=[l];function _(c,n,d,h,p,m){return t(),e("div",null,r)}const b=a(i,[["render",_]]);export{u as __pageData,b as default}; diff --git "a/assets/shoot_\346\236\204\345\233\276\345\275\242\345\274\217_\346\236\204\345\233\276\346\212\200\345\267\247.md.01544455.js" "b/assets/shoot_\346\236\204\345\233\276\345\275\242\345\274\217_\346\236\204\345\233\276\346\212\200\345\267\247.md.9d941cfa.js" similarity index 98% rename from "assets/shoot_\346\236\204\345\233\276\345\275\242\345\274\217_\346\236\204\345\233\276\346\212\200\345\267\247.md.01544455.js" rename to "assets/shoot_\346\236\204\345\233\276\345\275\242\345\274\217_\346\236\204\345\233\276\346\212\200\345\267\247.md.9d941cfa.js" index c1236853..4fb889bf 100644 --- "a/assets/shoot_\346\236\204\345\233\276\345\275\242\345\274\217_\346\236\204\345\233\276\346\212\200\345\267\247.md.01544455.js" +++ "b/assets/shoot_\346\236\204\345\233\276\345\275\242\345\274\217_\346\236\204\345\233\276\346\212\200\345\267\247.md.9d941cfa.js" @@ -1 +1 @@ -import{_ as t,o as a,c as e,Q as s}from"./chunks/framework.b6910bb2.js";const r="/vitePress-blob/assets/2.1e53a7e0.png",o="/vitePress-blob/assets/3.e934bbc6.png",i="/vitePress-blob/assets/4.ec3f3077.png",l="/vitePress-blob/assets/5.4c018a0f.png",n="/vitePress-blob/assets/6.f0091254.png",c="/vitePress-blob/assets/7.3dfb1ea6.png",p="/vitePress-blob/assets/8.4e45aae0.png",h="/vitePress-blob/assets/9.9c5e8ea1.png",_="/vitePress-blob/assets/10.f2b0f532.png",b="/vitePress-blob/assets/11.dce03aa0.png",d="/vitePress-blob/assets/12.d9df251e.png",m="/vitePress-blob/assets/13.677bfe09.png",g="/vitePress-blob/assets/14.3340cffc.png",P="/vitePress-blob/assets/15.58d84e53.png",f="/vitePress-blob/assets/16.7ce6920c.png",B=JSON.parse('{"title":"构图技巧","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/构图形式/构图技巧.md","filePath":"shoot/构图形式/构图技巧.md","lastUpdated":1736065650000}'),u={name:"shoot/构图形式/构图技巧.md"},x=s('

构图技巧

点构图

中心点构图

将主题放在中心,突出主题

中心点构图

中心点构图

三分点构图

三分点构图

alt text

线构图

对称线构图

可以展示对称的秩序感,常用于风光、风光 + 人文 alt text

alt text

alt text

对角线构图

alt text

alt text

引导线构图

充分利用场景中的线条,用于引导观者的注意力落在主体上,并让画面产生深度和透视感 alt text

面构图

前景构图

例如通过树叶、花草、植物等做前景

alt text

框架构图

拍摄中利用框架将主体框起来,突出主体

alt text

alt text

留白构图

留有空白,让画面更加简洁,突出主题

alt text

alt text

',31),q=[x];function v(k,T,S,$,A,N){return a(),e("div",null,q)}const C=t(u,[["render",v]]);export{B as __pageData,C as default}; +import{_ as t,o as a,c as e,Q as s}from"./chunks/framework.b6910bb2.js";const r="/vitePress-blob/assets/2.1e53a7e0.png",o="/vitePress-blob/assets/3.e934bbc6.png",i="/vitePress-blob/assets/4.ec3f3077.png",l="/vitePress-blob/assets/5.4c018a0f.png",n="/vitePress-blob/assets/6.f0091254.png",c="/vitePress-blob/assets/7.3dfb1ea6.png",p="/vitePress-blob/assets/8.4e45aae0.png",h="/vitePress-blob/assets/9.9c5e8ea1.png",_="/vitePress-blob/assets/10.f2b0f532.png",b="/vitePress-blob/assets/11.dce03aa0.png",d="/vitePress-blob/assets/12.d9df251e.png",m="/vitePress-blob/assets/13.677bfe09.png",g="/vitePress-blob/assets/14.3340cffc.png",P="/vitePress-blob/assets/15.58d84e53.png",f="/vitePress-blob/assets/16.7ce6920c.png",B=JSON.parse('{"title":"构图技巧","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/构图形式/构图技巧.md","filePath":"shoot/构图形式/构图技巧.md","lastUpdated":1736839070000}'),u={name:"shoot/构图形式/构图技巧.md"},x=s('

构图技巧

点构图

中心点构图

将主题放在中心,突出主题

中心点构图

中心点构图

三分点构图

三分点构图

alt text

线构图

对称线构图

可以展示对称的秩序感,常用于风光、风光 + 人文 alt text

alt text

alt text

对角线构图

alt text

alt text

引导线构图

充分利用场景中的线条,用于引导观者的注意力落在主体上,并让画面产生深度和透视感 alt text

面构图

前景构图

例如通过树叶、花草、植物等做前景

alt text

框架构图

拍摄中利用框架将主体框起来,突出主体

alt text

alt text

留白构图

留有空白,让画面更加简洁,突出主题

alt text

alt text

',31),q=[x];function v(k,T,S,$,A,N){return a(),e("div",null,q)}const C=t(u,[["render",v]]);export{B as __pageData,C as default}; diff --git "a/assets/shoot_\346\236\204\345\233\276\345\275\242\345\274\217_\346\236\204\345\233\276\346\212\200\345\267\247.md.01544455.lean.js" "b/assets/shoot_\346\236\204\345\233\276\345\275\242\345\274\217_\346\236\204\345\233\276\346\212\200\345\267\247.md.9d941cfa.lean.js" similarity index 94% rename from "assets/shoot_\346\236\204\345\233\276\345\275\242\345\274\217_\346\236\204\345\233\276\346\212\200\345\267\247.md.01544455.lean.js" rename to "assets/shoot_\346\236\204\345\233\276\345\275\242\345\274\217_\346\236\204\345\233\276\346\212\200\345\267\247.md.9d941cfa.lean.js" index be575ebd..6fb8ffbc 100644 --- "a/assets/shoot_\346\236\204\345\233\276\345\275\242\345\274\217_\346\236\204\345\233\276\346\212\200\345\267\247.md.01544455.lean.js" +++ "b/assets/shoot_\346\236\204\345\233\276\345\275\242\345\274\217_\346\236\204\345\233\276\346\212\200\345\267\247.md.9d941cfa.lean.js" @@ -1 +1 @@ -import{_ as t,o as a,c as e,Q as s}from"./chunks/framework.b6910bb2.js";const r="/vitePress-blob/assets/2.1e53a7e0.png",o="/vitePress-blob/assets/3.e934bbc6.png",i="/vitePress-blob/assets/4.ec3f3077.png",l="/vitePress-blob/assets/5.4c018a0f.png",n="/vitePress-blob/assets/6.f0091254.png",c="/vitePress-blob/assets/7.3dfb1ea6.png",p="/vitePress-blob/assets/8.4e45aae0.png",h="/vitePress-blob/assets/9.9c5e8ea1.png",_="/vitePress-blob/assets/10.f2b0f532.png",b="/vitePress-blob/assets/11.dce03aa0.png",d="/vitePress-blob/assets/12.d9df251e.png",m="/vitePress-blob/assets/13.677bfe09.png",g="/vitePress-blob/assets/14.3340cffc.png",P="/vitePress-blob/assets/15.58d84e53.png",f="/vitePress-blob/assets/16.7ce6920c.png",B=JSON.parse('{"title":"构图技巧","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/构图形式/构图技巧.md","filePath":"shoot/构图形式/构图技巧.md","lastUpdated":1736065650000}'),u={name:"shoot/构图形式/构图技巧.md"},x=s("",31),q=[x];function v(k,T,S,$,A,N){return a(),e("div",null,q)}const C=t(u,[["render",v]]);export{B as __pageData,C as default}; +import{_ as t,o as a,c as e,Q as s}from"./chunks/framework.b6910bb2.js";const r="/vitePress-blob/assets/2.1e53a7e0.png",o="/vitePress-blob/assets/3.e934bbc6.png",i="/vitePress-blob/assets/4.ec3f3077.png",l="/vitePress-blob/assets/5.4c018a0f.png",n="/vitePress-blob/assets/6.f0091254.png",c="/vitePress-blob/assets/7.3dfb1ea6.png",p="/vitePress-blob/assets/8.4e45aae0.png",h="/vitePress-blob/assets/9.9c5e8ea1.png",_="/vitePress-blob/assets/10.f2b0f532.png",b="/vitePress-blob/assets/11.dce03aa0.png",d="/vitePress-blob/assets/12.d9df251e.png",m="/vitePress-blob/assets/13.677bfe09.png",g="/vitePress-blob/assets/14.3340cffc.png",P="/vitePress-blob/assets/15.58d84e53.png",f="/vitePress-blob/assets/16.7ce6920c.png",B=JSON.parse('{"title":"构图技巧","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/构图形式/构图技巧.md","filePath":"shoot/构图形式/构图技巧.md","lastUpdated":1736839070000}'),u={name:"shoot/构图形式/构图技巧.md"},x=s("",31),q=[x];function v(k,T,S,$,A,N){return a(),e("div",null,q)}const C=t(u,[["render",v]]);export{B as __pageData,C as default}; diff --git "a/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\345\273\266\350\277\237\346\221\204\345\275\261.md.7117a224.js" "b/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\345\273\266\350\277\237\346\221\204\345\275\261.md.abf24cee.js" similarity index 97% rename from "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\345\273\266\350\277\237\346\221\204\345\275\261.md.7117a224.js" rename to "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\345\273\266\350\277\237\346\221\204\345\275\261.md.abf24cee.js" index 093b6123..7fc76992 100644 --- "a/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\345\273\266\350\277\237\346\221\204\345\275\261.md.7117a224.js" +++ "b/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\345\273\266\350\277\237\346\221\204\345\275\261.md.abf24cee.js" @@ -1 +1 @@ -import{_ as e,o as a,c as t,Q as o}from"./chunks/framework.b6910bb2.js";const s="/vitePress-blob/assets/2.be68a57d.png",r="/vitePress-blob/assets/4.9aaf818a.png",i="/vitePress-blob/assets/3.038403c4.png",g=JSON.parse('{"title":"延迟摄影","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/自我实战总结/延迟摄影.md","filePath":"shoot/自我实战总结/延迟摄影.md","lastUpdated":1736065650000}'),l={name:"shoot/自我实战总结/延迟摄影.md"},n=o('

延迟摄影

如何拍摄

  1. 调节到视频模式

  2. 打开 Menu 菜单,在第一大项的第4小项中找到延时短片,打开这一功能。之后,回到主屏幕,我们发现左上角摄影机图标旁多出了一个时针图案和一组4位数字,这代表延时摄影已打开成功,准备进入拍摄。 Menu 菜单

  3. 此时,主屏幕下方的三个参数均可手动调整,从左到右依次是快门、光圈、感光度ISO Menu 菜单

  4. 自动曝光设置 固定第一帧: 固定第一帧指的是延时短片将始终以第一帧曝光的参数拍摄下去,中途不会发生改变,适用于光源亮度变化不大的场景 每一帧: 延时短片将由相机依据环境光线的变化,自动测光来决定每一帧的曝光值,适用于光源有较大改变的场景,如日转夜或夜转日 (ISO 感光度要设置成 AUTO) 自动曝光

参考文章

  1. https://www.xiaohongshu.com/explore/651389a3000000001e023777
',5),_=[n];function c(h,p,d,m,u,b){return a(),t("div",null,_)}const x=e(l,[["render",c]]);export{g as __pageData,x as default}; +import{_ as e,o as a,c as t,Q as o}from"./chunks/framework.b6910bb2.js";const s="/vitePress-blob/assets/2.be68a57d.png",r="/vitePress-blob/assets/4.9aaf818a.png",i="/vitePress-blob/assets/3.038403c4.png",g=JSON.parse('{"title":"延迟摄影","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/自我实战总结/延迟摄影.md","filePath":"shoot/自我实战总结/延迟摄影.md","lastUpdated":1736839070000}'),l={name:"shoot/自我实战总结/延迟摄影.md"},n=o('

延迟摄影

如何拍摄

  1. 调节到视频模式

  2. 打开 Menu 菜单,在第一大项的第4小项中找到延时短片,打开这一功能。之后,回到主屏幕,我们发现左上角摄影机图标旁多出了一个时针图案和一组4位数字,这代表延时摄影已打开成功,准备进入拍摄。 Menu 菜单

  3. 此时,主屏幕下方的三个参数均可手动调整,从左到右依次是快门、光圈、感光度ISO Menu 菜单

  4. 自动曝光设置 固定第一帧: 固定第一帧指的是延时短片将始终以第一帧曝光的参数拍摄下去,中途不会发生改变,适用于光源亮度变化不大的场景 每一帧: 延时短片将由相机依据环境光线的变化,自动测光来决定每一帧的曝光值,适用于光源有较大改变的场景,如日转夜或夜转日 (ISO 感光度要设置成 AUTO) 自动曝光

参考文章

  1. https://www.xiaohongshu.com/explore/651389a3000000001e023777
',5),_=[n];function c(h,p,d,m,u,b){return a(),t("div",null,_)}const x=e(l,[["render",c]]);export{g as __pageData,x as default}; diff --git "a/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\345\273\266\350\277\237\346\221\204\345\275\261.md.7117a224.lean.js" "b/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\345\273\266\350\277\237\346\221\204\345\275\261.md.abf24cee.lean.js" similarity index 89% rename from "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\345\273\266\350\277\237\346\221\204\345\275\261.md.7117a224.lean.js" rename to "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\345\273\266\350\277\237\346\221\204\345\275\261.md.abf24cee.lean.js" index 368d99eb..7d6602bf 100644 --- "a/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\345\273\266\350\277\237\346\221\204\345\275\261.md.7117a224.lean.js" +++ "b/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\345\273\266\350\277\237\346\221\204\345\275\261.md.abf24cee.lean.js" @@ -1 +1 @@ -import{_ as e,o as a,c as t,Q as o}from"./chunks/framework.b6910bb2.js";const s="/vitePress-blob/assets/2.be68a57d.png",r="/vitePress-blob/assets/4.9aaf818a.png",i="/vitePress-blob/assets/3.038403c4.png",g=JSON.parse('{"title":"延迟摄影","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/自我实战总结/延迟摄影.md","filePath":"shoot/自我实战总结/延迟摄影.md","lastUpdated":1736065650000}'),l={name:"shoot/自我实战总结/延迟摄影.md"},n=o("",5),_=[n];function c(h,p,d,m,u,b){return a(),t("div",null,_)}const x=e(l,[["render",c]]);export{g as __pageData,x as default}; +import{_ as e,o as a,c as t,Q as o}from"./chunks/framework.b6910bb2.js";const s="/vitePress-blob/assets/2.be68a57d.png",r="/vitePress-blob/assets/4.9aaf818a.png",i="/vitePress-blob/assets/3.038403c4.png",g=JSON.parse('{"title":"延迟摄影","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/自我实战总结/延迟摄影.md","filePath":"shoot/自我实战总结/延迟摄影.md","lastUpdated":1736839070000}'),l={name:"shoot/自我实战总结/延迟摄影.md"},n=o("",5),_=[n];function c(h,p,d,m,u,b){return a(),t("div",null,_)}const x=e(l,[["render",c]]);export{g as __pageData,x as default}; diff --git "a/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\230\237\347\251\272.md.cbaec0d7.js" "b/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\230\237\347\251\272.md.79082746.js" similarity index 97% rename from "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\230\237\347\251\272.md.cbaec0d7.js" rename to "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\230\237\347\251\272.md.79082746.js" index aad56912..8f5d3337 100644 --- "a/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\230\237\347\251\272.md.cbaec0d7.js" +++ "b/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\230\237\347\251\272.md.79082746.js" @@ -1 +1 @@ -import{_ as a,o as e,c as t,Q as o}from"./chunks/framework.b6910bb2.js";const r="/vitePress-blob/assets/5.469845e2.png",s="/vitePress-blob/assets/6.c6e8eb70.png",f=JSON.parse('{"title":"拍摄星空","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/自我实战总结/拍摄星空.md","filePath":"shoot/自我实战总结/拍摄星空.md","lastUpdated":1736065650000}'),h={name:"shoot/自我实战总结/拍摄星空.md"},i=o('

拍摄星空

大光圈、感光度、长曝光,这是拍摄星空的三大要素。

大光圈

f1.8

感光度

ISO:2000 以上(感光度越大、图片越亮,但也容易出现噪点)

手动对焦

手动对焦到某一颗星星上,然后锁定对焦

长曝光

快门速度 10秒以上, 连续拍10张以上后期堆栈(拍得多更好)

脚架

需要使用三脚架

星空

星空参数

参考文章

  1. https://www.xiaohongshu.com/explore/6507b277000000001e02de1a
',16),l=[i];function n(c,d,p,_,b,m){return e(),t("div",null,l)}const q=a(h,[["render",n]]);export{f as __pageData,q as default}; +import{_ as a,o as e,c as t,Q as o}from"./chunks/framework.b6910bb2.js";const r="/vitePress-blob/assets/5.469845e2.png",s="/vitePress-blob/assets/6.c6e8eb70.png",f=JSON.parse('{"title":"拍摄星空","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/自我实战总结/拍摄星空.md","filePath":"shoot/自我实战总结/拍摄星空.md","lastUpdated":1736839070000}'),h={name:"shoot/自我实战总结/拍摄星空.md"},i=o('

拍摄星空

大光圈、感光度、长曝光,这是拍摄星空的三大要素。

大光圈

f1.8

感光度

ISO:2000 以上(感光度越大、图片越亮,但也容易出现噪点)

手动对焦

手动对焦到某一颗星星上,然后锁定对焦

长曝光

快门速度 10秒以上, 连续拍10张以上后期堆栈(拍得多更好)

脚架

需要使用三脚架

星空

星空参数

参考文章

  1. https://www.xiaohongshu.com/explore/6507b277000000001e02de1a
',16),l=[i];function n(c,d,p,_,b,m){return e(),t("div",null,l)}const q=a(h,[["render",n]]);export{f as __pageData,q as default}; diff --git "a/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\230\237\347\251\272.md.cbaec0d7.lean.js" "b/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\230\237\347\251\272.md.79082746.lean.js" similarity index 88% rename from "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\230\237\347\251\272.md.cbaec0d7.lean.js" rename to "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\230\237\347\251\272.md.79082746.lean.js" index 84775324..35b44b0b 100644 --- "a/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\230\237\347\251\272.md.cbaec0d7.lean.js" +++ "b/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\230\237\347\251\272.md.79082746.lean.js" @@ -1 +1 @@ -import{_ as a,o as e,c as t,Q as o}from"./chunks/framework.b6910bb2.js";const r="/vitePress-blob/assets/5.469845e2.png",s="/vitePress-blob/assets/6.c6e8eb70.png",f=JSON.parse('{"title":"拍摄星空","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/自我实战总结/拍摄星空.md","filePath":"shoot/自我实战总结/拍摄星空.md","lastUpdated":1736065650000}'),h={name:"shoot/自我实战总结/拍摄星空.md"},i=o("",16),l=[i];function n(c,d,p,_,b,m){return e(),t("div",null,l)}const q=a(h,[["render",n]]);export{f as __pageData,q as default}; +import{_ as a,o as e,c as t,Q as o}from"./chunks/framework.b6910bb2.js";const r="/vitePress-blob/assets/5.469845e2.png",s="/vitePress-blob/assets/6.c6e8eb70.png",f=JSON.parse('{"title":"拍摄星空","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/自我实战总结/拍摄星空.md","filePath":"shoot/自我实战总结/拍摄星空.md","lastUpdated":1736839070000}'),h={name:"shoot/自我实战总结/拍摄星空.md"},i=o("",16),l=[i];function n(c,d,p,_,b,m){return e(),t("div",null,l)}const q=a(h,[["render",n]]);export{f as __pageData,q as default}; diff --git "a/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\234\210\344\272\256.md.155f2cd6.js" "b/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\234\210\344\272\256.md.6fbf4de5.js" similarity index 92% rename from "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\234\210\344\272\256.md.155f2cd6.js" rename to "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\234\210\344\272\256.md.6fbf4de5.js" index 3a0b2173..1de00e41 100644 --- "a/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\234\210\344\272\256.md.155f2cd6.js" +++ "b/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\234\210\344\272\256.md.6fbf4de5.js" @@ -1 +1 @@ -import{_ as t,o as a,c as o,k as e,a as s}from"./chunks/framework.b6910bb2.js";const x=JSON.parse('{"title":"拍摄月亮","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/自我实战总结/拍摄月亮.md","filePath":"shoot/自我实战总结/拍摄月亮.md","lastUpdated":1736065650000}'),n={name:"shoot/自我实战总结/拍摄月亮.md"},r=e("h1",{id:"拍摄月亮",tabindex:"-1"},[s("拍摄月亮 "),e("a",{class:"header-anchor",href:"#拍摄月亮","aria-label":'Permalink to "拍摄月亮"'},"​")],-1),l=e("ul",null,[e("li",null,"光圈一般我喜欢小光圈 在 f8 - f10 之间"),e("li",null,"需要使用长焦镜头,例如 70-200mm")],-1),c=[r,l];function _(d,i,h,m,p,f){return a(),o("div",null,c)}const k=t(n,[["render",_]]);export{x as __pageData,k as default}; +import{_ as t,o as a,c as o,k as e,a as s}from"./chunks/framework.b6910bb2.js";const x=JSON.parse('{"title":"拍摄月亮","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/自我实战总结/拍摄月亮.md","filePath":"shoot/自我实战总结/拍摄月亮.md","lastUpdated":1736839070000}'),n={name:"shoot/自我实战总结/拍摄月亮.md"},r=e("h1",{id:"拍摄月亮",tabindex:"-1"},[s("拍摄月亮 "),e("a",{class:"header-anchor",href:"#拍摄月亮","aria-label":'Permalink to "拍摄月亮"'},"​")],-1),l=e("ul",null,[e("li",null,"光圈一般我喜欢小光圈 在 f8 - f10 之间"),e("li",null,"需要使用长焦镜头,例如 70-200mm")],-1),c=[r,l];function _(d,i,h,m,p,f){return a(),o("div",null,c)}const k=t(n,[["render",_]]);export{x as __pageData,k as default}; diff --git "a/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\234\210\344\272\256.md.155f2cd6.lean.js" "b/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\234\210\344\272\256.md.6fbf4de5.lean.js" similarity index 92% rename from "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\234\210\344\272\256.md.155f2cd6.lean.js" rename to "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\234\210\344\272\256.md.6fbf4de5.lean.js" index 3a0b2173..1de00e41 100644 --- "a/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\234\210\344\272\256.md.155f2cd6.lean.js" +++ "b/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\346\234\210\344\272\256.md.6fbf4de5.lean.js" @@ -1 +1 @@ -import{_ as t,o as a,c as o,k as e,a as s}from"./chunks/framework.b6910bb2.js";const x=JSON.parse('{"title":"拍摄月亮","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/自我实战总结/拍摄月亮.md","filePath":"shoot/自我实战总结/拍摄月亮.md","lastUpdated":1736065650000}'),n={name:"shoot/自我实战总结/拍摄月亮.md"},r=e("h1",{id:"拍摄月亮",tabindex:"-1"},[s("拍摄月亮 "),e("a",{class:"header-anchor",href:"#拍摄月亮","aria-label":'Permalink to "拍摄月亮"'},"​")],-1),l=e("ul",null,[e("li",null,"光圈一般我喜欢小光圈 在 f8 - f10 之间"),e("li",null,"需要使用长焦镜头,例如 70-200mm")],-1),c=[r,l];function _(d,i,h,m,p,f){return a(),o("div",null,c)}const k=t(n,[["render",_]]);export{x as __pageData,k as default}; +import{_ as t,o as a,c as o,k as e,a as s}from"./chunks/framework.b6910bb2.js";const x=JSON.parse('{"title":"拍摄月亮","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/自我实战总结/拍摄月亮.md","filePath":"shoot/自我实战总结/拍摄月亮.md","lastUpdated":1736839070000}'),n={name:"shoot/自我实战总结/拍摄月亮.md"},r=e("h1",{id:"拍摄月亮",tabindex:"-1"},[s("拍摄月亮 "),e("a",{class:"header-anchor",href:"#拍摄月亮","aria-label":'Permalink to "拍摄月亮"'},"​")],-1),l=e("ul",null,[e("li",null,"光圈一般我喜欢小光圈 在 f8 - f10 之间"),e("li",null,"需要使用长焦镜头,例如 70-200mm")],-1),c=[r,l];function _(d,i,h,m,p,f){return a(),o("div",null,c)}const k=t(n,[["render",_]]);export{x as __pageData,k as default}; diff --git "a/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\347\203\237\350\212\261.md.481cbdad.js" "b/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\347\203\237\350\212\261.md.bff2e64b.js" similarity index 93% rename from "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\347\203\237\350\212\261.md.481cbdad.js" rename to "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\347\203\237\350\212\261.md.bff2e64b.js" index 864f7737..62b750d0 100644 --- "a/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\347\203\237\350\212\261.md.481cbdad.js" +++ "b/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\347\203\237\350\212\261.md.bff2e64b.js" @@ -1 +1 @@ -import{_ as t,o as s,c as a,k as e,a as o}from"./chunks/framework.b6910bb2.js";const l="/vitePress-blob/assets/1.00dedace.jpg",k=JSON.parse('{"title":"拍摄烟花","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/自我实战总结/拍摄烟花.md","filePath":"shoot/自我实战总结/拍摄烟花.md","lastUpdated":1736065650000}'),n={name:"shoot/自我实战总结/拍摄烟花.md"},r=e("h1",{id:"拍摄烟花",tabindex:"-1"},[o("拍摄烟花 "),e("a",{class:"header-anchor",href:"#拍摄烟花","aria-label":'Permalink to "拍摄烟花"'},"​")],-1),c=e("ul",null,[e("li",null,"快门速度一般比较长可以出现拉线效果,例如 0.5s/1s"),e("li",null,"ISO 感光度一般在 500 以下,这样背景会黑一些"),e("li",null,"光圈一般我喜欢小光圈 在 f10 以后")],-1),_=e("p",null,[e("img",{src:l,alt:"快门速度"})],-1),i=[r,c,_];function d(h,p,m,u,f,x){return s(),a("div",null,i)}const P=t(n,[["render",d]]);export{k as __pageData,P as default}; +import{_ as t,o as s,c as a,k as e,a as o}from"./chunks/framework.b6910bb2.js";const l="/vitePress-blob/assets/1.00dedace.jpg",k=JSON.parse('{"title":"拍摄烟花","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/自我实战总结/拍摄烟花.md","filePath":"shoot/自我实战总结/拍摄烟花.md","lastUpdated":1736839070000}'),n={name:"shoot/自我实战总结/拍摄烟花.md"},r=e("h1",{id:"拍摄烟花",tabindex:"-1"},[o("拍摄烟花 "),e("a",{class:"header-anchor",href:"#拍摄烟花","aria-label":'Permalink to "拍摄烟花"'},"​")],-1),c=e("ul",null,[e("li",null,"快门速度一般比较长可以出现拉线效果,例如 0.5s/1s"),e("li",null,"ISO 感光度一般在 500 以下,这样背景会黑一些"),e("li",null,"光圈一般我喜欢小光圈 在 f10 以后")],-1),_=e("p",null,[e("img",{src:l,alt:"快门速度"})],-1),i=[r,c,_];function d(h,p,m,u,f,x){return s(),a("div",null,i)}const P=t(n,[["render",d]]);export{k as __pageData,P as default}; diff --git "a/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\347\203\237\350\212\261.md.481cbdad.lean.js" "b/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\347\203\237\350\212\261.md.bff2e64b.lean.js" similarity index 93% rename from "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\347\203\237\350\212\261.md.481cbdad.lean.js" rename to "assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\347\203\237\350\212\261.md.bff2e64b.lean.js" index 864f7737..62b750d0 100644 --- "a/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\347\203\237\350\212\261.md.481cbdad.lean.js" +++ "b/assets/shoot_\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223_\346\213\215\346\221\204\347\203\237\350\212\261.md.bff2e64b.lean.js" @@ -1 +1 @@ -import{_ as t,o as s,c as a,k as e,a as o}from"./chunks/framework.b6910bb2.js";const l="/vitePress-blob/assets/1.00dedace.jpg",k=JSON.parse('{"title":"拍摄烟花","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/自我实战总结/拍摄烟花.md","filePath":"shoot/自我实战总结/拍摄烟花.md","lastUpdated":1736065650000}'),n={name:"shoot/自我实战总结/拍摄烟花.md"},r=e("h1",{id:"拍摄烟花",tabindex:"-1"},[o("拍摄烟花 "),e("a",{class:"header-anchor",href:"#拍摄烟花","aria-label":'Permalink to "拍摄烟花"'},"​")],-1),c=e("ul",null,[e("li",null,"快门速度一般比较长可以出现拉线效果,例如 0.5s/1s"),e("li",null,"ISO 感光度一般在 500 以下,这样背景会黑一些"),e("li",null,"光圈一般我喜欢小光圈 在 f10 以后")],-1),_=e("p",null,[e("img",{src:l,alt:"快门速度"})],-1),i=[r,c,_];function d(h,p,m,u,f,x){return s(),a("div",null,i)}const P=t(n,[["render",d]]);export{k as __pageData,P as default}; +import{_ as t,o as s,c as a,k as e,a as o}from"./chunks/framework.b6910bb2.js";const l="/vitePress-blob/assets/1.00dedace.jpg",k=JSON.parse('{"title":"拍摄烟花","description":"","frontmatter":{},"headers":[],"relativePath":"shoot/自我实战总结/拍摄烟花.md","filePath":"shoot/自我实战总结/拍摄烟花.md","lastUpdated":1736839070000}'),n={name:"shoot/自我实战总结/拍摄烟花.md"},r=e("h1",{id:"拍摄烟花",tabindex:"-1"},[o("拍摄烟花 "),e("a",{class:"header-anchor",href:"#拍摄烟花","aria-label":'Permalink to "拍摄烟花"'},"​")],-1),c=e("ul",null,[e("li",null,"快门速度一般比较长可以出现拉线效果,例如 0.5s/1s"),e("li",null,"ISO 感光度一般在 500 以下,这样背景会黑一些"),e("li",null,"光圈一般我喜欢小光圈 在 f10 以后")],-1),_=e("p",null,[e("img",{src:l,alt:"快门速度"})],-1),i=[r,c,_];function d(h,p,m,u,f,x){return s(),a("div",null,i)}const P=t(n,[["render",d]]);export{k as __pageData,P as default}; diff --git "a/guide/DDD \351\242\206\345\237\237\351\251\261\345\212\250\350\256\276\350\256\241/index.html" "b/guide/DDD \351\242\206\345\237\237\351\251\261\345\212\250\350\256\276\350\256\241/index.html" index 049895d9..8089fc37 100644 --- "a/guide/DDD \351\242\206\345\237\237\351\251\261\345\212\250\350\256\276\350\256\241/index.html" +++ "b/guide/DDD \351\242\206\345\237\237\351\251\261\345\212\250\350\256\276\350\256\241/index.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

领域驱动设计

定义

- +
Skip to content

领域驱动设计

定义

+ \ No newline at end of file diff --git "a/guide/Fabric.js/\345\246\202\344\275\225\345\256\236\347\216\260\347\273\204\351\200\211.html" "b/guide/Fabric.js/Fabric \344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.html" similarity index 69% rename from "guide/Fabric.js/\345\246\202\344\275\225\345\256\236\347\216\260\347\273\204\351\200\211.html" rename to "guide/Fabric.js/Fabric \344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.html" index f05d745a..e6dbffed 100644 --- "a/guide/Fabric.js/\345\246\202\344\275\225\345\256\236\347\216\260\347\273\204\351\200\211.html" +++ "b/guide/Fabric.js/Fabric \344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.html" @@ -3,7 +3,7 @@ - 个人知识库 | 个人知识库 + Fabric 中的性能优化 | 个人知识库 @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content
- +
Skip to content
+ \ No newline at end of file diff --git "a/guide/Fabric.js/\345\206\205\351\203\250\347\273\223\346\236\204\346\246\202\350\277\260.html" "b/guide/Fabric.js/\345\206\205\351\203\250\347\273\223\346\236\204\346\246\202\350\277\260.html" index 686d4292..d0154f81 100644 --- "a/guide/Fabric.js/\345\206\205\351\203\250\347\273\223\346\236\204\346\246\202\350\277\260.html" +++ "b/guide/Fabric.js/\345\206\205\351\203\250\347\273\223\346\236\204\346\246\202\350\277\260.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

内部结构概述

使用场景上看

从使用场景上看,Fabric.js 通过对象的形式来创建图形 (源码上也充满了面向对象设计的身影)

对于使用者而言,不需要关心底层 Canvas 是如何实现的,只需要关心如何创建这些图像对象即可。

Canvas 开发原理

Fabric 底层是通过 Canvas 进行绘制的,因此需先简单了解下 Canvas 的开发原理。

Canvas 开发的本质其实很简单,可以想象下这种少儿画板:

初始状态图

Canvas 的绘制其实就是在不断在画布上绘制图形,然后擦出的过程。

Fabric.js 源码解析

基本原理

fabric.js 在初始化的时候会将你指定的 Canvas 元素(叫做 lowerCanvas)外面包裹上一 层 div 元素, 然后内部会插入另外一个上层的 Canvas 元素(叫做 upperCanvas),这两 个 Canvas 有如下区别

  • upperCanvas 动态层Canvas,上层画布,只处理交互相关以及事件绑定

  • lowerCanvas 静态层Canvas,是用来绘制图形

alt text

大体结构

当我们引入 fabric.js 后,再控制台打印 fabric ,可以看到如下结构:

初始状态图

可以看到,fabric 是一个对象,里面包含了多个属性,这些属性指向不同的模块。

Fabric的源码也是相当易读的,大部分情况下一个文件就代表一个模块,源码结构如下:

alt text

模块结构图

基于上述源码结构,可以很方便的绘制出 Fabric 的大概的一个模块结构图:

alt text

至此,我们大致了解了 Fabric 的内部结构,接下来,我们将通过一个简单的例子,来了解下 Fabric 是如何绘制一个图形。

参考

- +
Skip to content

内部结构概述

使用场景上看

从使用场景上看,Fabric.js 通过对象的形式来创建图形 (源码上也充满了面向对象设计的身影)

对于使用者而言,不需要关心底层 Canvas 是如何实现的,只需要关心如何创建这些图像对象即可。

Canvas 开发原理

Fabric 底层是通过 Canvas 进行绘制的,因此需先简单了解下 Canvas 的开发原理。

Canvas 开发的本质其实很简单,可以想象下这种少儿画板:

初始状态图

Canvas 的绘制其实就是在不断在画布上绘制图形,然后擦出的过程。

Fabric.js 源码解析

基本原理

fabric.js 在初始化的时候会将你指定的 Canvas 元素(叫做 lowerCanvas)外面包裹上一 层 div 元素, 然后内部会插入另外一个上层的 Canvas 元素(叫做 upperCanvas),这两 个 Canvas 有如下区别

  • upperCanvas 动态层Canvas,上层画布,只处理交互相关以及事件绑定

  • lowerCanvas 静态层Canvas,是用来绘制图形

alt text

大体结构

当我们引入 fabric.js 后,再控制台打印 fabric ,可以看到如下结构:

初始状态图

可以看到,fabric 是一个对象,里面包含了多个属性,这些属性指向不同的模块。

Fabric的源码也是相当易读的,大部分情况下一个文件就代表一个模块,源码结构如下:

alt text

模块结构图

基于上述源码结构,可以很方便的绘制出 Fabric 的大概的一个模块结构图:

alt text

至此,我们大致了解了 Fabric 的内部结构,接下来,我们将通过一个简单的例子,来了解下 Fabric 是如何绘制一个图形。

参考

+ \ No newline at end of file diff --git "a/guide/Fabric.js/\345\237\272\346\234\254\346\246\202\345\277\265.html" "b/guide/Fabric.js/\345\237\272\346\234\254\346\246\202\345\277\265.html" index 70234bd0..4b626b23 100644 --- "a/guide/Fabric.js/\345\237\272\346\234\254\346\246\202\345\277\265.html" +++ "b/guide/Fabric.js/\345\237\272\346\234\254\346\246\202\345\277\265.html" @@ -12,7 +12,7 @@ - + @@ -24,12 +24,12 @@ -
Skip to content

简介

Fabric.js 是一个强大而简单的 Canvas 画布库

可以提供简单的 API 来创建和编辑复杂的图形

基本使用

创建画布

html
<!-- HTML -->
+    
Skip to content

简介

Fabric.js 是一个强大而简单的 Canvas 画布库

可以提供简单的 API 来创建和编辑复杂的图形

基本使用

创建画布

html
<!-- HTML -->
 <canvas id="canvas"></canvas>
<!-- HTML -->
 <canvas id="canvas"></canvas>
js
// JavaScript
 const canvas = new fabric.Canvas('canvas');
// JavaScript
-const canvas = new fabric.Canvas('canvas');

添加图形到画布

js
canvas.add(new fabric.Rect({ left: 100, top: 100, fill: 'red', width: 100, height: 100 }));
canvas.add(new fabric.Rect({ left: 100, top: 100, fill: 'red', width: 100, height: 100 }));

通过案例可以感受到,如果你需要用 HTML Canvas 来绘制一些东西,并且这些东西可以响应用户的交互,比如:拖动、变形、旋转等 操作。 那用 fabric.js 是非常合适的,因为它内部不仅实现了 Canvas 对象模型,还将一些常用的交互操作封装好了,可以说是开箱即用。

- +const canvas = new fabric.Canvas('canvas');

添加图形到画布

js
canvas.add(new fabric.Rect({ left: 100, top: 100, fill: 'red', width: 100, height: 100 }));
canvas.add(new fabric.Rect({ left: 100, top: 100, fill: 'red', width: 100, height: 100 }));

通过案例可以感受到,如果你需要用 HTML Canvas 来绘制一些东西,并且这些东西可以响应用户的交互,比如:拖动、变形、旋转等 操作。 那用 fabric.js 是非常合适的,因为它内部不仅实现了 Canvas 对象模型,还将一些常用的交互操作封装好了,可以说是开箱即用。

+ \ No newline at end of file diff --git "a/guide/Fabric.js/\345\246\202\344\275\225\345\256\236\347\216\260\345\205\203\347\264\240\347\232\204\345\271\263\347\247\273\343\200\201\346\227\213\350\275\254\343\200\201\347\274\251\346\224\276.html" "b/guide/Fabric.js/\345\246\202\344\275\225\345\256\236\347\216\260\345\205\203\347\264\240\347\232\204\345\271\263\347\247\273\343\200\201\346\227\213\350\275\254\343\200\201\347\274\251\346\224\276.html" index 25c57335..85a32830 100644 --- "a/guide/Fabric.js/\345\246\202\344\275\225\345\256\236\347\216\260\345\205\203\347\264\240\347\232\204\345\271\263\347\247\273\343\200\201\346\227\213\350\275\254\343\200\201\347\274\251\346\224\276.html" +++ "b/guide/Fabric.js/\345\246\202\344\275\225\345\256\236\347\216\260\345\205\203\347\264\240\347\232\204\345\271\263\347\247\273\343\200\201\346\227\213\350\275\254\343\200\201\347\274\251\346\224\276.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

如何实现元素的平移、旋转、缩放

在上一节中,我们知道了元素是如何渲染到 Canvas 的画布上,这一节我们将了解如何对元素进行平移、旋转、缩放。

如何平移、旋转、缩放元素

在上一节我们知道,元素都继承自 fabric.Object,而 fabric.Object 的 render 方法都由各个元素实现。那么对于元素的平移、旋转、缩放,我们又需要如何抽离出通用的逻辑呢?

不妨思考一下,这个案例

假设要在 (100, 100) 的地方绘制一个 50*50 的矩形,并将其放大 2 倍,之后旋转 45°,该怎么画呢?

正常逻辑是:

  1. 手动算下宽高 100 * 100

  2. 手动算下旋转之后各个顶点的坐标

  3. 连接四个顶点,绘制矩形

也就是我们先手动先计算出各个转化后的坐标,通过坐标绘制出图形。

但在 Canvas 中,要改掉这种绘制的思想,而是要通过并善用变换坐标系来绘制物体

思路如下: 1 绘制一个 50*50 的矩形

2 转化坐标系

代码如下:

js
ctx.save(); // 之前提到过了,你要修改 ctx 上的一些配置或者画一个物体,最好先 save 一下,这是个好习惯
+    
Skip to content

如何实现元素的平移、旋转、缩放

在上一节中,我们知道了元素是如何渲染到 Canvas 的画布上,这一节我们将了解如何对元素进行平移、旋转、缩放。

如何平移、旋转、缩放元素

在上一节我们知道,元素都继承自 fabric.Object,而 fabric.Object 的 render 方法都由各个元素实现。那么对于元素的平移、旋转、缩放,我们又需要如何抽离出通用的逻辑呢?

不妨思考一下,这个案例

假设要在 (100, 100) 的地方绘制一个 50*50 的矩形,并将其放大 2 倍,之后旋转 45°,该怎么画呢?

正常逻辑是:

  1. 手动算下宽高 100 * 100

  2. 手动算下旋转之后各个顶点的坐标

  3. 连接四个顶点,绘制矩形

也就是我们先手动先计算出各个转化后的坐标,通过坐标绘制出图形。

但在 Canvas 中,要改掉这种绘制的思想,而是要通过并善用变换坐标系来绘制物体

思路如下: 1 绘制一个 50*50 的矩形

2 转化坐标系

代码如下:

js
ctx.save(); // 之前提到过了,你要修改 ctx 上的一些配置或者画一个物体,最好先 save 一下,这是个好习惯
 ctx.translate(100, 100); // 此时原点已经变到了 (100, 100) 的地方
 ctx.scale(2, 2); // 坐标系放大两倍
 ctx.rotate(Util.degreesToRadians(45)); // 注意 canvas 中用的都是弧度(弧度 / 2 * Math.PI = 角度 / 360),所以需要简单换算下
@@ -115,8 +115,8 @@
    // canvas.zoomToPoint({ x: opt.e.offsetX, y: opt.e.offsetY }, zoom);
    opt.e.preventDefault();
    opt.e.stopPropagation();
-});

演示效果👇:

alt text

演示地址: https://enson0131.github.io/mini-fabric-whiteboard/

仓库案例

https://github.com/enson0131/mini-fabric-whiteboard/blob/master/mini-fabric/src/fabric/FabricObject.ts#L212

参考文档

- +});

演示效果👇:

alt text

演示地址: https://enson0131.github.io/mini-fabric-whiteboard/

仓库案例

https://github.com/enson0131/mini-fabric-whiteboard/blob/master/mini-fabric/src/fabric/FabricObject.ts#L212

参考文档

+ \ No newline at end of file diff --git "a/guide/Fabric.js/\345\246\202\344\275\225\345\256\236\347\216\260\346\241\206\351\200\211.html" "b/guide/Fabric.js/\345\246\202\344\275\225\345\256\236\347\216\260\346\241\206\351\200\211.html" new file mode 100644 index 00000000..340c29cc --- /dev/null +++ "b/guide/Fabric.js/\345\246\202\344\275\225\345\256\236\347\216\260\346\241\206\351\200\211.html" @@ -0,0 +1,30 @@ + + + + + + 如何实现框选 | 个人知识库 + + + + + + + + + + + + + + + + + + + +
Skip to content

如何实现框选

判断俩个图形是否相交

判断俩个线段是否相交

向量叉乘 的方式判断

参考文章

+ + + + \ No newline at end of file diff --git "a/guide/Fabric.js/\345\246\202\344\275\225\345\256\236\347\216\260\347\202\271\351\200\211\347\211\251\344\275\223.html" "b/guide/Fabric.js/\345\246\202\344\275\225\345\256\236\347\216\260\347\202\271\351\200\211\347\211\251\344\275\223.html" index 84a5fbb6..f010bdf7 100644 --- "a/guide/Fabric.js/\345\246\202\344\275\225\345\256\236\347\216\260\347\202\271\351\200\211\347\211\251\344\275\223.html" +++ "b/guide/Fabric.js/\345\246\202\344\275\225\345\256\236\347\216\260\347\202\271\351\200\211\347\211\251\344\275\223.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

如何实现点选物体

包围盒

常见的有 OBB、AABB、球模型包围盒

OBB 包围盒

AABB 包围盒

点射法

优化

1 记录最近的这个物体,下次再次判断这个物体。 2 判断是否是透明区域

- +
Skip to content

如何实现点选物体

上一节,我们了解了

包围盒

常见的有 OBB、AABB、球模型包围盒

OBB 包围盒

AABB 包围盒

点射法

优化

1 记录最近的这个物体,下次再次判断这个物体。 2 判断是否是透明区域

+ \ No newline at end of file diff --git "a/guide/Fabric.js/\345\246\202\344\275\225\347\273\230\345\210\266\344\270\200\344\270\252\345\233\276\345\275\242.html" "b/guide/Fabric.js/\345\246\202\344\275\225\347\273\230\345\210\266\344\270\200\344\270\252\345\233\276\345\275\242.html" index 5b2f726b..212e451d 100644 --- "a/guide/Fabric.js/\345\246\202\344\275\225\347\273\230\345\210\266\344\270\200\344\270\252\345\233\276\345\275\242.html" +++ "b/guide/Fabric.js/\345\246\202\344\275\225\347\273\230\345\210\266\344\270\200\344\270\252\345\233\276\345\275\242.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

如何绘制一个图形

上一节中,大致了解了 Fabric 的内部结构,并且知道 Fabric.js 拥有俩层 Canvas。

上层 Canvas 称之为动态层,用于处理交互相关以及事件绑定

下层 Canvas 称之为静态层,用于获取数据,通过数据绘制图形

之前有编写过 React 的同学应该对父子组件的单向数据流有一定的印象,而 Fabric.js 的分层结构设计就很类似于这种单向数据流思想,

动态层用于获取数据输出到静态层进行渲染。

在这一节中,我们将以绘制矩形为例,了解 Fabric.js 是如何实现矩形的绘制的?

Fabric.js 是如何绘制矩形的

在 Fabric.js 中绘制矩形也非常简单,只需要初始化画布,添加矩形对象即可,代码如下👇:

html
<!-- html -->
+    
Skip to content

如何绘制一个图形

上一节中,大致了解了 Fabric 的内部结构,并且知道 Fabric.js 拥有俩层 Canvas。

上层 Canvas 称之为动态层,用于处理交互相关以及事件绑定

下层 Canvas 称之为静态层,用于获取数据,通过数据绘制图形

之前有编写过 React 的同学应该对父子组件的单向数据流有一定的印象,而 Fabric.js 的分层结构设计就很类似于这种单向数据流思想,

动态层用于获取数据输出到静态层进行渲染。

在这一节中,我们将以绘制矩形为例,了解 Fabric.js 是如何实现矩形的绘制的?

Fabric.js 是如何绘制矩形的

在 Fabric.js 中绘制矩形也非常简单,只需要初始化画布,添加矩形对象即可,代码如下👇:

html
<!-- html -->
 <canvas id="canvas"></canvas>
<!-- html -->
 <canvas id="canvas"></canvas>
js
// js
 const canvas = new fabric.Canvas('canvas');
@@ -77,8 +77,8 @@
     for (i = 0, len = objects.length; i < len; ++i) {
     objects[i] && objects[i].render(ctx);
     }
-}

实例化对象

虽然我们已经明白了 canvas 的绘制原理,但是一个对象(2d元素)到底是怎么绘制到 canvas 上去的,它们的移动怎么实现的?具体细节我们还不是很清楚。

fabric.Object 是元素的根类型,因此需要从 fabric.Object 根类型看起了。

由于 fabric 中的 2d 元素都是以面向对象的形式实现的, 而 Rect (矩形类) 继承 fabric.Object,实现了自己的 render 方法,

我们将通过一张 UML 类图更直观的方便我们的理解。

alt text

从图中可以看出各个元素类将实现自己的 render 方法,通过调用 render 方法,将各自图形绘制到 Canvas 中。

而在执行 canvas.add 的时候,会将元素对象 push 到一个数据数组内,进行遍历渲染。

参考

- +}

实例化对象

虽然我们已经明白了 canvas 的绘制原理,但是一个对象(2d元素)到底是怎么绘制到 canvas 上去的,它们的移动怎么实现的?具体细节我们还不是很清楚。

fabric.Object 是元素的根类型,因此需要从 fabric.Object 根类型看起了。

由于 fabric 中的 2d 元素都是以面向对象的形式实现的, 而 Rect (矩形类) 继承 fabric.Object,实现了自己的 render 方法,

我们将通过一张 UML 类图更直观的方便我们的理解。

alt text

从图中可以看出各个元素类将实现自己的 render 方法,通过调用 render 方法,将各自图形绘制到 Canvas 中。

而在执行 canvas.add 的时候,会将元素对象 push 到一个数据数组内,进行遍历渲染。

参考

+ \ No newline at end of file diff --git "a/guide/Fabric.js/\345\246\202\344\275\225\350\260\203\350\257\225Fabric.html" "b/guide/Fabric.js/\345\246\202\344\275\225\350\260\203\350\257\225Fabric.html" index 4ae93e51..22796a11 100644 --- "a/guide/Fabric.js/\345\246\202\344\275\225\350\260\203\350\257\225Fabric.html" +++ "b/guide/Fabric.js/\345\246\202\344\275\225\350\260\203\350\257\225Fabric.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

如何调试 Fabric.js

为了更好的学习 Fabric.js 的源码,我们需要了解如何调试 Fabric.js。

第一步:下载源码

首先,我们需要下载 Fabric.js 的源码。Fabric.js 的源码托管在 GitHub 上,我们可以通过 git clone 命令将源码下载到本地。

bash
git clone https://github.com/fabricjs/fabricjs.com.git
git clone https://github.com/fabricjs/fabricjs.com.git

第二步:安装依赖

Fabric.js 使用 Jekyll 来提供页面服务,因此我们需要安装 Jekyll

Jekyll 提供了多种安装方式,在下将以 macOS 为例

Step 1: Install Homebrew

bash
/bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)"
/bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)"

Step 2: Install chruby and the latest Ruby with ruby-install

bash
brew install chruby ruby-install xz
brew install chruby ruby-install xz

Step 3: Install Ruby

bash
ruby-install ruby 3.1.3
ruby-install ruby 3.1.3

Step 4: 检查 ruby 是否安装成功

bash
ruby -v
ruby -v

它应该显示 ruby​​ 3.1.3p185(2022-11-24 修订版 1a6b16756e)或更新版本。 alt text

Step 5: 安装 Jekyll

bash
gem install jekyll
gem install jekyll

Step 6: 运行环境 安装完成后,在项目仓库根目录中运行命令 jekyll serve 即可.在此命令的控制台输出中,您将看到 Server address: <base_url> ,在浏览器中访问此 base_url ,您将看到 fabricjs.com 上的内容以及本地更改应用。

- +
Skip to content

如何调试 Fabric.js

为了更好的学习 Fabric.js 的源码,我们需要了解如何调试 Fabric.js。

第一步:下载源码

首先,我们需要下载 Fabric.js 的源码。Fabric.js 的源码托管在 GitHub 上,我们可以通过 git clone 命令将源码下载到本地。

bash
git clone https://github.com/fabricjs/fabricjs.com.git
git clone https://github.com/fabricjs/fabricjs.com.git

第二步:安装依赖

Fabric.js 使用 Jekyll 来提供页面服务,因此我们需要安装 Jekyll

Jekyll 提供了多种安装方式,在下将以 macOS 为例

Step 1: Install Homebrew

bash
/bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)"
/bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)"

Step 2: Install chruby and the latest Ruby with ruby-install

bash
brew install chruby ruby-install xz
brew install chruby ruby-install xz

Step 3: Install Ruby

bash
ruby-install ruby 3.1.3
ruby-install ruby 3.1.3

Step 4: 检查 ruby 是否安装成功

bash
ruby -v
ruby -v

它应该显示 ruby​​ 3.1.3p185(2022-11-24 修订版 1a6b16756e)或更新版本。 alt text

Step 5: 安装 Jekyll

bash
gem install jekyll
gem install jekyll

Step 6: 运行环境 安装完成后,在项目仓库根目录中运行命令 jekyll serve 即可.在此命令的控制台输出中,您将看到 Server address: <base_url> ,在浏览器中访问此 base_url ,您将看到 fabricjs.com 上的内容以及本地更改应用。

+ \ No newline at end of file diff --git "a/guide/Node\347\233\270\345\205\263/\346\246\202\350\246\201.html" "b/guide/Node\347\233\270\345\205\263/\346\246\202\350\246\201.html" index 14dc2c5c..652a5a24 100644 --- "a/guide/Node\347\233\270\345\205\263/\346\246\202\350\246\201.html" +++ "b/guide/Node\347\233\270\345\205\263/\346\246\202\350\246\201.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content
- +
Skip to content
+ \ No newline at end of file diff --git a/guide/React/Essential Knowledge for Schedule about React.html b/guide/React/Essential Knowledge for Schedule about React.html index 9082fb95..34fae571 100644 --- a/guide/React/Essential Knowledge for Schedule about React.html +++ b/guide/React/Essential Knowledge for Schedule about React.html @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

[React Scheduler] Essential Knowledge for Scheduling API

Frame

Introduction

This article summarizes key points about Scheduling API (e.g., requestAnimationFrame, requestIdleCallback, setTimeout, MessageChannel, microTask ( Promise, MutationObserver )).

Screen Refresh Rate

The screen refresh rate is the number of times the screen is updated per second. The screen refresh rate is usually 60Hz, which means the screen is updated 60 times per second.

In The browser, it will try to match the screen refresh rate with the frame rate of the web page.

If the frame rate is higher than 60HZ, the browser will skip some frames.

If the frame rate is lower than the screen refresh rate, the browser will render the same frame multiple times, which will cause the screen to block.

The budgeted time what is rendering per frame is 16.66 ms (1 second /60), so when writing code, be careful to take lower than 16ms of work per frame. Within each frame, the browser does the following:

  1. Execute Macro Task、Micro Task、Event Handler and so on.
  2. Execute requestAnimationFrame.
  3. Update Render Tree
  4. Execute requestIdleCallback if there is time left.
html
<!DOCTYPE html>
+    
Skip to content

[React Scheduler] Essential Knowledge for Scheduling API

Frame

Introduction

This article summarizes key points about Scheduling API (e.g., requestAnimationFrame, requestIdleCallback, setTimeout, MessageChannel, microTask ( Promise, MutationObserver )).

Screen Refresh Rate

The screen refresh rate is the number of times the screen is updated per second. The screen refresh rate is usually 60Hz, which means the screen is updated 60 times per second.

In The browser, it will try to match the screen refresh rate with the frame rate of the web page.

If the frame rate is higher than 60HZ, the browser will skip some frames.

If the frame rate is lower than the screen refresh rate, the browser will render the same frame multiple times, which will cause the screen to block.

The budgeted time what is rendering per frame is 16.66 ms (1 second /60), so when writing code, be careful to take lower than 16ms of work per frame. Within each frame, the browser does the following:

  1. Execute Macro Task、Micro Task、Event Handler and so on.
  2. Execute requestAnimationFrame.
  3. Update Render Tree
  4. Execute requestIdleCallback if there is time left.
html
<!DOCTYPE html>
 <html lang="en">
   <head>
     <meta charset="utf-8" />
@@ -235,8 +235,8 @@
   count = 0;
   window.requestAnimationFrame(rafCallback);
 }
-window.requestAnimationFrame(rafCallback);

In the above example, the browser will block because control is returned to the browser only after the microtask is executed.

Summarize

requestAnimationFramerequestIdleCallbacksetTimeoutMessageChannel can used to schedule tasks in the browser, becaus they are returning control to the browser after the task is executed.

However, Microtasks are continuously executed until the microtask queue on the event loop is empty which means a higher-pri task such as one for user interaction would be blocked by React.

QA

Why is requestAnimationFrame unsuitable for React Sheduler?

  • Instability: requestAnimationFrame is executed before the next repaint, which means unexpected behavior in lower frame rates. However, others posts a message event and performs a small amount of work (5ms) before returning control to the browser. At the end of the event, if there's work left over, it posts another message event.

  • Not trigger: requestAnimationFrame calls are paused in most browsers when running in background tabs or hidden iframes, in order to improve performance and battery life.

Why is webwork unsuitable for React Sheduler?

Browser layout is currently only available from the main thread.

Reference

The demo corresponding to this section can be found here.

- +window.requestAnimationFrame(rafCallback);

In the above example, the browser will block because control is returned to the browser only after the microtask is executed.

Summarize

requestAnimationFramerequestIdleCallbacksetTimeoutMessageChannel can used to schedule tasks in the browser, becaus they are returning control to the browser after the task is executed.

However, Microtasks are continuously executed until the microtask queue on the event loop is empty which means a higher-pri task such as one for user interaction would be blocked by React.

QA

Why is requestAnimationFrame unsuitable for React Sheduler?

  • Instability: requestAnimationFrame is executed before the next repaint, which means unexpected behavior in lower frame rates. However, others posts a message event and performs a small amount of work (5ms) before returning control to the browser. At the end of the event, if there's work left over, it posts another message event.

  • Not trigger: requestAnimationFrame calls are paused in most browsers when running in background tabs or hidden iframes, in order to improve performance and battery life.

Why is webwork unsuitable for React Sheduler?

Browser layout is currently only available from the main thread.

Reference

The demo corresponding to this section can be found here.

+ \ No newline at end of file diff --git "a/guide/React/React\344\270\216Vue\347\232\204\345\214\272\345\210\253.html" "b/guide/React/React\344\270\216Vue\347\232\204\345\214\272\345\210\253.html" index e7f3b781..4e1e604b 100644 --- "a/guide/React/React\344\270\216Vue\347\232\204\345\214\272\345\210\253.html" +++ "b/guide/React/React\344\270\216Vue\347\232\204\345\214\272\345\210\253.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content
- +
Skip to content
+ \ No newline at end of file diff --git "a/guide/React/React\344\270\255\347\232\204\344\272\213\344\273\266\346\234\272\345\210\266.html" "b/guide/React/React\344\270\255\347\232\204\344\272\213\344\273\266\346\234\272\345\210\266.html" index 4c3ed06c..ae4cd230 100644 --- "a/guide/React/React\344\270\255\347\232\204\344\272\213\344\273\266\346\234\272\345\210\266.html" +++ "b/guide/React/React\344\270\255\347\232\204\344\272\213\344\273\266\346\234\272\345\210\266.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

React 事件机制

概要

为了抹平各个浏览器之间的差异以及统一维护和管理事件,React 自行实现了合成事件,用来模拟原生事件的行为 在 React 16 之前,React 的合成事件放在了 document 的冒泡阶段 在 React 16 之后,为了兼容各个版本,React 的合成事件放在了 根节点 的捕获和冒泡阶段

具体实现

- +
Skip to content

React 事件机制

概要

为了抹平各个浏览器之间的差异以及统一维护和管理事件,React 自行实现了合成事件,用来模拟原生事件的行为 在 React 16 之前,React 的合成事件放在了 document 的冒泡阶段 在 React 16 之后,为了兼容各个版本,React 的合成事件放在了 根节点 的捕获和冒泡阶段

具体实现

+ \ No newline at end of file diff --git "a/guide/React/React\344\270\255\347\232\204\345\274\202\345\270\270\346\234\272\345\210\266.html" "b/guide/React/React\344\270\255\347\232\204\345\274\202\345\270\270\346\234\272\345\210\266.html" index 947cc801..be482dad 100644 --- "a/guide/React/React\344\270\255\347\232\204\345\274\202\345\270\270\346\234\272\345\210\266.html" +++ "b/guide/React/React\344\270\255\347\232\204\345\274\202\345\270\270\346\234\272\345\210\266.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content
- +
Skip to content
+ \ No newline at end of file diff --git "a/guide/React/React\344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.html" "b/guide/React/React\344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.html" index fca1a6d5..53931ddb 100644 --- "a/guide/React/React\344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.html" +++ "b/guide/React/React\344\270\255\347\232\204\346\200\247\350\203\275\344\274\230\345\214\226.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

React 中的性能优化

在 React 中当我们的父组件更新时,将渲染整个子组件树,这样会造成很大的性能开销,所以我们需要对组件进行优化,避免不必要的渲染。

常见的性能优化手段有:

  1. 减少不必要的渲染

    1. 使用 useMemo 缓存数据、使用 useCallback 缓存函数
    2. 使用 React.memo 缓存组件
    3. 合理的使用 Key
    4. 使用 Fragment 避免额外标记
    5. 通过 Suspense 和 Lazy 拆分组件
  2. 通用方案

    1. 在组件销毁的时候清除定时器/事件
    2. 频繁切换时使用 display: none; 避免不必要的性能开销
    3. 为组件创建错误边界

1 使用 useMemo 缓存数据、使用 useCallback 缓存函数

jsx
/**
+    
Skip to content

React 中的性能优化

在 React 中当我们的父组件更新时,将渲染整个子组件树,这样会造成很大的性能开销,所以我们需要对组件进行优化,避免不必要的渲染。

常见的性能优化手段有:

  1. 减少不必要的渲染

    1. 使用 useMemo 缓存数据、使用 useCallback 缓存函数
    2. 使用 React.memo 缓存组件
    3. 合理的使用 Key
    4. 使用 Fragment 避免额外标记
    5. 通过 Suspense 和 Lazy 拆分组件
  2. 通用方案

    1. 在组件销毁的时候清除定时器/事件
    2. 频繁切换时使用 display: none; 避免不必要的性能开销
    3. 为组件创建错误边界

1 使用 useMemo 缓存数据、使用 useCallback 缓存函数

jsx
/**
  * 使用 useMemo 缓存数据,类似于 Vue 的 computed 计算属性
  * 使用 useCallback 缓存函数
  * @returns
@@ -459,8 +459,8 @@
 
     return this.props.children;
   }
-}

参考文章

  1. https://juejin.cn/post/6965747225154732069#heading-15
  2. https://cloud.tencent.com/developer/article/1810002
- +}

参考文章

  1. https://juejin.cn/post/6965747225154732069#heading-15
  2. https://cloud.tencent.com/developer/article/1810002
+ \ No newline at end of file diff --git "a/guide/React/React\344\270\272\344\273\200\344\271\210\350\246\201\344\275\277\347\224\250JSX.html" "b/guide/React/React\344\270\272\344\273\200\344\271\210\350\246\201\344\275\277\347\224\250JSX.html" index 4c432aa1..5d2c4a93 100644 --- "a/guide/React/React\344\270\272\344\273\200\344\271\210\350\246\201\344\275\277\347\224\250JSX.html" +++ "b/guide/React/React\344\270\272\344\273\200\344\271\210\350\246\201\344\275\277\347\224\250JSX.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

React 为什么要使用 JSX ?

JSX 是 JavaScript 语法拓展,结构类似于 XML,它可以在 JavaScript 中直接使用 HTML 标签,这样可以方便开发者编写组件,提高开发效率。

没有JSX时,也可以使用 React.createElement 实现一个组件,但是这样的代码可读性很差,而且不方便复用,所以 React 推荐使用 JSX。

- +
Skip to content

React 为什么要使用 JSX ?

JSX 是 JavaScript 语法拓展,结构类似于 XML,它可以在 JavaScript 中直接使用 HTML 标签,这样可以方便开发者编写组件,提高开发效率。

没有JSX时,也可以使用 React.createElement 实现一个组件,但是这样的代码可读性很差,而且不方便复用,所以 React 推荐使用 JSX。

+ \ No newline at end of file diff --git "a/guide/React/React\346\230\257\344\273\200\344\271\210.html" "b/guide/React/React\346\230\257\344\273\200\344\271\210.html" index 5307a5ec..303c4152 100644 --- "a/guide/React/React\346\230\257\344\273\200\344\271\210.html" +++ "b/guide/React/React\346\230\257\344\273\200\344\271\210.html" @@ -11,7 +11,7 @@ - + @@ -23,12 +23,12 @@ -
Skip to content

React 是什么?

React 是一个用于构建用户界面的 JavaScript 库, 通过组件化的方式解决视图层复用的问题。

它的核心思路是: 声明式、组件化、通用型

声明式

相比 JQ 的命令式编程: $(body).css('color', 'red');, 声明式编程更加直观、便于复用:

jsx
const Body = (props) => {
+    
Skip to content

React 是什么?

React 是一个用于构建用户界面的 JavaScript 库, 通过组件化的方式解决视图层复用的问题。

它的核心思路是: 声明式、组件化、通用型

声明式

相比 JQ 的命令式编程: $(body).css('color', 'red');, 声明式编程更加直观、便于复用:

jsx
const Body = (props) => {
    return <div style={{color: 'red'}}>{props.children}</div>
 }
const Body = (props) => {
    return <div style={{color: 'red'}}>{props.children}</div>
-}

组件化

可以降低功能之间的耦合、提高功能的内聚性,便于复用

通用性

虚拟DOM的实现,保证了跨平台和可移植性

缺点

没有提供全链路的解决方法,比如 router、数据仓库,需要借助第三方库

- +}

组件化

可以降低功能之间的耦合、提高功能的内聚性,便于复用

通用性

虚拟DOM的实现,保证了跨平台和可移植性

缺点

没有提供全链路的解决方法,比如 router、数据仓库,需要借助第三方库

+ \ No newline at end of file diff --git "a/guide/React/React\346\230\257\345\246\202\344\275\225\346\270\262\346\237\223\347\232\204.html" "b/guide/React/React\346\230\257\345\246\202\344\275\225\346\270\262\346\237\223\347\232\204.html" index da134c6d..370bdc1f 100644 --- "a/guide/React/React\346\230\257\345\246\202\344\275\225\346\270\262\346\237\223\347\232\204.html" +++ "b/guide/React/React\346\230\257\345\246\202\344\275\225\346\270\262\346\237\223\347\232\204.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

React 是如何渲染的

概要

React 的渲染过程可以分成 2 大阶段, 分别是调和阶段和提交阶段。 调和阶段可以分成 beginWork 阶段、 completeWork 阶段。 在 beginWork 阶段中,React 会根据新生成的 ReactElement 对象和旧的 Fiber 节点进行对比,判断是否可以复用旧的 Fiber 节点并对 Fiber 进行标记。 在 completeWork 节点中,会自底向上构建副作用链表,用来记录需要更新的节点,生成的 DOM 节点会挂载在 Fiber 的 stateNode 属性上。

提交阶段主要分成:操作 DOM 前阶段、操作 DOM 阶段、操作 DOM 后阶段。

React 16 以前

在浏览器中 js 线程与渲染线程是互斥的,如果 js 线程长期占用着浏览器的主线程,那么界面将长时间不更新,在动画等一些场景下会造成卡顿效果。

因为 Stack Reconciler 是一个同步的递归过程,随着业务复杂度增加,Stack Reconciler 需要的调和时间会变长,

这意味着 js 将长时间占用浏览器,进而导致页面卡顿

Stack Reconciler

React 16 以后

将同步执行的 Stack Reconciler 替换成了异步可中断的 Fiber Reconciler

Fiber Reconciler

在更新时,每个任务会被赋予一个优先级,当任务抵达调度器时,高优先级的任务会更快抵达协调器,如有新的更高优先级的任务进入调度器时,当前协调器的任务就会被中断,更高优先级的任务将进入 reconciler

Fiber Reconciler

新的架构会导致部分生命周期重复执行:

  • componentWillMount
  • componentWillUpdate
  • showComponentUpdate
  • componentWillReceiveProps

Render Performance

从首次渲染的调用栈来看,React 的渲染过程主要分为以下几个步骤:

    1. Mount 阶段
    1. Render 阶段
    1. Commit 阶段

Mount 阶段

当执行 ReactDOM.render 时会直接调用 legacyRenderSubtreeIntoContainer 方法

legacyRenderSubtreeIntoContainer 方法

会创建 reactRootContainer 对象(也就是挂载的容器对象), reactRootContainer 对象的 _internalRoot 会指向 fiberRoot (FiberRootNode 类),也就是根节点对象。

legacyRenderSubtreeIntoContainer 最终返回挂载组件( App组件 )的实例对象。

Mount 渲染过程

FiberRootNode对象的描述

在 fiberRoot 对象( FiberRootNode类 )中的 current 属性将指向 rootFiber 对象 (根节点Fiber,即 FiberNode 实例)

fiberRoot对象指向 rootFiber 根节点Fiber

挂载创建关系图

updateContainer 函数

updateContainer 函数主要有以下 3 件事:

    1. 获取当前节点的优先级 lane
    1. 结合 lane 创建当前 Fiber 节点 update 对象,并将其入队
    1. 调度当前节点 (rootFiber 节点)

updateContainer

scheduleUpdateOnFiber 函数

scheduleUpdateOnFiber 函数中会获取 Fiber 节点的mode属性判断是否走同步渲染还是异步渲染的逻辑,在 React17 中首次渲染走的是同步渲染的逻辑

scheduleUpdateOnFiber

这里可能有小伙伴会问,Fiber架构不就是异步渲染的么? 我想说的是,Fiber架构的设计初衷确实是为了异步渲染而设计的,但是 Fiber 架构并不能和异步渲染画上等号,我们不难发现,Fiber 架构同时兼容了同步渲染和异步渲染,如下图,决定同步还是异步取决于 mode

mode 决定同步异步

Render 阶段

performSyncWorkOnRoot 函数

核心逻辑在 renderRootSync 函数中

renderRootSync 函数

核心方法有俩个 prepareFreshStackworkLoopSync 函数

renderRootSync 逻辑

prepareFreshStack 函数

主要是有个方法 createWorkInProgress , 用来构建 workInProgress 双缓冲树,通过 alternate 相互指向

createWorkInProgress 构建 work-in-progress 树

当建立好双缓冲树的关系后,我们不难得到以下的关系图

双缓冲树关系图

workLoopSync 函数

当我们构建完 workInProgress Tree 的根节点时,建立 current tree 和 workInProgess Tree 的关联关系后,将进入 workLoopSync 调和阶段。

反复判断 workInProgress 是否为空,如果不为空,就执行 performUnitOfWork 方法 workLoopSync 逻辑

performUnitOfWork 函数

performUnitOfWork 函数的作用是 beginWork 优先创建子节点。 completeUnitOfWork 创建完子节点后判断是否有兄弟节点,有则创建兄弟节点,无则继续向上遍历父节点,直到遍历到根节点为止

performUnitOfWork 逻辑

beginWork 函数

参考文档

1 https://zhuanlan.zhihu.com/p/385319664 2

- +
Skip to content

React 是如何渲染的

概要

React 的渲染过程可以分成 2 大阶段, 分别是调和阶段和提交阶段。 调和阶段可以分成 beginWork 阶段、 completeWork 阶段。 在 beginWork 阶段中,React 会根据新生成的 ReactElement 对象和旧的 Fiber 节点进行对比,判断是否可以复用旧的 Fiber 节点并对 Fiber 进行标记。 在 completeWork 节点中,会自底向上构建副作用链表,用来记录需要更新的节点,生成的 DOM 节点会挂载在 Fiber 的 stateNode 属性上。

提交阶段主要分成:操作 DOM 前阶段、操作 DOM 阶段、操作 DOM 后阶段。

React 16 以前

在浏览器中 js 线程与渲染线程是互斥的,如果 js 线程长期占用着浏览器的主线程,那么界面将长时间不更新,在动画等一些场景下会造成卡顿效果。

因为 Stack Reconciler 是一个同步的递归过程,随着业务复杂度增加,Stack Reconciler 需要的调和时间会变长,

这意味着 js 将长时间占用浏览器,进而导致页面卡顿

Stack Reconciler

React 16 以后

将同步执行的 Stack Reconciler 替换成了异步可中断的 Fiber Reconciler

Fiber Reconciler

在更新时,每个任务会被赋予一个优先级,当任务抵达调度器时,高优先级的任务会更快抵达协调器,如有新的更高优先级的任务进入调度器时,当前协调器的任务就会被中断,更高优先级的任务将进入 reconciler

Fiber Reconciler

新的架构会导致部分生命周期重复执行:

  • componentWillMount
  • componentWillUpdate
  • showComponentUpdate
  • componentWillReceiveProps

Render Performance

从首次渲染的调用栈来看,React 的渲染过程主要分为以下几个步骤:

    1. Mount 阶段
    1. Render 阶段
    1. Commit 阶段

Mount 阶段

当执行 ReactDOM.render 时会直接调用 legacyRenderSubtreeIntoContainer 方法

legacyRenderSubtreeIntoContainer 方法

会创建 reactRootContainer 对象(也就是挂载的容器对象), reactRootContainer 对象的 _internalRoot 会指向 fiberRoot (FiberRootNode 类),也就是根节点对象。

legacyRenderSubtreeIntoContainer 最终返回挂载组件( App组件 )的实例对象。

Mount 渲染过程

FiberRootNode对象的描述

在 fiberRoot 对象( FiberRootNode类 )中的 current 属性将指向 rootFiber 对象 (根节点Fiber,即 FiberNode 实例)

fiberRoot对象指向 rootFiber 根节点Fiber

挂载创建关系图

updateContainer 函数

updateContainer 函数主要有以下 3 件事:

    1. 获取当前节点的优先级 lane
    1. 结合 lane 创建当前 Fiber 节点 update 对象,并将其入队
    1. 调度当前节点 (rootFiber 节点)

updateContainer

scheduleUpdateOnFiber 函数

scheduleUpdateOnFiber 函数中会获取 Fiber 节点的mode属性判断是否走同步渲染还是异步渲染的逻辑,在 React17 中首次渲染走的是同步渲染的逻辑

scheduleUpdateOnFiber

这里可能有小伙伴会问,Fiber架构不就是异步渲染的么? 我想说的是,Fiber架构的设计初衷确实是为了异步渲染而设计的,但是 Fiber 架构并不能和异步渲染画上等号,我们不难发现,Fiber 架构同时兼容了同步渲染和异步渲染,如下图,决定同步还是异步取决于 mode

mode 决定同步异步

Render 阶段

performSyncWorkOnRoot 函数

核心逻辑在 renderRootSync 函数中

renderRootSync 函数

核心方法有俩个 prepareFreshStackworkLoopSync 函数

renderRootSync 逻辑

prepareFreshStack 函数

主要是有个方法 createWorkInProgress , 用来构建 workInProgress 双缓冲树,通过 alternate 相互指向

createWorkInProgress 构建 work-in-progress 树

当建立好双缓冲树的关系后,我们不难得到以下的关系图

双缓冲树关系图

workLoopSync 函数

当我们构建完 workInProgress Tree 的根节点时,建立 current tree 和 workInProgess Tree 的关联关系后,将进入 workLoopSync 调和阶段。

反复判断 workInProgress 是否为空,如果不为空,就执行 performUnitOfWork 方法 workLoopSync 逻辑

performUnitOfWork 函数

performUnitOfWork 函数的作用是 beginWork 优先创建子节点。 completeUnitOfWork 创建完子节点后判断是否有兄弟节点,有则创建兄弟节点,无则继续向上遍历父节点,直到遍历到根节点为止

performUnitOfWork 逻辑

beginWork 函数

参考文档

1 https://zhuanlan.zhihu.com/p/385319664 2

+ \ No newline at end of file diff --git "a/guide/React/React\347\273\204\344\273\266\346\230\257\345\246\202\344\275\225\351\200\232\344\277\241\347\232\204.html" "b/guide/React/React\347\273\204\344\273\266\346\230\257\345\246\202\344\275\225\351\200\232\344\277\241\347\232\204.html" index 233b07ec..76f2bcdf 100644 --- "a/guide/React/React\347\273\204\344\273\266\346\230\257\345\246\202\344\275\225\351\200\232\344\277\241\347\232\204.html" +++ "b/guide/React/React\347\273\204\344\273\266\346\230\257\345\246\202\344\275\225\351\200\232\344\277\241\347\232\204.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

React 组件是如何通信的

- +
Skip to content

React 组件是如何通信的

+ \ No newline at end of file diff --git "a/guide/React/diff\347\256\227\346\263\225.html" "b/guide/React/diff\347\256\227\346\263\225.html" index 82bb5437..be81f310 100644 --- "a/guide/React/diff\347\256\227\346\263\225.html" +++ "b/guide/React/diff\347\256\227\346\263\225.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

React 中的 Diff 算法

概要

React Diff 算法主要分单节点和多节点 Diff 算法 在单节点 Diff 算法中,会对比 key 和 tag 是否可以复用 如果 key 和 tag 都相同时,复用当前的 Fiber 节点,标记其他 Fiber 节点为删除 如果 key 不相同,标记删除当前的 Fiber 节点,对比下一个节点 如果 key 相同、tag 不同,则标记删除所有 Fiber 节点,生成新的 Fiber 节点

在多节点 Diff 中,有俩个 for 循环,第一个 for 循环判断可以复用的节点,记录最后可复用节点的 lastIndex 位置

通过 Map 建立节点和下标的对应关系,如果新节点的key可以在 Map 中找到,则复用旧节点,并判断新节点下标和 lastIndex 下标的关系, 如果新节点下标 > lastIndex 下标,说明只需要更新节点,不需要移动位置,更新 lastIndex。 如果新节点下标 < lastIndex,说明需要更新移动位置,不需要更新 lastIndex。

单节点diff算法 (render方法创建的element元素子节点只有一个)

1. 旧节点不存在

  1. 如果旧节点不存在,则直接新增节点

2. 旧节点存在

1 判断 key 和 tag 是否相同,相同则复用节点,更新属性 2 如果 key 相同但 tag 不相同,则删除所有节点 3 如果 key、tag 不同,则不需要复用,旧节点标记为删除

多节点diff算法 (render方法创建的element元素子节点有多个)

多节点diff有俩次 for 循环, 第一次 for 循环判断元素是否需要更新,第二次 for 循环判断元素是否需要移动位置

1 第一次 for 循环比较key和tag, 如果相同则复用, 并用 lastIndex 标记当前可以复用的节点位置

2 遇到不相同时, 跳出第一层 for 循环, 创建一个 Map 对象, 存储旧节点的 key 和 index

  • key不同导致不可复用,立即跳出整个遍历,第一轮遍历结束
  • key 相同 type 不同导致不可复用,会将 oldFiber 标记为 DELETION,并继续遍历

3 在 Map 对象中查找新节点的 key, 如果存在, 则说明新节点可以复用旧节点, 并且判断是否需要移动位置

4 如果 index > lastIndex, 则不需要移动位置, 更新 lastIndex

5 如果 index < lastIndex, 则需要移动位置, 不需要更新 lastIndex

- +
Skip to content

React 中的 Diff 算法

概要

React Diff 算法主要分单节点和多节点 Diff 算法 在单节点 Diff 算法中,会对比 key 和 tag 是否可以复用 如果 key 和 tag 都相同时,复用当前的 Fiber 节点,标记其他 Fiber 节点为删除 如果 key 不相同,标记删除当前的 Fiber 节点,对比下一个节点 如果 key 相同、tag 不同,则标记删除所有 Fiber 节点,生成新的 Fiber 节点

在多节点 Diff 中,有俩个 for 循环,第一个 for 循环判断可以复用的节点,记录最后可复用节点的 lastIndex 位置

通过 Map 建立节点和下标的对应关系,如果新节点的key可以在 Map 中找到,则复用旧节点,并判断新节点下标和 lastIndex 下标的关系, 如果新节点下标 > lastIndex 下标,说明只需要更新节点,不需要移动位置,更新 lastIndex。 如果新节点下标 < lastIndex,说明需要更新移动位置,不需要更新 lastIndex。

单节点diff算法 (render方法创建的element元素子节点只有一个)

1. 旧节点不存在

  1. 如果旧节点不存在,则直接新增节点

2. 旧节点存在

1 判断 key 和 tag 是否相同,相同则复用节点,更新属性 2 如果 key 相同但 tag 不相同,则删除所有节点 3 如果 key、tag 不同,则不需要复用,旧节点标记为删除

多节点diff算法 (render方法创建的element元素子节点有多个)

多节点diff有俩次 for 循环, 第一次 for 循环判断元素是否需要更新,第二次 for 循环判断元素是否需要移动位置

1 第一次 for 循环比较key和tag, 如果相同则复用, 并用 lastIndex 标记当前可以复用的节点位置

2 遇到不相同时, 跳出第一层 for 循环, 创建一个 Map 对象, 存储旧节点的 key 和 index

  • key不同导致不可复用,立即跳出整个遍历,第一轮遍历结束
  • key 相同 type 不同导致不可复用,会将 oldFiber 标记为 DELETION,并继续遍历

3 在 Map 对象中查找新节点的 key, 如果存在, 则说明新节点可以复用旧节点, 并且判断是否需要移动位置

4 如果 index > lastIndex, 则不需要移动位置, 更新 lastIndex

5 如果 index < lastIndex, 则需要移动位置, 不需要更新 lastIndex

+ \ No newline at end of file diff --git "a/guide/React/react\345\274\202\345\270\270\346\234\272\345\210\266.html" "b/guide/React/react\345\274\202\345\270\270\346\234\272\345\210\266.html" index baa82d69..ab82086f 100644 --- "a/guide/React/react\345\274\202\345\270\270\346\234\272\345\210\266.html" +++ "b/guide/React/react\345\274\202\345\270\270\346\234\272\345\210\266.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content
- +
Skip to content
+ \ No newline at end of file diff --git "a/guide/React/setState\346\230\257\345\220\214\346\255\245\350\277\230\346\230\257\345\274\202\346\255\245\347\232\204.html" "b/guide/React/setState\346\230\257\345\220\214\346\255\245\350\277\230\346\230\257\345\274\202\346\255\245\347\232\204.html" index a34b4412..c2241110 100644 --- "a/guide/React/setState\346\230\257\345\220\214\346\255\245\350\277\230\346\230\257\345\274\202\346\255\245\347\232\204.html" +++ "b/guide/React/setState\346\230\257\345\220\214\346\255\245\350\277\230\346\230\257\345\274\202\346\255\245\347\232\204.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

setState 是同步还是异步的?

概要

setState 用于状态变更,触发组件的重新渲染,更新视图。

在 React 18 以前,setState 同步还是异步取决于是否处于 isBatchUpdates 字段,true 的话就是异步,否则就是同步

在 React 18 版本所有的 setState 都是异步批量操作

setState

js
// React 15
+    
Skip to content

setState 是同步还是异步的?

概要

setState 用于状态变更,触发组件的重新渲染,更新视图。

在 React 18 以前,setState 同步还是异步取决于是否处于 isBatchUpdates 字段,true 的话就是异步,否则就是同步

在 React 18 版本所有的 setState 都是异步批量操作

setState

js
// React 15
 function enqueueUpdate (component) {
   // isBatchingUpdates 判断当前是否处理批量更新操作
   if (!batchingStrategy.isBatchingUpdates) { // 锁管理器
@@ -331,8 +331,8 @@
   }
 }

最终将输出

this.state.count0 0
 this.state.count1 1
this.state.count0 0
-this.state.count1 1

参考资料

- +this.state.count1 1

参考资料

+ \ No newline at end of file diff --git "a/guide/React/\344\273\200\344\271\210\346\230\257Fiber.html" "b/guide/React/\344\273\200\344\271\210\346\230\257Fiber.html" index 341b97ec..224e81ce 100644 --- "a/guide/React/\344\273\200\344\271\210\346\230\257Fiber.html" +++ "b/guide/React/\344\273\200\344\271\210\346\230\257Fiber.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

对 Fiber 的理解

在 React 15 中通过 递归 的形式进行对比,找到需要更新的节点,并同步更新它,在这段时间一直占据着浏览器主线程,可能会给用户带来卡顿的感受(在渲染进程中,js线程和渲染线程是互斥的

在 React 15 以后引入了 Fiber 架构,将对比的过程变成了异步可中断的过程,让出浏览器的使用权,让浏览器处理更高优先级的事情

Fiber 的调和过程(Reconciler)由分成了 beginWork 阶段 和 completeUnitOfWork 阶段。

beginWork 阶段自顶向下,根据当前工作的 Fiber 节点最新的 React Element 子元素与旧 Fiber 节点进行对比,决定是否需要复用旧 Fiber 节点并标记 Fiber 节点是否有副作用。

compeleteUnitOfWork 阶段自底向上构建副作用链表,生成的 DOM 节点挂在 Fiber 的 stateNode 属性

- +
Skip to content

对 Fiber 的理解

在 React 15 中通过 递归 的形式进行对比,找到需要更新的节点,并同步更新它,在这段时间一直占据着浏览器主线程,可能会给用户带来卡顿的感受(在渲染进程中,js线程和渲染线程是互斥的

在 React 15 以后引入了 Fiber 架构,将对比的过程变成了异步可中断的过程,让出浏览器的使用权,让浏览器处理更高优先级的事情

Fiber 的调和过程(Reconciler)由分成了 beginWork 阶段 和 completeUnitOfWork 阶段。

beginWork 阶段自顶向下,根据当前工作的 Fiber 节点最新的 React Element 子元素与旧 Fiber 节点进行对比,决定是否需要复用旧 Fiber 节点并标记 Fiber 节点是否有副作用。

compeleteUnitOfWork 阶段自底向上构建副作用链表,生成的 DOM 节点挂在 Fiber 的 stateNode 属性

+ \ No newline at end of file diff --git "a/guide/React/\345\246\202\344\275\225\350\256\276\350\256\241React\347\273\204\344\273\266.html" "b/guide/React/\345\246\202\344\275\225\350\256\276\350\256\241React\347\273\204\344\273\266.html" index 3a601bdc..34a549d7 100644 --- "a/guide/React/\345\246\202\344\275\225\350\256\276\350\256\241React\347\273\204\344\273\266.html" +++ "b/guide/React/\345\246\202\344\275\225\350\256\276\350\256\241React\347\273\204\344\273\266.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

如何设计 React 组件

设计分类

  • 展示组件: 只负责 UI 展示,不涉及业务逻辑

  • 业务组件: 复用负责业务逻辑

展示组件

展示组件一般受制于 props, 具有通用性、复用性

业务组件

业务组件一般受制于 state, 具有复用性

组件分类

- +
Skip to content

如何设计 React 组件

设计分类

  • 展示组件: 只负责 UI 展示,不涉及业务逻辑

  • 业务组件: 复用负责业务逻辑

展示组件

展示组件一般受制于 props, 具有通用性、复用性

业务组件

业务组件一般受制于 state, 具有复用性

组件分类

+ \ No newline at end of file diff --git "a/guide/React/\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.html" "b/guide/React/\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.html" index 0190fcd0..829a00d8 100644 --- "a/guide/React/\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.html" +++ "b/guide/React/\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content
- +
Skip to content
+ \ No newline at end of file diff --git "a/guide/Redux/\346\272\220\347\240\201\350\247\243\350\257\273.html" "b/guide/Redux/\346\272\220\347\240\201\350\247\243\350\257\273.html" index 5d87a57c..ef4cf43c 100644 --- "a/guide/Redux/\346\272\220\347\240\201\350\247\243\350\257\273.html" +++ "b/guide/Redux/\346\272\220\347\240\201\350\247\243\350\257\273.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content
- +
Skip to content
+ \ No newline at end of file diff --git "a/guide/Redux/\350\256\276\350\256\241\347\220\206\345\277\265.html" "b/guide/Redux/\350\256\276\350\256\241\347\220\206\345\277\265.html" index 59a0c1c3..97aab379 100644 --- "a/guide/Redux/\350\256\276\350\256\241\347\220\206\345\277\265.html" +++ "b/guide/Redux/\350\256\276\350\256\241\347\220\206\345\277\265.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content
- +
Skip to content
+ \ No newline at end of file diff --git "a/guide/ai/Embedding\344\271\213\345\212\240\350\275\275\346\225\260\346\215\256.html" "b/guide/ai/Embedding\344\271\213\345\212\240\350\275\275\346\225\260\346\215\256.html" index a8d9f254..43833bb1 100644 --- "a/guide/ai/Embedding\344\271\213\345\212\240\350\275\275\346\225\260\346\215\256.html" +++ "b/guide/ai/Embedding\344\271\213\345\212\240\350\275\275\346\225\260\346\215\256.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

Embedding 之加载数据

因为 RAG 本质是给 Chat Bot 额外挂在数据源,而数据源存在的形式是多种多样的,有可能是文件/网页/数据库/代码等等其情况。

针对这个情况,langchain 提供了一系列开箱即用的数据加载器 (loader),方便用户快速加载数据。

Document 对象

Document 对象你可以理解成 langchain 对所有类型的数据的一个统一抽象,其中包含

pageContent 文本内容,即文档对象对应的文本数据 metadata 元数据,文本数据对应的元数据,例如 原始文档的标题、页数等信息,可以用于后面 Retriever 基于此进行筛选。

TypeScript
// Document 对象的定义
+    
Skip to content

Embedding 之加载数据

因为 RAG 本质是给 Chat Bot 额外挂在数据源,而数据源存在的形式是多种多样的,有可能是文件/网页/数据库/代码等等其情况。

针对这个情况,langchain 提供了一系列开箱即用的数据加载器 (loader),方便用户快速加载数据。

Document 对象

Document 对象你可以理解成 langchain 对所有类型的数据的一个统一抽象,其中包含

pageContent 文本内容,即文档对象对应的文本数据 metadata 元数据,文本数据对应的元数据,例如 原始文档的标题、页数等信息,可以用于后面 Retriever 基于此进行筛选。

TypeScript
// Document 对象的定义
 interface Document {
   pageContent: string;
   metadata: Record<string, any>;
@@ -103,8 +103,8 @@
 const apiKey = process.env["SEARCH_KEY"]
 const question = "什么 github copliot"
 const searchLoader = new SearchApiLoader({ q: question, apiKey, engine: "google" });
-const searchRes = await searchLoader.load();

输出结果如下:👇

输出结果

参考文档

- +const searchRes = await searchLoader.load();

输出结果如下:👇

输出结果

参考文档

+ \ No newline at end of file diff --git "a/guide/ai/Embedding\344\271\213\345\244\247\350\247\204\346\250\241\346\225\260\346\215\256\346\213\206\345\210\206.html" "b/guide/ai/Embedding\344\271\213\345\244\247\350\247\204\346\250\241\346\225\260\346\215\256\346\213\206\345\210\206.html" index 23d5bf06..9024ab38 100644 --- "a/guide/ai/Embedding\344\271\213\345\244\247\350\247\204\346\250\241\346\225\260\346\215\256\346\213\206\345\210\206.html" +++ "b/guide/ai/Embedding\344\271\213\345\244\247\350\247\204\346\250\241\346\225\260\346\215\256\346\213\206\345\210\206.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

Embedding 之大规模数据拆分

受限于常见 LLM 的上下文大小,例如 gpt3.5t 是 16k、gpt4t 是 128k,我们并不能把完整的数据整个塞到对话的上下文中。

即使数据源接近于 LLM 的上下文窗口大小,llm 在读取数据时很容易出现分神,或者忽略其中部分细节的问题。

因此,我们需要对数据进行拆分,然后将最关联的内容输入给 LLM,以便 LLM 能够更好的理解数据。

分割数据

对于分割来说,将数据分割成具有独立逻辑的段落是比较好的选择,段落分割的质量越高,意味着 RAG 的效果越好,LLM 返回数据的质量就越好。

langchain 目前提供的切分工具有:

  • Recursive: 根据给定的切分字符(例如 \n\n、\n等),递归的切分
  • HTML: 根据 html 特定字符进行切分
  • Markdown: 根据 md 的特定字符进行切分
  • Code: 根据不同编程语言的特定字符进行切分
  • Token: 根据文本块的 token 数据进行切分
  • Character: 根据用户给定的字符进行切割

RecursiveCharacterTextSplitter

RecursiveCharacterTextSplitter 默认的分隔符列表是 ["\n\n", "\n", " ", ""], 你可以理解为它将文本分割后,在根据设置的 chunk 大小进行组装。

最影响切分质量的就是两个参数:

  • chunkSize: 定义切分块的大小
  • chunkOverlap: 定义块与块之间重叠的大小。较大的重叠可能会导致分割的内容重复,较小的重叠可能会导致分割的内容不完整。
js
// RecursiveCharacterTextSplitter 对文本进行分割
+    
Skip to content

Embedding 之大规模数据拆分

受限于常见 LLM 的上下文大小,例如 gpt3.5t 是 16k、gpt4t 是 128k,我们并不能把完整的数据整个塞到对话的上下文中。

即使数据源接近于 LLM 的上下文窗口大小,llm 在读取数据时很容易出现分神,或者忽略其中部分细节的问题。

因此,我们需要对数据进行拆分,然后将最关联的内容输入给 LLM,以便 LLM 能够更好的理解数据。

分割数据

对于分割来说,将数据分割成具有独立逻辑的段落是比较好的选择,段落分割的质量越高,意味着 RAG 的效果越好,LLM 返回数据的质量就越好。

langchain 目前提供的切分工具有:

  • Recursive: 根据给定的切分字符(例如 \n\n、\n等),递归的切分
  • HTML: 根据 html 特定字符进行切分
  • Markdown: 根据 md 的特定字符进行切分
  • Code: 根据不同编程语言的特定字符进行切分
  • Token: 根据文本块的 token 数据进行切分
  • Character: 根据用户给定的字符进行切割

RecursiveCharacterTextSplitter

RecursiveCharacterTextSplitter 默认的分隔符列表是 ["\n\n", "\n", " ", ""], 你可以理解为它将文本分割后,在根据设置的 chunk 大小进行组装。

最影响切分质量的就是两个参数:

  • chunkSize: 定义切分块的大小
  • chunkOverlap: 定义块与块之间重叠的大小。较大的重叠可能会导致分割的内容重复,较小的重叠可能会导致分割的内容不完整。
js
// RecursiveCharacterTextSplitter 对文本进行分割
 // https://chunkviz.up.railway.app/
 import { RecursiveCharacterTextSplitter } from 'langchain/text_splitter';
 import { TextLoader } from 'langchain/document_loaders/fs/text';
@@ -53,8 +53,8 @@
 
 const splitDocs = await splitter.splitDocuments(docs);
 
-console.log(splitDocs);

输出结果

- +console.log(splitDocs);

输出结果

+ \ No newline at end of file diff --git "a/guide/ai/OutputParser\346\236\204\345\273\272\346\240\274\345\274\217\345\214\226\350\276\223\345\207\272.html" "b/guide/ai/OutputParser\346\236\204\345\273\272\346\240\274\345\274\217\345\214\226\350\276\223\345\207\272.html" index 73da0b24..7bdda3be 100644 --- "a/guide/ai/OutputParser\346\236\204\345\273\272\346\240\274\345\274\217\345\214\226\350\276\223\345\207\272.html" +++ "b/guide/ai/OutputParser\346\236\204\345\273\272\346\240\274\345\274\217\345\214\226\350\276\223\345\207\272.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

OutputParser

OutputParser是一个用于解析模型输出的工具类,它可以帮助用户将模型输出解析为用户需要的格式。

String Output Parser

StringOutputParser是OutputParser的一个实现类,它可以将模型输出解析为字符串。

前面的例子中,我们使用了StringOutputParser来解析模型输出,这里就不过多赘述了。

StructuredOutputParser (结构化的输出)

StructuredOutputParser是OutputParser的另一个实现类,它可以将模型输出解析为结构化的数据。

例如,我们可以将模型输出解析为JSON格式的数据,这样我们就可以更方便地处理模型输出。

下面是一个使用StructuredOutputParser的例子:

js
import { StructuredOutputParser } from "langchain/output_parsers";
+    
Skip to content

OutputParser

OutputParser是一个用于解析模型输出的工具类,它可以帮助用户将模型输出解析为用户需要的格式。

String Output Parser

StringOutputParser是OutputParser的一个实现类,它可以将模型输出解析为字符串。

前面的例子中,我们使用了StringOutputParser来解析模型输出,这里就不过多赘述了。

StructuredOutputParser (结构化的输出)

StructuredOutputParser是OutputParser的另一个实现类,它可以将模型输出解析为结构化的数据。

例如,我们可以将模型输出解析为JSON格式的数据,这样我们就可以更方便地处理模型输出。

下面是一个使用StructuredOutputParser的例子:

js
import { StructuredOutputParser } from "langchain/output_parsers";
 import { PromptTemplate } from "@langchain/core/prompts";
 
 const parser = StructuredOutputParser.fromNamesAndDescriptions({
@@ -215,8 +215,8 @@
 const fixParser = OutputFixingParser.fromLLM(model, parser);
 const output = await fixParser.parse(JSON.stringify(wrongOutput));
 
-console.log(output);

可能会有朋友问,如果我把用户的问题也给 fixParser,这样不就得到一个正确的答案和正确的格式了么? 在我们的 demo 中当然是可以的,但实际工程中,引导 llm 返回数据的 prompt 可能非常巨大,非常消耗 token,我们使用 fixParser 就是用较少的成本去修复这个输出,来节约重复调用的成本。所以把原文也给 fixParser 的话,就达不到成本节约的目的了。

在进一步节约成本的背景下,我们是可以用对 GPT4 的错误输出用 GPT3.5 的 fixer 来修复,甚至是用一些开源模型来进行修复,因为在这个场景下,并不需要模型具有太高的质量,通过多模型的协同来降低成本。

- +console.log(output);

可能会有朋友问,如果我把用户的问题也给 fixParser,这样不就得到一个正确的答案和正确的格式了么? 在我们的 demo 中当然是可以的,但实际工程中,引导 llm 返回数据的 prompt 可能非常巨大,非常消耗 token,我们使用 fixParser 就是用较少的成本去修复这个输出,来节约重复调用的成本。所以把原文也给 fixParser 的话,就达不到成本节约的目的了。

在进一步节约成本的背景下,我们是可以用对 GPT4 的错误输出用 GPT3.5 的 fixer 来修复,甚至是用一些开源模型来进行修复,因为在这个场景下,并不需要模型具有太高的质量,通过多模型的协同来降低成本。

+ \ No newline at end of file diff --git "a/guide/ai/RAG \346\243\200\347\264\242\345\242\236\345\274\272\347\224\237\346\210\220\347\232\204\346\265\201\347\250\213.html" "b/guide/ai/RAG \346\243\200\347\264\242\345\242\236\345\274\272\347\224\237\346\210\220\347\232\204\346\265\201\347\250\213.html" index 9ac22ad4..7f1c0fad 100644 --- "a/guide/ai/RAG \346\243\200\347\264\242\345\242\236\345\274\272\347\224\237\346\210\220\347\232\204\346\265\201\347\250\213.html" +++ "b/guide/ai/RAG \346\243\200\347\264\242\345\242\236\345\274\272\347\224\237\346\210\220\347\232\204\346\265\201\347\250\213.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

RAG 检索增强生成的流程

RAG 其全称是 Retrieval Augmented Generation,可以被翻译成 检索增强生成技术,从标题上也能了解其核心的流程 检索 => 增强 => 生成。

LLM 的局限

在介绍 RAG 之前,我们先来了解一下 LLM 的局限性。

首先是幻觉问题(hallucination),LLM 底层还不具备真正的逻辑推理能力,是根据大量的数据进行概率性预测,所以在某些情况下,LLM 会生成一些不合理的答案。

其次是对领域知识的欠缺,造成这个问题主要是俩个原因,第一个是对知识的更新慢,另一个是对专业领域的知识训练样本不足导致。

RAG 的优势

当我们了解了 LLM 的局限性后,RAG 会尽可能提供与答案相关的上下文,来增强它正确输出的可能性。

RAG 的优势主要体现在以下几个方面:

  1. 用户输入提问
  2. 检索:根据用户提问对 向量数据库 进行相似性检测,查找与回答用户问题最相关的内容
  3. 增强:根据检索的结果,生成 prompt。 一般都会涉及 “仅依赖下述信息源来回答问题” 这种限制 LLM 参考信息源的语句,来减少幻想,让回答更加聚焦
  4. 生成:将增强后的 prompt 传递给 LLM,返回数据给用户

所以 RAG 就是哪里有问题解决哪里,既然大模型无法获得最新和内部的数据集,那我们就使用外挂的向量数据库为 LLM 提供最新和内部的数据库。既然大模型有幻想问题,我们就将回答问题所需要的信息和知识编码到上下文中,强制大模型只参考这些内容进行回答。

注意:LLM 是逻辑推理引擎,而不是信息引擎。所以,由外挂的向量数据库提供最有效的知识,然后由 LLM 根据知识进行推理,提供有价值的回复。

RAG 流程

输出结果

后续章节,在下将针对 RAG 的各个流程做详细的介绍。

- +
Skip to content

RAG 检索增强生成的流程

RAG 其全称是 Retrieval Augmented Generation,可以被翻译成 检索增强生成技术,从标题上也能了解其核心的流程 检索 => 增强 => 生成。

LLM 的局限

在介绍 RAG 之前,我们先来了解一下 LLM 的局限性。

首先是幻觉问题(hallucination),LLM 底层还不具备真正的逻辑推理能力,是根据大量的数据进行概率性预测,所以在某些情况下,LLM 会生成一些不合理的答案。

其次是对领域知识的欠缺,造成这个问题主要是俩个原因,第一个是对知识的更新慢,另一个是对专业领域的知识训练样本不足导致。

RAG 的优势

当我们了解了 LLM 的局限性后,RAG 会尽可能提供与答案相关的上下文,来增强它正确输出的可能性。

RAG 的优势主要体现在以下几个方面:

  1. 用户输入提问
  2. 检索:根据用户提问对 向量数据库 进行相似性检测,查找与回答用户问题最相关的内容
  3. 增强:根据检索的结果,生成 prompt。 一般都会涉及 “仅依赖下述信息源来回答问题” 这种限制 LLM 参考信息源的语句,来减少幻想,让回答更加聚焦
  4. 生成:将增强后的 prompt 传递给 LLM,返回数据给用户

所以 RAG 就是哪里有问题解决哪里,既然大模型无法获得最新和内部的数据集,那我们就使用外挂的向量数据库为 LLM 提供最新和内部的数据库。既然大模型有幻想问题,我们就将回答问题所需要的信息和知识编码到上下文中,强制大模型只参考这些内容进行回答。

注意:LLM 是逻辑推理引擎,而不是信息引擎。所以,由外挂的向量数据库提供最有效的知识,然后由 LLM 根据知识进行推理,提供有价值的回复。

RAG 流程

输出结果

后续章节,在下将针对 RAG 的各个流程做详细的介绍。

+ \ No newline at end of file diff --git "a/guide/ai/Retriever \345\270\270\350\247\201\347\232\204\344\274\230\345\214\226\346\226\271\345\274\217.html" "b/guide/ai/Retriever \345\270\270\350\247\201\347\232\204\344\274\230\345\214\226\346\226\271\345\274\217.html" index ae47044f..8dbf66c6 100644 --- "a/guide/ai/Retriever \345\270\270\350\247\201\347\232\204\344\274\230\345\214\226\346\226\271\345\274\217.html" +++ "b/guide/ai/Retriever \345\270\270\350\247\201\347\232\204\344\274\230\345\214\226\346\226\271\345\274\217.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

Retriever 常见的优化方式

上一节我们通过厂商提供的 Embedding 算法去制作匹配向量,通过 Facebook 提供的 Faiss 作为向量数据库,将数据存储到向量数据库中。

接下来我们将优化 Retriever 检索数据的方式

MultiQueryRetriever

MultiQueryRetriever 思路,或者说其他解决 llm 缺陷的思路基本都是一致的:加入更多 llm。

而 MultiQueryRetriever 是其中比较简单的一种解决方案,它使用 LLM 去将用户的输入改写成多个不同写法,从不同的角度来表达同一个意思,来克服因为关键词或者细微措词导致检索效果差的问题。

js
// MultiQueryRetriever: 
+    
Skip to content

Retriever 常见的优化方式

上一节我们通过厂商提供的 Embedding 算法去制作匹配向量,通过 Facebook 提供的 Faiss 作为向量数据库,将数据存储到向量数据库中。

接下来我们将优化 Retriever 检索数据的方式

MultiQueryRetriever

MultiQueryRetriever 思路,或者说其他解决 llm 缺陷的思路基本都是一致的:加入更多 llm。

而 MultiQueryRetriever 是其中比较简单的一种解决方案,它使用 LLM 去将用户的输入改写成多个不同写法,从不同的角度来表达同一个意思,来克服因为关键词或者细微措词导致检索效果差的问题。

js
// MultiQueryRetriever: 
 import "dotenv/config";
 import { FaissStore } from "@langchain/community/vectorstores/faiss";
 import { BaiduQianfanEmbeddings } from "@langchain/community/embeddings/baidu_qianfan";
@@ -113,8 +113,8 @@
     minSimilarityScore: 0.4, // 相似度
     maxK: 5, // 最大 K 值
     kIncrement: 1, // K 值每次增加的步长
-});

输出结果

参考文章

- +});

输出结果

参考文章

+ \ No newline at end of file diff --git "a/guide/ai/Retriever\344\271\213\345\220\221\351\207\217\346\225\260\346\215\256\345\272\223.html" "b/guide/ai/Retriever\344\271\213\345\220\221\351\207\217\346\225\260\346\215\256\345\272\223.html" index 3c225f4b..f138cf08 100644 --- "a/guide/ai/Retriever\344\271\213\345\220\221\351\207\217\346\225\260\346\215\256\345\272\223.html" +++ "b/guide/ai/Retriever\344\271\213\345\220\221\351\207\217\346\225\260\346\215\256\345\272\223.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

Retriever 之向量数据库

经过前面的学习,我们知道了如何对 数据进行加载和切割,接下来我们就要学习如何将数据通过 Embedding 算法转化为向量加载到向量数据库中。

Embedding

这里我们用最简单的词袋(words bag)模型来描述一下最简单的 embedding 过程,让大家更具象化的理解这个。

简单地说,词袋模型首先将一篇文章拆分成一个个单词,然后将其放入袋子里面。

例如我们有十篇文章,我们可以将文章拆分成一个个单词,然后统计单词出现的次数

js
第一篇文章:
+    
Skip to content

Retriever 之向量数据库

经过前面的学习,我们知道了如何对 数据进行加载和切割,接下来我们就要学习如何将数据通过 Embedding 算法转化为向量加载到向量数据库中。

Embedding

这里我们用最简单的词袋(words bag)模型来描述一下最简单的 embedding 过程,让大家更具象化的理解这个。

简单地说,词袋模型首先将一篇文章拆分成一个个单词,然后将其放入袋子里面。

例如我们有十篇文章,我们可以将文章拆分成一个个单词,然后统计单词出现的次数

js
第一篇文章:
 enson: 10  cool: 5  handsome: 8
 
 第二篇文章:
@@ -85,8 +85,8 @@
 const retriever = vectorStore.asRetriever(2); // 获取最相关的俩个文档片段
 const res = await retriever.invoke("茴香豆是做什么用的");
 
-console.log(res);

输出结果

- +console.log(res);

输出结果

+ \ No newline at end of file diff --git "a/guide/ai/langchain\345\277\253\351\200\237\345\205\245\351\227\250.html" "b/guide/ai/langchain\345\277\253\351\200\237\345\205\245\351\227\250.html" index 6f6b33a2..2882c053 100644 --- "a/guide/ai/langchain\345\277\253\351\200\237\345\205\245\351\227\250.html" +++ "b/guide/ai/langchain\345\277\253\351\200\237\345\205\245\351\227\250.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

LangChain 快速入门

安装

要安装这个包,你可以使用 npm 或 yarn:

bash
yarn add langchain
yarn add langchain

安装环境

node >= 18.x

什么是 LCEL (LangChain Expression Language)

LangChain Expression Language is a way to create arbitrary custom chains. It is built on the Runnable protocol.

LCEL 无论是 python 还是 js 版本都在主推的新设计,能创建自定义的链,它是基于 Runnable 协议构建的。

通过 LangChain 加载大模型

本地大模型

在 mac 平台下,推荐用 ollama,使用简单,下载好模型后,点击这个 app,就会自动在 http://localhost:11434 起一个 llm 的服务。

安装 llama3 模型

可以通过 ollama list 查看所有的模型,然后通过 ollama pull 安装模型。

bash
# 模型列表参考:https://ollama.com/library
+    
Skip to content

LangChain 快速入门

安装

要安装这个包,你可以使用 npm 或 yarn:

bash
yarn add langchain
yarn add langchain

安装环境

node >= 18.x

什么是 LCEL (LangChain Expression Language)

LangChain Expression Language is a way to create arbitrary custom chains. It is built on the Runnable protocol.

LCEL 无论是 python 还是 js 版本都在主推的新设计,能创建自定义的链,它是基于 Runnable 协议构建的。

通过 LangChain 加载大模型

本地大模型

在 mac 平台下,推荐用 ollama,使用简单,下载好模型后,点击这个 app,就会自动在 http://localhost:11434 起一个 llm 的服务。

安装 llama3 模型

可以通过 ollama list 查看所有的模型,然后通过 ollama pull 安装模型。

bash
# 模型列表参考:https://ollama.com/library
 ollama pull [model]
# 模型列表参考:https://ollama.com/library
 ollama pull [model]

目前下载了一个 ollama3 的模型

初始状态图

当我们在本地启动了一个模型后,我们就可以通过 LangChain 来调用这个模型了。

js
import { Ollama } from '@langchain/community/llms/ollama';
 
@@ -93,8 +93,8 @@
 
 const res = await ernieTurbo.invoke(messages);
 
-console.log(res);

输出结果

⚠️注意: 如果出现 API 调用次数有所限制,需要在 https://console.bce.baidu.com/qianfan/ais/console/onlineService 开通对应的模型服务。

LCEL 有什么优势

LCEL 从底层设计的目标就是支持 从原型到生产 完整流程不需要修改任何代码,也就是我们在写的任何原型代码不需要太多的改变就能支持生产级别的各种特性(比如并行、steaming 等),具体来说会有这些优势:

  • 并行: 只要是整个 chain 中有可以并行的步骤就会自动的并行,来减少使用时的延迟。
  • 自动的重试和 fallback: 大部分 chain 的组成部分都有自动的重试(比如因为网络原因的失败)和回退机制,来解决很多请求的出错问题。
  • 对 chain 中间结果的访问,在旧的写法中很难访问中间的结果,而 LCEL 中可以方便的通过访问中间结果来进行调试和记录。
  • LCEL 会自动支持 LangSimith 进行可视化和记录。

一条 Chain 组成的每个模块都是继承自 Runnable 这个接口,而一条 Chain 也是继承自这个接口,所以一条 Chain 也可以很自然的成为另一个 Chain 的一个模块。

任意的 Runnable 对象,都有几个常用的标准调用接口:

  • invoke 基础调用
  • batch 批量调用
  • stream 流式返回结果
  • streamLog 流式返回结果,并返回中间的运行结果

参考文章

- +console.log(res);

输出结果

⚠️注意: 如果出现 API 调用次数有所限制,需要在 https://console.bce.baidu.com/qianfan/ais/console/onlineService 开通对应的模型服务。

LCEL 有什么优势

LCEL 从底层设计的目标就是支持 从原型到生产 完整流程不需要修改任何代码,也就是我们在写的任何原型代码不需要太多的改变就能支持生产级别的各种特性(比如并行、steaming 等),具体来说会有这些优势:

  • 并行: 只要是整个 chain 中有可以并行的步骤就会自动的并行,来减少使用时的延迟。
  • 自动的重试和 fallback: 大部分 chain 的组成部分都有自动的重试(比如因为网络原因的失败)和回退机制,来解决很多请求的出错问题。
  • 对 chain 中间结果的访问,在旧的写法中很难访问中间的结果,而 LCEL 中可以方便的通过访问中间结果来进行调试和记录。
  • LCEL 会自动支持 LangSimith 进行可视化和记录。

一条 Chain 组成的每个模块都是继承自 Runnable 这个接口,而一条 Chain 也是继承自这个接口,所以一条 Chain 也可以很自然的成为另一个 Chain 的一个模块。

任意的 Runnable 对象,都有几个常用的标准调用接口:

  • invoke 基础调用
  • batch 批量调用
  • stream 流式返回结果
  • streamLog 流式返回结果,并返回中间的运行结果

参考文章

+ \ No newline at end of file diff --git "a/guide/ai/\344\273\200\344\271\210\346\230\257langchain.html" "b/guide/ai/\344\273\200\344\271\210\346\230\257langchain.html" index adb6f81d..1a5a6df0 100644 --- "a/guide/ai/\344\273\200\344\271\210\346\230\257langchain.html" +++ "b/guide/ai/\344\273\200\344\271\210\346\230\257langchain.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

什么是 LangChain

LangChain 是一个用于开发由大型语言模型 (LLMs) 驱动的应用程序的框架。

为什么是 LangChain

因为 LLM 的 API 只是提供了一个非常基础的调用方式,当我们需要构建一个复杂的 Chat Bot 时,就需要考虑如何保存聊天的上下文、网络搜索、加载 PDF 等工程问题, 而LangChain 提供了一种解决方案,让开发者可以专注于业务逻辑的开发。

足够的流行度和认可度,目前已经在 Github 获得 83k star,并且其上升速度非常恐怖:

Star History Chart

而 LangChain.js 并不是 Python 版本的套壳,而是一个完整的团队从 0 开始构建的生态,足以看出官方对 JavaScript 生态的重视:

Star History Chart

基于此,在下将以 LangChain.js 为例,去使用大模型领域最流行的框架去构建应用,感受大模型的魅力。

支持的环境

  • Node.js (ESM and CommonJS) - 18.x, 19.x, 20.x
  • Cloudflare Workers Cloudflare
  • Vercel / Next.js (Browser, Serverless and Edge functions)
  • Supabase Edge Functions Supabase
  • Browser
  • Deno

相关文档

- +
Skip to content

什么是 LangChain

LangChain 是一个用于开发由大型语言模型 (LLMs) 驱动的应用程序的框架。

为什么是 LangChain

因为 LLM 的 API 只是提供了一个非常基础的调用方式,当我们需要构建一个复杂的 Chat Bot 时,就需要考虑如何保存聊天的上下文、网络搜索、加载 PDF 等工程问题, 而LangChain 提供了一种解决方案,让开发者可以专注于业务逻辑的开发。

足够的流行度和认可度,目前已经在 Github 获得 83k star,并且其上升速度非常恐怖:

Star History Chart

而 LangChain.js 并不是 Python 版本的套壳,而是一个完整的团队从 0 开始构建的生态,足以看出官方对 JavaScript 生态的重视:

Star History Chart

基于此,在下将以 LangChain.js 为例,去使用大模型领域最流行的框架去构建应用,感受大模型的魅力。

支持的环境

  • Node.js (ESM and CommonJS) - 18.x, 19.x, 20.x
  • Cloudflare Workers Cloudflare
  • Vercel / Next.js (Browser, Serverless and Edge functions)
  • Supabase Edge Functions Supabase
  • Browser
  • Deno

相关文档

+ \ No newline at end of file diff --git "a/guide/ai/\346\236\204\345\273\272\345\217\257\345\244\215\347\224\250\347\232\204PromptTemplate.html" "b/guide/ai/\346\236\204\345\273\272\345\217\257\345\244\215\347\224\250\347\232\204PromptTemplate.html" index 6c27fe44..3d78ebd5 100644 --- "a/guide/ai/\346\236\204\345\273\272\345\217\257\345\244\215\347\224\250\347\232\204PromptTemplate.html" +++ "b/guide/ai/\346\236\204\345\273\272\345\217\257\345\244\215\347\224\250\347\232\204PromptTemplate.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

构建可复用的 PromptTemplate

Prompt 是大模型的核心,传统的方式一般是通过字符串或者字符串模版来构建 Prompt,但是这种方式不够灵活,也不够易用。为了解决这个问题,LangChain 引入了 PromptTemplate,它是一个可以复用的 Prompt 模版,可以通过参数化的方式来构建 Prompt。

基础的 Prompt 模版使用

js
// 基础的 Prompt 模版使用
+    
Skip to content

构建可复用的 PromptTemplate

Prompt 是大模型的核心,传统的方式一般是通过字符串或者字符串模版来构建 Prompt,但是这种方式不够灵活,也不够易用。为了解决这个问题,LangChain 引入了 PromptTemplate,它是一个可以复用的 Prompt 模版,可以通过参数化的方式来构建 Prompt。

基础的 Prompt 模版使用

js
// 基础的 Prompt 模版使用
 import { PromptTemplate } from "@langchain/core/prompts";
 
 const greetingPrompt = new PromptTemplate({
@@ -147,8 +147,8 @@
     text: "你好,世界",
 })
 
-console.log(res);

组合多个 Prompt 模版

可以通过 PipelinePromptTemplate 组合多个 Prompt 模版,这样可以更好的复用和管理 Prompt 模版。

在 PipelinePromptTemplate 有两个核心的概念:

  • pipelinePrompts,一组 object,每个 object 表示 prompt 运行后赋值给 name 变量
  • finalPrompt,表示最终输出的 prompt
- +console.log(res);

组合多个 Prompt 模版

可以通过 PipelinePromptTemplate 组合多个 Prompt 模版,这样可以更好的复用和管理 Prompt 模版。

在 PipelinePromptTemplate 有两个核心的概念:

  • pipelinePrompts,一组 object,每个 object 表示 prompt 运行后赋值给 name 变量
  • finalPrompt,表示最终输出的 prompt
+ \ No newline at end of file diff --git "a/guide/canvas/Canvas\345\260\272\345\257\270\345\217\212\345\210\206\350\276\250\347\216\207\347\237\253\346\255\243.html" "b/guide/canvas/Canvas\345\260\272\345\257\270\345\217\212\345\210\206\350\276\250\347\216\207\347\237\253\346\255\243.html" index f2e22d9b..e694bc6f 100644 --- "a/guide/canvas/Canvas\345\260\272\345\257\270\345\217\212\345\210\206\350\276\250\347\216\207\347\237\253\346\255\243.html" +++ "b/guide/canvas/Canvas\345\260\272\345\257\270\345\217\212\345\210\206\350\276\250\347\216\207\347\237\253\346\255\243.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

Canvas 尺寸与分辨率矫正

前言

Canvas 的默认大小为 300 像素 ×150 像素(宽 × 高,像素的单位是 px)

html
<canvas id="tutorial" width="150" height="150"></canvas>
<canvas id="tutorial" width="150" height="150"></canvas>

canvas 看起来和 img 元素很相像,唯一的不同就是它并没有 src 和 alt 属性。实际上,canvas 标签只有两个属性: width 和 height。

正文

设置画布的 css 大小

canvas 可以使用CSS来定义大小,但在绘制时图像会伸缩以适应它的框架尺寸:如果 CSS 的尺寸与初始画布的比例不一致,它会出现扭曲。

假设一个场景,当我们给 canvas 设置了 css 的宽高 500px * 500px, 但是我们在 canvas 中绘制的图形是 100px * 100px, 那么最终的图形会被拉伸,如下图所示:

html
<canvas id="test-canvas"></canvas>
+    
Skip to content

Canvas 尺寸与分辨率矫正

前言

Canvas 的默认大小为 300 像素 ×150 像素(宽 × 高,像素的单位是 px)

html
<canvas id="tutorial" width="150" height="150"></canvas>
<canvas id="tutorial" width="150" height="150"></canvas>

canvas 看起来和 img 元素很相像,唯一的不同就是它并没有 src 和 alt 属性。实际上,canvas 标签只有两个属性: width 和 height。

正文

设置画布的 css 大小

canvas 可以使用CSS来定义大小,但在绘制时图像会伸缩以适应它的框架尺寸:如果 CSS 的尺寸与初始画布的比例不一致,它会出现扭曲。

假设一个场景,当我们给 canvas 设置了 css 的宽高 500px * 500px, 但是我们在 canvas 中绘制的图形是 100px * 100px, 那么最终的图形会被拉伸,如下图所示:

html
<canvas id="test-canvas"></canvas>
 <script>
     const canvas = document.getElementById('test-canvas');
     const ctx = canvas.getContext('2d');
@@ -163,8 +163,8 @@
 // 这时候我们还需要重新缩放 ctx
 const ctx = canvas.getContext('2d');
 ctx.setTransform(1, 0, 0, 1, 0, 0); // scale 前先恢复变换矩阵,不然会重复 scale
-ctx.scale(dpr, dpr);

参考文档

- +ctx.scale(dpr, dpr);

参考文档

+ \ No newline at end of file diff --git "a/guide/canvas/\345\217\257\350\247\206\345\214\272\345\237\237\345\206\205\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.html" "b/guide/canvas/\345\217\257\350\247\206\345\214\272\345\237\237\345\206\205\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.html" index 84785db5..25156284 100644 --- "a/guide/canvas/\345\217\257\350\247\206\345\214\272\345\237\237\345\206\205\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.html" +++ "b/guide/canvas/\345\217\257\350\247\206\345\214\272\345\237\237\345\206\205\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

可视区域内渲染提高 Canvas 的书写性能

前言

上一节我们通过离屏渲染提高了 Canvas 的渲染性能,但是离屏渲染也有一些缺点,比如会增加内存的使用,而且在某些场景下,离屏渲染的性能并不会比直接在 Canvas 上绘制要高。本节我们将介绍如何通过可视区域内渲染提高 Canvas 的书写性能。

前提

一般我们说的可视区域内渲染,是指在 Canvas 上只绘制可视区域内的内容,而不是绘制整个 Canvas 的内容。这样做的好处是可以减少 Canvas 的绘制区域,从而提高 Canvas 的渲染性能。

对于 Canvas 而言,无法无限制地扩大 Canvas 的面积,因此浏览器对 Canvas 的大小也有一定的限制。从 MDN 文档 可知, 在 Chrome 浏览器中,Canvas 的大小限制为 32767px * 32767px。由于 Canvas 的大小限制,因此我们在实现 无限画布 的功能时,不能无限拓展 Canvas 的大小,可以通过坐标的切换,来实现无限画布的功能。

实现无限画布

实现思路

记初始坐标A (x, y), 横向滚动距离为 scrollX, 纵向滚动距离为 scrollY

在初始状态下, scrollX、scrollY 均为 0

初始状态图

假设现在,我们在水平方向向右滚动了scrollX,垂直方向向下滚动scrollY。那么滚动后的坐标就是

x1 = x - scrollX

y1 = y - scrollY

这里大家可能会有疑惑,为什么是减法呢?因为向下滚动后,绘制的图形应该是往上移动的,因此我们需要减去滚动的距离。

滚动后的状态图

在代码中,我们可以通过监听 Canvas 的 WheelEvent 事件,来获取滚动的距离,然后根据上面的公式计算出滚动后的坐标,最后重新绘制 Canvas。

具体实现如下

jsx
ctx.clearRect(0, 0, window.innerWidth, window.innerHeight);
+    
Skip to content

可视区域内渲染提高 Canvas 的书写性能

前言

上一节我们通过离屏渲染提高了 Canvas 的渲染性能,但是离屏渲染也有一些缺点,比如会增加内存的使用,而且在某些场景下,离屏渲染的性能并不会比直接在 Canvas 上绘制要高。本节我们将介绍如何通过可视区域内渲染提高 Canvas 的书写性能。

前提

一般我们说的可视区域内渲染,是指在 Canvas 上只绘制可视区域内的内容,而不是绘制整个 Canvas 的内容。这样做的好处是可以减少 Canvas 的绘制区域,从而提高 Canvas 的渲染性能。

对于 Canvas 而言,无法无限制地扩大 Canvas 的面积,因此浏览器对 Canvas 的大小也有一定的限制。从 MDN 文档 可知, 在 Chrome 浏览器中,Canvas 的大小限制为 32767px * 32767px。由于 Canvas 的大小限制,因此我们在实现 无限画布 的功能时,不能无限拓展 Canvas 的大小,可以通过坐标的切换,来实现无限画布的功能。

实现无限画布

实现思路

记初始坐标A (x, y), 横向滚动距离为 scrollX, 纵向滚动距离为 scrollY

在初始状态下, scrollX、scrollY 均为 0

初始状态图

假设现在,我们在水平方向向右滚动了scrollX,垂直方向向下滚动scrollY。那么滚动后的坐标就是

x1 = x - scrollX

y1 = y - scrollY

这里大家可能会有疑惑,为什么是减法呢?因为向下滚动后,绘制的图形应该是往上移动的,因此我们需要减去滚动的距离。

滚动后的状态图

在代码中,我们可以通过监听 Canvas 的 WheelEvent 事件,来获取滚动的距离,然后根据上面的公式计算出滚动后的坐标,最后重新绘制 Canvas。

具体实现如下

jsx
ctx.clearRect(0, 0, window.innerWidth, window.innerHeight);
 ctx.save();
 ctx.translate(scrollX, scrollY);
 // 绘制相关的逻辑
@@ -225,8 +225,8 @@
       ctx.restore();
     },
     []
-);

实现效果

canvas

具体代码

参考文章

- +);

实现效果

canvas

具体代码

参考文章

+ \ No newline at end of file diff --git "a/guide/canvas/\345\246\202\344\275\225\345\234\250Canvas\347\224\273\346\235\277\344\270\212\350\207\252\347\224\261\344\271\246\345\206\231.html" "b/guide/canvas/\345\246\202\344\275\225\345\234\250Canvas\347\224\273\346\235\277\344\270\212\350\207\252\347\224\261\344\271\246\345\206\231.html" index ffa428c3..f4bead7b 100644 --- "a/guide/canvas/\345\246\202\344\275\225\345\234\250Canvas\347\224\273\346\235\277\344\270\212\350\207\252\347\224\261\344\271\246\345\206\231.html" +++ "b/guide/canvas/\345\246\202\344\275\225\345\234\250Canvas\347\224\273\346\235\277\344\270\212\350\207\252\347\224\261\344\271\246\345\206\231.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

如何实现一个自由绘制的 Canvas 画板

前言

承接上文,当我们了解了如何去设置一个 Canvas 尺寸,并通过分辨率对 Canvas 进行矫正后,我们就可以开始实现一个自由绘制的 Canvas 画板了。

正文

实现思路

自由画笔的实现可以通过:

    1. 监听鼠标事件
    1. 将鼠标移动的点记录下来
    1. 将这些点连成线,就可以实现自由画笔了。

实现代码

html
<!DOCTYPE html>
+    
Skip to content

如何实现一个自由绘制的 Canvas 画板

前言

承接上文,当我们了解了如何去设置一个 Canvas 尺寸,并通过分辨率对 Canvas 进行矫正后,我们就可以开始实现一个自由绘制的 Canvas 画板了。

正文

实现思路

自由画笔的实现可以通过:

    1. 监听鼠标事件
    1. 将鼠标移动的点记录下来
    1. 将这些点连成线,就可以实现自由画笔了。

实现代码

html
<!DOCTYPE html>
 <html lang="en">
 
 <head>
@@ -357,8 +357,8 @@
       addPoint(e); // 将鼠标移动的点添加到points数组中
       render(ctx, points); // 绘制
       updatePointCounter(points.length);
-  }));

之前绘制一条直线需要近 40 个点 canvas

通过点稀释,我们发现画一条直线只需要 8 个点就可以了,这样就大大提高了性能。 canvas

虽然点稀释可以减少绘制的点数,提高性能,但在绘制曲线的时候,我们会发现曲线的圆滑度不够,这是因为我们的点数太少了,我们可以通过贝塞尔曲线来优化我们的 Canvas 书写。

canvas

参考

- + }));

之前绘制一条直线需要近 40 个点 canvas

通过点稀释,我们发现画一条直线只需要 8 个点就可以了,这样就大大提高了性能。 canvas

虽然点稀释可以减少绘制的点数,提高性能,但在绘制曲线的时候,我们会发现曲线的圆滑度不够,这是因为我们的点数太少了,我们可以通过贝塞尔曲线来优化我们的 Canvas 书写。

canvas

参考

+ \ No newline at end of file diff --git "a/guide/canvas/\351\200\232\350\277\207 OffscreenCanvas + Worker \346\217\220\351\253\230\344\271\246\345\206\231\346\200\247\350\203\275.html" "b/guide/canvas/\351\200\232\350\277\207 OffscreenCanvas + Worker \346\217\220\351\253\230\344\271\246\345\206\231\346\200\247\350\203\275.html" index 5811721f..298e53b5 100644 --- "a/guide/canvas/\351\200\232\350\277\207 OffscreenCanvas + Worker \346\217\220\351\253\230\344\271\246\345\206\231\346\200\247\350\203\275.html" +++ "b/guide/canvas/\351\200\232\350\277\207 OffscreenCanvas + Worker \346\217\220\351\253\230\344\271\246\345\206\231\346\200\247\350\203\275.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

通过 OffscreenCanvas + Worker 提高书写性能

前言

上一节我们通过离屏渲染提高了 Canvas 的渲染性能,但是在绘制的过程中,我们会发现,当绘制的图形越来越多时,Canvas 的渲染性能会越来越差,这是因为我们在绘制图形会阻塞主线程,如果主线程中还有其他的任务也会表现出卡顿的效果,Canvas 的渲染性能越来越差。

这节我们将通过 OffscreenCanvas + Worker 将绘制图形的任务放到 Worker 中进行,避免阻塞主线程,从而提高 Canvas 的渲染性能。

实现思路

在 worker 线程中是无法操作 DOM 的,但 OffscreenCanvas 可以在 worker 线程中进行操作,因此我们可以通过 OffscreenCanvas 将绘制图形的任务放到 worker 线程中进行。这样可以减少主线程的任务,从而提高书写性能。

具体实现

创建 OffscreenCanvas

js
const canvas = document.getElementById('draw');
+    
Skip to content

通过 OffscreenCanvas + Worker 提高书写性能

前言

上一节我们通过离屏渲染提高了 Canvas 的渲染性能,但是在绘制的过程中,我们会发现,当绘制的图形越来越多时,Canvas 的渲染性能会越来越差,这是因为我们在绘制图形会阻塞主线程,如果主线程中还有其他的任务也会表现出卡顿的效果,Canvas 的渲染性能越来越差。

这节我们将通过 OffscreenCanvas + Worker 将绘制图形的任务放到 Worker 中进行,避免阻塞主线程,从而提高 Canvas 的渲染性能。

实现思路

在 worker 线程中是无法操作 DOM 的,但 OffscreenCanvas 可以在 worker 线程中进行操作,因此我们可以通过 OffscreenCanvas 将绘制图形的任务放到 worker 线程中进行。这样可以减少主线程的任务,从而提高书写性能。

具体实现

创建 OffscreenCanvas

js
const canvas = document.getElementById('draw');
 const offScreenCanvas = canvas.transferControlToOffscreen(); // 将 canvas 转换为 offScreenCanvas
const canvas = document.getElementById('draw');
 const offScreenCanvas = canvas.transferControlToOffscreen(); // 将 canvas 转换为 offScreenCanvas

将 offScreenCanvas 传递给 worker 线程

js
const worker = new Worker('./worker.js'); // 创建一个 webWorker
 const offScreenCanvas = canvas.transferControlToOffscreen(); // 将 canvas 转换为 offScreenCanvas
@@ -145,8 +145,8 @@
         x,
         y,
     });
-}

实现效果

初始状态图

具体代码

参考文章

- +}

实现效果

初始状态图

具体代码

参考文章

+ \ No newline at end of file diff --git "a/guide/canvas/\351\200\232\350\277\207\344\270\212\344\270\213\345\210\206\345\261\202\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.html" "b/guide/canvas/\351\200\232\350\277\207\344\270\212\344\270\213\345\210\206\345\261\202\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.html" index 3d407e9d..76f35540 100644 --- "a/guide/canvas/\351\200\232\350\277\207\344\270\212\344\270\213\345\210\206\345\261\202\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.html" +++ "b/guide/canvas/\351\200\232\350\277\207\344\270\212\344\270\213\345\210\206\345\261\202\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

通过上下分层优化 Canvas 书写性能

前言

上一节,我们通过点稀疏的方式,优化了 Canvas 的绘制性能。但在书写的过程中,因为点的数量减少导致书写的效果不够理想,因此通过贝塞尔曲线的方式,将点连接起来,形成平滑的曲线,从而达到书写的效果。 这节,我们将通过上下分层的方式,优化 Canvas 的书写性能。

基本思路

这里我将 Canvas 分为上下两层,上层称之为动态层,下层称之为静态层,动态层用于书写,静态层用于显示书写的效果。当书写的时候,将书写的点绘制到上层,当书写完成后,将上层的内容绘制到下层,然后清空上层的内容,这样就可以达到书写的效果。

因为 Canvas 每次渲染都会将整个 Canvas 清空,所以我们需要将静态层的内容保存下来,然后在每次书写的时候,只需要渲染当前动态层的内容即可。

实现

创建 2 个 Canvas

首先创建 2 个 Canvas, 上层 Canvas 用于书写, 书写完成后将上层 Canvas 的内容绘制到下层 Canvas 中, 然后清空上层 Canvas 的内容, 这样就可以达到书写的效果。

这样的好处在于, 每次书写的时候, 只需要渲染上层 Canvas 的书写内容即可, 不需要每次都渲染整个 Canvas, 从而达到优化性能的目的。

html
<style>
+    
Skip to content

通过上下分层优化 Canvas 书写性能

前言

上一节,我们通过点稀疏的方式,优化了 Canvas 的绘制性能。但在书写的过程中,因为点的数量减少导致书写的效果不够理想,因此通过贝塞尔曲线的方式,将点连接起来,形成平滑的曲线,从而达到书写的效果。 这节,我们将通过上下分层的方式,优化 Canvas 的书写性能。

基本思路

这里我将 Canvas 分为上下两层,上层称之为动态层,下层称之为静态层,动态层用于书写,静态层用于显示书写的效果。当书写的时候,将书写的点绘制到上层,当书写完成后,将上层的内容绘制到下层,然后清空上层的内容,这样就可以达到书写的效果。

因为 Canvas 每次渲染都会将整个 Canvas 清空,所以我们需要将静态层的内容保存下来,然后在每次书写的时候,只需要渲染当前动态层的内容即可。

实现

创建 2 个 Canvas

首先创建 2 个 Canvas, 上层 Canvas 用于书写, 书写完成后将上层 Canvas 的内容绘制到下层 Canvas 中, 然后清空上层 Canvas 的内容, 这样就可以达到书写的效果。

这样的好处在于, 每次书写的时候, 只需要渲染上层 Canvas 的书写内容即可, 不需要每次都渲染整个 Canvas, 从而达到优化性能的目的。

html
<style>
     #draw {
         border: 1px solid black;
         position: absolute;
@@ -109,8 +109,8 @@
         points = []; // 绘制完毕后,清空points数组
         renderLowerCanvas(ctx, ctxContent);
     });
-</script>

演示效果

这里我将动静 Canvas 分成了左右 2 个部分,左边是动态层 Canvas,右边是静态层 Canvas,可以看到,当书写的时候,只有左边的 Canvas 会有书写的效果,右边的 Canvas 不会有书写的效果,当书写完成后,左边的 Canvas 会将书写的内容绘制到右边的 Canvas 中,然后清空左边的 Canvas 的内容,这样就可以达到书写的效果。

canvas

当我们把 2 个 Canvas 整合在一起后,具体的效果如下

canvas

具体代码

https://github.com/enson0131/learn/blob/main/Canvas/白板相关/上下分层绘制.html

- +</script>

演示效果

这里我将动静 Canvas 分成了左右 2 个部分,左边是动态层 Canvas,右边是静态层 Canvas,可以看到,当书写的时候,只有左边的 Canvas 会有书写的效果,右边的 Canvas 不会有书写的效果,当书写完成后,左边的 Canvas 会将书写的内容绘制到右边的 Canvas 中,然后清空左边的 Canvas 的内容,这样就可以达到书写的效果。

canvas

当我们把 2 个 Canvas 整合在一起后,具体的效果如下

canvas

具体代码

https://github.com/enson0131/learn/blob/main/Canvas/白板相关/上下分层绘制.html

+ \ No newline at end of file diff --git "a/guide/canvas/\351\200\232\350\277\207\347\246\273\345\261\217\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.html" "b/guide/canvas/\351\200\232\350\277\207\347\246\273\345\261\217\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.html" index 8a3f5fbe..7e502020 100644 --- "a/guide/canvas/\351\200\232\350\277\207\347\246\273\345\261\217\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.html" +++ "b/guide/canvas/\351\200\232\350\277\207\347\246\273\345\261\217\346\270\262\346\237\223\346\217\220\351\253\230Canvas\344\271\246\345\206\231\346\200\247\350\203\275.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

通过离屏渲染提高 Canvas 书写性能

前言

前面我们通过上下分层的方式,优化了 Canvas 的书写性能,接下来我们通过离屏渲染的方式,进一步优化 Canvas 的书写性能。

基本思路

在书写的过程中,每绘制一笔都需要不断地调用 Canvas 的 API,重新渲染整个 Canvas,这样就会导致性能的浪费。 而离屏渲染则是将 绘制内容存储到离屏的 Canvas 中,相当于一个缓冲区,然后将需要绘制的画面在离屏的 Canvas 缓冲好,最后将离屏的 Canvas 转化成图片,渲染到屏幕上,这样就可以达到优化性能的目的。

实现

创建离屏 Canvas

思路如下: 基于上一节的基础,我们改写 render 函数,如果是离屏渲染的话,将绘制的内容存储到离屏的 Canvas 中,然后将离屏的 Canvas 缓存起来,下次绘制的时候,如果命中缓存的话,就直接使用缓存的 Canvas,从而达到优化性能的目的。

操作如下:

  • 1 在执行 render 函数之前,先判断是否存在缓存的 Canvas,如果存在的话,就直接使用缓存的 Canvas
  • 2 如果命中缓存,使用离屏 Canvas 转化成图片进行绘制
  • 3 如果不存在缓存的 Canvas,就创建一个离屏的 Canvas,然后将绘制的内容存储到离屏的 Canvas 中,最后将离屏的 Canvas 缓存起来
html
<script>
+    
Skip to content

通过离屏渲染提高 Canvas 书写性能

前言

前面我们通过上下分层的方式,优化了 Canvas 的书写性能,接下来我们通过离屏渲染的方式,进一步优化 Canvas 的书写性能。

基本思路

在书写的过程中,每绘制一笔都需要不断地调用 Canvas 的 API,重新渲染整个 Canvas,这样就会导致性能的浪费。 而离屏渲染则是将 绘制内容存储到离屏的 Canvas 中,相当于一个缓冲区,然后将需要绘制的画面在离屏的 Canvas 缓冲好,最后将离屏的 Canvas 转化成图片,渲染到屏幕上,这样就可以达到优化性能的目的。

实现

创建离屏 Canvas

思路如下: 基于上一节的基础,我们改写 render 函数,如果是离屏渲染的话,将绘制的内容存储到离屏的 Canvas 中,然后将离屏的 Canvas 缓存起来,下次绘制的时候,如果命中缓存的话,就直接使用缓存的 Canvas,从而达到优化性能的目的。

操作如下:

  • 1 在执行 render 函数之前,先判断是否存在缓存的 Canvas,如果存在的话,就直接使用缓存的 Canvas
  • 2 如果命中缓存,使用离屏 Canvas 转化成图片进行绘制
  • 3 如果不存在缓存的 Canvas,就创建一个离屏的 Canvas,然后将绘制的内容存储到离屏的 Canvas 中,最后将离屏的 Canvas 缓存起来
html
<script>
         const elementWithCanvasCache = new WeakMap(); // 用于存储离屏 Canvas 的缓存
         const generateOffScreenCanvas = (points) => {
             const padding = 20; // 避免笔记被 Canvas 
@@ -317,8 +317,8 @@
               y: minY * dpr - padding
           });
       }
-</script>

实现的效果

canvas

具体代码

https://github.com/enson0131/learn/blob/main/Canvas/白板相关/性能优化之离屏渲染及缓存.html

- +</script>

实现的效果

canvas

具体代码

https://github.com/enson0131/learn/blob/main/Canvas/白板相关/性能优化之离屏渲染及缓存.html

+ \ No newline at end of file diff --git "a/guide/canvas/\351\200\232\350\277\207\350\264\235\345\241\236\345\260\224\346\233\262\347\272\277\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.html" "b/guide/canvas/\351\200\232\350\277\207\350\264\235\345\241\236\345\260\224\346\233\262\347\272\277\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.html" index 995e2b2f..d8b8312c 100644 --- "a/guide/canvas/\351\200\232\350\277\207\350\264\235\345\241\236\345\260\224\346\233\262\347\272\277\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.html" +++ "b/guide/canvas/\351\200\232\350\277\207\350\264\235\345\241\236\345\260\224\346\233\262\347\272\277\344\274\230\345\214\226Canvas\344\271\246\345\206\231\346\200\247\350\203\275.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

通过贝塞尔曲线解决 Canvas 书写的圆滑问题

前言

书接上回,当我们实现了一个自由书写的 Canvas 画板后,我们发现采集的点越少,画出来的圈圈越不圆润,这时候我们可以使用贝塞尔曲线来优化书写。

canvas

正文

贝塞尔曲线

贝塞尔曲线是一种数学曲线,常见的有: 线性贝塞尔曲线、二次·贝塞尔曲线、三次贝塞尔曲线等, 它可以用来实现平滑曲线,我们可以通过贝塞尔曲线以相对圆滑的方式来优化我们的 Canvas 书写。

优化思路

我们可以通过贝塞尔曲线来优化我们的 Canvas 书写性能,将笔直的线变的更加圆滑弯曲。

在 Canvas2D API 中有一个方法叫做 quadraticCurveTo 用于创建二次贝塞尔曲线。它需要两个点:一个控制点和一个终点。曲线从当前的绘图位置开始绘制,并以控制点为参考,向终点绘制曲线。

这个方法的语法是:context.quadraticCurveTo(cp1x, cp1y, x, y);

其中:

cp1x 和 cp1y 是控制点坐标。
+    
Skip to content

通过贝塞尔曲线解决 Canvas 书写的圆滑问题

前言

书接上回,当我们实现了一个自由书写的 Canvas 画板后,我们发现采集的点越少,画出来的圈圈越不圆润,这时候我们可以使用贝塞尔曲线来优化书写。

canvas

正文

贝塞尔曲线

贝塞尔曲线是一种数学曲线,常见的有: 线性贝塞尔曲线、二次·贝塞尔曲线、三次贝塞尔曲线等, 它可以用来实现平滑曲线,我们可以通过贝塞尔曲线以相对圆滑的方式来优化我们的 Canvas 书写。

优化思路

我们可以通过贝塞尔曲线来优化我们的 Canvas 书写性能,将笔直的线变的更加圆滑弯曲。

在 Canvas2D API 中有一个方法叫做 quadraticCurveTo 用于创建二次贝塞尔曲线。它需要两个点:一个控制点和一个终点。曲线从当前的绘图位置开始绘制,并以控制点为参考,向终点绘制曲线。

这个方法的语法是:context.quadraticCurveTo(cp1x, cp1y, x, y);

其中:

cp1x 和 cp1y 是控制点坐标。
 x 和 y 是终点坐标。
 

这个方法不会直接绘制曲线,而是将曲线添加到当前的路径中。要在画布上实际绘制曲线,你需要使用 stroke 或 fill 方法

优化实现

1 采集到所有的点集 2 前一个点作为控制点,当前点作为终点,绘制二次贝塞尔曲线

html
<!DOCTYPE html>
 <html lang="en">
@@ -367,8 +367,8 @@
         }
     </script>
 </body>
-</html>

将两个点的中点作为控制点的选择通常用于创建平滑的曲线,该方法可以确保曲线通过两个点并且在两个点之间有一个平滑的拐角。 方法的依据是曲线的切线在控制点处与曲线的切线在两个点的中点处平行,从而使曲线过渡更加平滑。

canvas

参考文章

- +</html>

将两个点的中点作为控制点的选择通常用于创建平滑的曲线,该方法可以确保曲线通过两个点并且在两个点之间有一个平滑的拐角。 方法的依据是曲线的切线在控制点处与曲线的切线在两个点的中点处平行,从而使曲线过渡更加平滑。

canvas

参考文章

+ \ No newline at end of file diff --git "a/guide/css\347\233\270\345\205\263/BFC.html" "b/guide/css\347\233\270\345\205\263/BFC.html" index c2f41f09..ac8a3e91 100644 --- "a/guide/css\347\233\270\345\205\263/BFC.html" +++ "b/guide/css\347\233\270\345\205\263/BFC.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

BFC

块级格式化上下文

条件

  1. 根元素 (html)
  2. 浮动元素
  3. position 为绝对定位元素 (absolute/fixed)
  4. display: inline-block、inline-flex等
  5. overflow: auto/scroll/hidden (overflow 值不为 visible 或 clip 的块级元素)

特点

  1. BFC 内部元素不影响外部元素
  2. 计算高度需要计算浮动元素
  3. BFC 区域不会与浮动容器发生重叠
  4. 在 BFC 中上下相邻的俩个容器的 margin 会重叠
  5. 每个元素的左 margin 值和容器的左 border 相接触

作用

  1. 清除浮动带来的高度塌陷问题
  2. 解决俩个元素的 margin 重叠问题
  3. 创建自适应的俩栏布局
- +
Skip to content

BFC

块级格式化上下文

条件

  1. 根元素 (html)
  2. 浮动元素
  3. position 为绝对定位元素 (absolute/fixed)
  4. display: inline-block、inline-flex等
  5. overflow: auto/scroll/hidden (overflow 值不为 visible 或 clip 的块级元素)

特点

  1. BFC 内部元素不影响外部元素
  2. 计算高度需要计算浮动元素
  3. BFC 区域不会与浮动容器发生重叠
  4. 在 BFC 中上下相邻的俩个容器的 margin 会重叠
  5. 每个元素的左 margin 值和容器的左 border 相接触

作用

  1. 清除浮动带来的高度塌陷问题
  2. 解决俩个元素的 margin 重叠问题
  3. 创建自适应的俩栏布局
+ \ No newline at end of file diff --git "a/guide/css\347\233\270\345\205\263/display.html" "b/guide/css\347\233\270\345\205\263/display.html" index 6f7211f8..30216030 100644 --- "a/guide/css\347\233\270\345\205\263/display.html" +++ "b/guide/css\347\233\270\345\205\263/display.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

display

none

不显示元素,脱离文档流

inline 行内元素

不能设置宽高、不换行

水平 padding、margin 有效

垂直方向 padding、margin 无效

line-block

可以设置宽高、不换行

block

块级元素,可以设置宽高,独占一行

flex

弹性布局

table

元素作为块级表格元素使用

inherit

继承父元素 display 值

常见的问题

line-block、inline 都会有空格问题

因为浏览器会将换行符当空格字符处理

解决方案:

  1. 使用 font-size 时,可通过设置 font-size: 0、letter-spacing、word-spacing 解决
  2. 使用弹性布局
  3. 使用 margin 负值

浏览器对于空格的默认表现

  1. 元素的头尾的空白符会直接忽略
  2. 内容中间有多个空格,会被合并成一个空格
js
<!DOCTYPE html>
+    
Skip to content

display

none

不显示元素,脱离文档流

inline 行内元素

不能设置宽高、不换行

水平 padding、margin 有效

垂直方向 padding、margin 无效

line-block

可以设置宽高、不换行

block

块级元素,可以设置宽高,独占一行

flex

弹性布局

table

元素作为块级表格元素使用

inherit

继承父元素 display 值

常见的问题

line-block、inline 都会有空格问题

因为浏览器会将换行符当空格字符处理

解决方案:

  1. 使用 font-size 时,可通过设置 font-size: 0、letter-spacing、word-spacing 解决
  2. 使用弹性布局
  3. 使用 margin 负值

浏览器对于空格的默认表现

  1. 元素的头尾的空白符会直接忽略
  2. 内容中间有多个空格,会被合并成一个空格
js
<!DOCTYPE html>
 <html lang="en">
 
 <head>
@@ -115,8 +115,8 @@
     <li>元素的头尾的空白符会直接忽略</li>
     <li>内容中间有多个空格,会被合并成一个空格</li>
   </ul>
-</body>
- +</body>
+ \ No newline at end of file diff --git "a/guide/css\347\233\270\345\205\263/display\343\200\201float\343\200\201position\347\232\204\345\205\263\347\263\273.html" "b/guide/css\347\233\270\345\205\263/display\343\200\201float\343\200\201position\347\232\204\345\205\263\347\263\273.html" index d8db87a9..581552ae 100644 --- "a/guide/css\347\233\270\345\205\263/display\343\200\201float\343\200\201position\347\232\204\345\205\263\347\263\273.html" +++ "b/guide/css\347\233\270\345\205\263/display\343\200\201float\343\200\201position\347\232\204\345\205\263\347\263\273.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

display、float、position的关系

  • 如果元素的 display 为 none 直接隐藏
  • 如果元素的 position 为 absolute 或者 fixed 时,float 失效,display 会提升块级元素
  • 如果元素的 position 为 relative, 会在浮动后的元素进行定位,display 会被提升成块级元素
  • 如果是根元素,display 会被提升为块级元素
  • 浮动元素会将 display 提升为块级元素
html
<!DOCTYPE html>
+    
Skip to content

display、float、position的关系

  • 如果元素的 display 为 none 直接隐藏
  • 如果元素的 position 为 absolute 或者 fixed 时,float 失效,display 会提升块级元素
  • 如果元素的 position 为 relative, 会在浮动后的元素进行定位,display 会被提升成块级元素
  • 如果是根元素,display 会被提升为块级元素
  • 浮动元素会将 display 提升为块级元素
html
<!DOCTYPE html>
 <html lang="en">
 
 <head>
@@ -87,8 +87,8 @@
     </div>
 </body>
 
-</html>
- +</html>
+ \ No newline at end of file diff --git "a/guide/css\347\233\270\345\205\263/display\343\200\201visibility\343\200\201opacity\345\214\272\345\210\253.html" "b/guide/css\347\233\270\345\205\263/display\343\200\201visibility\343\200\201opacity\345\214\272\345\210\253.html" index 286196da..318d4549 100644 --- "a/guide/css\347\233\270\345\205\263/display\343\200\201visibility\343\200\201opacity\345\214\272\345\210\253.html" +++ "b/guide/css\347\233\270\345\205\263/display\343\200\201visibility\343\200\201opacity\345\214\272\345\210\253.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

display、visibility、opacity区别

  • 占据空间

    • opacity、visibility 占据空间,不会引起回流,但是会重绘
    • display 不占据空间,但会引起页面的回流和重绘
  • 绑定事件

    • display、visibility 不会触发绑定事件
    • opacity 会触发绑定事件

display: none 和 visibility: hidden 的区别

  • 从渲染树上看
    • display: none 不存在渲染树中
    • visibility: hidden 的元素存在渲染树中,还会占据空间
  • 从继承上看
    • display 不会被继承
    • visibility 会被继承
  • 从渲染上看
    • display 会影响回流重绘
    • visibility 只会引起重绘
- +
Skip to content

display、visibility、opacity区别

  • 占据空间

    • opacity、visibility 占据空间,不会引起回流,但是会重绘
    • display 不占据空间,但会引起页面的回流和重绘
  • 绑定事件

    • display、visibility 不会触发绑定事件
    • opacity 会触发绑定事件

display: none 和 visibility: hidden 的区别

  • 从渲染树上看
    • display: none 不存在渲染树中
    • visibility: hidden 的元素存在渲染树中,还会占据空间
  • 从继承上看
    • display 不会被继承
    • visibility 会被继承
  • 从渲染上看
    • display 会影响回流重绘
    • visibility 只会引起重绘
+ \ No newline at end of file diff --git "a/guide/css\347\233\270\345\205\263/position.html" "b/guide/css\347\233\270\345\205\263/position.html" index 7ff9919c..6adb7913 100644 --- "a/guide/css\347\233\270\345\205\263/position.html" +++ "b/guide/css\347\233\270\345\205\263/position.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

Position

static [ˈstætɪk]

默认值,没有定位,元素出现在正常的文档流中,会忽略 top,bottom,left,right 或者 z-index 声明

relative

生成相对定位元素,相对于其原来位置进行定位,元素的位置通过 left、top、right、bottom 属性规定

relative

元素定位永远相对于元素自身位置,和其他元素没有关系,也不会影响其他元素。

absolute

生成绝对定位的元素,相对于 static 定位以外的一个父元素进行定位

元素位置通过 left、top、right、bottom 属性规定

relative

浏览器会递归查找该元素的所有父元素,如果找到一个设置了非 static 的定位元素,就以该元素为基准定位,如果没有找到,就以浏览器边界定位

但是它具有破坏性,会导致其他元素位置的变化

fixed

生成绝对定位的元素,指定元素相对于屏幕视口的位置来制定元素位置。

relative

但是它具有破坏性,会导致其他元素位置的变化

sticky [ˈstɪki]

相对它的最近滚动祖先(nearest scrolling ancestor)和 containing block (最近块级祖先 nearest block-level ancestor),包括table-related元素,基于top, right, bottom, 和 left的值进行偏移。偏移值不会影响任何其他元素的位置。

inherit

规定从父元素继承 position 属性的值

js
<!DOCTYPE html>
+    
Skip to content

Position

static [ˈstætɪk]

默认值,没有定位,元素出现在正常的文档流中,会忽略 top,bottom,left,right 或者 z-index 声明

relative

生成相对定位元素,相对于其原来位置进行定位,元素的位置通过 left、top、right、bottom 属性规定

relative

元素定位永远相对于元素自身位置,和其他元素没有关系,也不会影响其他元素。

absolute

生成绝对定位的元素,相对于 static 定位以外的一个父元素进行定位

元素位置通过 left、top、right、bottom 属性规定

relative

浏览器会递归查找该元素的所有父元素,如果找到一个设置了非 static 的定位元素,就以该元素为基准定位,如果没有找到,就以浏览器边界定位

但是它具有破坏性,会导致其他元素位置的变化

fixed

生成绝对定位的元素,指定元素相对于屏幕视口的位置来制定元素位置。

relative

但是它具有破坏性,会导致其他元素位置的变化

sticky [ˈstɪki]

相对它的最近滚动祖先(nearest scrolling ancestor)和 containing block (最近块级祖先 nearest block-level ancestor),包括table-related元素,基于top, right, bottom, 和 left的值进行偏移。偏移值不会影响任何其他元素的位置。

inherit

规定从父元素继承 position 属性的值

js
<!DOCTYPE html>
 <html lang="en">
 
 <head>
@@ -135,8 +135,8 @@
     <div class="subbottom"></div>
   </div>
 </body>
-</html>
- +</html>
+ \ No newline at end of file diff --git "a/guide/css\347\233\270\345\205\263/\345\257\271 line-height \347\232\204\347\220\206\350\247\243\345\217\212\345\205\266\350\265\213\345\200\274\346\226\271\345\274\217.html" "b/guide/css\347\233\270\345\205\263/\345\257\271 line-height \347\232\204\347\220\206\350\247\243\345\217\212\345\205\266\350\265\213\345\200\274\346\226\271\345\274\217.html" index 4a3b84be..79cea729 100644 --- "a/guide/css\347\233\270\345\205\263/\345\257\271 line-height \347\232\204\347\220\206\350\247\243\345\217\212\345\205\266\350\265\213\345\200\274\346\226\271\345\274\217.html" +++ "b/guide/css\347\233\270\345\205\263/\345\257\271 line-height \347\232\204\347\220\206\350\247\243\345\217\212\345\205\266\350\265\213\345\200\274\346\226\271\345\274\217.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

对 line-height 的理解及其赋值方式

概念

line-height 指一行文本的高度,包含了字间距,实际上是下一行基线到上一行基线的距离

line-height 和 height 都能撑开元素高度 (如果 line-height 和 height 一致时可以实现单行文字的垂直居中)

计算方式

带单位的px

line-height 为 固定值

纯数字

直接 line-height 会把比例传递给后代

例如,父级行高为 1.5,子元素字体为 18px,则子元素行高为 1.5 * 18 = 27px

百分比

父元素将计算后的值传递给后代

例如,父元素行高为 200%,父元素字体为 18px,那么子元素行高为 18 * 200% = 36px

- +
Skip to content

对 line-height 的理解及其赋值方式

概念

line-height 指一行文本的高度,包含了字间距,实际上是下一行基线到上一行基线的距离

line-height 和 height 都能撑开元素高度 (如果 line-height 和 height 一致时可以实现单行文字的垂直居中)

计算方式

带单位的px

line-height 为 固定值

纯数字

直接 line-height 会把比例传递给后代

例如,父级行高为 1.5,子元素字体为 18px,则子元素行高为 1.5 * 18 = 27px

百分比

父元素将计算后的值传递给后代

例如,父元素行高为 200%,父元素字体为 18px,那么子元素行高为 18 * 200% = 36px

+ \ No newline at end of file diff --git "a/guide/css\347\233\270\345\205\263/\346\246\202\350\246\201.html" "b/guide/css\347\233\270\345\205\263/\346\246\202\350\246\201.html" index 04c6ad13..49a1cd24 100644 --- "a/guide/css\347\233\270\345\205\263/\346\246\202\350\246\201.html" +++ "b/guide/css\347\233\270\345\205\263/\346\246\202\350\246\201.html" @@ -12,7 +12,7 @@ - + @@ -24,8 +24,8 @@ -
Skip to content
- +
Skip to content
+ \ No newline at end of file diff --git "a/guide/css\347\233\270\345\205\263/\351\232\220\350\227\217\345\205\203\347\264\240\347\232\204\346\226\271\346\263\225.html" "b/guide/css\347\233\270\345\205\263/\351\232\220\350\227\217\345\205\203\347\264\240\347\232\204\346\226\271\346\263\225.html" index a271f27a..ca78b625 100644 --- "a/guide/css\347\233\270\345\205\263/\351\232\220\350\227\217\345\205\203\347\264\240\347\232\204\346\226\271\346\263\225.html" +++ "b/guide/css\347\233\270\345\205\263/\351\232\220\350\227\217\345\205\203\347\264\240\347\232\204\346\226\271\346\263\225.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

隐藏元素的方法

    1. display: none - 不会显示在渲染树上,不占据空间,无法监听事件
    1. visibility: hidden - 占据空间,无法监听事件
    1. opacity: 0 - 占据空间,可以监听事件
    1. transform: scale(0, 0)
    1. 绝对定位
    1. z-index 为负数
- +
Skip to content

隐藏元素的方法

    1. display: none - 不会显示在渲染树上,不占据空间,无法监听事件
    1. visibility: hidden - 占据空间,无法监听事件
    1. opacity: 0 - 占据空间,可以监听事件
    1. transform: scale(0, 0)
    1. 绝对定位
    1. z-index 为负数
+ \ No newline at end of file diff --git "a/guide/javaScript\347\233\270\345\205\263/DOM\347\233\270\345\205\263.html" "b/guide/javaScript\347\233\270\345\205\263/DOM\347\233\270\345\205\263.html" index 45163d04..3eee8a67 100644 --- "a/guide/javaScript\347\233\270\345\205\263/DOM\347\233\270\345\205\263.html" +++ "b/guide/javaScript\347\233\270\345\205\263/DOM\347\233\270\345\205\263.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

DOM 相关

如何阻止事件的冒泡和默认事件

js
event.stopPropagation(); // 阻止事件冒泡
+    
Skip to content

DOM 相关

如何阻止事件的冒泡和默认事件

js
event.stopPropagation(); // 阻止事件冒泡
 event.preventDefalut(); // 阻止默认事件
 event.stopImmediatePropagation(); // 阻止监听同一事件的其他事件监听器被调用
event.stopPropagation(); // 阻止事件冒泡
 event.preventDefalut(); // 阻止默认事件
@@ -39,8 +39,8 @@
 
 var imoocList = document.getElementByClassName('imooc'); // 查询类名 imooc 类名集合
 
-var imoocList = docuemnt.querySelectorAll('.imooc'); // 查询类名 imooc 的集合

高度相关

  • offsetHeight: 包含 padding、content、border 和 滚动条
  • clientHeight: 包含 padding 和 content
  • scrollHeight: 包含 padding 和 内容高度

判断是否有滚动条 scrollHeight >= clientHeight

顶部高度

  • offsetTop: 元素顶部到最近父元素顶部的距离,和有木有滚动条没有关系
  • scrollTop: 滚动条滚动的距离

鼠标事件

按下

  • click: 点击鼠标左键或按下回车键调用
  • dblclick: 双击鼠标左键调用
  • mousedown: 鼠标被按下(左键/右键)时触发,不能通过键盘触发。
  • mouseup: 鼠标按键被松开时触发,不能通过键盘触发。

移动

  • mouseover: 鼠标移入目标元素。鼠标移入子元素也会触发
  • mouseout: 鼠标移出目标元素。鼠标移出子元素也会触发
  • mouseenter: 鼠标移入目标元素。鼠标移入子元素不会触发
  • mouseleave: 鼠标移出目标元素。鼠标移出子元素不会触发
  • mousemove: 鼠标在目标元素内移动时触发,不能通过键盘触发。
- +var imoocList = docuemnt.querySelectorAll('.imooc'); // 查询类名 imooc 的集合

高度相关

  • offsetHeight: 包含 padding、content、border 和 滚动条
  • clientHeight: 包含 padding 和 content
  • scrollHeight: 包含 padding 和 内容高度

判断是否有滚动条 scrollHeight >= clientHeight

顶部高度

  • offsetTop: 元素顶部到最近父元素顶部的距离,和有木有滚动条没有关系
  • scrollTop: 滚动条滚动的距离

鼠标事件

按下

  • click: 点击鼠标左键或按下回车键调用
  • dblclick: 双击鼠标左键调用
  • mousedown: 鼠标被按下(左键/右键)时触发,不能通过键盘触发。
  • mouseup: 鼠标按键被松开时触发,不能通过键盘触发。

移动

  • mouseover: 鼠标移入目标元素。鼠标移入子元素也会触发
  • mouseout: 鼠标移出目标元素。鼠标移出子元素也会触发
  • mouseenter: 鼠标移入目标元素。鼠标移入子元素不会触发
  • mouseleave: 鼠标移出目标元素。鼠标移出子元素不会触发
  • mousemove: 鼠标在目标元素内移动时触发,不能通过键盘触发。
+ \ No newline at end of file diff --git "a/guide/javaScript\347\233\270\345\205\263/ES6\347\233\270\345\205\263.html" "b/guide/javaScript\347\233\270\345\205\263/ES6\347\233\270\345\205\263.html" index 1c1608ca..5c1bac7c 100644 --- "a/guide/javaScript\347\233\270\345\205\263/ES6\347\233\270\345\205\263.html" +++ "b/guide/javaScript\347\233\270\345\205\263/ES6\347\233\270\345\205\263.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

ES6相关

var、let、const 的区别

1 作用域 let、const 存在块级作用域 var 不存在块级作用域

2 const 常量 定义时赋值

3 变量提升 let/const 声明的变量需要声明后使用 var 声明的变量可以声明前使用

4 重复声明 let/const 不能重复声明 var 可以重复声明

5 暂时性死区 let/const 声明的变量不能在声明前使用

暂时性死区

箭头函数和普通函数的区别

1 箭头函数没有 arguments

2 箭头函数没有自己的 this this 由定义的所在父级上下文决定

3 箭头函数继承来的 this 不会改变 call、apply、bind 方法不能改变箭头函数的 this 指向

js
var id = 'global';
+    
Skip to content

ES6相关

var、let、const 的区别

1 作用域 let、const 存在块级作用域 var 不存在块级作用域

2 const 常量 定义时赋值

3 变量提升 let/const 声明的变量需要声明后使用 var 声明的变量可以声明前使用

4 重复声明 let/const 不能重复声明 var 可以重复声明

5 暂时性死区 let/const 声明的变量不能在声明前使用

暂时性死区

箭头函数和普通函数的区别

1 箭头函数没有 arguments

2 箭头函数没有自己的 this this 由定义的所在父级上下文决定

3 箭头函数继承来的 this 不会改变 call、apply、bind 方法不能改变箭头函数的 this 指向

js
var id = 'global';
 var obj = {
     id: 'obj',
     a: function () {
@@ -123,8 +123,8 @@
 let test = onWatch(obj);
 
 test.a // getting a!
-test.a = 2 // setting a!
- +test.a = 2 // setting a!
+ \ No newline at end of file diff --git "a/guide/javaScript\347\233\270\345\205\263/JS\345\274\202\345\270\270\346\215\225\350\216\267\346\234\272\345\210\266.html" "b/guide/javaScript\347\233\270\345\205\263/JS\345\274\202\345\270\270\346\215\225\350\216\267\346\234\272\345\210\266.html" index 5c3c2dd9..32cd526b 100644 --- "a/guide/javaScript\347\233\270\345\205\263/JS\345\274\202\345\270\270\346\215\225\350\216\267\346\234\272\345\210\266.html" +++ "b/guide/javaScript\347\233\270\345\205\263/JS\345\274\202\345\270\270\346\215\225\350\216\267\346\234\272\345\210\266.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content
- +
Skip to content
+ \ No newline at end of file diff --git "a/guide/javaScript\347\233\270\345\205\263/JavaScript\346\211\247\350\241\214\346\234\272\345\210\266.html" "b/guide/javaScript\347\233\270\345\205\263/JavaScript\346\211\247\350\241\214\346\234\272\345\210\266.html" index ad70f6e0..314d38ff 100644 --- "a/guide/javaScript\347\233\270\345\205\263/JavaScript\346\211\247\350\241\214\346\234\272\345\210\266.html" +++ "b/guide/javaScript\347\233\270\345\205\263/JavaScript\346\211\247\350\241\214\346\234\272\345\210\266.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

JavaScript 执行机制

先编译后执行

编译 -> 创建上下文、创建变量环境、创建词法环境 -> 将上下文压入执行调用栈执行代码

- +
Skip to content

JavaScript 执行机制

先编译后执行

编译 -> 创建上下文、创建变量环境、创建词法环境 -> 将上下文压入执行调用栈执行代码

+ \ No newline at end of file diff --git "a/guide/javaScript\347\233\270\345\205\263/JavaScript\347\274\226\350\257\221\346\234\272\345\210\266.html" "b/guide/javaScript\347\233\270\345\205\263/JavaScript\347\274\226\350\257\221\346\234\272\345\210\266.html" index f37c6a40..288011b3 100644 --- "a/guide/javaScript\347\233\270\345\205\263/JavaScript\347\274\226\350\257\221\346\234\272\345\210\266.html" +++ "b/guide/javaScript\347\233\270\345\205\263/JavaScript\347\274\226\350\257\221\346\234\272\345\210\266.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

JavaScript 编译机制

code -> 词法分析、语法分析 -> AST -> 生成字节码 -> 解释执行

- +
Skip to content

JavaScript 编译机制

code -> 词法分析、语法分析 -> AST -> 生成字节码 -> 解释执行

+ \ No newline at end of file diff --git "a/guide/javaScript\347\233\270\345\205\263/Promise.html" "b/guide/javaScript\347\233\270\345\205\263/Promise.html" index bc5e4470..79de6896 100644 --- "a/guide/javaScript\347\233\270\345\205\263/Promise.html" +++ "b/guide/javaScript\347\233\270\345\205\263/Promise.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

Promise 相关

定义

Promise 有 3 种状态:

  • 等待状态 pending
  • 成功状态 resolved
  • 失败状态 rejected

当 Promise 状态是 Pending 时,可以转换为 Resolved 或者 Rejected 状态,一旦状态转换,就不会再改变。

resolved、rejected 是异步函数

then 默认返回一个 Promise 对象,第一个参数是 resolved 的回调函数,第二个参数是 rejected 的回调函数 catch 会返回一个 Promise 对象,catch 正常返回 resolved 状态,里面报错返回 rejected 状态

Promise.resolve

Promise.resolve(value) 返回一个成功状态的 promise 对象

js
/*
+    
Skip to content

Promise 相关

定义

Promise 有 3 种状态:

  • 等待状态 pending
  • 成功状态 resolved
  • 失败状态 rejected

当 Promise 状态是 Pending 时,可以转换为 Resolved 或者 Rejected 状态,一旦状态转换,就不会再改变。

resolved、rejected 是异步函数

then 默认返回一个 Promise 对象,第一个参数是 resolved 的回调函数,第二个参数是 rejected 的回调函数 catch 会返回一个 Promise 对象,catch 正常返回 resolved 状态,里面报错返回 rejected 状态

Promise.resolve

Promise.resolve(value) 返回一个成功状态的 promise 对象

js
/*
   如果 .then 函数参数不是一个函数,那么会将其封装成 v => v 函数, 这里的 v 是上一个 resolve 的返回值
 */
 
@@ -81,8 +81,8 @@
             })
         }
     })
-}

Promise.any

参数是一个数组,返回第一个成功的值

Promise.finally

不管 Promise 返回什么状态都会执行

常见的问题

Promise 解决了什么问题

  • 解决了回调地狱的问题
- +}

Promise.any

参数是一个数组,返回第一个成功的值

Promise.finally

不管 Promise 返回什么状态都会执行

常见的问题

Promise 解决了什么问题

  • 解决了回调地狱的问题
+ \ No newline at end of file diff --git "a/guide/javaScript\347\233\270\345\205\263/index.html" "b/guide/javaScript\347\233\270\345\205\263/index.html" index 5702a0b1..1daed8dc 100644 --- "a/guide/javaScript\347\233\270\345\205\263/index.html" +++ "b/guide/javaScript\347\233\270\345\205\263/index.html" @@ -12,7 +12,7 @@ - + @@ -24,8 +24,8 @@ -
Skip to content
- +
Skip to content
+ \ No newline at end of file diff --git "a/guide/javaScript\347\233\270\345\205\263/\344\272\213\344\273\266\345\276\252\347\216\257\346\234\272\345\210\266.html" "b/guide/javaScript\347\233\270\345\205\263/\344\272\213\344\273\266\345\276\252\347\216\257\346\234\272\345\210\266.html" index aa4ca9ab..19f87d2b 100644 --- "a/guide/javaScript\347\233\270\345\205\263/\344\272\213\344\273\266\345\276\252\347\216\257\346\234\272\345\210\266.html" +++ "b/guide/javaScript\347\233\270\345\205\263/\344\272\213\344\273\266\345\276\252\347\216\257\346\234\272\345\210\266.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

事件循环机制

执行 js 代码时,会将代码分成同步代码和异步代码, 同步任务会进入主线程执行,异步任务又分成宏任务和微任务 宏任务会进入宏任务队列,微任务会进入微任务队列 当主线程的代码执行完后,会查看是否有微任务队列,有则执行,无则进入下一个宏任务

- +
Skip to content

事件循环机制

执行 js 代码时,会将代码分成同步代码和异步代码, 同步任务会进入主线程执行,异步任务又分成宏任务和微任务 宏任务会进入宏任务队列,微任务会进入微任务队列 当主线程的代码执行完后,会查看是否有微任务队列,有则执行,无则进入下一个宏任务

+ \ No newline at end of file diff --git "a/guide/javaScript\347\233\270\345\205\263/\345\236\203\345\234\276\345\233\236\346\224\266\346\234\272\345\210\266.html" "b/guide/javaScript\347\233\270\345\205\263/\345\236\203\345\234\276\345\233\236\346\224\266\346\234\272\345\210\266.html" index b694ac01..55f6c99a 100644 --- "a/guide/javaScript\347\233\270\345\205\263/\345\236\203\345\234\276\345\233\236\346\224\266\346\234\272\345\210\266.html" +++ "b/guide/javaScript\347\233\270\345\205\263/\345\236\203\345\234\276\345\233\236\346\224\266\346\234\272\345\210\266.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

垃圾回收机制

在 JS 的垃圾回收机制中,分成新生代和老生代 新生代采用的是 GC 算法,将新生代划分成活动对象区域和空闲区域,例如声明的对象会被放入活动对象区域,当活动对象区域填满时, 会进行一次垃圾回收,将存活的对象移入空闲区域并进行碎片空间整理,完成后再将空闲区域和活动对象区域进行翻转,依次循环回收。

当进行 2次 垃圾回收后,存活的对象会被放入老生代中,

老生代采用的是标记清除和标记整理算法,递归遍历堆内存的变量,将不需要使用的变量标记为垃圾数据,将需要使用的变量标记成活动变量,清除掉垃圾,然后进行标记整理碎片空间。

- +
Skip to content

垃圾回收机制

在 JS 的垃圾回收机制中,分成新生代和老生代 新生代采用的是 GC 算法,将新生代划分成活动对象区域和空闲区域,例如声明的对象会被放入活动对象区域,当活动对象区域填满时, 会进行一次垃圾回收,将存活的对象移入空闲区域并进行碎片空间整理,完成后再将空闲区域和活动对象区域进行翻转,依次循环回收。

当进行 2次 垃圾回收后,存活的对象会被放入老生代中,

老生代采用的是标记清除和标记整理算法,递归遍历堆内存的变量,将不需要使用的变量标记为垃圾数据,将需要使用的变量标记成活动变量,清除掉垃圾,然后进行标记整理碎片空间。

+ \ No newline at end of file diff --git "a/guide/javaScript\347\233\270\345\205\263/\345\237\272\347\241\200\346\246\202\345\277\265.html" "b/guide/javaScript\347\233\270\345\205\263/\345\237\272\347\241\200\346\246\202\345\277\265.html" index 5bdc7875..a6a3f280 100644 --- "a/guide/javaScript\347\233\270\345\205\263/\345\237\272\347\241\200\346\246\202\345\277\265.html" +++ "b/guide/javaScript\347\233\270\345\205\263/\345\237\272\347\241\200\346\246\202\345\277\265.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

3 基础概念

3.1 闭包

执行外部函数返回内部函数后,虽然外部函数已经弹出调用栈了,但是内部函数对外部函数变量的引用依然保存在内存中,那么内部函数和这些变量的集合就叫做闭包。

闭包的好处: 1. 私有变量在内存中持久化

闭包的坏处: 1. 使用不当会造成内存泄漏

3.2 原型

在 js 中对象是由构造函数创建的,构造函数中会有一个 prototype 属性,指向一个对象,这个对象包含了由该构造函数创建的实例所共享的属性和方法, 由该构造函数创建的实例可以通过 proto 属性指向这个对象,这个对象就是我们所说的原型。

当想要访问对象的某个属性时,如果在当前对象查找不到,就会往该对象的原型查找,对象的原型也会有属于他的原型对象,如此循环,直到 null 停止,这就是我们所说的原型链

隐式原型: proto 显示原型: prototype

相关方法: 1. hasOwnProperty() 判断属性是否是实例自身的属性 2. Object.getPrototypeOf() 获取实例的原型

3.3 作用域

作用域分成全局作用域函数作用域块级作用域,它标识着一个变量是否合法 (编译过程中就已经确认了)

当查询一个变量时,如果当前作用域查询不到,会往上一级作用域查找,如此循环,直到全局作用域,这就是我们所说的作用域链

3.4 执行上下文

从类型上看

  1. 全局执行上下文
  2. 函数执行上下文
  3. eval 执行上下文

从生命周期上看

  1. 创建阶段
    • this 绑定
    • 创建词法环境 (let、const会被提升到词法环境)
    • 创建变量环境 (var 声明的变量会被提升到变量环境)
  2. 执行阶段
    • 对变量进行赋值,执行代码
  3. 回收阶段
    • 当执行上下文弹出调用站后,会对上下文进行回收

执行上下文栈: 当 JS 执行代码时,首先遇到全局代码,会创建一个全局执行上下文并压入执行栈中,当遇到函数调用时,就会为该函数创建一个新的函数执行上下文压入栈中, 引擎会执行位于执行上下文栈顶的函数,当函数执行完后,执行上下文会从栈中弹出,继续执行下一个上下文,当所有代码都执行完毕后,从栈中弹出全局执行上下文

执行上下文

3.5 类数组对象

一个拥有 length 属性和若干索引属性的对象就是类数组对象。

将类数组对象转化成数组的方法:

1 Array.from(arrayLike)
+    
Skip to content

3 基础概念

3.1 闭包

执行外部函数返回内部函数后,虽然外部函数已经弹出调用栈了,但是内部函数对外部函数变量的引用依然保存在内存中,那么内部函数和这些变量的集合就叫做闭包。

闭包的好处: 1. 私有变量在内存中持久化

闭包的坏处: 1. 使用不当会造成内存泄漏

3.2 原型

在 js 中对象是由构造函数创建的,构造函数中会有一个 prototype 属性,指向一个对象,这个对象包含了由该构造函数创建的实例所共享的属性和方法, 由该构造函数创建的实例可以通过 proto 属性指向这个对象,这个对象就是我们所说的原型。

当想要访问对象的某个属性时,如果在当前对象查找不到,就会往该对象的原型查找,对象的原型也会有属于他的原型对象,如此循环,直到 null 停止,这就是我们所说的原型链

隐式原型: proto 显示原型: prototype

相关方法: 1. hasOwnProperty() 判断属性是否是实例自身的属性 2. Object.getPrototypeOf() 获取实例的原型

3.3 作用域

作用域分成全局作用域函数作用域块级作用域,它标识着一个变量是否合法 (编译过程中就已经确认了)

当查询一个变量时,如果当前作用域查询不到,会往上一级作用域查找,如此循环,直到全局作用域,这就是我们所说的作用域链

3.4 执行上下文

从类型上看

  1. 全局执行上下文
  2. 函数执行上下文
  3. eval 执行上下文

从生命周期上看

  1. 创建阶段
    • this 绑定
    • 创建词法环境 (let、const会被提升到词法环境)
    • 创建变量环境 (var 声明的变量会被提升到变量环境)
  2. 执行阶段
    • 对变量进行赋值,执行代码
  3. 回收阶段
    • 当执行上下文弹出调用站后,会对上下文进行回收

执行上下文栈: 当 JS 执行代码时,首先遇到全局代码,会创建一个全局执行上下文并压入执行栈中,当遇到函数调用时,就会为该函数创建一个新的函数执行上下文压入栈中, 引擎会执行位于执行上下文栈顶的函数,当函数执行完后,执行上下文会从栈中弹出,继续执行下一个上下文,当所有代码都执行完毕后,从栈中弹出全局执行上下文

执行上下文

3.5 类数组对象

一个拥有 length 属性和若干索引属性的对象就是类数组对象。

将类数组对象转化成数组的方法:

1 Array.from(arrayLike)
 
 2 Array.prototype.slice.call(arrayLike)
 
@@ -60,8 +60,8 @@
     }
 }
 
-fn(); // undefined
- +fn(); // undefined
+ \ No newline at end of file diff --git "a/guide/javaScript\347\233\270\345\205\263/\345\255\227\347\254\246\344\270\262\345\270\270\350\247\201\347\232\204API.html" "b/guide/javaScript\347\233\270\345\205\263/\345\255\227\347\254\246\344\270\262\345\270\270\350\247\201\347\232\204API.html" index f598d8bd..e9379b20 100644 --- "a/guide/javaScript\347\233\270\345\205\263/\345\255\227\347\254\246\344\270\262\345\270\270\350\247\201\347\232\204API.html" +++ "b/guide/javaScript\347\233\270\345\205\263/\345\255\227\347\254\246\344\270\262\345\270\270\350\247\201\347\232\204API.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content
- +
Skip to content
+ \ No newline at end of file diff --git "a/guide/javaScript\347\233\270\345\205\263/\345\256\232\344\271\211.html" "b/guide/javaScript\347\233\270\345\205\263/\345\256\232\344\271\211.html" index 8d9a7a56..e3000856 100644 --- "a/guide/javaScript\347\233\270\345\205\263/\345\256\232\344\271\211.html" +++ "b/guide/javaScript\347\233\270\345\205\263/\345\256\232\344\271\211.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

1 定义

JavaScript 是一门动态语言,可以不需要定义变量类型

JavaScript 是一门弱类型语言,存在隐式类型转化

JavaScript 是一门解释型语言,在运行程序时动态编译

解释型语言特点: 1 每次运行都需要将源代码转化成机器码并执行,效率低 2 只要平台提供相应的解析器,就可以运行源代码,跨平台

V8 执行一段代码的流程图 流程

- +
Skip to content

1 定义

JavaScript 是一门动态语言,可以不需要定义变量类型

JavaScript 是一门弱类型语言,存在隐式类型转化

JavaScript 是一门解释型语言,在运行程序时动态编译

解释型语言特点: 1 每次运行都需要将源代码转化成机器码并执行,效率低 2 只要平台提供相应的解析器,就可以运行源代码,跨平台

V8 执行一段代码的流程图 流程

+ \ No newline at end of file diff --git "a/guide/javaScript\347\233\270\345\205\263/\345\257\271\350\261\241\345\270\270\350\247\201\347\232\204API.html" "b/guide/javaScript\347\233\270\345\205\263/\345\257\271\350\261\241\345\270\270\350\247\201\347\232\204API.html" index e0aac19e..6a10787b 100644 --- "a/guide/javaScript\347\233\270\345\205\263/\345\257\271\350\261\241\345\270\270\350\247\201\347\232\204API.html" +++ "b/guide/javaScript\347\233\270\345\205\263/\345\257\271\350\261\241\345\270\270\350\247\201\347\232\204API.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content
- +
Skip to content
+ \ No newline at end of file diff --git "a/guide/javaScript\347\233\270\345\205\263/\346\225\260\346\215\256\347\261\273\345\236\213.html" "b/guide/javaScript\347\233\270\345\205\263/\346\225\260\346\215\256\347\261\273\345\236\213.html" index b6f43698..b4257864 100644 --- "a/guide/javaScript\347\233\270\345\205\263/\346\225\260\346\215\256\347\261\273\345\236\213.html" +++ "b/guide/javaScript\347\233\270\345\205\263/\346\225\260\346\215\256\347\261\273\345\236\213.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

2 数据类型

2.1 数据类型

一共有 7 种基本数据类型

  • 1 Number - 基于 IEEE 754 标准实现,采用双精度 64 位二进制格式, -(2 ^ 63 - 1) ~ 2 ^ 63 - 1
  • 2 Boolean - 只有 true 和 false
  • 3 Undefined - 没有定义的值
  • 4 Null - 空值
  • 5 String - 字符串
  • 6 Symbol - 唯一不可修改的值
  • 7 BigInt - 大整数类型

一种引用类型

  • Object - 引用类型,存储在堆中

2.2 类型检测

2.2.1 typeof

判断基础数据类型,除了 null 也可以判断 function

js
typeof (function() {}) // function
+    
Skip to content

2 数据类型

2.1 数据类型

一共有 7 种基本数据类型

  • 1 Number - 基于 IEEE 754 标准实现,采用双精度 64 位二进制格式, -(2 ^ 63 - 1) ~ 2 ^ 63 - 1
  • 2 Boolean - 只有 true 和 false
  • 3 Undefined - 没有定义的值
  • 4 Null - 空值
  • 5 String - 字符串
  • 6 Symbol - 唯一不可修改的值
  • 7 BigInt - 大整数类型

一种引用类型

  • Object - 引用类型,存储在堆中

2.2 类型检测

2.2.1 typeof

判断基础数据类型,除了 null 也可以判断 function

js
typeof (function() {}) // function
 typeof (() => {}) // function
typeof (function() {}) // function
 typeof (() => {}) // function

但数组、对象、null 都会返回 object

js
typeof [] // object
 typeof {} // object
@@ -117,8 +117,8 @@
 a.length; // 3
 a.toUpperCase(); // "ABC"

3 undefined 和 null 的区别

undefined 表示未定义,一般变量声明了但还没有定义的时候返回 undefined null 表示空值,常用在对象初始化的场景

区别如下:

  • typeof null 返回的是 ‘object’, 因为在早期 javaScript 设计时,类型通过 1~3位标识进行存储,对象和null一样都是 000
  • typeof undefined 返回的是 ‘undefined’
  • undefined 不是保留字,可以被赋值,通常我们会用 void 0 表示 undefined,null 是保留字,不能被赋值

4 isNaN 和 Number.isNaN 的区别

  • isNaN 不会判断是否为数字,会先将参数转化为数字,再判断是否为 NaN
  • Number.isNaN 会判断是否是数字,是数字后才会判断是否是 NaN
js
isNaN('a'); // true
 Number.isNaN('a'); // false
isNaN('a'); // true
-Number.isNaN('a'); // false
- +Number.isNaN('a'); // false
+ \ No newline at end of file diff --git "a/guide/javaScript\347\233\270\345\205\263/\346\225\260\347\273\204\345\270\270\350\247\201\347\232\204API.html" "b/guide/javaScript\347\233\270\345\205\263/\346\225\260\347\273\204\345\270\270\350\247\201\347\232\204API.html" index 67680ef0..0087f151 100644 --- "a/guide/javaScript\347\233\270\345\205\263/\346\225\260\347\273\204\345\270\270\350\247\201\347\232\204API.html" +++ "b/guide/javaScript\347\233\270\345\205\263/\346\225\260\347\273\204\345\270\270\350\247\201\347\232\204API.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content
- +
Skip to content
+ \ No newline at end of file diff --git "a/guide/javaScript\347\233\270\345\205\263/\346\226\260\347\232\204\350\277\220\347\256\227\347\254\246.html" "b/guide/javaScript\347\233\270\345\205\263/\346\226\260\347\232\204\350\277\220\347\256\227\347\254\246.html" index 67f98996..81341dc5 100644 --- "a/guide/javaScript\347\233\270\345\205\263/\346\226\260\347\232\204\350\277\220\347\256\227\347\254\246.html" +++ "b/guide/javaScript\347\233\270\345\205\263/\346\226\260\347\232\204\350\277\220\347\256\227\347\254\246.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content
- +
Skip to content
+ \ No newline at end of file diff --git "a/guide/javaScript\347\233\270\345\205\263/\346\255\243\345\210\231\350\241\250\350\276\276\345\274\217.html" "b/guide/javaScript\347\233\270\345\205\263/\346\255\243\345\210\231\350\241\250\350\276\276\345\274\217.html" index ec5ef528..0e4caaa6 100644 --- "a/guide/javaScript\347\233\270\345\205\263/\346\255\243\345\210\231\350\241\250\350\276\276\345\274\217.html" +++ "b/guide/javaScript\347\233\270\345\205\263/\346\255\243\345\210\231\350\241\250\350\276\276\345\274\217.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content
- +
Skip to content
+ \ No newline at end of file diff --git "a/guide/javaScript\347\233\270\345\205\263/\347\273\247\346\211\277\347\233\270\345\205\263.html" "b/guide/javaScript\347\233\270\345\205\263/\347\273\247\346\211\277\347\233\270\345\205\263.html" index 982a890b..fe5e1ba8 100644 --- "a/guide/javaScript\347\233\270\345\205\263/\347\273\247\346\211\277\347\233\270\345\205\263.html" +++ "b/guide/javaScript\347\233\270\345\205\263/\347\273\247\346\211\277\347\233\270\345\205\263.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

继承

原型链继承

子类的原型对象是父类的实例对象

优点:

  • 简单易懂 缺点:
  • 1 实例化子类时,没办法向父类的构造函数传值
  • 2 所有子类实例共享一个父类,当某个子类修改了父类的属性时,其他子类也会受到影响
js
function Parent() {
+    
Skip to content

继承

原型链继承

子类的原型对象是父类的实例对象

优点:

  • 简单易懂 缺点:
  • 1 实例化子类时,没办法向父类的构造函数传值
  • 2 所有子类实例共享一个父类,当某个子类修改了父类的属性时,其他子类也会受到影响
js
function Parent() {
     this.name = 'parent';
 }
 
@@ -163,8 +163,8 @@
 }
 
 const children = new Children();
-children.getName(); // parent
- +children.getName(); // parent
+ \ No newline at end of file diff --git a/guide/webpack/Loader.html b/guide/webpack/Loader.html index 9c7209a8..c47542a5 100644 --- a/guide/webpack/Loader.html +++ b/guide/webpack/Loader.html @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

Loader

loader 是一个加载器,让 webpack 拥有加载和解析非 JavaScript 文件的能力

类型

  1. pre - 前置
  2. normal - 普通
  3. inline - 行内
  4. post - 后置
js
// 定义在 require 请求内部的叫做行内 loader
+    
Skip to content

Loader

loader 是一个加载器,让 webpack 拥有加载和解析非 JavaScript 文件的能力

类型

  1. pre - 前置
  2. normal - 普通
  3. inline - 行内
  4. post - 后置
js
// 定义在 require 请求内部的叫做行内 loader
 const a = require('inline1-loader!inline2-loader!a.js');
// 定义在 require 请求内部的叫做行内 loader
 const a = require('inline1-loader!inline2-loader!a.js');
  • 如果是 ! 作为前缀,将禁用 normal loader,例如 !inline1-loader!inline2-loader!a.js
  • 如果是 !! 作为前缀,将禁用所有 loader,例如 !!inline1-loader!inline2-loader!a.js
  • 如果是 -! 作为前缀,将禁用所有 loader 但不包含 post loader,例如 -!inline1-loader!inline2-loader!a.js

pitch

一个 loader 中存在 normal (必须) 和 pitch (可选)

先从左往右执行 loader 的 pitch 方法,在从右往左执行 loader 的默认方法

js
function preLoader(source) {
   console.log("pre1");
@@ -97,8 +97,8 @@
   console.log(`post1 pitch`);
 };
 
-// post1 pitch -> inline1 pitch -> normal1 pitch -> pre1 pitch -> pre1 -> normal1 -> inline1 -> post1

pitch loader

⚠️注意: 某一个 loader 的 pitch 方法返回了值,那么直接跳到默认方法阶段

例如 loader2 的 pitch 返回了值,则直接跳过当前 loader 后续的阶段 pitch loader

- +// post1 pitch -> inline1 pitch -> normal1 pitch -> pre1 pitch -> pre1 -> normal1 -> inline1 -> post1

pitch loader

⚠️注意: 某一个 loader 的 pitch 方法返回了值,那么直接跳到默认方法阶段

例如 loader2 的 pitch 返回了值,则直接跳过当前 loader 后续的阶段 pitch loader

+ \ No newline at end of file diff --git a/guide/webpack/index.html b/guide/webpack/index.html index 06789b35..dc0538db 100644 --- a/guide/webpack/index.html +++ b/guide/webpack/index.html @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content
- +
Skip to content
+ \ No newline at end of file diff --git a/guide/webpack/mini-webpack.html b/guide/webpack/mini-webpack.html index 2deb4a5f..b4204ca7 100644 --- a/guide/webpack/mini-webpack.html +++ b/guide/webpack/mini-webpack.html @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content
- +
Skip to content
+ \ No newline at end of file diff --git "a/guide/webpack/\346\236\204\345\273\272\346\265\201\347\250\213.html" "b/guide/webpack/\346\236\204\345\273\272\346\265\201\347\250\213.html" index f056ec75..8d33f353 100644 --- "a/guide/webpack/\346\236\204\345\273\272\346\265\201\347\250\213.html" +++ "b/guide/webpack/\346\236\204\345\273\272\346\265\201\347\250\213.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

构建流程

工作流程

webpack 的工作流程是一个串行的过程,从启动到结束会依次执行以下流程:

  1. 初始化参数:从配置文件和 Shell 语句中读取与合并参数,得出最终的参数;
  2. 开始编译:用上一步得到的参数初始化 Compiler 对象,加载所有配置的插件,执行对象的 run 方法开始执行编译;
  3. 确定入口:根据配置中的 entry 确定入口文件;
  4. 编译模块:从入口文件出发,调用所有配置的 Loader 对模块进行编译,再找出该模块依赖的模块,得到了每个模块被编译后的 最终内容 以及他们之间的 依赖关系图
  5. 输出资源:根据入口和模块之间的依赖关系,组装成一个个包含多个模块的 Chunk,再把每个 Chunk 转换成一个单独的文件加入到输出列表,这步是可以修改输出内容的最后机会;
  6. 输出完成:在确定好输出内容后,根据配置确定输出的路径和文件名,把文件内容写入到文件系统。

常见的问题

Module、Chunk、Bundle 的区别

module: 模块,一个模块就是一个文件

chunk: 代码块,一个 chunk 可以由多个模块组合而成

bundle: webpack 打包出来的文件

Loader 和 Plugin

loader 是一个加载器,让 webpack 拥有加载和解析非 JavaScript 文件的能力

Plugin 是一个插件,是对 webpack 功能的扩展,让 webpack 具有更高的灵活性

Compiler 和 Compilation 的区别

Compiler 对象在 webpack 启动时候被实例化,可以访问当前运行的 webpack 配置,包括 entry、output、loader 等配置,它是全局唯一的。

Plugin 的 apply 方法会传入一个 Compiler 对象,通过这个 Compiler 对象可以注册各种钩子函数,执行插件任务,也可以通过该对象获取配置信息。

js
Compiler.plugin('emit', function(compilation, callback) {
+    
Skip to content

构建流程

工作流程

webpack 的工作流程是一个串行的过程,从启动到结束会依次执行以下流程:

  1. 初始化参数:从配置文件和 Shell 语句中读取与合并参数,得出最终的参数;
  2. 开始编译:用上一步得到的参数初始化 Compiler 对象,加载所有配置的插件,执行对象的 run 方法开始执行编译;
  3. 确定入口:根据配置中的 entry 确定入口文件;
  4. 编译模块:从入口文件出发,调用所有配置的 Loader 对模块进行编译,再找出该模块依赖的模块,得到了每个模块被编译后的 最终内容 以及他们之间的 依赖关系图
  5. 输出资源:根据入口和模块之间的依赖关系,组装成一个个包含多个模块的 Chunk,再把每个 Chunk 转换成一个单独的文件加入到输出列表,这步是可以修改输出内容的最后机会;
  6. 输出完成:在确定好输出内容后,根据配置确定输出的路径和文件名,把文件内容写入到文件系统。

常见的问题

Module、Chunk、Bundle 的区别

module: 模块,一个模块就是一个文件

chunk: 代码块,一个 chunk 可以由多个模块组合而成

bundle: webpack 打包出来的文件

Loader 和 Plugin

loader 是一个加载器,让 webpack 拥有加载和解析非 JavaScript 文件的能力

Plugin 是一个插件,是对 webpack 功能的扩展,让 webpack 具有更高的灵活性

Compiler 和 Compilation 的区别

Compiler 对象在 webpack 启动时候被实例化,可以访问当前运行的 webpack 配置,包括 entry、output、loader 等配置,它是全局唯一的。

Plugin 的 apply 方法会传入一个 Compiler 对象,通过这个 Compiler 对象可以注册各种钩子函数,执行插件任务,也可以通过该对象获取配置信息。

js
Compiler.plugin('emit', function(compilation, callback) {
     // emit 是异步 hook, 在生成之前输出到目录之前调用
  });
Compiler.plugin('emit', function(compilation, callback) {
     // emit 是异步 hook, 在生成之前输出到目录之前调用
@@ -47,8 +47,8 @@
       config: [__filename], // 当配置修改时,缓存失效
     },
   },
-};

4 sideEffects

  • webpack4 需要配置 sideEffects: false
  • webpack5 可以根据源代码静态分析,自动将模块标记成无副作用。

5 支持 import 加载异步模块

6 支持模块联邦

参考文章

- +};

4 sideEffects

  • webpack4 需要配置 sideEffects: false
  • webpack5 可以根据源代码静态分析,自动将模块标记成无副作用。

5 支持 import 加载异步模块

6 支持模块联邦

参考文章

+ \ No newline at end of file diff --git "a/guide/webpack/\346\250\241\345\235\227\350\201\224\351\202\246.html" "b/guide/webpack/\346\250\241\345\235\227\350\201\224\351\202\246.html" index a3a4a973..77421170 100644 --- "a/guide/webpack/\346\250\241\345\235\227\350\201\224\351\202\246.html" +++ "b/guide/webpack/\346\250\241\345\235\227\350\201\224\351\202\246.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

模块联邦

目的

更好的复用应用块或者库

每个应用块都是独立构建,被称之为容器

Module Federation

一个被引用的容器被称之为 remote

引用者被称之为 host

配置参数

字段类型含义
namestring必传值,即输出的模块名,被远程引用时的路径为 ${name}/$
libraryobject声明全局变量的方式,name为 umd 的name
filenamestring构建输出的文件名称
remotesobject(host 使用)远程引用的应用名及其别名的映射,使用时以 key 值作为 name
exposesobject(remote 使用)被引用时可暴露的资源路径以其别名
sharedobject与其他应用之间可以共享的第三方依赖,使你的代码中不用重复加载同一份依赖

remote 配置

js
// webpack.config.js
+    
Skip to content

模块联邦

目的

更好的复用应用块或者库

每个应用块都是独立构建,被称之为容器

Module Federation

一个被引用的容器被称之为 remote

引用者被称之为 host

配置参数

字段类型含义
namestring必传值,即输出的模块名,被远程引用时的路径为 ${name}/$
libraryobject声明全局变量的方式,name为 umd 的name
filenamestring构建输出的文件名称
remotesobject(host 使用)远程引用的应用名及其别名的映射,使用时以 key 值作为 name
exposesobject(remote 使用)被引用时可暴露的资源路径以其别名
sharedobject与其他应用之间可以共享的第三方依赖,使你的代码中不用重复加载同一份依赖

remote 配置

js
// webpack.config.js
 const path = require("path");
 const webpack = require("webpack");
 const HtmlWebpackPlugin = require("html-webpack-plugin");
@@ -203,8 +203,8 @@
        */
     }),
   ],
-};

host 和 remote 的区别在于: host 通过 remotes 引入模块 remote 通过 exposes 导出模块

如果是存在 shared 共享代码块得通过 bootstrap 动态引入 - https://webpack.js.org/concepts/module-federation/

项目应用图

项目应用图

案例代码

案例代码

原理

项目应用图

remote 通过 exposes 导出模块 host 通过 remotes 引入远程模块

若存在共享依赖会存储到同一个变量中(require.S), 这个变量(require.S) 存放各个作用域下的共享模块, 可以通过 作用域名(scopeName)、包名(key)、版本 (version) 再去获取对应的共享模块合如 modules 中

案例代码

- +};

host 和 remote 的区别在于: host 通过 remotes 引入模块 remote 通过 exposes 导出模块

如果是存在 shared 共享代码块得通过 bootstrap 动态引入 - https://webpack.js.org/concepts/module-federation/

项目应用图

项目应用图

案例代码

案例代码

原理

项目应用图

remote 通过 exposes 导出模块 host 通过 remotes 引入远程模块

若存在共享依赖会存储到同一个变量中(require.S), 这个变量(require.S) 存放各个作用域下的共享模块, 可以通过 作用域名(scopeName)、包名(key)、版本 (version) 再去获取对应的共享模块合如 modules 中

案例代码

+ \ No newline at end of file diff --git "a/guide/webpack/\347\203\255\346\233\264\346\226\260\345\216\237\347\220\206.html" "b/guide/webpack/\347\203\255\346\233\264\346\226\260\345\216\237\347\220\206.html" index 64c9cd10..4bc78726 100644 --- "a/guide/webpack/\347\203\255\346\233\264\346\226\260\345\216\237\347\220\206.html" +++ "b/guide/webpack/\347\203\255\346\233\264\346\226\260\345\216\237\347\220\206.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content
- +
Skip to content
+ \ No newline at end of file diff --git "a/guide/\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263/\346\246\202\350\246\201.html" "b/guide/\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263/\346\246\202\350\246\201.html" index b343b4fb..c04b975c 100644 --- "a/guide/\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263/\346\246\202\350\246\201.html" +++ "b/guide/\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263/\346\246\202\350\246\201.html" @@ -12,7 +12,7 @@ - + @@ -24,8 +24,8 @@ -
Skip to content
- +
Skip to content
+ \ No newline at end of file diff --git "a/guide/\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263/\346\265\217\350\247\210\345\231\250\345\206\205\346\240\270.html" "b/guide/\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263/\346\265\217\350\247\210\345\231\250\345\206\205\346\240\270.html" index b9d53756..89e339d5 100644 --- "a/guide/\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263/\346\265\217\350\247\210\345\231\250\345\206\205\346\240\270.html" +++ "b/guide/\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263/\346\265\217\350\247\210\345\231\250\345\206\205\346\240\270.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

浏览器内核

  • Trident (IE)
  • Gecko (Firefox)
  • Webkit (Safari)
  • Blink (Chrome、Opera) Blink 是 Webkit 的一个分支
- +
Skip to content

浏览器内核

  • Trident (IE)
  • Gecko (Firefox)
  • Webkit (Safari)
  • Blink (Chrome、Opera) Blink 是 Webkit 的一个分支
+ \ No newline at end of file diff --git "a/guide/\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263/\346\265\217\350\247\210\345\231\250\345\256\211\345\205\250.html" "b/guide/\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263/\346\265\217\350\247\210\345\231\250\345\256\211\345\205\250.html" index bada4aa6..b4a43f15 100644 --- "a/guide/\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263/\346\265\217\350\247\210\345\231\250\345\256\211\345\205\250.html" +++ "b/guide/\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263/\346\265\217\350\247\210\345\231\250\345\256\211\345\205\250.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

浏览器安全

页面安全

同源策略

协议、域名、端口号都相同,才是同源。

限制

  1. 无法进行 DOM 操作
  2. 无法获取 Cookie、LocalStorage 等数据
  3. 无法向不同源发请求

开放

  1. 默认页面中可以引用任意第三方资源

    引入 CSP (内容安全策略)

    渲染流程图

  2. 只能请求同源的接口

    引入 CORS (跨域资源共享)

    • 简单请求 (HEAD/POST/GET请求)

      1. 浏览器在请求头上添加 Oirgin 字段,该字段用来说明请求来自那个源,服务器可以根据这个值决定是否同意这次请求
      2. 当服务器收到请求后,根据 Origin 判断是否在许可范围内
      3. 如果不在范围内,服务器会返回一个正常的 HTTP 响应,浏览器发现该响应头没有包含 Access-Control-Allow-Origin 字段,就会抛出一个错误,被 XMLHttpRequest 的 onerror 回调函数捕获 (由于正常响应,其状态码是 200,所以该错误不能通过状态码识别)
      4. 如果 Origin 在指定范围内,服务器返回的响应会多几个头信息字段
        • Access-Control-Allow-Origin(必须) 判断源是否可以跨域,该值要么是请求的 Origin,要么是 *

        • Access-Control-Allow-Credentials 是否允许发送 Cookie,默认为 false 如果 Access-Control-Allow-Origin 的值为 *,则无法发送 Cookie

        • Access-Control-Expose-Headers 制定其他的头信息字段可以暴露给客户端 因为默认情况下,只有 6 个字段可以被暴露,其他的都会被过滤掉

          简单请求

    • 非简单请求 (PUT/DELETE 或者 Content-type字段类型是 application/json)

      1. 浏览器发起 Option 预检请求
      2. 服务器收到预检请求后,检查
        • Origin 发起请求的源信息
        • Access-Control-Request-Method 浏览器会发起请求的方法
        • Access-Control-Request-Headers 请求额外发送的头信息字段
      3. 如果服务器否定预检请求,会返回一个正常的 HTTP 响应,但是没有任何的 CORS 相关的头信息字段,这是浏览器会认定服务器不同意预检,触发错误
      4. 如果服务器同意预检请求,会返回一个正常的 HTTP 响应,但是会多几个 CORS 相关的头信息字段
        • Access-Control-Allow-Origin (必须)
        • Access-Control-Allow-Methods (必须)
        • Access-Control-Allow-Credentials
        • Access-Control-Expose-Headers
        • Access-Control-Max-Age - 预检请求的有效期, 单位秒
        • Access-Control-Allow-Headers

      非简单请求

  3. 只能操作同源的DOM

    引入了跨文档消息机制 postMessage

XSS 攻击

跨域脚本攻击。

类型

  1. 存储型

    脚本存储在服务器,用户访问时,脚本从服务器返回,浏览器执行脚本

  2. 反射型

    在文章链接参数上存放了一段恶意脚本,用户访问时,脚本从参数中取出,浏览器执行脚本

  3. DOM 型

    比如在搜索的时候,添加了一段脚本, 如果网站的搜索功能没有对用户输入进行适当的过滤和转义,搜索结果页面可能会将恶意的脚本代码直接插入到 HTML 中,并在用户浏览器中执行

    web 资源在传输或者用户在使用的过程中修改了 Web 页面的数据

预防

  1. 对输入的内容进行截取或者转义(encode)
  2. 限制输入的长度和类型
  3. 引入内容安全策略 (CSP)
  4. 对 Cookie 设置 HttpOnly 属性

CSRF 攻击

跨站请求伪造。

比如诱导用户点击链接,访问网址后,网址通过用户的登录信息伪造请求

必要条件

  1. 站点有 CSRF 漏洞
  2. 用户登录了
  3. 用户点击了诱导链接

预防

  1. 给 Cookie 新增 SameSite 属性
    • Strict: 第三方站点无法发送 Cookie
    • Lax (默认): 第三方站点如果是 GET 请求可以发送 Cookie
    • None 不做限制 Cookie的书写
  2. 通过请求的 Origin or Referer 字段判断请求合法性
  3. 在请求地址中添加 token 并验证

浏览器安全

将渲染进程放在沙箱中

  • 无法持久存储
  • 无法访问网络
  • 无法监听用户交互
  • 需要通过 IPC 与浏览器主进程通信,由浏览器进程处理后,再将结果传递给渲染进程

网络安全

HTTPS、重放攻击、验签

  • 基本属性:

    • name
    • value
  • 访问性:

    • expire
    • path
    • domain
  • 安全性:

    • secure
    • httpOnly
    • sameSite
- +
Skip to content

浏览器安全

页面安全

同源策略

协议、域名、端口号都相同,才是同源。

限制

  1. 无法进行 DOM 操作
  2. 无法获取 Cookie、LocalStorage 等数据
  3. 无法向不同源发请求

开放

  1. 默认页面中可以引用任意第三方资源

    引入 CSP (内容安全策略)

    渲染流程图

  2. 只能请求同源的接口

    引入 CORS (跨域资源共享)

    • 简单请求 (HEAD/POST/GET请求)

      1. 浏览器在请求头上添加 Oirgin 字段,该字段用来说明请求来自那个源,服务器可以根据这个值决定是否同意这次请求
      2. 当服务器收到请求后,根据 Origin 判断是否在许可范围内
      3. 如果不在范围内,服务器会返回一个正常的 HTTP 响应,浏览器发现该响应头没有包含 Access-Control-Allow-Origin 字段,就会抛出一个错误,被 XMLHttpRequest 的 onerror 回调函数捕获 (由于正常响应,其状态码是 200,所以该错误不能通过状态码识别)
      4. 如果 Origin 在指定范围内,服务器返回的响应会多几个头信息字段
        • Access-Control-Allow-Origin(必须) 判断源是否可以跨域,该值要么是请求的 Origin,要么是 *

        • Access-Control-Allow-Credentials 是否允许发送 Cookie,默认为 false 如果 Access-Control-Allow-Origin 的值为 *,则无法发送 Cookie

        • Access-Control-Expose-Headers 制定其他的头信息字段可以暴露给客户端 因为默认情况下,只有 6 个字段可以被暴露,其他的都会被过滤掉

          简单请求

    • 非简单请求 (PUT/DELETE 或者 Content-type字段类型是 application/json)

      1. 浏览器发起 Option 预检请求
      2. 服务器收到预检请求后,检查
        • Origin 发起请求的源信息
        • Access-Control-Request-Method 浏览器会发起请求的方法
        • Access-Control-Request-Headers 请求额外发送的头信息字段
      3. 如果服务器否定预检请求,会返回一个正常的 HTTP 响应,但是没有任何的 CORS 相关的头信息字段,这是浏览器会认定服务器不同意预检,触发错误
      4. 如果服务器同意预检请求,会返回一个正常的 HTTP 响应,但是会多几个 CORS 相关的头信息字段
        • Access-Control-Allow-Origin (必须)
        • Access-Control-Allow-Methods (必须)
        • Access-Control-Allow-Credentials
        • Access-Control-Expose-Headers
        • Access-Control-Max-Age - 预检请求的有效期, 单位秒
        • Access-Control-Allow-Headers

      非简单请求

  3. 只能操作同源的DOM

    引入了跨文档消息机制 postMessage

XSS 攻击

跨域脚本攻击。

类型

  1. 存储型

    脚本存储在服务器,用户访问时,脚本从服务器返回,浏览器执行脚本

  2. 反射型

    在文章链接参数上存放了一段恶意脚本,用户访问时,脚本从参数中取出,浏览器执行脚本

  3. DOM 型

    比如在搜索的时候,添加了一段脚本, 如果网站的搜索功能没有对用户输入进行适当的过滤和转义,搜索结果页面可能会将恶意的脚本代码直接插入到 HTML 中,并在用户浏览器中执行

    web 资源在传输或者用户在使用的过程中修改了 Web 页面的数据

预防

  1. 对输入的内容进行截取或者转义(encode)
  2. 限制输入的长度和类型
  3. 引入内容安全策略 (CSP)
  4. 对 Cookie 设置 HttpOnly 属性

CSRF 攻击

跨站请求伪造。

比如诱导用户点击链接,访问网址后,网址通过用户的登录信息伪造请求

必要条件

  1. 站点有 CSRF 漏洞
  2. 用户登录了
  3. 用户点击了诱导链接

预防

  1. 给 Cookie 新增 SameSite 属性
    • Strict: 第三方站点无法发送 Cookie
    • Lax (默认): 第三方站点如果是 GET 请求可以发送 Cookie
    • None 不做限制 Cookie的书写
  2. 通过请求的 Origin or Referer 字段判断请求合法性
  3. 在请求地址中添加 token 并验证

浏览器安全

将渲染进程放在沙箱中

  • 无法持久存储
  • 无法访问网络
  • 无法监听用户交互
  • 需要通过 IPC 与浏览器主进程通信,由浏览器进程处理后,再将结果传递给渲染进程

网络安全

HTTPS、重放攻击、验签

  • 基本属性:

    • name
    • value
  • 访问性:

    • expire
    • path
    • domain
  • 安全性:

    • secure
    • httpOnly
    • sameSite
+ \ No newline at end of file diff --git "a/guide/\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263/\346\265\217\350\247\210\345\231\250\346\270\262\346\237\223\346\265\201\347\250\213.html" "b/guide/\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263/\346\265\217\350\247\210\345\231\250\346\270\262\346\237\223\346\265\201\347\250\213.html" index d1e3add5..ba21f5ab 100644 --- "a/guide/\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263/\346\265\217\350\247\210\345\231\250\346\270\262\346\237\223\346\265\201\347\250\213.html" +++ "b/guide/\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263/\346\265\217\350\247\210\345\231\250\346\270\262\346\237\223\346\265\201\347\250\213.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

浏览器渲染流程

浏览器渲染流程

  1. 发出请求到页面首次绘制

    • 第一阶段: 页面提交请求到服务器响应,这时候页面还是之前的页面
    • 第二阶段: 获取到响应数据提交到渲染进程,进行 HTML 解析、CSS 加载、JS 加载、JS 执行、CSSOM 解析、布局树生成、页面绘制
    • 第三阶段: 等首次加载完成后,页面一点点被渲染
  2. HTML 解析

  3. 生成 CSSOM 树

    CSS 不会阻塞 HTML 解析,但是会阻塞页面渲染,因为要生成渲染树

  4. 生成布局树

    去除不显示的节点,计算样式

  5. 分层和合成机制

    • 分层: 分层树在布局树之后,分层树的每一个节点都是图层,如没有,则和父节点同一个图层

    • 绘制阶段: 根据图层在绘制阶段生成绘制指令

    • 光栅化: 根据绘制指令,将每个图层都绘制成一张图片

    • 合成: 合成线程将多张图片合成一张图片,然后显示在屏幕上 (由合成线程完成,不影响主线程)

      • 优化操作1: 合成线程内会维护一个光栅化线程池,将绘制指令列表提交到 GPU 进行光栅化,生成位图,放在内存中
      • 优化操作2: 分块,合成线程将图层分块,优先渲染离屏幕最近的图块
      • 合成线程
  6. 页面显示

常见的问题

1 从输入 URL 到页面渲染完成,发生了什么?

  1. 浏览器会根据用户输入的内容判断是关键字还是URL
  2. 如果是关键字,会将其组成成带有搜索关键字的URL,通过IPC进程通信发送给网络进程
  3. 网络进程发起请求前,会判断是否命中强缓存,如命中直接返回存储资源
  4. 否则发起请求,根据 DNS 解析获取域名对应的 IP 地址,进行 TCP、HTTP 连接
  5. 服务端收到请求后,会判断是否命中协商缓存,如命中则返回304状态码
  6. 如返回的是301、302状态码,浏览器会根据响应头返回的location字段,进行重定向
  7. 如是正常返回资源类型,浏览器会根据 content-type 对资源做相对应的操作,如果是下载类型则进行下载,如果是html类型则会提交到渲染进程进行解析
  8. 解析 HTML,转换成浏览器能识别的 DOM 树
  9. 解析 CSS,转化成浏览器能识别的 CSS 样式树
  10. 根据 DOM 和 CSS 样式树,通过 布局计算生成 布局树
  11. 根据布局树上的分层属性(z-index),生成分层树
  12. 根据分层树,生成绘制指令列表
  13. 渲染主线程会将绘制指令列表提交给合成线程
  14. 合成线程会维护一个光栅化线程池,将绘制指令列表提交到 GPU 进行光栅化,生成位图,放在内存中
  15. 合成线程通信浏览器进程,浏览器进程将内存中的数据输出到显卡的后缓存区,在下一帧绘制之前,显卡的后缓冲区与前缓冲区对换,显示屏读取前缓冲区数据,显示到屏幕上

整体流程

2 回流和重绘有什么区别

回流:元素尺寸、定位改变,可能会影响到其他元素的位置

重绘:元素外观改变,如背景色、颜色,元素尺寸位置不改变且不影响其他元素

减少回流的方法:

  1. 集中修改
    • 修改样式,使用 class
    • 修改前将 DOM 改成 display: none,修改后再显示
    • 使用 DocumentFragment
    • resize、scroll 事件,使用防抖
  2. 使用 BFC
    • 隔离内部元素对外部元素的影响
  3. 脱离文档流
    • position: absolute、fixed
    • float
  4. 提升合成图层
    • CSS3 属性: will-change、transform: translate3d(0);
  5. 不是用 offsetHeight、getBoundingClientRect
    • 使用 intersectionObserver API - 判断元素是否在可视区域内

BFC: 块级格式化上下文 特点: BFC 内部元素不会影响到外部元素 形成条件:

  1. HTML 元素
  2. overflow: hidden、auto、scroll
  3. position: absolute、fixed
  4. float 元素
  5. display: inline-block、flex、inline-flex等

3 渲染流程图

渲染流程图

JS 会阻塞 HTML 的解析和渲染

CSS 不会阻塞 HTML 解析,但会阻塞 DOM 渲染,还会阻塞 JS 的执行

为什么 CSS 会阻塞 JS 的执行?

因为 JS 可能会操作 DOM 节点和 CSS 样式,因此浏览器为了获取到最新的 CSS 样式,样式表会在后面的 JS 执行前先加载执行完毕,所以 CSS 会阻塞 JS 的执行

- +
Skip to content

浏览器渲染流程

浏览器渲染流程

  1. 发出请求到页面首次绘制

    • 第一阶段: 页面提交请求到服务器响应,这时候页面还是之前的页面
    • 第二阶段: 获取到响应数据提交到渲染进程,进行 HTML 解析、CSS 加载、JS 加载、JS 执行、CSSOM 解析、布局树生成、页面绘制
    • 第三阶段: 等首次加载完成后,页面一点点被渲染
  2. HTML 解析

  3. 生成 CSSOM 树

    CSS 不会阻塞 HTML 解析,但是会阻塞页面渲染,因为要生成渲染树

  4. 生成布局树

    去除不显示的节点,计算样式

  5. 分层和合成机制

    • 分层: 分层树在布局树之后,分层树的每一个节点都是图层,如没有,则和父节点同一个图层

    • 绘制阶段: 根据图层在绘制阶段生成绘制指令

    • 光栅化: 根据绘制指令,将每个图层都绘制成一张图片

    • 合成: 合成线程将多张图片合成一张图片,然后显示在屏幕上 (由合成线程完成,不影响主线程)

      • 优化操作1: 合成线程内会维护一个光栅化线程池,将绘制指令列表提交到 GPU 进行光栅化,生成位图,放在内存中
      • 优化操作2: 分块,合成线程将图层分块,优先渲染离屏幕最近的图块
      • 合成线程
  6. 页面显示

常见的问题

1 从输入 URL 到页面渲染完成,发生了什么?

  1. 浏览器会根据用户输入的内容判断是关键字还是URL
  2. 如果是关键字,会将其组成成带有搜索关键字的URL,通过IPC进程通信发送给网络进程
  3. 网络进程发起请求前,会判断是否命中强缓存,如命中直接返回存储资源
  4. 否则发起请求,根据 DNS 解析获取域名对应的 IP 地址,进行 TCP、HTTP 连接
  5. 服务端收到请求后,会判断是否命中协商缓存,如命中则返回304状态码
  6. 如返回的是301、302状态码,浏览器会根据响应头返回的location字段,进行重定向
  7. 如是正常返回资源类型,浏览器会根据 content-type 对资源做相对应的操作,如果是下载类型则进行下载,如果是html类型则会提交到渲染进程进行解析
  8. 解析 HTML,转换成浏览器能识别的 DOM 树
  9. 解析 CSS,转化成浏览器能识别的 CSS 样式树
  10. 根据 DOM 和 CSS 样式树,通过 布局计算生成 布局树
  11. 根据布局树上的分层属性(z-index),生成分层树
  12. 根据分层树,生成绘制指令列表
  13. 渲染主线程会将绘制指令列表提交给合成线程
  14. 合成线程会维护一个光栅化线程池,将绘制指令列表提交到 GPU 进行光栅化,生成位图,放在内存中
  15. 合成线程通信浏览器进程,浏览器进程将内存中的数据输出到显卡的后缓存区,在下一帧绘制之前,显卡的后缓冲区与前缓冲区对换,显示屏读取前缓冲区数据,显示到屏幕上

整体流程

2 回流和重绘有什么区别

回流:元素尺寸、定位改变,可能会影响到其他元素的位置

重绘:元素外观改变,如背景色、颜色,元素尺寸位置不改变且不影响其他元素

减少回流的方法:

  1. 集中修改
    • 修改样式,使用 class
    • 修改前将 DOM 改成 display: none,修改后再显示
    • 使用 DocumentFragment
    • resize、scroll 事件,使用防抖
  2. 使用 BFC
    • 隔离内部元素对外部元素的影响
  3. 脱离文档流
    • position: absolute、fixed
    • float
  4. 提升合成图层
    • CSS3 属性: will-change、transform: translate3d(0);
  5. 不是用 offsetHeight、getBoundingClientRect
    • 使用 intersectionObserver API - 判断元素是否在可视区域内

BFC: 块级格式化上下文 特点: BFC 内部元素不会影响到外部元素 形成条件:

  1. HTML 元素
  2. overflow: hidden、auto、scroll
  3. position: absolute、fixed
  4. float 元素
  5. display: inline-block、flex、inline-flex等

3 渲染流程图

渲染流程图

JS 会阻塞 HTML 的解析和渲染

CSS 不会阻塞 HTML 解析,但会阻塞 DOM 渲染,还会阻塞 JS 的执行

为什么 CSS 会阻塞 JS 的执行?

因为 JS 可能会操作 DOM 节点和 CSS 样式,因此浏览器为了获取到最新的 CSS 样式,样式表会在后面的 JS 执行前先加载执行完毕,所以 CSS 会阻塞 JS 的执行

+ \ No newline at end of file diff --git "a/guide/\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263/\346\265\217\350\247\210\345\231\250\347\274\223\345\255\230.html" "b/guide/\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263/\346\265\217\350\247\210\345\231\250\347\274\223\345\255\230.html" index de00f68a..4b39c5e2 100644 --- "a/guide/\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263/\346\265\217\350\247\210\345\231\250\347\274\223\345\255\230.html" +++ "b/guide/\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263/\346\265\217\350\247\210\345\231\250\347\274\223\345\255\230.html" @@ -11,7 +11,7 @@ - + @@ -23,7 +23,7 @@ -
Skip to content

浏览器缓存

缓存

从 HTTP 缓存策略来说

  • 强缓存
  • 协商缓存

从浏览器的缓存位置来说

  • Service Worker Cache (必须是 Https)
  • Memory Cache (内存缓存)
  • Disk Cache (硬盘缓存)
  • Push Cache (Http2.0)

本地存储

请求时携带,用于信息持久化

Local Storage 本地存储

用于长期不变化的数据,只能存储字符串

Session Storage 会话存储

用于存储当前会话的数据,只能存储字符串

indexedDB

非关系型数据库

Service Worker

运行在浏览器背后的独立线程(必须 HTTPS 才能使用)

js
// main.js
+    
Skip to content

浏览器缓存

缓存

从 HTTP 缓存策略来说

  • 强缓存
  • 协商缓存

从浏览器的缓存位置来说

  • Service Worker Cache (必须是 Https)
  • Memory Cache (内存缓存)
  • Disk Cache (硬盘缓存)
  • Push Cache (Http2.0)

本地存储

请求时携带,用于信息持久化

Local Storage 本地存储

用于长期不变化的数据,只能存储字符串

Session Storage 会话存储

用于存储当前会话的数据,只能存储字符串

indexedDB

非关系型数据库

Service Worker

运行在浏览器背后的独立线程(必须 HTTPS 才能使用)

js
// main.js
 /* 判断当前浏览器是否支持serviceWorker */
 if ('serviceWorker' in navigator) {
   /* 当页面加载完成就创建一个 serviceWorker */
@@ -115,8 +115,8 @@
       return response;
     })()
   );
-});
- +});
+ \ No newline at end of file diff --git "a/guide/\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263/\346\265\217\350\247\210\345\231\250\350\277\233\347\250\213\346\236\266\346\236\204.html" "b/guide/\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263/\346\265\217\350\247\210\345\231\250\350\277\233\347\250\213\346\236\266\346\236\204.html" index 3f0113e5..0231cbc7 100644 --- "a/guide/\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263/\346\265\217\350\247\210\345\231\250\350\277\233\347\250\213\346\236\266\346\236\204.html" +++ "b/guide/\346\265\217\350\247\210\345\231\250\347\233\270\345\205\263/\346\265\217\350\247\210\345\231\250\350\277\233\347\250\213\346\236\266\346\236\204.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

浏览器进程架构

进程与线程

进程

进程是一个程序的实例

进程是系统进行资源调度和分配的最小单位

进程之间相互独立,通过 IPC 进行通信

线程

线程是 CPU 调度的最小单位

线程之间共享进程的数据

线程里面会有协程,但只能运行一个

浏览器架构

1 浏览器主进程

用户交互、文件存储、页面显示、子进程管理

2 渲染进程

解析资源、执行 js

渲染进程被浏览器放在安全沙箱里面,因此渲染进程的资源是通过网络获取

  • 渲染线程: 渲染页面
  • JS 线程:解析执行 js
  • 事件触发线程: 事件循环
  • 定时器线程:定时器相关
  • 异步 http 请求线程: 请求相关

3 GPU 进程

UI界面的绘制

4 网络进程

网络资源加载

5 插件进程

管理插件

- +
Skip to content

浏览器进程架构

进程与线程

进程

进程是一个程序的实例

进程是系统进行资源调度和分配的最小单位

进程之间相互独立,通过 IPC 进行通信

线程

线程是 CPU 调度的最小单位

线程之间共享进程的数据

线程里面会有协程,但只能运行一个

浏览器架构

1 浏览器主进程

用户交互、文件存储、页面显示、子进程管理

2 渲染进程

解析资源、执行 js

渲染进程被浏览器放在安全沙箱里面,因此渲染进程的资源是通过网络获取

  • 渲染线程: 渲染页面
  • JS 线程:解析执行 js
  • 事件触发线程: 事件循环
  • 定时器线程:定时器相关
  • 异步 http 请求线程: 请求相关

3 GPU 进程

UI界面的绘制

4 网络进程

网络资源加载

5 插件进程

管理插件

+ \ No newline at end of file diff --git "a/guide/\347\275\221\347\273\234\347\233\270\345\205\263/CDN.html" "b/guide/\347\275\221\347\273\234\347\233\270\345\205\263/CDN.html" index e8c62d9c..60cce8e0 100644 --- "a/guide/\347\275\221\347\273\234\347\233\270\345\205\263/CDN.html" +++ "b/guide/\347\275\221\347\273\234\347\233\270\345\205\263/CDN.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

CDN

定义

内容分发网络,利用最靠近用户的服务器,将资源更快更可靠的给到用户

组成

  1. 分发服务系统

    最小单位是 Cache 设备

    • 响应用户请求
    • 与源站点进行资源同步
  2. 负载均衡系统

    负责对用户的请求进行调度

  3. 运营管理系统

    负责业务层面与外界系统的交互所需要的收集、整理、交付等工作 (为了做闭环)

    运营管理子系统 网络管理子系统

作用

托管 web 资源,加速 web 资源的获取速度

性能方面

  1. 用户从最近的 CDN 站点获取资源,访问速度更快
  2. 突破 TCP 只有同域名下只有 6 个的限制
  3. 减少源服务器的负担

安全方面

  1. 针对 DDos: 通过监控分析异常流量,限制请求频率
  2. 针对 MITM: 从源服务器到 CDN 节点到 ISP (Internet Service Provider, 网络运营提供商),全链路 HTTPS 通信

工作原理

没有使用 CDN 过程

  1. 浏览器通过 DNS 对域名进行解析,得到此域名对应的 IP 地址
  2. 浏览器根据得到的 IP 地址,向域名的服务主机发送数据请求
  3. 服务器向浏览器返回响应数据

使用了 CDN 过程

  1. 获取负载均衡服务器 IP 地址 (自底向上)

      1. 数据的 URL 先进行 DNS 系统解析,发现该 URL 对应的是一个 CDN 专用的 DNS 服务器,DNS 系统就会将域名解析权交给 CNAME 指向的 CDN 专用的 DNS 服务器
      1. CDN 专用的 DNS 服务器将 CDN 全局负载均衡设备 IP 地址返回给用户
      1. 用户向 CDN 的全局负载均衡设备发起数据请求
  2. 获取目标缓存服务器 IP 地址 (自顶向下)

      1. CDN 全局负载均衡设备根据用户 IP 地址以及用户请求的 URL,选择一台用户所属区域的负载均衡设备
      1. 区域负责均衡设备选择一台适合的缓存服务器来提供服务,将该缓存服务器的 IP 地址返回给全局负载均衡设备
      1. 全局负载均衡设备把服务器 IP 地址返回给用户
  3. 用户发起请求获取

    • 用户向该缓存服务器发起请求,缓存服务器响应用户的请求,将用户所需内容发送支用户终端 (如果缓存服务器没有用户想要的内容,那么缓存服务器会穿透 CDN 集群往源服务器获取设备)

流程

使用场景

  1. 使用第三方 CDN 服务 (例如开源框架可以使用第三方 CDN 服务,如 React、Vue、JQuery)
  2. 使用 CDN 进行静态资源的缓存 (将网站的资源放在 CDN 上)
  3. 直播传送 (直播本质也是使用流媒体进行传送,CDN 也支持流媒体传送,所以直播完全可以使用 CDN 来提高访问速度)
- +
Skip to content

CDN

定义

内容分发网络,利用最靠近用户的服务器,将资源更快更可靠的给到用户

组成

  1. 分发服务系统

    最小单位是 Cache 设备

    • 响应用户请求
    • 与源站点进行资源同步
  2. 负载均衡系统

    负责对用户的请求进行调度

  3. 运营管理系统

    负责业务层面与外界系统的交互所需要的收集、整理、交付等工作 (为了做闭环)

    运营管理子系统 网络管理子系统

作用

托管 web 资源,加速 web 资源的获取速度

性能方面

  1. 用户从最近的 CDN 站点获取资源,访问速度更快
  2. 突破 TCP 只有同域名下只有 6 个的限制
  3. 减少源服务器的负担

安全方面

  1. 针对 DDos: 通过监控分析异常流量,限制请求频率
  2. 针对 MITM: 从源服务器到 CDN 节点到 ISP (Internet Service Provider, 网络运营提供商),全链路 HTTPS 通信

工作原理

没有使用 CDN 过程

  1. 浏览器通过 DNS 对域名进行解析,得到此域名对应的 IP 地址
  2. 浏览器根据得到的 IP 地址,向域名的服务主机发送数据请求
  3. 服务器向浏览器返回响应数据

使用了 CDN 过程

  1. 获取负载均衡服务器 IP 地址 (自底向上)

      1. 数据的 URL 先进行 DNS 系统解析,发现该 URL 对应的是一个 CDN 专用的 DNS 服务器,DNS 系统就会将域名解析权交给 CNAME 指向的 CDN 专用的 DNS 服务器
      1. CDN 专用的 DNS 服务器将 CDN 全局负载均衡设备 IP 地址返回给用户
      1. 用户向 CDN 的全局负载均衡设备发起数据请求
  2. 获取目标缓存服务器 IP 地址 (自顶向下)

      1. CDN 全局负载均衡设备根据用户 IP 地址以及用户请求的 URL,选择一台用户所属区域的负载均衡设备
      1. 区域负责均衡设备选择一台适合的缓存服务器来提供服务,将该缓存服务器的 IP 地址返回给全局负载均衡设备
      1. 全局负载均衡设备把服务器 IP 地址返回给用户
  3. 用户发起请求获取

    • 用户向该缓存服务器发起请求,缓存服务器响应用户的请求,将用户所需内容发送支用户终端 (如果缓存服务器没有用户想要的内容,那么缓存服务器会穿透 CDN 集群往源服务器获取设备)

流程

使用场景

  1. 使用第三方 CDN 服务 (例如开源框架可以使用第三方 CDN 服务,如 React、Vue、JQuery)
  2. 使用 CDN 进行静态资源的缓存 (将网站的资源放在 CDN 上)
  3. 直播传送 (直播本质也是使用流媒体进行传送,CDN 也支持流媒体传送,所以直播完全可以使用 CDN 来提高访问速度)
+ \ No newline at end of file diff --git "a/guide/\347\275\221\347\273\234\347\233\270\345\205\263/DNS.html" "b/guide/\347\275\221\347\273\234\347\233\270\345\205\263/DNS.html" index ed578fd5..ccf3d7da 100644 --- "a/guide/\347\275\221\347\273\234\347\233\270\345\205\263/DNS.html" +++ "b/guide/\347\275\221\347\273\234\347\233\270\345\205\263/DNS.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

DNS

定义

域名解析系统

工作原理 (自下而上)

例如查找 www.baidu.com 的 ip 地址

  1. 查找缓存
      1. 检查浏览器缓存
      1. 检查操作系统缓存 (常见的有 host 文件)
      1. 检查路由器缓存
  2. 如何缓存查找到,会向 ISP (网络服务提供商) 的本地 DNS 服务器查询
  3. 如果本地 DNS 服务器没有找到,会向根域名服务器请求解析,分为以下几步:(自顶向下)
      1. 根服务器返回顶级域名服务器(如.com、.cn、.org等)地址,例如该例子中的 .com 的地址
      1. 接着向顶级瑜域名服务器发送请求,然后会返回次级域名服务器的地址,例如该例子会返回 .baidu 的地址
      1. 接着向次级域名服务器发送请求,会返回通过三级域名地址的 IP,例如本例子返回的 www.baidu.com 的地址
    • 4.Local DNS Server会缓存结果,并返回给用户,缓存在系统中
- +
Skip to content

DNS

定义

域名解析系统

工作原理 (自下而上)

例如查找 www.baidu.com 的 ip 地址

  1. 查找缓存
      1. 检查浏览器缓存
      1. 检查操作系统缓存 (常见的有 host 文件)
      1. 检查路由器缓存
  2. 如何缓存查找到,会向 ISP (网络服务提供商) 的本地 DNS 服务器查询
  3. 如果本地 DNS 服务器没有找到,会向根域名服务器请求解析,分为以下几步:(自顶向下)
      1. 根服务器返回顶级域名服务器(如.com、.cn、.org等)地址,例如该例子中的 .com 的地址
      1. 接着向顶级瑜域名服务器发送请求,然后会返回次级域名服务器的地址,例如该例子会返回 .baidu 的地址
      1. 接着向次级域名服务器发送请求,会返回通过三级域名地址的 IP,例如本例子返回的 www.baidu.com 的地址
    • 4.Local DNS Server会缓存结果,并返回给用户,缓存在系统中
+ \ No newline at end of file diff --git "a/guide/\347\275\221\347\273\234\347\233\270\345\205\263/GET\350\257\267\346\261\202\345\222\214POST\350\257\267\346\261\202\347\232\204\345\214\272\345\210\253.html" "b/guide/\347\275\221\347\273\234\347\233\270\345\205\263/GET\350\257\267\346\261\202\345\222\214POST\350\257\267\346\261\202\347\232\204\345\214\272\345\210\253.html" index 4e7a7f24..40890eee 100644 --- "a/guide/\347\275\221\347\273\234\347\233\270\345\205\263/GET\350\257\267\346\261\202\345\222\214POST\350\257\267\346\261\202\347\232\204\345\214\272\345\210\253.html" +++ "b/guide/\347\275\221\347\273\234\347\233\270\345\205\263/GET\350\257\267\346\261\202\345\222\214POST\350\257\267\346\261\202\347\232\204\345\214\272\345\210\253.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

GET 请求和 POST 请求的区别

  1. 功能上: GET 请求常用于获取数据,POST 请求常用于提交数据
  2. 参数上: GET 请求参数放在 URL 上,POST 请求参数放在请求体上
  3. 参数类型上: POST 请求支持更多的参数类型
  4. 安全性上: POST 请求更加安全
  5. 是否有缓存上: GET 请求会被缓存
  6. 请求长度上: 由于浏览器存在 URL 长度限制,因此会影响 GET 请求的长度
- +
Skip to content

GET 请求和 POST 请求的区别

  1. 功能上: GET 请求常用于获取数据,POST 请求常用于提交数据
  2. 参数上: GET 请求参数放在 URL 上,POST 请求参数放在请求体上
  3. 参数类型上: POST 请求支持更多的参数类型
  4. 安全性上: POST 请求更加安全
  5. 是否有缓存上: GET 请求会被缓存
  6. 请求长度上: 由于浏览器存在 URL 长度限制,因此会影响 GET 请求的长度
+ \ No newline at end of file diff --git "a/guide/\347\275\221\347\273\234\347\233\270\345\205\263/HTTP.html" "b/guide/\347\275\221\347\273\234\347\233\270\345\205\263/HTTP.html" index 51215319..38dad2d8 100644 --- "a/guide/\347\275\221\347\273\234\347\233\270\345\205\263/HTTP.html" +++ "b/guide/\347\275\221\347\273\234\347\233\270\345\205\263/HTTP.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

HTTP

HTTP 0.9

早期做学术研究使用,只是单纯的文本传输(只有请求行、没有请求头和请求体)

HTTP1.0

支持图片、视频传输,但传输效率太低

核心需求:支持多种类型的文件下载(引入了请求头和响应头)

HTTP1.1

持久连接 keep-alive

在此之前一个 HTTP 请求需要建立一个 TCP 连接

目前浏览器中对于 同一个域名,默认允许同时建立 6 个 TCP 连接

管道化连接

将串行请求改成并行请求

支持虚拟主机

HTTP 根据 Host 去区分同一 IP 物理机子上的不同虚拟主机

对动态生成的内容提供完美支持

Chunk transfer 机制

Cookie、完全机制

主要问题

    1. HTTP/1.1 对头阻塞问题 在 HTTP 中,因为前一条请求因某些原因没有及时返回,会导致后面的请求无法发起,容易导致对头阻塞(串行)。虽然 HTTP1.1 引入了管道化技术尝试解决队头阻塞问题,但服务端还是需要按顺序处理请求并返回,因此阻塞问题依然存在
    1. 带宽利用率不理想
    • 原因:
      • TCP 的慢启动: 传输数据的速度从慢到快
      • 同时开启多个 TCP 连接,那么这些连接会竞争固定的带宽(带宽不足时,TCP 传输速度会变慢)

HTTP/1.0 与 HTTP/1.1 的 区别?

1 连接方面

持久化连接: HTTP1.1 支持持久化连接,减少每次 HTTP 请求都需要建立 TCP 连接的耗时

管道化连接:提高请求效率

2 缓存方面

HTTP1.1 新增了协商缓存的字段 ETag 比 if-None-Match 更加紧准

3 新增 Host 字段

根据 Host 区分同一 IP 服务器上的不同网站

之前只能通过 IP 区分网站

4 请求资源方面

添加范围请求 (206 状态码)

5 新增了请求方法

PUT、HEAD、OPTIONS 等

HTTP/2.0

一个域名只有一个 TCP 长连接传输数据 (避规 HTTP 1.1 的 TCP 慢启动和多个 TCP 连接时竞争带宽资源)

多路复用机制

通过引入二进制分帧层,实现多路复用机制

整体流程整体流程

  1. 浏览器发起请求,经过二进制分帧层将请求拆分成一个个带请求id的帧,发送给服务端
  2. 服务端收到这些帧后,根据请求id组装成完整请求信息,服务端处理完请求后,将响应请求发送到二进制分帧层,经过二进制分帧层转化为一个个带有响应id的帧发送给浏览器
  3. 浏览器收到这些帧后,会组合成一个完整的响应请求

实现基础: HTTP2 是一个二进制协议,在 HTTP/1.1 版中,报文的头信息必须是文本(ASCII 编码),数据体可以是文本,也可以是二进制。HTTP/2 则是一个彻底的二进制协议,头信息和数据体都是二进制,并且统称为"帧",可以分为头信息帧和数据帧。 帧的概念是它实现多路复用的基础

头部压缩

HTTP/2 实现了头信息压缩,由于 HTTP 1.1 协议不带状态,每次请求都必须附上所有信息。所以,请求的很多字段都是重复的,比如 Cookie 和 User Agent ,一模一样的内容,每次请求都必须附带,这会浪费很多带宽,也影响速度。HTTP/2 对这一点做了优化,引入了头信息压缩机制。一方面,头信息使用 gzip 或 compress 压缩后再发送;另一方面,客户端和服务器同时维护一张头信息表,所有字段都会存入这个表,生成一个索引号,以后就不发送同样字段了,只发送索引号,这样就能提高速度了。

设置请求优先级

服务器推送

存在的问题

TCP 协议本身存在的问题:

    1. TCP 的队头阻塞 TCP 传输过程中把一份数据分成多个数据包进行传输,当某个数据包没有按顺序返回,接收端会一直保持连接等待数据包返回,这就阻塞了后续数据包的传输 整体流程
    1. TCP 建立连接需要耗时

优点

多路复用技术能有效利用带宽(只有一个TCP连接),缓解 TCP 慢启动带来的问题,解决了 HTTP 队头阻塞问题,同时还支持设置优先级、服务器推送、头部压缩极大的提高了 HTTP 的传输效率

http2 与 http1 的区别?

  1. 为了解决 HTTP1.1 的队头阻塞问题引入了二进制分帧层
  2. 只建立一个 TCP 连接,解决 HTTP1.1中 TCP 慢启动以及多个 TCP 之间相互竞争带宽问题
  3. 头部压缩,HTTP2 请求头部采用头部压缩,减少数据内容的大小,提高效率
  4. 服务端推送: 服务端可以向客户端推送资源

HTTP3.0

采用UDP + QUIC协议

  1. UDP 可以减少连接耗费的 RTT,加快数据传输
  2. 使用 QUIC 协议,可以使用多路复用、TLS、可靠性传输等 TCP 协议的特点

存在的问题:

  1. 规范的制定和落地有着较大的差异(比如官方的QUIC和谷歌的QUIC协议有较大差异)
  2. 是对底层协议的改造,落地成本高。

HTTP1.0/1.1/2.0有什么区别?

这道题应该从HTTP的发展历史的角度出发

  1. http1.0 在 http0.9 时代新增了图片、视频的传输,同时新增了请求、响应头,支持文件下载,但请求的效率低,因为每个http请求需要一次TCP连接
  2. HTTP1.1 引入了缓存策略、支持长连接keep-alive等,支持PUT/DELETE/OPTION方法,提高了HTTP的传输效率,但并发请求时会有一个HTTP的头部阻塞问题
  3. HTTP2 通过引入二进制分帧层、采用多路复用避规了HTTP的头部阻塞问题,还支持头部压缩,提高HTTP的传输效率,服务端推送

RTT

浏览器发送数据包到服务器 + 浏览器接收到服务器确认接收的数据包的往返时间,成为RTT

- +
Skip to content

HTTP

HTTP 0.9

早期做学术研究使用,只是单纯的文本传输(只有请求行、没有请求头和请求体)

HTTP1.0

支持图片、视频传输,但传输效率太低

核心需求:支持多种类型的文件下载(引入了请求头和响应头)

HTTP1.1

持久连接 keep-alive

在此之前一个 HTTP 请求需要建立一个 TCP 连接

目前浏览器中对于 同一个域名,默认允许同时建立 6 个 TCP 连接

管道化连接

将串行请求改成并行请求

支持虚拟主机

HTTP 根据 Host 去区分同一 IP 物理机子上的不同虚拟主机

对动态生成的内容提供完美支持

Chunk transfer 机制

Cookie、完全机制

主要问题

    1. HTTP/1.1 对头阻塞问题 在 HTTP 中,因为前一条请求因某些原因没有及时返回,会导致后面的请求无法发起,容易导致对头阻塞(串行)。虽然 HTTP1.1 引入了管道化技术尝试解决队头阻塞问题,但服务端还是需要按顺序处理请求并返回,因此阻塞问题依然存在
    1. 带宽利用率不理想
    • 原因:
      • TCP 的慢启动: 传输数据的速度从慢到快
      • 同时开启多个 TCP 连接,那么这些连接会竞争固定的带宽(带宽不足时,TCP 传输速度会变慢)

HTTP/1.0 与 HTTP/1.1 的 区别?

1 连接方面

持久化连接: HTTP1.1 支持持久化连接,减少每次 HTTP 请求都需要建立 TCP 连接的耗时

管道化连接:提高请求效率

2 缓存方面

HTTP1.1 新增了协商缓存的字段 ETag 比 if-None-Match 更加紧准

3 新增 Host 字段

根据 Host 区分同一 IP 服务器上的不同网站

之前只能通过 IP 区分网站

4 请求资源方面

添加范围请求 (206 状态码)

5 新增了请求方法

PUT、HEAD、OPTIONS 等

HTTP/2.0

一个域名只有一个 TCP 长连接传输数据 (避规 HTTP 1.1 的 TCP 慢启动和多个 TCP 连接时竞争带宽资源)

多路复用机制

通过引入二进制分帧层,实现多路复用机制

整体流程整体流程

  1. 浏览器发起请求,经过二进制分帧层将请求拆分成一个个带请求id的帧,发送给服务端
  2. 服务端收到这些帧后,根据请求id组装成完整请求信息,服务端处理完请求后,将响应请求发送到二进制分帧层,经过二进制分帧层转化为一个个带有响应id的帧发送给浏览器
  3. 浏览器收到这些帧后,会组合成一个完整的响应请求

实现基础: HTTP2 是一个二进制协议,在 HTTP/1.1 版中,报文的头信息必须是文本(ASCII 编码),数据体可以是文本,也可以是二进制。HTTP/2 则是一个彻底的二进制协议,头信息和数据体都是二进制,并且统称为"帧",可以分为头信息帧和数据帧。 帧的概念是它实现多路复用的基础

头部压缩

HTTP/2 实现了头信息压缩,由于 HTTP 1.1 协议不带状态,每次请求都必须附上所有信息。所以,请求的很多字段都是重复的,比如 Cookie 和 User Agent ,一模一样的内容,每次请求都必须附带,这会浪费很多带宽,也影响速度。HTTP/2 对这一点做了优化,引入了头信息压缩机制。一方面,头信息使用 gzip 或 compress 压缩后再发送;另一方面,客户端和服务器同时维护一张头信息表,所有字段都会存入这个表,生成一个索引号,以后就不发送同样字段了,只发送索引号,这样就能提高速度了。

设置请求优先级

服务器推送

存在的问题

TCP 协议本身存在的问题:

    1. TCP 的队头阻塞 TCP 传输过程中把一份数据分成多个数据包进行传输,当某个数据包没有按顺序返回,接收端会一直保持连接等待数据包返回,这就阻塞了后续数据包的传输 整体流程
    1. TCP 建立连接需要耗时

优点

多路复用技术能有效利用带宽(只有一个TCP连接),缓解 TCP 慢启动带来的问题,解决了 HTTP 队头阻塞问题,同时还支持设置优先级、服务器推送、头部压缩极大的提高了 HTTP 的传输效率

http2 与 http1 的区别?

  1. 为了解决 HTTP1.1 的队头阻塞问题引入了二进制分帧层
  2. 只建立一个 TCP 连接,解决 HTTP1.1中 TCP 慢启动以及多个 TCP 之间相互竞争带宽问题
  3. 头部压缩,HTTP2 请求头部采用头部压缩,减少数据内容的大小,提高效率
  4. 服务端推送: 服务端可以向客户端推送资源

HTTP3.0

采用UDP + QUIC协议

  1. UDP 可以减少连接耗费的 RTT,加快数据传输
  2. 使用 QUIC 协议,可以使用多路复用、TLS、可靠性传输等 TCP 协议的特点

存在的问题:

  1. 规范的制定和落地有着较大的差异(比如官方的QUIC和谷歌的QUIC协议有较大差异)
  2. 是对底层协议的改造,落地成本高。

HTTP1.0/1.1/2.0有什么区别?

这道题应该从HTTP的发展历史的角度出发

  1. http1.0 在 http0.9 时代新增了图片、视频的传输,同时新增了请求、响应头,支持文件下载,但请求的效率低,因为每个http请求需要一次TCP连接
  2. HTTP1.1 引入了缓存策略、支持长连接keep-alive等,支持PUT/DELETE/OPTION方法,提高了HTTP的传输效率,但并发请求时会有一个HTTP的头部阻塞问题
  3. HTTP2 通过引入二进制分帧层、采用多路复用避规了HTTP的头部阻塞问题,还支持头部压缩,提高HTTP的传输效率,服务端推送

RTT

浏览器发送数据包到服务器 + 浏览器接收到服务器确认接收的数据包的往返时间,成为RTT

+ \ No newline at end of file diff --git "a/guide/\347\275\221\347\273\234\347\233\270\345\205\263/HTTPS.html" "b/guide/\347\275\221\347\273\234\347\233\270\345\205\263/HTTPS.html" index 6dfa8a48..fc076469 100644 --- "a/guide/\347\275\221\347\273\234\347\233\270\345\205\263/HTTPS.html" +++ "b/guide/\347\275\221\347\273\234\347\233\270\345\205\263/HTTPS.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

HTTPS

HTTP 和 HTTPS 的区别

  1. HTTP 是超文本传输协议,明文传输、简单、无状态,而 HTTPS 是在 HTTP 基础上增加了 SSL 协议,更加安全
  2. HTTP 协议默认端口是 80,HTTPS 协议默认端口是 443
  3. HTTPS 需要 CA 证书、相对于 HTTP 费用更高

HTTPS 演进过程

第一版对称加密 - 不安全

流程

非对称加密 - 传输效率低

流程

对称加密与非对称加密结合 - 存在中间人攻击

流程

通过数字证书校验网站的真实性和获取网站的公钥

之前传输的是密钥,这种方式传输的是装有密钥的保险箱,就算获取到了保险箱,也没有保险箱的钥匙🔑

流程

  1. 浏览器将对称加密方法列表、非对称加密方法列表、随机数 A 传输给服务端
  2. 服务器接受后,将对称加密、非对称加密方法、服务器生成的随机数 B 、数字证书发送给浏览器
  3. 浏览器接受后,验证证书的可靠性,并获取证书内的非对称加密的公钥
  4. 浏览器通过 2 个随机数生成新的随机数 C,并通过非对成加密的公钥对随机数 C 进行加密发送给服务器
  5. 服务器确认后,服务器与浏览器通过 3 个随机数生成对称加密密钥,进行数据传输

如何验证证书的可靠性

  1. 浏览器获取到证书后,通过 CA 相同的 Hash 算法对证书信息进行加密得到摘要 A
  2. 通过 CA 的公钥对证书内的数字签名进行解密,获取到摘要 B
  3. 如果 AB 相同,则证书可靠

什么是 HTTPS 中间人攻击?如何预防?

  1. 先说下 HTTPS 传输过程
  2. 客户端和服务端通信之间,新增一个中间人,伪造 CA 证书和加密数据,获取服务器和客户端的通信信息

流程

如何预防?

使用正规厂商的第三方证书

- +
Skip to content

HTTPS

HTTP 和 HTTPS 的区别

  1. HTTP 是超文本传输协议,明文传输、简单、无状态,而 HTTPS 是在 HTTP 基础上增加了 SSL 协议,更加安全
  2. HTTP 协议默认端口是 80,HTTPS 协议默认端口是 443
  3. HTTPS 需要 CA 证书、相对于 HTTP 费用更高

HTTPS 演进过程

第一版对称加密 - 不安全

流程

非对称加密 - 传输效率低

流程

对称加密与非对称加密结合 - 存在中间人攻击

流程

通过数字证书校验网站的真实性和获取网站的公钥

之前传输的是密钥,这种方式传输的是装有密钥的保险箱,就算获取到了保险箱,也没有保险箱的钥匙🔑

流程

  1. 浏览器将对称加密方法列表、非对称加密方法列表、随机数 A 传输给服务端
  2. 服务器接受后,将对称加密、非对称加密方法、服务器生成的随机数 B 、数字证书发送给浏览器
  3. 浏览器接受后,验证证书的可靠性,并获取证书内的非对称加密的公钥
  4. 浏览器通过 2 个随机数生成新的随机数 C,并通过非对成加密的公钥对随机数 C 进行加密发送给服务器
  5. 服务器确认后,服务器与浏览器通过 3 个随机数生成对称加密密钥,进行数据传输

如何验证证书的可靠性

  1. 浏览器获取到证书后,通过 CA 相同的 Hash 算法对证书信息进行加密得到摘要 A
  2. 通过 CA 的公钥对证书内的数字签名进行解密,获取到摘要 B
  3. 如果 AB 相同,则证书可靠

什么是 HTTPS 中间人攻击?如何预防?

  1. 先说下 HTTPS 传输过程
  2. 客户端和服务端通信之间,新增一个中间人,伪造 CA 证书和加密数据,获取服务器和客户端的通信信息

流程

如何预防?

使用正规厂商的第三方证书

+ \ No newline at end of file diff --git "a/guide/\347\275\221\347\273\234\347\233\270\345\205\263/\345\223\215\345\272\224\346\212\245\346\226\207.html" "b/guide/\347\275\221\347\273\234\347\233\270\345\205\263/\345\223\215\345\272\224\346\212\245\346\226\207.html" index 89aafa1c..e6adfe9c 100644 --- "a/guide/\347\275\221\347\273\234\347\233\270\345\205\263/\345\223\215\345\272\224\346\212\245\346\226\207.html" +++ "b/guide/\347\275\221\347\273\234\347\233\270\345\205\263/\345\223\215\345\272\224\346\212\245\346\226\207.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

响应报文

由响应行、响应头、响应体组成

响应行

由版本、状态码、原因语句组成

状态码

  • 1xx: 请求已被接受,需要继续处理

  • 2xx: 请求已成功处理

    • 200 请求正常处理、命中强缓存
    • 204 请求处理成功,但没有资源返回
    • 206 客户端进行了范围请求,服务端成功执行了这部分 GET 请求
  • 3xx:客户端需要采取进一步操作才能完成

    • 301 永久重定向
    • 302 临时重定向
    • 304 命中协商缓存
  • 4xx: 客户端错误

    • 400 请求存在语法错误
    • 401 用户登录权限不通过
    • 403 用户登录了,但操作权限不通过
    • 404 资源不存在
    • 405 请求行中制定的方法不能被用于请求相应的资源
  • 5xx: 服务端错误

    • 500 服务端报错
    • 504 服务/网关超时

流程

响应头

响应内容相关

  • content-type 内容类型 Content-Type: text/plain;charset=UTF-8
    • 常见的有:
      • application/x-www-form-urlencoded: 浏览器的原生 form 表单,如果不设置 enctype 属性,那么最终就会以 application/x-www-form-urlencoded 方式提交数据。该种方式提交的数据放在 body 里面,数据按照 key1=val1&key2=val2 的方式进行编码,key 和 val 都进行了 URL转码
      • application/json: 服务器消息主体是序列化后的 JSON 字符串
      • multipart/form-data: 该种方式也是一个常见的 POST 提交方式,通常表单上传文件时使用该种方式。
  • content-length 内容长度

缓存相关

  • Cache-Control
  • Last-Modified
  • ETag

客户端相关

  • Set-Cookie: isGray=true;

跨域相关

  • 简单请求

    • Access-Control-Allow-Origin
  • 非简单请求

    • Access-Control-Allow-Methods
    • Access-Control-Allow-Headers
    • Access-Control-Max-Age
  • Cookie 相关

    • Access-Control-Allow-Credentials: true;

时间相关

Date: Mon, 21 Mar 2022 03:36:53 GMT

响应体

- +
Skip to content

响应报文

由响应行、响应头、响应体组成

响应行

由版本、状态码、原因语句组成

状态码

  • 1xx: 请求已被接受,需要继续处理

  • 2xx: 请求已成功处理

    • 200 请求正常处理、命中强缓存
    • 204 请求处理成功,但没有资源返回
    • 206 客户端进行了范围请求,服务端成功执行了这部分 GET 请求
  • 3xx:客户端需要采取进一步操作才能完成

    • 301 永久重定向
    • 302 临时重定向
    • 304 命中协商缓存
  • 4xx: 客户端错误

    • 400 请求存在语法错误
    • 401 用户登录权限不通过
    • 403 用户登录了,但操作权限不通过
    • 404 资源不存在
    • 405 请求行中制定的方法不能被用于请求相应的资源
  • 5xx: 服务端错误

    • 500 服务端报错
    • 504 服务/网关超时

流程

响应头

响应内容相关

  • content-type 内容类型 Content-Type: text/plain;charset=UTF-8
    • 常见的有:
      • application/x-www-form-urlencoded: 浏览器的原生 form 表单,如果不设置 enctype 属性,那么最终就会以 application/x-www-form-urlencoded 方式提交数据。该种方式提交的数据放在 body 里面,数据按照 key1=val1&key2=val2 的方式进行编码,key 和 val 都进行了 URL转码
      • application/json: 服务器消息主体是序列化后的 JSON 字符串
      • multipart/form-data: 该种方式也是一个常见的 POST 提交方式,通常表单上传文件时使用该种方式。
  • content-length 内容长度

缓存相关

  • Cache-Control
  • Last-Modified
  • ETag

客户端相关

  • Set-Cookie: isGray=true;

跨域相关

  • 简单请求

    • Access-Control-Allow-Origin
  • 非简单请求

    • Access-Control-Allow-Methods
    • Access-Control-Allow-Headers
    • Access-Control-Max-Age
  • Cookie 相关

    • Access-Control-Allow-Credentials: true;

时间相关

Date: Mon, 21 Mar 2022 03:36:53 GMT

响应体

+ \ No newline at end of file diff --git "a/guide/\347\275\221\347\273\234\347\233\270\345\205\263/\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.html" "b/guide/\347\275\221\347\273\234\347\233\270\345\205\263/\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.html" index c5c6520d..cc45faa9 100644 --- "a/guide/\347\275\221\347\273\234\347\233\270\345\205\263/\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.html" +++ "b/guide/\347\275\221\347\273\234\347\233\270\345\205\263/\345\270\270\350\247\201\347\232\204\351\227\256\351\242\230.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content
- +
Skip to content
+ \ No newline at end of file diff --git "a/guide/\347\275\221\347\273\234\347\233\270\345\205\263/\346\246\202\350\246\201.html" "b/guide/\347\275\221\347\273\234\347\233\270\345\205\263/\346\246\202\350\246\201.html" index 0bb3e6fa..020ca04f 100644 --- "a/guide/\347\275\221\347\273\234\347\233\270\345\205\263/\346\246\202\350\246\201.html" +++ "b/guide/\347\275\221\347\273\234\347\233\270\345\205\263/\346\246\202\350\246\201.html" @@ -12,7 +12,7 @@ - + @@ -24,8 +24,8 @@ -
Skip to content
- +
Skip to content
+ \ No newline at end of file diff --git "a/guide/\347\275\221\347\273\234\347\233\270\345\205\263/\350\257\267\346\261\202\346\212\245\346\226\207.html" "b/guide/\347\275\221\347\273\234\347\233\270\345\205\263/\350\257\267\346\261\202\346\212\245\346\226\207.html" index da0e8850..f6b34c62 100644 --- "a/guide/\347\275\221\347\273\234\347\233\270\345\205\263/\350\257\267\346\261\202\346\212\245\346\226\207.html" +++ "b/guide/\347\275\221\347\273\234\347\233\270\345\205\263/\350\257\267\346\261\202\346\212\245\346\226\207.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

请求报文

由请求行、请求头、请求体组成

请求行

由请求方法、URI、协议版本组成

流程

请求方法

  1. GET 请求获取资源
  2. POST 提交数据
  3. PUT 更新资源
  4. DELETE 删除资源
  5. OPTIONS 查询针对请求 URL 指定的资源支持的方法(常用在非简单请求的预检上)
  6. HEAD 获取响应头
  7. CONNECT 要求用隧道协议链接代理
  8. TRACE 追踪请求经过的路径

URI 字段

协议版本

HTTP0.9、 HTTP1.0、HTTP1.1、HTTP2.0、HTTP3.0

请求头

接受内容相关

  1. Accept 浏览器接受的格式 Accept: application/json, text/plain, /
  2. Accept-Encoding 浏览器接受的压缩格式 Accept-Encoding: gzip, deflate, br
  3. Accept-Language 浏览器接受的语言 Accept-Language: zh-CN
  4. Accept-Charset 浏览器接受的字符集

强缓存相关

  1. Cache-Control

协商缓存相关

  1. If-Modified-Since 上一次访问时文件的更改时间 (存在校验问题,比如文件内容只是添加了空格,但是 Last-Modified 改变了)
  2. If-None-Match 上次访问的 ETag 信息 (如果 If-None-Match 和 ETag 一致,则命中 304 协商缓存)

请求域名相关

  1. HOST HTTP 请求的域名
  2. Origin 页面域名 (常用于防止 CSRF 攻击和跨域请求)
  3. Referer 发出请求的页面 URL

客户端相关

  • UA
  • COOKIE

连接相关

  • Connection Connection: keep-alive

流程

请求体

- +
Skip to content

请求报文

由请求行、请求头、请求体组成

请求行

由请求方法、URI、协议版本组成

流程

请求方法

  1. GET 请求获取资源
  2. POST 提交数据
  3. PUT 更新资源
  4. DELETE 删除资源
  5. OPTIONS 查询针对请求 URL 指定的资源支持的方法(常用在非简单请求的预检上)
  6. HEAD 获取响应头
  7. CONNECT 要求用隧道协议链接代理
  8. TRACE 追踪请求经过的路径

URI 字段

协议版本

HTTP0.9、 HTTP1.0、HTTP1.1、HTTP2.0、HTTP3.0

请求头

接受内容相关

  1. Accept 浏览器接受的格式 Accept: application/json, text/plain, /
  2. Accept-Encoding 浏览器接受的压缩格式 Accept-Encoding: gzip, deflate, br
  3. Accept-Language 浏览器接受的语言 Accept-Language: zh-CN
  4. Accept-Charset 浏览器接受的字符集

强缓存相关

  1. Cache-Control

协商缓存相关

  1. If-Modified-Since 上一次访问时文件的更改时间 (存在校验问题,比如文件内容只是添加了空格,但是 Last-Modified 改变了)
  2. If-None-Match 上次访问的 ETag 信息 (如果 If-None-Match 和 ETag 一致,则命中 304 协商缓存)

请求域名相关

  1. HOST HTTP 请求的域名
  2. Origin 页面域名 (常用于防止 CSRF 攻击和跨域请求)
  3. Referer 发出请求的页面 URL

客户端相关

  • UA
  • COOKIE

连接相关

  • Connection Connection: keep-alive

流程

请求体

+ \ No newline at end of file diff --git "a/guide/\347\275\221\347\273\234\347\233\270\345\205\263/\350\267\250\345\237\237\350\257\267\346\261\202.html" "b/guide/\347\275\221\347\273\234\347\233\270\345\205\263/\350\267\250\345\237\237\350\257\267\346\261\202.html" index 908ddf91..c5e03c44 100644 --- "a/guide/\347\275\221\347\273\234\347\233\270\345\205\263/\350\267\250\345\237\237\350\257\267\346\261\202.html" +++ "b/guide/\347\275\221\347\273\234\347\233\270\345\205\263/\350\267\250\345\237\237\350\257\267\346\261\202.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

跨域请求

简单请求

若请求满足所有下述条件,则该请求可视为简单请求:

  1. HEAD/GET/POST 请求
  2. Content-Type: text/plain、multipart/form-data、application/x-www-form-urlencoded
  3. 除了被用户代理自动设置的标头字段,剩下的请求头是: Accept、Accept-Language、Content-Language、Content-Type、Range

请求过程

  1. 发起请求,请求头会带上 Origin 字段,该字段用来说明请求来自哪个源(协议 + 域名 + 端口),服务器根据这个值决定是否同意这次请求
  2. 当服务器接收到请求后,根据 Origin 判断是否在允许的范围内
  3. 如果不在范围内,服务器会返回一个正常的 HTTP 响应,浏览器发现响应头信息没有 Access-Control-Allow-Origin 字段,就知道出错了,从而抛出错误,被 XML 的 onerror 回调函数捕获。(注意:由于正常响应,其状态码为200,因此该错误不能通过状态码识别)
  4. 如果 Origin 指定的域名在范围内,服务器返回的响应会多出几个头信息字段(Access-Control-Allow-Origin、Access-Control-Allow-Credentials、Access-Control-Expose-Header 等)

流程

非简单请求

请求方法是 PUT、DELETE 或者 Content-Type 是 application/json 类型

请求过程 (源、请求头、方法、缓存、Cookie)

  1. 浏览器发起 Option 预检请求
  2. 服务器收到预检请求以后,检查了 Origin、Access-Control-Request-Method 和 Access-Control-Request-Headers 字段后,确认允许跨域请求,就可以做出回应
    • Origin (必须): 发起请求的源信息
    • Access-Control-Request-Method(必须): 用来列出浏览器的 CORS 请求会用到哪些 HTTP 方法
    • Access-Control-Request-Headers 跨域请求而外的请求头字段
  3. 如果服务器预检请求,会返回一个正常的 HTTP 回应,但没有任何的 CORS 相关的头信息字段,这时浏览器就会认定服务器不同意预检请求,触发错误。
  4. 如果服务器通过了预检请求,以后每次浏览器正常的 CORS 请求就跟简单请求一样
    • Access-Control-Allow-Origin(必须): 运行哪些源跨域,如果是 * 无法携带 Cookie
    • Access-Control-Allow-Method(必须): 服务器支持哪些请求
    • Access-Control-Allow-Header: 服务器支持的头信息字段
    • Access-Control-Max-Age: 预检请求的有效期,单位秒

流程

  1. 响应头设置 Access-Control-Allow-Credentials 等于 true
  2. Access-Control-Allow-Origin 不能设置成 * (Cookie 的 SameSite 属性如果是 Lax 可能也会导致带不上去)
  3. 前端设置 withCredentials: true

常见的问题

cookie 一般用于登录验证,存储用户信息,大小为 4KB,会随着网络请求携带给服务端

session 一般存储在服务端,常用于与 Cookie 配合做登录检验

  • cookie 是 HTTP 的内容,token 是自定义的数据
  • cookie 可以默认存储在浏览器中,token 需要自行存储
  • token 没有跨域限制,cookie 存在跨域限制
  • token 常用于 CSRF 或者 JWT(JSON WEB TOKEN)
  • Cookie 常于 Session 配合,做用户登录鉴权
  • 基本属性:

    • name
    • value
  • 访问性:

    • expire
    • path
    • domain
  • 安全性:

    • secure
    • httpOnly
    • sameSite

Session 和 JWT 哪个更合适?

  • Session

    • 优点:
      1. 易于学习
      2. 用户信息存储在服务器,可以快速封禁某个用户
    • 缺点:
      1. 占用服务器资源,硬件成本高
      2. 多进程、多服务器时,不好同步 (需要第三方缓存, Redis)
      3. Session 需要配合 Cookie 使用,cookie 有域名限制
  • Json Web Token

    • 优点:
      1. 存储在客户端,不占用服务端资源
      2. 易于同步
      3. 没有跨域限制
    • 缺点:
      1. 无法快速封禁用户
      2. Token 不安全,一但秘钥被泄漏,容易窃取用户信息
      3. Token很大,影响请求体积
  • 使用场景:

    • 用户信息安全 -> 使用 Session
    • 没有特殊要求 -> 使用 JWT

如何实现 SSO 单点登录

Cookie 默认跨域不共享

可以通过设置 Cookie 的 domain 为相同的主域名,即可共享 Cookie

比如 www.baidu.com、image.baidu.com 主域名是相同的,设置 cookie domain 为主域名,即可共享 cookie

流程

如果主域名不一致

  1. 使用 SSO 第三方登录, 获取 ticket 返回给 A、B 系统

流程

  1. OAuth2.0

第三方登录(例如微信扫码登录)

流程

- +
Skip to content

跨域请求

简单请求

若请求满足所有下述条件,则该请求可视为简单请求:

  1. HEAD/GET/POST 请求
  2. Content-Type: text/plain、multipart/form-data、application/x-www-form-urlencoded
  3. 除了被用户代理自动设置的标头字段,剩下的请求头是: Accept、Accept-Language、Content-Language、Content-Type、Range

请求过程

  1. 发起请求,请求头会带上 Origin 字段,该字段用来说明请求来自哪个源(协议 + 域名 + 端口),服务器根据这个值决定是否同意这次请求
  2. 当服务器接收到请求后,根据 Origin 判断是否在允许的范围内
  3. 如果不在范围内,服务器会返回一个正常的 HTTP 响应,浏览器发现响应头信息没有 Access-Control-Allow-Origin 字段,就知道出错了,从而抛出错误,被 XML 的 onerror 回调函数捕获。(注意:由于正常响应,其状态码为200,因此该错误不能通过状态码识别)
  4. 如果 Origin 指定的域名在范围内,服务器返回的响应会多出几个头信息字段(Access-Control-Allow-Origin、Access-Control-Allow-Credentials、Access-Control-Expose-Header 等)

流程

非简单请求

请求方法是 PUT、DELETE 或者 Content-Type 是 application/json 类型

请求过程 (源、请求头、方法、缓存、Cookie)

  1. 浏览器发起 Option 预检请求
  2. 服务器收到预检请求以后,检查了 Origin、Access-Control-Request-Method 和 Access-Control-Request-Headers 字段后,确认允许跨域请求,就可以做出回应
    • Origin (必须): 发起请求的源信息
    • Access-Control-Request-Method(必须): 用来列出浏览器的 CORS 请求会用到哪些 HTTP 方法
    • Access-Control-Request-Headers 跨域请求而外的请求头字段
  3. 如果服务器预检请求,会返回一个正常的 HTTP 回应,但没有任何的 CORS 相关的头信息字段,这时浏览器就会认定服务器不同意预检请求,触发错误。
  4. 如果服务器通过了预检请求,以后每次浏览器正常的 CORS 请求就跟简单请求一样
    • Access-Control-Allow-Origin(必须): 运行哪些源跨域,如果是 * 无法携带 Cookie
    • Access-Control-Allow-Method(必须): 服务器支持哪些请求
    • Access-Control-Allow-Header: 服务器支持的头信息字段
    • Access-Control-Max-Age: 预检请求的有效期,单位秒

流程

  1. 响应头设置 Access-Control-Allow-Credentials 等于 true
  2. Access-Control-Allow-Origin 不能设置成 * (Cookie 的 SameSite 属性如果是 Lax 可能也会导致带不上去)
  3. 前端设置 withCredentials: true

常见的问题

cookie 一般用于登录验证,存储用户信息,大小为 4KB,会随着网络请求携带给服务端

session 一般存储在服务端,常用于与 Cookie 配合做登录检验

  • cookie 是 HTTP 的内容,token 是自定义的数据
  • cookie 可以默认存储在浏览器中,token 需要自行存储
  • token 没有跨域限制,cookie 存在跨域限制
  • token 常用于 CSRF 或者 JWT(JSON WEB TOKEN)
  • Cookie 常于 Session 配合,做用户登录鉴权
  • 基本属性:

    • name
    • value
  • 访问性:

    • expire
    • path
    • domain
  • 安全性:

    • secure
    • httpOnly
    • sameSite

Session 和 JWT 哪个更合适?

  • Session

    • 优点:
      1. 易于学习
      2. 用户信息存储在服务器,可以快速封禁某个用户
    • 缺点:
      1. 占用服务器资源,硬件成本高
      2. 多进程、多服务器时,不好同步 (需要第三方缓存, Redis)
      3. Session 需要配合 Cookie 使用,cookie 有域名限制
  • Json Web Token

    • 优点:
      1. 存储在客户端,不占用服务端资源
      2. 易于同步
      3. 没有跨域限制
    • 缺点:
      1. 无法快速封禁用户
      2. Token 不安全,一但秘钥被泄漏,容易窃取用户信息
      3. Token很大,影响请求体积
  • 使用场景:

    • 用户信息安全 -> 使用 Session
    • 没有特殊要求 -> 使用 JWT

如何实现 SSO 单点登录

Cookie 默认跨域不共享

可以通过设置 Cookie 的 domain 为相同的主域名,即可共享 Cookie

比如 www.baidu.com、image.baidu.com 主域名是相同的,设置 cookie domain 为主域名,即可共享 cookie

流程

如果主域名不一致

  1. 使用 SSO 第三方登录, 获取 ticket 返回给 A、B 系统

流程

  1. OAuth2.0

第三方登录(例如微信扫码登录)

流程

+ \ No newline at end of file diff --git a/hashmap.json b/hashmap.json index a51756db..b64e24f2 100644 --- a/hashmap.json +++ b/hashmap.json @@ -1 +1 @@ -{"guide_react_react与vue的区别.md":"aefdc566","guide_node相关_概要.md":"6d2f3b80","guide_react_react为什么要使用jsx.md":"62407e1e","guide_react_react是什么.md":"471a44bf","guide_fabric.js_如何实现点选物体.md":"01b61e2b","guide_react_什么是fiber.md":"29e5a130","guide_fabric.js_内部结构概述.md":"be67fc71","guide_react_react异常机制.md":"95438d51","guide_react_react中的性能优化.md":"6ad45a79","guide_canvas_可视区域内渲染提高canvas书写性能.md":"e953eb67","guide_ai_outputparser构建格式化输出.md":"d681019e","guide_fabric.js_如何绘制一个图形.md":"78353ed8","guide_fabric.js_基本概念.md":"711f70e5","guide_react_react是如何渲染的.md":"ee09ffde","guide_fabric.js_如何调试fabric.md":"e3e1288a","guide_fabric.js_如何实现组选.md":"3e1b07e1","guide_react_如何设计react组件.md":"7bfc2acc","guide_ai_retriever之向量数据库.md":"6b2ddb25","guide_网络相关_响应报文.md":"cc269f40","shoot_构图形式_构图技巧.md":"01544455","shoot_自我实战总结_延迟摄影.md":"7117a224","guide_ddd 领域驱动设计_index.md":"b9fd2eae","guide_ai_retriever 常见的优化方式.md":"86e88704","guide_ai_rag 检索增强生成的流程.md":"c97a9cbd","shoot_自我实战总结_拍摄烟花.md":"481cbdad","guide_ai_构建可复用的prompttemplate.md":"25d3a453","guide_react_react中的事件机制.md":"4c13470e","guide_react_setstate是同步还是异步的.md":"d2fb1cb9","guide_ai_langchain快速入门.md":"ab114f06","guide_css相关_display.md":"3c174711","guide_css相关_bfc.md":"97daca69","guide_canvas_通过上下分层优化canvas书写性能.md":"81bf21d6","guide_canvas_通过 offscreencanvas _ worker 提高书写性能.md":"09ce6c22","guide_react_react中的异常机制.md":"b181d518","guide_css相关_display、visibility、opacity区别.md":"66ff32b5","guide_ai_什么是langchain.md":"a8c31c8d","guide_redux_源码解读.md":"a19e3f7c","guide_canvas_通过离屏渲染提高canvas书写性能.md":"5cb7017a","guide_react_常见的问题.md":"681d5e24","guide_redux_设计理念.md":"c290b375","guide_ai_embedding之大规模数据拆分.md":"8b24253c","guide_css相关_对 line-height 的理解及其赋值方式.md":"838b7cb9","shoot_自我实战总结_拍摄月亮.md":"155f2cd6","guide_css相关_概要.md":"6367bf5b","shoot_自我实战总结_拍摄星空.md":"cbaec0d7","guide_javascript相关_dom相关.md":"7690a9dd","guide_css相关_隐藏元素的方法.md":"6677bfcf","guide_javascript相关_js异常捕获机制.md":"b6816c6f","guide_javascript相关_es6相关.md":"4615c5ea","guide_javascript相关_javascript执行机制.md":"f9a311e1","guide_javascript相关_javascript编译机制.md":"4fefe971","guide_javascript相关_promise.md":"54f111a9","guide_javascript相关_index.md":"41467648","guide_javascript相关_事件循环机制.md":"0b3536a3","guide_javascript相关_垃圾回收机制.md":"7fe4a5e0","guide_canvas_通过贝塞尔曲线优化canvas书写性能.md":"61a93128","guide_javascript相关_基础概念.md":"3cedefb2","guide_javascript相关_字符串常见的api.md":"c4f598fd","guide_javascript相关_定义.md":"05754f7e","guide_javascript相关_对象常见的api.md":"c8d02937","guide_javascript相关_数组常见的api.md":"9011f496","guide_javascript相关_新的运算符.md":"c349c9ff","guide_javascript相关_正则表达式.md":"e5d049d6","guide_react_essential knowledge for schedule about react.md":"5abf05d0","guide_浏览器相关_浏览器内核.md":"8232bb8e","guide_网络相关_常见的问题.md":"0601ae85","guide_canvas_canvas尺寸及分辨率矫正.md":"e573b0a5","guide_网络相关_概要.md":"a9f4d298","guide_网络相关_跨域请求.md":"bcf87cbb","index.md":"b3d414ec","guide_网络相关_请求报文.md":"c7f4f8d0","guide_webpack_模块联邦.md":"9176a16d","shoot_基础概念_iso感光度.md":"2f26bf17","shoot_基础概念_光圈.md":"ed2aed3b","guide_webpack_index.md":"fcc0bdf4","shoot_实战技巧_index.md":"5e1171b3","guide_webpack_构建流程.md":"e14ee84d","guide_浏览器相关_概要.md":"053a8f89","shoot_实战技巧_如何拍花的黑背景.md":"fb63b890","guide_webpack_热更新原理.md":"f90caa71","shoot_实战技巧_亮调人像.md":"1d135fd4","guide_fabric.js_如何实现元素的平移、旋转、缩放.md":"493fc2f7","shoot_实战技巧_冷暖对比人像.md":"70050c0c","guide_react_diff算法.md":"e12213ca","shoot_基础概念_快门速度.md":"8bdd3100","shoot_实战技巧_如何拍雨丝.md":"e607e8e7","shoot_实战技巧_逆光人像.md":"0e2e6697","shoot_实战技巧_暗调人像.md":"09e0e873","guide_webpack_mini-webpack.md":"938949bc","guide_webpack_loader.md":"daa9d3ad","shoot_实战技巧_拍摄梦幻光斑.md":"498fd396","guide_浏览器相关_浏览器安全.md":"73ccefca","guide_react_react组件是如何通信的.md":"a82d5c10","guide_css相关_position.md":"839d8549","guide_ai_embedding之加载数据.md":"a848af48","guide_浏览器相关_浏览器渲染流程.md":"f3c634f1","guide_网络相关_dns.md":"584b66a4","guide_浏览器相关_浏览器进程架构.md":"5bd0b06d","guide_网络相关_cdn.md":"0fd3f4e9","guide_网络相关_get请求和post请求的区别.md":"2f19dc13","guide_浏览器相关_浏览器缓存.md":"4f6289a9","guide_网络相关_http.md":"5112c2aa","guide_css相关_display、float、position的关系.md":"0e4b5524","guide_网络相关_https.md":"efe51aa4","guide_javascript相关_继承相关.md":"7058e93b","guide_canvas_如何在canvas画板上自由书写.md":"a3f6d57a","guide_javascript相关_数据类型.md":"b73ffe80"} +{"guide_react_react是什么.md":"ee0c3664","guide_react_react是如何渲染的.md":"f058feee","guide_react_react组件是如何通信的.md":"a0c59c32","guide_react_diff算法.md":"4f2b0e59","guide_redux_设计理念.md":"00f4264a","guide_react_如何设计react组件.md":"b6076bcd","guide_ai_构建可复用的prompttemplate.md":"2d5ec969","guide_ai_什么是langchain.md":"487089c2","guide_fabric.js_fabric 中的性能优化.md":"9993b631","guide_react_常见的问题.md":"5afef26f","guide_canvas_canvas尺寸及分辨率矫正.md":"6671b487","guide_javascript相关_事件循环机制.md":"d507e357","guide_浏览器相关_浏览器渲染流程.md":"08093b9d","guide_浏览器相关_浏览器内核.md":"6fdfde81","guide_浏览器相关_浏览器安全.md":"ff5d442d","guide_浏览器相关_浏览器进程架构.md":"4157b63e","guide_网络相关_cdn.md":"426d00a8","guide_浏览器相关_浏览器缓存.md":"3b8afe38","guide_网络相关_http.md":"2fd00196","shoot_自我实战总结_拍摄星空.md":"79082746","guide_redux_源码解读.md":"e0e871fd","guide_网络相关_get请求和post请求的区别.md":"675258cb","guide_网络相关_dns.md":"dc1c7d42","guide_webpack_index.md":"8c12b279","guide_react_react为什么要使用jsx.md":"40e22e85","guide_canvas_通过上下分层优化canvas书写性能.md":"fc9344af","guide_css相关_display.md":"6d34245c","guide_css相关_display、visibility、opacity区别.md":"2678884b","guide_webpack_模块联邦.md":"a4b55cfe","guide_javascript相关_dom相关.md":"b5647b3c","guide_浏览器相关_概要.md":"8b5cc56c","guide_css相关_bfc.md":"fbec5d1d","guide_javascript相关_js异常捕获机制.md":"bd59bbe3","guide_fabric.js_内部结构概述.md":"5690116d","guide_javascript相关_javascript执行机制.md":"b95eb7c2","guide_fabric.js_如何实现元素的平移、旋转、缩放.md":"140dd9a6","guide_fabric.js_如何绘制一个图形.md":"fc479c77","guide_css相关_概要.md":"9e3e6a15","guide_css相关_隐藏元素的方法.md":"3f129790","guide_css相关_position.md":"b4f4db42","guide_css相关_对 line-height 的理解及其赋值方式.md":"7f0ca022","guide_css相关_display、float、position的关系.md":"f469d683","guide_node相关_概要.md":"5e89b8cf","guide_fabric.js_如何调试fabric.md":"b88e0c09","guide_react_essential knowledge for schedule about react.md":"8a28d74e","guide_fabric.js_基本概念.md":"302ccee7","guide_react_react中的异常机制.md":"ac78472e","guide_react_react中的事件机制.md":"d11dda49","guide_ddd 领域驱动设计_index.md":"d79253fb","guide_react_react异常机制.md":"39386d10","guide_react_什么是fiber.md":"9a75b68a","guide_ai_embedding之加载数据.md":"6565d23d","guide_react_react与vue的区别.md":"11b5926a","guide_网络相关_概要.md":"3fbc149d","guide_ai_embedding之大规模数据拆分.md":"64da80d7","guide_网络相关_https.md":"1a833174","guide_ai_retriever之向量数据库.md":"0af240f5","guide_网络相关_请求报文.md":"dd8a47bc","guide_javascript相关_基础概念.md":"53cfc405","guide_javascript相关_字符串常见的api.md":"699074e1","guide_javascript相关_定义.md":"ea2dc8ec","guide_javascript相关_对象常见的api.md":"6057b79b","guide_ai_retriever 常见的优化方式.md":"a17d2dce","guide_webpack_构建流程.md":"61545129","guide_webpack_loader.md":"8ee4eae1","guide_javascript相关_正则表达式.md":"df2c1874","guide_javascript相关_数组常见的api.md":"452ffdf7","guide_fabric.js_如何实现点选物体.md":"4b2066fc","guide_ai_rag 检索增强生成的流程.md":"58fac56a","guide_网络相关_响应报文.md":"b46d175f","guide_网络相关_跨域请求.md":"848f5d06","guide_react_setstate是同步还是异步的.md":"10d004c3","guide_javascript相关_新的运算符.md":"c8bd4cf5","guide_网络相关_常见的问题.md":"fee49d4c","guide_fabric.js_如何实现框选.md":"382e7c32","shoot_基础概念_iso感光度.md":"1329ace6","shoot_基础概念_光圈.md":"5071eb5f","guide_canvas_通过离屏渲染提高canvas书写性能.md":"86c812db","index.md":"dd3d0939","guide_javascript相关_javascript编译机制.md":"4e21f748","shoot_实战技巧_如何拍雨丝.md":"7e47cf94","shoot_实战技巧_拍摄梦幻光斑.md":"8ca08bfb","shoot_实战技巧_index.md":"e5c3a073","shoot_实战技巧_亮调人像.md":"55937770","shoot_构图形式_构图技巧.md":"9d941cfa","shoot_实战技巧_暗调人像.md":"740aab34","shoot_实战技巧_逆光人像.md":"57538260","guide_canvas_通过 offscreencanvas _ worker 提高书写性能.md":"74068538","guide_javascript相关_promise.md":"9dcd3d04","guide_javascript相关_数据类型.md":"77b639a5","shoot_自我实战总结_拍摄月亮.md":"6fbf4de5","guide_ai_langchain快速入门.md":"e65bc2d1","shoot_实战技巧_冷暖对比人像.md":"852b078d","guide_ai_outputparser构建格式化输出.md":"1a2532c7","shoot_自我实战总结_延迟摄影.md":"abf24cee","guide_canvas_通过贝塞尔曲线优化canvas书写性能.md":"1acbc131","guide_react_react中的性能优化.md":"780419b2","guide_javascript相关_index.md":"6fa036d3","guide_javascript相关_继承相关.md":"86520b75","guide_canvas_如何在canvas画板上自由书写.md":"a9d3a092","shoot_自我实战总结_拍摄烟花.md":"bff2e64b","shoot_实战技巧_如何拍花的黑背景.md":"5e8263dc","guide_webpack_热更新原理.md":"87ba4ad8","shoot_基础概念_快门速度.md":"858c51e6","guide_webpack_mini-webpack.md":"7a56cd34","guide_canvas_可视区域内渲染提高canvas书写性能.md":"1828bbfc","guide_javascript相关_垃圾回收机制.md":"41875062","guide_javascript相关_es6相关.md":"a46321f6"} diff --git a/index.html b/index.html index 901016e8..32dcb111 100644 --- a/index.html +++ b/index.html @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

个人知识库

👋 欢迎来到我的博客,这里是我记录学习和生活的地方,希望能帮助到你

采用 VitePress 进行构建,为了有更好的阅读体验,请使用新版本浏览器

- +
Skip to content

个人知识库

👋 欢迎来到我的博客,这里是我记录学习和生活的地方,希望能帮助到你

采用 VitePress 进行构建,为了有更好的阅读体验,请使用新版本浏览器

+ \ No newline at end of file diff --git "a/shoot/\345\237\272\347\241\200\346\246\202\345\277\265/ISO\346\204\237\345\205\211\345\272\246.html" "b/shoot/\345\237\272\347\241\200\346\246\202\345\277\265/ISO\346\204\237\345\205\211\345\272\246.html" index 912cbb15..4a2d6102 100644 --- "a/shoot/\345\237\272\347\241\200\346\246\202\345\277\265/ISO\346\204\237\345\205\211\345\272\246.html" +++ "b/shoot/\345\237\272\347\241\200\346\246\202\345\277\265/ISO\346\204\237\345\205\211\345\272\246.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

ISO 感光度

传感器对光线的敏感程度

  • 感光度越大、图片越亮,但也容易出现噪点
- +
Skip to content

ISO 感光度

传感器对光线的敏感程度

  • 感光度越大、图片越亮,但也容易出现噪点
+ \ No newline at end of file diff --git "a/shoot/\345\237\272\347\241\200\346\246\202\345\277\265/\345\205\211\345\234\210.html" "b/shoot/\345\237\272\347\241\200\346\246\202\345\277\265/\345\205\211\345\234\210.html" index a60312e6..81ca8a1c 100644 --- "a/shoot/\345\237\272\347\241\200\346\246\202\345\277\265/\345\205\211\345\234\210.html" +++ "b/shoot/\345\237\272\347\241\200\346\246\202\345\277\265/\345\205\211\345\234\210.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

光圈

控制光圈大小

  1. 数字越大、光圈越小就越暗 - 图像越清晰
  2. 数字越小、光圈越大就越亮 - 光圈过大虚化效果明显
- +
Skip to content

光圈

控制光圈大小

  1. 数字越大、光圈越小就越暗 - 图像越清晰
  2. 数字越小、光圈越大就越亮 - 光圈过大虚化效果明显
+ \ No newline at end of file diff --git "a/shoot/\345\237\272\347\241\200\346\246\202\345\277\265/\345\277\253\351\227\250\351\200\237\345\272\246.html" "b/shoot/\345\237\272\347\241\200\346\246\202\345\277\265/\345\277\253\351\227\250\351\200\237\345\272\246.html" index bf6cb7be..5758ff52 100644 --- "a/shoot/\345\237\272\347\241\200\346\246\202\345\277\265/\345\277\253\351\227\250\351\200\237\345\272\246.html" +++ "b/shoot/\345\237\272\347\241\200\346\246\202\345\277\265/\345\277\253\351\227\250\351\200\237\345\272\246.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

快门速度

控制开启传感器的时间

  • 快门速度越短,越能凝固物体,常用在抓拍
  • 快门速度越长,容易得到拖影的图
    • 快门速度
    • 拍静止的水面 (一般需要借助三脚架)
- +
Skip to content

快门速度

控制开启传感器的时间

  • 快门速度越短,越能凝固物体,常用在抓拍
  • 快门速度越长,容易得到拖影的图
    • 快门速度
    • 拍静止的水面 (一般需要借助三脚架)
+ \ No newline at end of file diff --git "a/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/index.html" "b/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/index.html" index 2b5481ef..64ae7d4c 100644 --- "a/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/index.html" +++ "b/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/index.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ - - + + \ No newline at end of file diff --git "a/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\344\272\256\350\260\203\344\272\272\345\203\217.html" "b/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\344\272\256\350\260\203\344\272\272\345\203\217.html" index 6e58b3e9..25a721f7 100644 --- "a/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\344\272\256\350\260\203\344\272\272\345\203\217.html" +++ "b/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\344\272\256\350\260\203\344\272\272\345\203\217.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

亮调人像

百分之 70 都是浅色系

前置

  1. 背景是浅色
  2. 人物衣服也是浅色

如何拍摄

  1. 评价测光
  2. 可适当增加曝光补偿
- +
Skip to content

亮调人像

百分之 70 都是浅色系

前置

  1. 背景是浅色
  2. 人物衣服也是浅色

如何拍摄

  1. 评价测光
  2. 可适当增加曝光补偿
+ \ No newline at end of file diff --git "a/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\345\206\267\346\232\226\345\257\271\346\257\224\344\272\272\345\203\217.html" "b/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\345\206\267\346\232\226\345\257\271\346\257\224\344\272\272\345\203\217.html" index 08312a6e..1a134740 100644 --- "a/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\345\206\267\346\232\226\345\257\271\346\257\224\344\272\272\345\203\217.html" +++ "b/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\345\206\267\346\232\226\345\257\271\346\257\224\344\272\272\345\203\217.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

冷暖对比人像

人物和背景冷暖色系不同

场景

  1. 蓝调时刻 - 太阳刚落山

如何拍摄

  1. 暖色 - 可以通过补光灯
  2. 色温 - 3500 - 4000k
  3. 白平衡 - b2m2
- +
Skip to content

冷暖对比人像

人物和背景冷暖色系不同

场景

  1. 蓝调时刻 - 太阳刚落山

如何拍摄

  1. 暖色 - 可以通过补光灯
  2. 色温 - 3500 - 4000k
  3. 白平衡 - b2m2
+ \ No newline at end of file diff --git "a/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\345\246\202\344\275\225\346\213\215\350\212\261\347\232\204\351\273\221\350\203\214\346\231\257.html" "b/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\345\246\202\344\275\225\346\213\215\350\212\261\347\232\204\351\273\221\350\203\214\346\231\257.html" index b4751a86..ace762b6 100644 --- "a/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\345\246\202\344\275\225\346\213\215\350\212\261\347\232\204\351\273\221\350\203\214\346\231\257.html" +++ "b/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\345\246\202\344\275\225\346\213\215\350\212\261\347\232\204\351\273\221\350\203\214\346\231\257.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

如何拍花

如何拍摄

  1. 采用 点测光 突出花的亮度
  2. 背景采用 深色系 突出主体
- +
Skip to content

如何拍花

如何拍摄

  1. 采用 点测光 突出花的亮度
  2. 背景采用 深色系 突出主体
+ \ No newline at end of file diff --git "a/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\345\246\202\344\275\225\346\213\215\351\233\250\344\270\235.html" "b/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\345\246\202\344\275\225\346\213\215\351\233\250\344\270\235.html" index 0bada13a..709ad00c 100644 --- "a/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\345\246\202\344\275\225\346\213\215\351\233\250\344\270\235.html" +++ "b/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\345\246\202\344\275\225\346\213\215\351\233\250\344\270\235.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

如何拍雨丝

如何拍摄

灯位要求

需要俩盏灯

  • 逆光灯: 在拍摄物体后方 (容易拍摄到雨丝,而且物体会有光边效果)
  • 正光灯:在拍摄物体前方(避免物体逆光,变黑)

逆光灯亮度 > 主光灯

灯位要求

白平衡

如果拍摄物体偏黄,可以调低白平衡,使物体变白

快门速度

  • 快门速度越慢,雨水成丝
  • 快门速度越快,雨水成点
- +
Skip to content

如何拍雨丝

如何拍摄

灯位要求

需要俩盏灯

  • 逆光灯: 在拍摄物体后方 (容易拍摄到雨丝,而且物体会有光边效果)
  • 正光灯:在拍摄物体前方(避免物体逆光,变黑)

逆光灯亮度 > 主光灯

灯位要求

白平衡

如果拍摄物体偏黄,可以调低白平衡,使物体变白

快门速度

  • 快门速度越慢,雨水成丝
  • 快门速度越快,雨水成点
+ \ No newline at end of file diff --git "a/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\346\213\215\346\221\204\346\242\246\345\271\273\345\205\211\346\226\221.html" "b/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\346\213\215\346\221\204\346\242\246\345\271\273\345\205\211\346\226\221.html" index 8e256069..483756d1 100644 --- "a/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\346\213\215\346\221\204\346\242\246\345\271\273\345\205\211\346\226\221.html" +++ "b/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\346\213\215\346\221\204\346\242\246\345\271\273\345\205\211\346\226\221.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

拍摄梦幻光斑

如何拍摄

  • 采用大光圈镜头,有很好的背景虚化效果
  • 需要一个补光灯,减少避免人物逆光导致过黑
    • 拍摄物体在光斑前面
    • 对焦到物体上(也可以通过精细对焦,即放大后手动对焦)
- +
Skip to content

拍摄梦幻光斑

如何拍摄

  • 采用大光圈镜头,有很好的背景虚化效果
  • 需要一个补光灯,减少避免人物逆光导致过黑
    • 拍摄物体在光斑前面
    • 对焦到物体上(也可以通过精细对焦,即放大后手动对焦)
+ \ No newline at end of file diff --git "a/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\346\232\227\350\260\203\344\272\272\345\203\217.html" "b/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\346\232\227\350\260\203\344\272\272\345\203\217.html" index 6e880d3b..5f93bfff 100644 --- "a/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\346\232\227\350\260\203\344\272\272\345\203\217.html" +++ "b/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\346\232\227\350\260\203\344\272\272\345\203\217.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

暗调人像

百分之 70 都是黑色的

前置

  1. 背景是黑色
  2. 人物衣服也是黑色

如何拍摄

  1. 点测光人物面部,保证面部光亮
  2. 为了避免人物和背景重合,可以在人物背面打一束光
  3. 如果周围环境还是比较亮,可以减少一两档曝光补偿

参数参考

  • 光圈:f/2.8、曝光时间 1/125、ISO: 1600
- +
Skip to content

暗调人像

百分之 70 都是黑色的

前置

  1. 背景是黑色
  2. 人物衣服也是黑色

如何拍摄

  1. 点测光人物面部,保证面部光亮
  2. 为了避免人物和背景重合,可以在人物背面打一束光
  3. 如果周围环境还是比较亮,可以减少一两档曝光补偿

参数参考

  • 光圈:f/2.8、曝光时间 1/125、ISO: 1600
+ \ No newline at end of file diff --git "a/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\351\200\206\345\205\211\344\272\272\345\203\217.html" "b/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\351\200\206\345\205\211\344\272\272\345\203\217.html" index 82f368c6..eb991f3c 100644 --- "a/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\351\200\206\345\205\211\344\272\272\345\203\217.html" +++ "b/shoot/\345\256\236\346\210\230\346\212\200\345\267\247/\351\200\206\345\205\211\344\272\272\345\203\217.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

逆光人像

前置

  1. 逆光,光线较好(如夕阳)
  2. 背景建议干净,突出人物

如何拍摄

  1. 采用手动对焦,避免对焦有偏差
  2. 使用点测光,对人物面部最亮部分测光
  3. 前景补光,避免人像面部过暗

灯位要求

- +
Skip to content

逆光人像

前置

  1. 逆光,光线较好(如夕阳)
  2. 背景建议干净,突出人物

如何拍摄

  1. 采用手动对焦,避免对焦有偏差
  2. 使用点测光,对人物面部最亮部分测光
  3. 前景补光,避免人像面部过暗

灯位要求

+ \ No newline at end of file diff --git "a/shoot/\346\236\204\345\233\276\345\275\242\345\274\217/\346\236\204\345\233\276\346\212\200\345\267\247.html" "b/shoot/\346\236\204\345\233\276\345\275\242\345\274\217/\346\236\204\345\233\276\346\212\200\345\267\247.html" index 8e04a228..6ad51742 100644 --- "a/shoot/\346\236\204\345\233\276\345\275\242\345\274\217/\346\236\204\345\233\276\346\212\200\345\267\247.html" +++ "b/shoot/\346\236\204\345\233\276\345\275\242\345\274\217/\346\236\204\345\233\276\346\212\200\345\267\247.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

构图技巧

点构图

中心点构图

将主题放在中心,突出主题

中心点构图

中心点构图

三分点构图

三分点构图

alt text

线构图

对称线构图

可以展示对称的秩序感,常用于风光、风光 + 人文 alt text

alt text

alt text

对角线构图

alt text

alt text

引导线构图

充分利用场景中的线条,用于引导观者的注意力落在主体上,并让画面产生深度和透视感 alt text

面构图

前景构图

例如通过树叶、花草、植物等做前景

alt text

框架构图

拍摄中利用框架将主体框起来,突出主体

alt text

alt text

留白构图

留有空白,让画面更加简洁,突出主题

alt text

alt text

- +
Skip to content

构图技巧

点构图

中心点构图

将主题放在中心,突出主题

中心点构图

中心点构图

三分点构图

三分点构图

alt text

线构图

对称线构图

可以展示对称的秩序感,常用于风光、风光 + 人文 alt text

alt text

alt text

对角线构图

alt text

alt text

引导线构图

充分利用场景中的线条,用于引导观者的注意力落在主体上,并让画面产生深度和透视感 alt text

面构图

前景构图

例如通过树叶、花草、植物等做前景

alt text

框架构图

拍摄中利用框架将主体框起来,突出主体

alt text

alt text

留白构图

留有空白,让画面更加简洁,突出主题

alt text

alt text

+ \ No newline at end of file diff --git "a/shoot/\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223/\345\273\266\350\277\237\346\221\204\345\275\261.html" "b/shoot/\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223/\345\273\266\350\277\237\346\221\204\345\275\261.html" index 1b30b30f..92763078 100644 --- "a/shoot/\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223/\345\273\266\350\277\237\346\221\204\345\275\261.html" +++ "b/shoot/\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223/\345\273\266\350\277\237\346\221\204\345\275\261.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

延迟摄影

如何拍摄

  1. 调节到视频模式

  2. 打开 Menu 菜单,在第一大项的第4小项中找到延时短片,打开这一功能。之后,回到主屏幕,我们发现左上角摄影机图标旁多出了一个时针图案和一组4位数字,这代表延时摄影已打开成功,准备进入拍摄。 Menu 菜单

  3. 此时,主屏幕下方的三个参数均可手动调整,从左到右依次是快门、光圈、感光度ISO Menu 菜单

  4. 自动曝光设置 固定第一帧: 固定第一帧指的是延时短片将始终以第一帧曝光的参数拍摄下去,中途不会发生改变,适用于光源亮度变化不大的场景 每一帧: 延时短片将由相机依据环境光线的变化,自动测光来决定每一帧的曝光值,适用于光源有较大改变的场景,如日转夜或夜转日 (ISO 感光度要设置成 AUTO) 自动曝光

参考文章

  1. https://www.xiaohongshu.com/explore/651389a3000000001e023777
- +
Skip to content

延迟摄影

如何拍摄

  1. 调节到视频模式

  2. 打开 Menu 菜单,在第一大项的第4小项中找到延时短片,打开这一功能。之后,回到主屏幕,我们发现左上角摄影机图标旁多出了一个时针图案和一组4位数字,这代表延时摄影已打开成功,准备进入拍摄。 Menu 菜单

  3. 此时,主屏幕下方的三个参数均可手动调整,从左到右依次是快门、光圈、感光度ISO Menu 菜单

  4. 自动曝光设置 固定第一帧: 固定第一帧指的是延时短片将始终以第一帧曝光的参数拍摄下去,中途不会发生改变,适用于光源亮度变化不大的场景 每一帧: 延时短片将由相机依据环境光线的变化,自动测光来决定每一帧的曝光值,适用于光源有较大改变的场景,如日转夜或夜转日 (ISO 感光度要设置成 AUTO) 自动曝光

参考文章

  1. https://www.xiaohongshu.com/explore/651389a3000000001e023777
+ \ No newline at end of file diff --git "a/shoot/\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223/\346\213\215\346\221\204\346\230\237\347\251\272.html" "b/shoot/\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223/\346\213\215\346\221\204\346\230\237\347\251\272.html" index 70336a92..0d7df886 100644 --- "a/shoot/\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223/\346\213\215\346\221\204\346\230\237\347\251\272.html" +++ "b/shoot/\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223/\346\213\215\346\221\204\346\230\237\347\251\272.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

拍摄星空

大光圈、感光度、长曝光,这是拍摄星空的三大要素。

大光圈

f1.8

感光度

ISO:2000 以上(感光度越大、图片越亮,但也容易出现噪点)

手动对焦

手动对焦到某一颗星星上,然后锁定对焦

长曝光

快门速度 10秒以上, 连续拍10张以上后期堆栈(拍得多更好)

脚架

需要使用三脚架

星空

星空参数

参考文章

  1. https://www.xiaohongshu.com/explore/6507b277000000001e02de1a
- +
Skip to content

拍摄星空

大光圈、感光度、长曝光,这是拍摄星空的三大要素。

大光圈

f1.8

感光度

ISO:2000 以上(感光度越大、图片越亮,但也容易出现噪点)

手动对焦

手动对焦到某一颗星星上,然后锁定对焦

长曝光

快门速度 10秒以上, 连续拍10张以上后期堆栈(拍得多更好)

脚架

需要使用三脚架

星空

星空参数

参考文章

  1. https://www.xiaohongshu.com/explore/6507b277000000001e02de1a
+ \ No newline at end of file diff --git "a/shoot/\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223/\346\213\215\346\221\204\346\234\210\344\272\256.html" "b/shoot/\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223/\346\213\215\346\221\204\346\234\210\344\272\256.html" index 613d7cbf..c8644b6c 100644 --- "a/shoot/\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223/\346\213\215\346\221\204\346\234\210\344\272\256.html" +++ "b/shoot/\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223/\346\213\215\346\221\204\346\234\210\344\272\256.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

拍摄月亮

  • 光圈一般我喜欢小光圈 在 f8 - f10 之间
  • 需要使用长焦镜头,例如 70-200mm
- +
Skip to content

拍摄月亮

  • 光圈一般我喜欢小光圈 在 f8 - f10 之间
  • 需要使用长焦镜头,例如 70-200mm
+ \ No newline at end of file diff --git "a/shoot/\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223/\346\213\215\346\221\204\347\203\237\350\212\261.html" "b/shoot/\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223/\346\213\215\346\221\204\347\203\237\350\212\261.html" index 649fd875..bc9cf197 100644 --- "a/shoot/\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223/\346\213\215\346\221\204\347\203\237\350\212\261.html" +++ "b/shoot/\350\207\252\346\210\221\345\256\236\346\210\230\346\200\273\347\273\223/\346\213\215\346\221\204\347\203\237\350\212\261.html" @@ -11,7 +11,7 @@ - + @@ -23,8 +23,8 @@ -
Skip to content

拍摄烟花

  • 快门速度一般比较长可以出现拉线效果,例如 0.5s/1s
  • ISO 感光度一般在 500 以下,这样背景会黑一些
  • 光圈一般我喜欢小光圈 在 f10 以后

快门速度

- +
Skip to content

拍摄烟花

  • 快门速度一般比较长可以出现拉线效果,例如 0.5s/1s
  • ISO 感光度一般在 500 以下,这样背景会黑一些
  • 光圈一般我喜欢小光圈 在 f10 以后

快门速度

+ \ No newline at end of file