过去流行的是Angular,然后是React,现在是Vue
</ul>。
模板中的条件或循环语句该怎么办?且不说这可能从来都不是一个好主意,你可以只用JS来实现逻辑,然后使用上面的技术将结果插入到模板中。
事件
现在,我们有了基本的模板,那么该如何将事件绑定到DOM节点呢?这里也有几种选择:
HTML事件处理器代码可以被插入到HTML源代码中,但这并非最好的办法,因为指定的处理器只在指定的范围内可用。
事件处理器API可用于所有通过DOMAPI或HTML标记模板字面量函数创建的节点。
那么定制或业务事件该怎么办?如果我需要对应用程序的某个组件触发的一些事件作出反应该怎么办?这里也有多种处理方式:
自定义事件:你可以通过扩展EventTarget来创建自己的事件类,并派发或监听它们,就像“标准”事件一样。
理论上说,使用EventEmitter也是一种办法,但它很少被使用。
观察者模式:你可以构建自己的观察者,也可以考虑使用RxJs,它是这方面的标准。你只需要构建一个Subject,并在发生事件时通知所有订阅者,让订阅者对事件做出反应。
组件
虽说开发普通的应用程序不同于开发复杂的基础设施,但如果一些东西在系统中会多次出现,那么将它们设计成可重用组件仍然是一个好主意。无论你使用何种技术,也无论是业务还是技术,一定程度粒度的抽象仍然是有用的:将与同一业务概念相关的数据和规则封装成一个可重用的对象,或者构建可以在应用程序多个地方进行实例化的小部件,总归是个好主意。
OpenAI CEO:公司已不用API付费客户数据训练大语言模型:金色财经报道,ChatGPT开发公司OpenAI首席执行官Sam Altman表示,公司已经有一段时间没有使用付费客户的数据来训练人工智能大语言模型了;用户们显然希望我们不要使用他们的数据进行训练,所以我们已经改变了我们的计划,未来我们也不会这样做了。网络记录也显示,今年3月1日,OpenAI悄然更新了其服务条款,不会使用通过其API提交的任何数据来服务改进,包括AI模型训练,除非客户或组织选择加入。但OpenAI的新隐私和数据保护仅适用于使用该公司API服务的客户。[2023/5/6 14:46:49]
创建组件的方法有很多,具体视自己的需求而定。早在2017年,Mev-Rael就提出了很多技巧,用于处理JavaScript组件的状态、自定义属性和视图。当然,我们不要拘囿于别人推荐的技术,而是要先考虑自己的需求,然后再选择合适的技术。
除了标准的小部件组件,任何一个组件都应该能够:
将逻辑和视图拆分开。把它们混合在一起通常会导致代码不易于维护,还会降低灵活性。
参数化组件的行为或视图。
通过触发事件的形式通知订阅者组件中发生了某些事件。
同步:如果发生一些事件,组件应该能够进行重绘。这个使用反应式开发库可以很容易实现。
在任何情况下,无论你选择了什么样的设计策略,你的组件都必须能够提供一些HTML渲染结果。你可以使用包含HTML代码的字符串,但HTMLElement通常是更好的选择,而且性能更好。
CoinDesk专栏作家:美国不用担心在CBDC竞赛中落后:CoinDesk专栏作家JP Koning刊文称,美国不用担心在央行数字货币(CBDC)竞赛中落后。这是因为发行央行数字货币没有先发优势。对于许多产品来说,抢占先机对获得品牌优势很重要。但央行数字货币的特点是后发优势,而不是先发优势。最好还是坐下来,向那些缺乏耐心的央行学习。[2020/11/3 11:28:57]
此外,你可能希望使用来自第三方的外部组件。由于专有框架的流行程度较高,它们可以更大程度地利用社区开发的库和组。它们中的大多数实际上与纯JS实现的特性并没有太大不同,但问题是,它们缺乏互操作性,所以到最后你会发现自己需要的其实是纯JS或Web组件。
所幸的是,这样的库确实存在,比如VanillaJSToolkit,尽管可能不太常见。在Web组件方面,webcomponents.org列出了2000多个元素。甚至还有普通的Web组件,只是它们与我们要讨论的不太相关。
路由
在SPA中管理路由需要使用WebHistoryAPI。虽然这并不复杂,但你仍然可能希望将其委托给简单的路由器库,如Navigo。
你所要做的就是在路由时用一个DOM元素替换另一个DOM元素。
延迟加载
按需加载JavaScript代码是任何一个Web应用程序都需要考虑的问题。你一定不希望为了显示一个登录界面而加载全部的应用程序代码。
早在2009年,在Web框架出现之前,JamesBurke就发布了RequireJS来解决这个问题。从那时起,随着模块化的出现,出现了更多的技术。从ES6开始,我们可以动态加载代码。在Node中可以,在浏览器中也可以:
掌柜调查署丨王东临:去中心化数字身份不用担心账号被掠夺:在今日的掌柜调查署上,针对“互联网和区块链的数字身份有何不同之处?”的提问,YottaChain创始人王东临发言指出:互联网上的是中心化的数字身份,区块链上可以打造去中心化的数字身份。中心化的数字身份依赖于一个中心化的机构,这个机构可以掌控你的身份,甚至可以随时封你的账号。去中心化数字身份让用户自己掌握身份,再也不用担心被封号,也不用担心账号里的资产被中心化平台所掠夺。[2020/3/16]
那么如何将模块分拆到单独的文件中?打包器可以为你做这些工作。
需要注意的是,在导入路径里你应该只使用常量,否则打包器就无法猜到你想要加载什么,就会将所有可能的文件都打包在一个文件中。例如,awaitimport(./welcome/${moduleName})将把所有东西都打包到指定的目录中,因为打包器不知道变量moduleName在运行时会是什么。
原生应用程序
越来越多的框架为原生平台提供了运行、迁移或编译应用程序的方法,以便将它们作为独立应用程序部署到Android或iOS移动系统上。
除了考虑开发真正的原生应用程序之外,更普遍的解决方案是将Web应用程序嵌入到原生容器中,比如之前的PhoneGap或ApacheCordova,现在的NativeScript,或者像Electron这样的原生Web应用程序包装器,或者Electron的轻量级后继者Tauri。
服务器端渲染
很多框架在前端和后端运行的代码是相似的,这样更容易实现对SEO友好的服务器端渲染。
声音 | Joseph Young:不用担心BTC退回到6550美元:福布斯分析师Joseph Young发推文称,不用担心BTC退回到6550美元,它不会背叛我们。[2018/10/9]
这可能是一个又酷又便利的特性,但需要注意的是,它也可能导致服务器锁定。因此,在向应用程序引入框架锁定之前,你需要考虑它对项目、基础设施、客户端技术等方面的影响。
所幸的是,你也可以在不使用框架的情况下实现这个特性。
从服务器端渲染
采用普通的实现方案在一开始看起来很简单:不就是返回HTML吗?是的,你已经有现成的组件了,但是:
你还需要一个服务器端DOMAPI,因为默认情况下,服务器端不提供DOMAPI。
你的渲染组件不能假设是DOM是在客户端或服务器端,也就是说,不要使用全局DOM,因为在服务器端,每个请求都需要一个DOM。要做到这一点,你需要从应用程序上下文中选择DOM对象,而不是直接获取。
在客户端和服务器应用程序之间共享渲染组件有多种办法,比如将其发布在包存储库中,但最灵活的应该是让应用程序包引用monorepo中的模块。
添加交互性
然而,一旦HTML元素被转换成字符串,在这些元素上设置的所有事件处理器都丢失了。为了恢复交互性,你需要一些“补水”步骤,也就是注入脚本,让它们在客户端执行。框架因其普适性很难做到这一点。就拿影子DOM来说,它们不断尝试改进算法,希望能够以最聪明的方式做到这一点,但如果我们把问题缩小到应用程序层面,就会变得简单很多。
声音 | 陈军:联盟链可以不用发行代币进行激励:本期金色相对论中,对于无币区块链的共识及其技术如何应用在场景中,杭州量子大学区块链研究院联合执行院长陈军表示,无币不是没有Token,它是区块链记账的一部分,只是当没有资产上链的时候,Token就是代币,当资产上链时Token就是对资产的标记,这个时候如果再发行一个用来为这个资产定价的Token,那就是代币。
无币区块链应用在联盟链、私有链环境下,私有链肯定不需要共识,联盟链的也比较容易实现,通常是轮流记账,如果联盟太大,就需随机排队加上轮流记账,这个可以不用激励,只是解决公正公平问题。如果一定要激励,直接用收取的法币交易手续费来激励就可以。
所以在联盟链环境下一般都可以用法币做支付,不需要发行代币,但会有代表交易标的Token,方便查询和追溯。[2018/9/29]
当然,在普通的服务器应用程序中做到这一点也意味着需要将JS脚本注入到响应消息中。
普通的解决方案让你可以控制在哪里、什么时候以及附加哪些东西:你可以先只发送HTML,再加载基本的交互性JavaScript,然后加载更多,等等。
这比本文中提到的任何一个东西都简单,因为它们是应用程序代码,而不是通用的框架代码。
国际化
多年来,国际化问题都是通过库来处理的。要自己集成这些库也很容易,但你也可以选择自己实现一个,因为与通用库相比,自己的实现可以支持更简单、更有效的消息类型。
这里为你提供了:
类型检查:每个消息都有一个静态类型,所以IDE可以检查你是否使用了有效的消息属性,并为你提供自动补全功能。
翻译完整性检查:在为所有消息键提供所有语言的翻译之前,无法通过编译。
你所需要做的就是实例化与用户语言环境相关的消息类。通用库不会提供这种特定于业务的消息类型。
工具
如果你想要摆脱对强约束性软件技术栈的依赖,那你很可能也想摆脱对工具的依赖:你不希望只有靠着它们才能向前走。你不希望被一个你无法解决的构建问题所困扰。
话虽如此,你仍然很难避免使用这些工具。大多数情况下,你的产品代码必须以某种方式打成包,包括缩小体积、混淆、代码拆分、摇树优化、延迟加载、包含样式等。毫无疑问,现有的打包工具如Webpack、Parcel、ESBuild或Vite会做得比你更好。
你所能做的是:
尽可能少用转译。例如,使用TypeScript可能是件好事,但它会带来额外的复杂性,你的工具链中必须有相应的工具来处理这种复杂性。CSS也一样,特别是最新版本,不值得你用预处理器来处理它们。
尽可能少用工具。你用的工具越多,就越有可能出问题或无法满足你的需求。
如果确实需要使用工具,请选择最流行的工具,因为它们经过实战测试,更有可能满足你的需求。过早使用最新的打包工具可能会为你节省几秒钟的构建时间,但这些时间很可能都不够用来理解工具文档、处理bug或处理因缺乏支持而导致的问题。
最大的挑战
说到底,最大的挑战不是技术上的,而是关于人的:
你要走出舒适区。希望你终将能够明白,使用普通的解决方案并不是那么困难,框架的复杂性比它们带来的好处要大得多。此外,你可能会看到更多新的API,而且Web比你想象的更现代、更强大。
至于其他人,你可以尝试说服他们。他们可能不愿意这么做,因为任何人都不愿意开启自己从未尝试过的旅程。
其他人可能会跟你说:
“你要开发自己的框架”:不,我们要开发的是应用程序,而不是框架。
“你要写更多的代码”:也许,但也许不会太多,因为这需要与框架的样板代码进行比较。但不管怎样,需要加载的代码都会更少。
“你将不断地重新发明轮子”:当然不是。不使用框架是为了不遵循它们预定义的规则,但我们并没有忘记DRY原则,我们仍然可以使用经过实战测试的第三方库。
“你需要为每一个功能写更多的代码”:不,你可以遵循自己的规则,而不是使用框架样板代码。
“没有文档可看”:肯定不会有框架文档,但你仍然需要写应用程序文档。值得一提的是,使用模式有助于你自动文档化你的软件设计。你只需要关心应用程序的代码文档,而如果你多使用一个框架,就需要多看一份文档。
“不会有约束或模式来指导开发人员”:不,如果你确实需要约束,没有什么能阻止你。
“你会错过性能提升”,比如曾经被大肆炒作的虚拟Dom:不,因为需要这些“性能提升”的是框架本身,而不是应用程序。相反,通用框架更有可能错过一些可以通过自定义代码实现的性能提升。
你遇到这个问题是因为你没有使用框架。每一个问题都会被归咎于因为没有使用框架。因为大多数开发人员的经验是,所有正常运行的东西都使用了框架,默认情况下,不使用它们将被认为是有风险的。一旦出现问题,无论是否与不使用框架有关,这个假设都会被认为是正确的。他们忘记了在使用框架时也会遇到类似的问题。
“我们找不到开发者”:他们会说很难找到能够写纯JS代码的开发者。这句话是对的,也是错的。因为很多开发者会发现自己更习惯于使用框架。如果他们从来没有使用过或不了解基本的WebAPI,那么他们可能会对从零开始构建一个Web应用程序感到害怕。但是,如果你想要开发高质量的应用程序,就不应该去找这种类型的开发者。当然,现在找React开发者很容易,但你需要的不只是React开发者,而是优秀的开发者。
“你无法获得与框架相同的代码质量”。当然,框架或开发库通常是由行业里有经验的开发者编写的。但是,框架的代码主要与框架特定的活动相关,与你的应用程序无关。此外,即使使用了框架,你仍然可能做出糟糕的设计,写出糟糕的代码。应用程序的质量总是更多地取决于团队的质量,而不是因为缺少框架。
“你无法获得与框架相同的性能”:不,我们可以获得更好的性能。行业里关于框架采用了可以“提升性能”的复杂技术的说法就不在这里讨论了,因为它们可能主要被用来解决框架通用解决方案的性能缺陷。
毫无疑问,性能最好的框架是那些在普通代码之上添加层数较少的框架。框架的“优化”更多的是为了弥补框架本身的开销。
结??论
不使用框架构建Web应用程序并非意味着要自己构建框架,它是关于在不使用通用引擎的情况下开发应用程序,目的是:
避免散失控制和被隐含约束;
可以进行优化。
也就是只编写特定于应用程序的代码,包括使用开发库。你真正应该关注的框架是你自己的框架,也就是那个特定于应用程序的框架。这是真正的“专注于业务”,也是最有效的。
这并没有你想象的那么难,特别是有了现代标准的加持。
郑重声明: 本文版权归原作者所有, 转载文章仅为传播更多信息之目的, 如作者信息标记有误, 请第一时间联系我们修改或删除, 多谢。