13518219792

建站动态

根据您的个性需求进行定制 先人一步 抢占小程序红利时代

盘点Solid.js源码中的那些迷惑行为

前言

我研究 Solid.js 源码已经有一段时间了,在钻研的过程中我发现了其中的一些迷惑行为,在搞懂之后终于恍然大悟,忍不住想要分享给大家。不过这么说其实也不太准确,因为在严格意义上来讲 Solid.js 其实是被划分为了两个部分的。我只认真钻研了其中一个部分,所以也不能说钻研 Solid.js 源码,因为另外一个部分压根就不叫 Solid。

为改则等地区用户提供了全套网页设计制作服务,及改则网站建设行业解决方案。主营业务为网站设计制作、网站制作、改则网站设计,以传统方式定制建设网站,并提供域名空间备案等一条龙服务,秉承以专业、用心的态度为用户提供真诚的服务。我们深信只要达到每一位用户的要求,就会得到认可,从而选择与我们长期合作。这样,我们也可以走得更远!

两部分

有些同学看到这可能就会感到疑惑了,哪两个部分?Solid、.js?其实是这样:大家应该都听说过 Solid.js 是一个重编译、轻运行的框架吧,所以它可以被分为编译器和运行时两个部分。

那有人可能会问:你要是这么说的话那岂不是 Vue 也可以被分为两部分,毕竟 Vue 也有编译器和运行时,为什么从来没有人说过 Vue 是两部分组成的呢?是这样,Vue 的编译器和运行时全都放在了同一仓库内的 Monorepo 中:

你可以说 Vue2 和 Vue3 是两个部分,因为它俩被放在了两个不同的仓库中:

虽然它俩已经是两个不同的仓库了,但好歹也都是 vuejs 名下的吧:

而 Solid.js 的两部分不仅不在同一个仓库内,甚至连组织名都不一样:

一个是 solidjs/solid:

而另一个则是 ryansolid/dom-expressions:

ryan 是 Solid.js 作者的名字,所以 ryan + solid = ryansolid(有点迷,为啥不放在 solidjs 旗下非要单独开一个 ryansolid)

这个 dom-expressions 就是 Solid.js 的编译器,那为啥不像 Vue 编译器似的都放在同一个仓库内呢?因为 Vue 的编译器就是专门为 Vue 设计的,你啥时候看非 Vue项目中使用 xxx.vue 这样的写法过?

.vue 这种单文件组件就只有 Vue 使用,虽说其他框架也有单文件组件的概念并且有着类似的写法(如:xxx.svelte)但人家 Svelte也不会去用 Vue 的编译器去编译人家的 Svelte 组件。不过 Solid 不一样,Solid没自创一个 xxx.solid,而是明智的选择了 xxx.jsx。

SFC VS JSX

单文件组件和 jsx 各有利弊,不能说哪一方就一定比另一方更好。但对于一个声明式框架作者而言,选择单文件组件的好处是可以自定义各种语法,并且还可以牺牲一定的灵活性来换取更优的编译策略。缺点就是成本太高了,单单语法高亮和 TS 支持度这一方面就得写一个非常复杂的插件才能填平。

好在 Vue 的单文件组件插件 Volar 已经可以支持自定义自己的单文件组件插件了,这有效的降低了框架作者的开发成本。但 Solid 刚开始的时候还没有 Volar 呢(可以去看看 Volar 的源码有多复杂 这还仅仅只是一个插件就需要花费那么多时间和精力),甚至直到现在 Volar 也没个文档,就只有 Vue 那帮人在用 Volar(毕竟是他们自己研究的):

并且人家选择 jsx 也有可能并非是为了降低开发成本,而是单纯的钟意于 jsx 语法而已。那么为什么选择 jsx 会降低开发成本呢?首先就是不用自己写 parser、generator 等一堆编译相关的东西了,一个 babel 插件就能识别 jsx 语法。语法高亮、TS 支持度这方面更是不用操心,甚至用户都不需要为编辑器安装任何插件(何时听过 jsx 插件)。

并且由于 React 是全球占有率最高的框架,jsx 已被广泛接受(甚至连 Vue 都支持 jsx)但如果选择单文件组件的话又会产生有人喜欢这种写法有人喜欢那种写法的问题,比方说同样使用 sfc 的 Vue 和 Svelte,if-else 写法分别是这样:

{#if xxx}
  

{:else}
{/if}

有人喜欢上面那种写法就有人喜欢下面那种写法,众口难调,无论选择哪种写法可能都会导致另一部分的用户失望。而 jsx 就灵活的多了,if-else 想写成什么样都可以根据自己的喜好来:

if (xxx) {
  return 

} else { return
} // 或者 return xxx ?

:
// 亦或 let Title = 'h1' if (xxx) Title = 'div' return </code></pre><p>jsx 最大程度的融合了 js,正是因为它对 js 良好的兼容性才导致它的适用范围更广,而不是像 Vue、Svelte 那样只适用于自己的框架。</p></p><p>毕竟每种模板语言的 if-else、循环等功能写法都不太一样,当然 jsx 里的 if-else 也可以有各种千奇百怪的写法,但毕竟还是 js 写法,而不是自创的 ng-if、v-else、{:else if} {% for i in xxx %}等各种不互通的写法。</p></p><p>正是由于 jsx 的这个优势导致了很多非 React 框架(如:Preact、Stancil、Solid 等)用 jsx 也照样用的飞起,那么既然 jsx 可以不跟 React 绑定,那 Ryan 自创的 jsx编译策略也同样可以不跟 Solid 绑定啊对不对?</p></p><p>这是一款可以和 Solid.js 搭配使用的 babel 插件,也同样是一款可以和 MobX、和 Knockout、和 S.js、甚至和 Rx.js 搭配使用的插件,只要你有一款响应式系统,那么 dom-expressions 就可以为你提供 jsx 服务。</p> <h4>Solid.js</h4> <p>所以这才是 Ryan 没把 dom-expressions 放在 solidjs/solid 里的重要原因之一,但 Solid.js 又是一个注重编译的框架,没了 dom-expressions 还不行,所以只能说 Solid.js 是由两部分组成的。</p></p> </p> <h3>DOM Expressions</h3> </p><p>DOM Expressions 翻译过来就是 DOM 表达式的意思,有人可能会问那你标题为啥不写成《盘点 DOM Expressions 源码中的那些迷惑行为》?拜托!谁知道 DOM Expressions 到底是个什么鬼!</p></p><p>如果不是我苦口婆心的说了这么多,有几个能知道这玩意就是 Solid.js 的编译器,甭说国内了,就连国外都没几个知道 DOM Expressions的。你要说 Solid.js 那别人可能会竖起大拇指说声 Excellent,但你要说 DOM Expressions 那别人说的很可能就是 What the fuck is that? 了。不信你看它俩的对比:</p></p><p> </p><p> </p><p>再来看看 Ryan 在油管上亲自直播 DOM Expressions时的惨淡数据:</p></p><p> </p><p>这都没我随便写篇文章的点赞量高,信不信如果我把标题中的 Solid.js 换成了 DOM Expression 的话点赞量都不会有 Ryan 直播的数据好?好歹人家还是 Solid的作者,都只能获得如此惨淡的数据,那更别提我了。</p></p><p>言归正传,为了防止大家不知道 Solid.js 编译后的产物与 React 编译后的产物有何不同,我们先来写一段简单的 jsx:</p></p> <pre><code>import c from 'c' import xxx from 'xxx' export function Component () { return ( <div a="1" b={2} c={c} onClick={() => {}}> { 1 + 2 } { xxx } </div> ) }</code></pre> </p><p>React 编译产物:</p> <pre><code>import c from 'c'; import xxx from 'xxx'; import { jsxs as _jsxs } from "react/jsx-runtime"; export function Component() { return /*#__PURE__*/_jsxs("div", { a: "1", b: 2, c: c, onClick: () => {}, children: [1 + 2, xxx] }); }</code></pre><p>Solid 编译产物:</p></p> <pre><code>import { template as _$template } from "solid-js/web"; import { delegateEvents as _$delegateEvents } from "solid-js/web"; import { insert as _$insert } from "solid-js/web"; import { setAttribute as _$setAttribute } from "solid-js/web"; const _tmpl$ = /*#__PURE__*/_$template(`<div a="1" b="2">3`); import c from 'c'; import xxx from 'xxx'; export function Component() { return (() => { const _el$ = _tmpl$(), _el$2 = _el$.firstChild; _el$.$$click = () => {}; _$setAttribute(_el$, "c", c); _$insert(_el$, xxx, null); return _el$; })(); } _$delegateEvents(["click"]);</code></pre> </p><p>Solid 编译后的产物乍一看有点不太易读,我来给大家写一段伪代码,用来帮助大家快速理解 Solid 到底把那段 jsx 编译成了啥:</p></p> <pre><code>import c from 'c'; import xxx from 'xxx'; const template = doucment.createElement('template') template.innerHTML = '<div a="1" b="2">3</div>' const el = template.content.firstChild.cloneNode(true) // 大家可以简单的理解为 el 就是 <div a="1" b="2">3</div> export function Component() { return (() => { el.onclick = () => {}; el.setAttribute("c", c); el.insertBefore(xxx); return el; })(); }</code></pre> </p><p>这样看上去就清晰多了吧?直接编译成了真实的 DOM 操作,这也是它性能为何能够如此强悍的原因之一,没有中间商(虚拟DOM)赚差价。但大家有没有感觉有个地方看起来好像有点多此一举,就是那个自执行函数:</p> <pre><code>export function Component() { return (() => { el.onclick = () => {}; el.setAttribute("c", c); el.insertBefore(xxx); return el; })(); }</code></pre><p>为何不直接编译成这样:</p> <pre><code>export function Component() { el.onclick = () => {}; el.setAttribute("c", c); el.insertBefore(xxx); return el; }</code></pre><p>效果其实都是一样的,不信你试着运行下面这段代码:</p></p> <pre><code>let num = 1 console.log(num) // 1 num = (() => { return 1 })() console.log(num) // 还是 1 但感觉多了一个脱裤子放屁的步骤</code></pre> </p><p>看了源码才知道,原来看似多此一举的举动实则是有苦衷的。因为我们这是典型的站在上帝视角来审视编译后的代码,源码的做法是只对 jsx 进行遍历,在刚刚那种情况下所编译出来的代码确实不是最优解,但它能保证在各种的场景下都能正常运行。</p></p><p>我们来写一段比较罕见的代码大家就能明白过来怎么回事了:</p></p> <pre><code>if (<div a={value} onClick={() => {}} />) { // do something… }</code></pre> </p><p>当然这么写没有任何的意义,这是为了帮助大家理解为何 Solid 要把它的 jsx 编译成一段自执行函数才会写成这样的。我们来写一段伪代码,实际上 Solid 编译出来的并不是这样的代码,但相信大家能够明白其中的含义:</p> <pre><code><div a={value} notallow={() => {}} /> // 将会被编译成 const el = document.createElement('div') el.setAttribute('a', value) el.onclick = () => {}</code></pre><p>发现问题所在了么?原本 jsx 只有一行代码,但编译过后却变成三行了。所以如果不加一个自执行函数的话将会变成:</p></p> <pre><code>if (const el = document.createElement('div'); el.setAttribute('a', value); el.onclick = () => {}) { // do something… }</code></pre> </p><p>这很明显是错误的语法,if 括号里根本不能写成这样,会报错的!但如果把 if 括号里的代码放在自执行函数中那就没问题了:</p> <pre><code>if ((() => { const el = document.createElement('div') el.setAttribute('a', value) el.onclick = () => {} return el })()) { // do something… }</code></pre><p>我知道肯定有人会说把那三行代码提出去不就得了么:</p></p> <pre><code>const el = document.createElement('div') el.setAttribute('a', value) el.onclick = () => {} if (el) { // do something… }</code></pre> </p><p>还记得我之前说过的那句:我们是站在上帝视角来审判 Solid 编译后代码的么?理论上来说这么做确实可以,但编译成本无疑会高上许多,因为还要判断 jsx 到底写在了哪里,根据上下文的不同来生成不同的代码,但这样肯定没有只编译 jsx 而不管 jsx 到底是被写在了哪里来的方便。而且我们上述的那种方式也不是百分百没问题的,照样还是会有一些意想不到的场景:</p></p> <pre><code>for (let i = 0, j; j = <div a={i} />, i < 3; i++) { console.log(j) }</code></pre> </p><p>但假如按照我们那种策略来编译代码的话:</p></p> <pre><code>const el = document.createElement('div') el.setAttribute('a', i) for (let i = 0, j; j = el, i < 3; i++) { console.log(j) }</code></pre> </p><p>此时就会出现问题,因为 el 用到了变量 i,而 el 又被提到外面去了所以访问不到 i变量,所以 el 这几行代码必须要在 jsx 的原位置上才行,只有自执行函数能够做到这一点。由于 js 是一门极其灵活的语言,各种骚操作数不胜数,所以把编译后的代码全都加上一段自执行函数才是性价比最高并且最省事的选择之一。</p></p> </p> <h3>迷之叹号️</h3> </p><p>有次在用 playground.solidjs.com 编译 jsx 时惊奇的发现:</p></p><p> </p><p>不知大家看到这段 <h1>Hello, <!>!</h1> 时是什么感受,反正我的第一感觉就是出 bug 了,把我的叹号 ! 给编译成 <!> 了。</p></p><p>但令人摸不着头脑的是,这段代码完全可以正常运行,没有出现任何的 bug。随着测试的深入,发现其实并不是把我的叹号 ! 给编译成 <!> 了,只是恰巧在那个位置上我写了个叹号,就算不写叹号也照样会有这个 <!>的:</p></p><p> </p><p>发现没?<!> 出现的位置恰巧就是 {xxx} 的位置,我们在调试的时候发现最终生成的代码其实是这样:</p></p> <pre><code><h1>1<!---->2</h1></code></pre> </p><p>也就是说当我们 .innerHTML = '<!>' 的时候其实就相当于 .innerHTML = '' 了,很多人看到这个空注释节点以后肯定会联想到 Vue,当我们在 Vue 中使用 v-if="false" 时,按理说这个节点就已经不复存在了。但每当我们打开控制台时就会看到原本 v-if 的那个位置变成了这样:</p></p><p> </p><p>尤雨溪为何要留下一个看似没有任何意义的空注释节点呢?广大强迫症小伙伴们忍不了了,赶忙去 GitHub 里开个 issue 问尤雨溪:</p></p><p> </p><p>尤雨溪给出的答案是这样:</p></p><p> </p><p>那 Solid 加一个这玩意也是和 Vue 一样的原由么?随着对源码的深入,我发现它跟 Vue 的  原由并不一样,我们再来用一段伪代码来帮助大家理解 Solid 为什么需要一段空注释节点:</p> <pre><code><h1>1{xxx}2</h1> // 将会被编译成: const el = template('<h1>12</h1>') const el1 = el.firstChild // 1 const el2 = el1.nextSibling // const el3 = el2.nextSibling // 2 // 在空节点之前插入 xxx 而空节点恰好就在 1 2 之间 所以就相当于在 1 2 之间插入了 xxx el.insertBefore(xxx, el2)</code></pre><p>看懂了么,Solid 需要在 1 和 2 之间插入 xxx,如果不加这个空节点的话那就找不到该往哪插了:</p></p> <pre><code><h1>1{xxx}2</h1> // 假如编译成没有空节点的样子: const el = template('<h1>12</h1>') const el1 = el1.firstChild // 12 const el2 = el2.nextSibling // 没有兄弟节点了 只有一个子节点:12 el.insertBefore(xxx, 特么的往哪插?)</code></pre> </p><p>所以当大家在 playground.solidjs.com 中发现有 <!> 这种奇怪符号时,请不要觉得这是个 bug,这是为了留个占位符,方便 Solid 找到插入点。只不过大多数人都想不到,把这个 <!> 赋值给 innerHTML 后会在页面上生成一个 <!---->。</p></p> </p> <h3>迷之 ref</h3> </p><p>无论是 Vue 还是 React 都是用 ref 来获取 DOM 的,Solid 的整体 API 设计的与 React 较为相似,ref 自然也不例外:</p></p><p> </p><p>但它也有自己的小创新,就是 ref 既可以传函数也可以传普通变量。如果是函数的话就把 DOM 传进去,如果是普通变量的话就直接赋值:</p></p> <pre><code>// 伪代码 <h1 ref={title} /> // 将会编译成: const el = document.createElement('h1') typeof title === 'function' ? title(el) : title = el</code></pre> </p><p>但在查看源码时发现了一个未被覆盖到的情况:</p> <pre><code>// 简化后的源码 transformAttributes () { if (key === "ref") { let binding, isFunction = t.isIdentifier(value.expression) && (binding = path.scope.getBinding(value.expression.name)) && binding.kind === "const"; if (!isFunction && t.isLVal(value.expression)) { ... } else if (isFunction || t.isFunction(value.expression)) { ... } else if (t.isCallExpression(value.expression)) { ... } } }</code></pre><p>稍微给大家解释一下,这个 transformAttributes 是用来编译 jsx 上的属性的:</p></p><p> </p><p>当 key 等于 ref 时需要进行一些特殊处理,非常迷的一个命名就是这个 isFunction,看名字大家肯定会认为这个变量代表的是属性值是否为函数。我来用人话给大家翻译一下这个变量赋的值代表什么含义:t.isIdentifier(value.expression)的意思是这个 value 是否为变量名:</p></p><p> </p><p>比方说 ref={a} 中的 a 就是个变量名,但如果是 ref={1}、ref={() => {}}那就不是变量名,剩下那俩条件是判断这个变量名是否是 const 声明的。也就是说:</p></p> <pre><code>const isFunction = value 是个变量名 && 是用 const 声明的</code></pre> </p><p>这特么就能代表 value 是个 function 了?</p></p><p>在我眼里看来这个变量叫 isConst 还差不多,我们再来梳理一下这段逻辑:</p></p> <pre><code>// 简化后的源码 transformAttributes () { if (key === "ref") { const isConst = value is 常量 if (!isConst && t.isLVal(value.expression)) { ... } else if (isConst || t.isFunction(value.expression)) { ... } else if (t.isCallExpression(value.expression)) { ... } } }</code></pre> </p><p>接下来就是 if-else 条件判断里的条件了,再来翻译下,t.isLVal 代表的是:value 是否可以放在等号左侧,这是什么意思呢?一个例子就能让大家明白:</p></p> <pre><code>// 此时 key = 'ref'、value = () => {} <h1 ref={() => {}} /> // 现在我们需要写一个等号 看看 value 能不能放在等号的左侧: () => {} = xxx // 很明显这是错误的语法 所以 t.isLVal(value.expression) 是 false // 但假如写成这样: <h1 ref={a.b.c} /> a.b.c = xxx // 这是正确的语法 所以 t.isLVal(value.expression) 现在为 true</code></pre><p>明白了 t.isLVal 接下来就是 t.isFunction 了,这个从命名上就能看出来是判断是否为函数的。然后就是 t.isCallExpression,这是用来判断是否为函数调用的:</p></p> <pre><code>// 这就是 callExpression xxx()</code></pre> </p><p>翻译完了,接下来咱们就来分析一遍:</p><p>当 value 不是常量并且不能放在等号左侧时(这种情况有处理)</p><p>当 value 是常量或者是一个函数字面量时(这种情况有处理)</p><p>当 value 是一个正在调用的函数时(这种情况有处理)</p></p><p>不知大家看完这仨判断后有什么感悟,反正当我捋完这段逻辑的时候感觉有点迷,因为好像压根儿就没覆盖掉全部情况啊!咱们先这么分一下:value 肯定是变量名、字面量以及常量中的其中一种对吧?是常量的情况下有覆盖,不是常量时就有漏洞了,因为它用了个并且符号 &&,也就是说当 value 不是常量时必须还要同时满足不能放在等号左侧这种情况才会进入到这个判断中去,那假如我们写一个三元表达式或者二元表达式那岂不就哪个判断也没进么?不信我们来试一下:</p></p><p> </p><p>可以看到编译后的 abc 三个变量直接变暗了,哪都没有用到这仨变量,也就是说相当于吞掉了这段逻辑(毕竟哪个分支都没进就相当于没处理)不过有人可能会感到疑惑,三元表达式明明能放到等号左侧啊:</p></p><p> <p> </p><p>实际上并不是你想的那样,等号和三元表达式放在一起时有优先级关系,调整一下格式你就明白是怎样运行的了:</p> <pre><code>const _tmpl$ = /*#__PURE__*/_$template(`<h1>Hello`) a ? b : c = 1 // 实际上相当于 a ? b : (c = 1) // 相当于 if (a) { b } else { c = 1 }</code></pre><p>如果我们用括号来把优先级放在三元这边就会直接报错了:</p></p><p> <p> </p><p>二元表达式也是同理:</p><p> <p> </p><p>我想在 ref 里写成这样没毛病吧:</p></p> <pre><code><h1 ref={a || b} /></code></pre> </p><p>虽然这种写法比较少见,但这也不是你漏掉判断的理由呀!毕竟好多用 Solid.js 的人都是用过 React 的,他们会把在 React 那养成的习惯不自觉的带到 Solid.js 里来,而且这不也是 Solid.js 把 API 设计的尽可能与 React 有一定相似性的重要原因之一吗?</p></p><p>但人家在 React 没问题的写法到了你这就出问题了的话,是会非常影响你这框架的口碑的!而且在文档里还没有提到任何关于 ref 不能写表达式的说明:</p></p><p> </p><p>后来我仔细想了一下,发现还真不是他们不小心漏掉的,而是有意为之。至于为什么会有意为之那就要看它编译后的产物了:</p></p> <pre><code>// 伪代码 <div ref={a} /> // 将会被编译为: const el = template(`<div>`) typeof a === 'function' ? a(el) : a = el</code></pre> </p><p>其中咱们重点看 a = el 这段代码,a 就是我们写在 ref 里的,但假如我们给它换成一个二元表达式就会变成:</p> <pre><code>// 伪代码 <div ref={a || b} /> // 将会被编译为: const el = template(`<div>`) a || b = el</code></pre><p>a || b 不能放在等号左侧,所以源码中的 isLVal 就是为了过滤这种情况的。那为什么不能编译成:</p> <pre><code>(a = el) || (b = el)</code></pre><p>这么编译是错的,因为假如 a 为 false,a 就不应该被赋值,但实际上 a 会被赋值为 el:</p></p><p> <p> </p><p>所以要把二元编译成三元:</p><p> <p> </p><p>如果是并且符号就要编译成取反:</p></p> <pre><code>// 伪代码 <div ref={a && b} /> // 将会被编译为: const el = template(`<div>`) !a ? a = el : b = el</code></pre> </p><p>然后三元表达式以及嵌套三元表达式:</p></p> <pre><code><div ref={ Math.random() > 0.5 ? refFactory() && refArr[0] && (refTarget1 = refTarget2) && (refTarget1 > refTarget2) : refTarget1 ? refTarget2 : refTarget3 } /></code></pre> </p><p>当然可能并不会有人这么写,Solid 那帮人也是这么想的,所以就算了,太麻烦了,如果真要是有复杂的条件的话可以用函数:</p></p> <pre><code><div ref={ el => Math.random() > 0.5 ? refTarget1 = el : refTarget2 = el } /></code></pre> </p><p>就先不管 isLVal 为 false 的情况了,不过我还是觉得至少要在官网上提一嘴,不然真有人写成这样的时候又搜不到答案的话那多影响口碑啊!</p></p> <h3>总结</h3> </p><p>看过源码之后感觉有的地方设计的很巧妙,但有些地方又不是很严谨。也怪 jsx 太灵活了,不可能做判断把所有情况都做到面面俱到,当你要写一些在 React 里能运行的骚操作可能在 Solid 里就哑火了。</p> <br> 网站题目:盘点Solid.js源码中的那些迷惑行为 <br> 网页网址:<a href="http://cdbrznjsb.com/article/coggesc.html">http://cdbrznjsb.com/article/coggesc.html</a> </div> <div class="other"> <h3>其他资讯</h3> <ul> <li><a href="/article/djhjhse.html">工作总结VB.NET文件操作三个方法</a></li><li><a href="/article/djhjhdj.html">数据加密存储都包含哪些</a></li><li><a href="/article/djhjheg.html">云计算如何使企业的业务受益?</a></li><li><a href="/article/djhjhsd.html">windows系统如何查看计算机的基本信息/?windows常用消息</a></li><li><a href="/article/djhjhci.html">oppo手机连无线网老慢了为什么</a></li> </ul> </div> </div> <div class="oneE"> <div class="oneEa container wow fadeInUp"> <ul> <li> <dd><img src="/Public/Home/img/oe1.png" alt=""></dd> <h3>网站建设专属方案</h3> </li> <li> <dd><img src="/Public/Home/img/oe2.png" alt=""></dd> <h3>网站定制化设计</h3> </li> <li> <dd><img src="/Public/Home/img/oe3.png" alt=""></dd> <h3>7X24小时服务</h3> </li> <li> <dd><img src="/Public/Home/img/oe4.png" alt=""></dd> <h3>N对管家服务</h3> </li> </ul> </div> <div class="oneEb container wow fadeInUp"> <h2>让你的专属顾问为你服务</h2> <form action=""> <input type="text" placeholder="需求"> <input type="text" placeholder="输入你的联系方式(微信或电话号码)"> <button>立即联系</button> </form> </div> </div> <footer> <div class="foot container"> <div class="footl"> <img src="/Public/Home/img/logo.png" alt=""> <p>用前卫的视觉</p> <p>把握好每一个细节</p> </div> <div class="footc"> <dl> <dt>服务项目</dt> <dd><a href="">网站建设</a></dd> <dd><a href="">网站优化</a></dd> <dd><a href="">网站设计</a></dd> <dd><a href="">小程序开发</a></dd> <dd><a href="">电商平台</a></dd> </dl> <dl> <dt>客户案例</dt> <dd><a href="">网站案例</a></dd> <dd><a href="">优化案例</a></dd> <dd><a href="">外贸网站案例</a></dd> </dl> <dl> <dt>资讯中心</dt> <dd><a href="">建站动态</a></dd> <dd><a href="">网站知识</a></dd> <dd><a href="">网站运营</a></dd> </dl> <dl> <dt>快捷导航</dt> <dd><a href="">关于澄梦希喜</a></dd> <dd><a href="">联系方式</a></dd> </dl> </div> <div class="footr"> <h3>联系方式</h3> <p>地址:成都市太升南路288号锦天国际A幢1002号</p> <div class="tel"> <i><img src="/Public/Home/img/ftel.png" alt=""></i><a href="tel:13518219792">电话:13518219792</a> </div> </div> </div> <div class="yqlink container"> 标签: <a href="http://www.zsjierui.cn/" target="_blank">资阳</a> <a href="http://www.wzjierui.cn/" target="_blank">温江</a> <a href="http://www.ndjierui.cn/" target="_blank">南部</a> <a href="http://www.ptjierui.cn/" target="_blank">郫县</a> <a href="http://www.hzjierui.cn/" target="_blank">彭州</a> <a href="http://www.ncjierui.cn/" target="_blank">彭山</a> <a href="http://www.whjierui.cn/" target="_blank">乐山</a> <a href="http://www.ahjierui.cn/" target="_blank">简阳</a> <a href="http://www.csjierui.cn/" target="_blank">绵阳</a> <a href="http://www.qhjierui.cn/" target="_blank">德阳</a> <a href="http://www.scjierui.cn/" target="_blank">四川</a> <a href="http://www.tjjierui.cn/" target="_blank">什邡</a> <a href="http://www.tyjierui.cn/" target="_blank">绵竹</a> <a href="http://www.xzjierui.cn/" target="_blank">眉山</a> <a href="http://www.sxjierui.cn/" target="_blank">双流</a> <a href="http://www.ptruijie.cn/" target="_blank">新都</a> <a href="http://www.xjjierui.cn/" target="_blank">新津</a> <a href="http://www.jljierui.cn/" target="_blank">龙泉</a> <a href="http://www.gyruijie.cn/" target="_blank">广汉</a> <a href="http://www.csruizhi.cn/" target="_blank">崇州</a> <a href="http://www.zjjierui.cn/" target="_blank">广元</a> <a href="http://www.zzjierui.cn/" target="_blank">广安</a> <a href="http://www.hnjierui.cn/" target="_blank">巴中</a> <a href="http://www.fjjierui.cn/" target="_blank">达州</a> <a href="http://www.gyjierui.cn/" target="_blank">南充</a> <a href="http://www.fzjierui.cn/" target="_blank">遂宁</a> <a href="http://www.cdjierui.cn/" target="_blank">广安</a> <a href="http://www.jxjierui.cn/" target="_blank">内江</a> <a href="http://www.jxruijie.cn/" target="_blank">自贡</a> <a href="http://www.hyruijie.cn/" target="_blank">泸州</a> <a href="http://www.gzruizhi.cn/" target="_blank">宜宾</a> </div> <div class="copy container"> <div class="copyl"> © Copyright 2013-2026 四川澄梦希喜广告有限公司 <a href="https://beian.miit.gov.cn/" target="_blank" rel="nofollow" style="color:#FFFFFF">蜀ICP备2025175960号</a> 版权所有 <a href="https://www.cdcxhl.com/menu.html">网站地图</a> <a href="https://www.cdcxhl.com/articles/" rel="nofollow">其他文章分类</a> <a href="http://www.cdbrznjsb.com">成都澄梦希喜</a> </div> <div class="copyr"> <i><img src="/Public/Home/img/foot1.png" alt=""></i> <i><img src="/Public/Home/img/foot2.png" alt=""></i> <i><img src="/Public/Home/img/foot3.png" alt=""></i> <i><img src="/Public/Home/img/foot4.png" alt=""></i> </div> </div> <div class="bq_tag container"> 热门推荐: <a href="http://www.cdhuace.com/logo.html" target="_blank">LOGO设计</a><a href="http://www.jgbui.com/" target="_blank">彭州柴油发电机</a><a href="http://www.scfdjzulin.cn/" target="_blank">四川发电机租赁公司</a><a href="https://www.cdxwcx.com/city/yaan/" target="_blank">雅安做网站</a><a href="http://www.scjzmc.cn/" target="_blank">四川晃众门窗</a><a href="http://www.cdxwcx.cn/tuoguan/xibuxinxi.html" target="_blank">成都西信机房托管</a><a href="http://www.tjysf.cn/" target="_blank">郫县明安消防器材</a><a href="http://www.sqggao.com/" target="_blank">成都社区广告公司</a><a href="http://www.lzxfdj.com/" target="_blank">成都变电柜安装</a><a href="http://www.fszma.com/" target="_blank">成都企业级IDC服务</a><a href="http://www.cdhuace.com/vi.html" target="_blank">VI设计</a><a href="http://www.ncdkfx.com/" target="_blank">成都纯化水设备</a> </div> </footer> <div class="footbarline"></div> <div id="footbar" class="uin0"> <ul> <li class="on" data-href="/"><a><i><svg t="1638436981291" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="2991" width="48" height="48"><path d="M958.400956 451.54921c-0.058328-5.760191-2.597151-11.215436-6.965645-14.97097L524.345166 69.511143c-7.498788-6.445806-18.581194-6.445806-26.079982 0L309.582871 231.6755l0-102.017488c0-11.04966-8.901741-19.532869-19.951401-19.532869l-88.034009 0c-11.048637 0-19.928888 8.482185-19.928888 19.532869l0 211.954343L71.176063 436.57824c-4.423753 3.800559-6.967692 9.341762-6.967692 15.173584l0 105.500822c0 7.819083 4.554736 14.921851 11.660574 18.183128 2.670829 1.226944 5.51562 1.824555 8.343015 1.824555 4.699022 0 9.346879-1.654686 13.048177-4.836145l53.29788-45.825698 0 324.100516c0 60.677964 49.364291 110.042255 110.042255 110.042255L764.792447 960.741257c60.677964 0 110.042255-49.364291 110.042255-110.042255L874.834702 527.026228l51.585889 44.335764c5.955642 5.119601 14.356986 6.282077 21.481244 2.965541 7.122211-3.313465 11.645225-10.488889 11.565407-18.342764L958.400956 451.54921zM221.578538 150.034085l48.095391 0 0 115.941616-48.095391 41.336454L221.578538 150.034085zM570.718333 920.725892 436.666244 920.725892 436.666244 700.642404c0-11.031241 8.976442-20.007683 20.007683-20.007683l94.0357 0c11.031241 0 20.007683 8.976442 20.007683 20.007683L570.71731 920.725892zM834.818313 495.895207l0 354.803795c0 38.612413-31.414477 70.02689-70.02689 70.02689l-154.058748 0L610.732675 700.642404c0-33.096792-26.926256-60.023048-60.023048-60.023048l-94.0357 0c-33.096792 0-60.023048 26.926256-60.023048 60.023048l0 220.084511L260.59925 920.726915c-38.612413 0-70.02689-31.414477-70.02689-70.02689L190.57236 495.895207c0-1.172709-0.121773-2.314719-0.315178-3.432169l322.113255-276.958846 322.70268 277.348726C834.921667 493.848595 834.818313 494.858598 834.818313 495.895207zM525.411451 173.947727c-7.502881-6.445806-18.587334-6.446829-26.086122 0.00307L104.223736 513.663896l0-52.726875 407.081439-349.870436 407.176606 349.9523 0.521886 51.205219L525.411451 173.947727z" p-id="2992" fill="#2c2c2c"></path></svg><p>首页</p></i></a></li> <li><a href="tel:13518219792"><i><svg t="1638437906526" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="4519" width="48" height="48"><path d="M705.74 604.873333a53.4 53.4 0 0 0-75.426667 0l-37.713333 37.713334c-21.333333 21.333333-90.413333 0.1-150.846667-60.34S360.046667 452.76 381.413333 431.4l0.046667-0.046667 37.666667-37.666666a53.4 53.4 0 0 0 0-75.426667l-165.94-165.933333a53.393333 53.393333 0 0 0-75.42 0l-37.713334 37.713333c-27.866667 27.866667-44.84 64.52-50.46 108.946667-5.213333 41.206667-0.406667 87.42 14.28 137.333333C133.333333 536.586667 199.773333 642 290.9 733.1S487.42 890.666667 587.653333 920.126667c36.926667 10.86 71.813333 16.32 104.146667 16.32a264.333333 264.333333 0 0 0 33.213333-2.04c44.426667-5.62 81.08-22.593333 108.946667-50.46l37.713333-37.713334a53.393333 53.393333 0 0 0 0-75.42z m135.76 211.193334l-37.706667 37.713333c-42.58 42.573333-115.06 51.6-204.1 25.413333-93.506667-27.5-192.453333-90.1-278.62-176.266666s-148.766667-185.113333-176.266666-278.62c-26.186667-89.033333-17.16-161.52 25.413333-204.1l37.713333-37.706667a10.666667 10.666667 0 0 1 15.086667 0l165.933333 165.933333a10.666667 10.666667 0 0 1 0 15.086667l-37.713333 37.706667C329.113333 423.333333 324.666667 458.82 338.766667 501.073333c12.426667 37.273333 38.286667 76.813333 72.813333 111.333334s74.073333 60.386667 111.333333 72.813333c16.213333 5.406667 31.42 8.08 45.26 8.08 22.233333 0 40.946667-6.913333 54.586667-20.553333l37.706667-37.713334a10.666667 10.666667 0 0 1 15.086666 0l165.933334 165.933334a10.666667 10.666667 0 0 1 0.013333 15.1zM576 234.666667a21.333333 21.333333 0 0 1 21.333333-21.333334 213.333333 213.333333 0 0 1 213.333334 213.333334 21.333333 21.333333 0 0 1-42.666667 0c0-94.106667-76.56-170.666667-170.666667-170.666667a21.333333 21.333333 0 0 1-21.333333-21.333333z m0 128a21.333333 21.333333 0 0 1 21.333333-21.333334 85.426667 85.426667 0 0 1 85.333334 85.333334 21.333333 21.333333 0 0 1-42.666667 0 42.713333 42.713333 0 0 0-42.666667-42.666667 21.333333 21.333333 0 0 1-21.333333-21.333333z m362.666667 64a21.333333 21.333333 0 0 1-42.666667 0c0-164.666667-134-298.666667-298.666667-298.666667a21.333333 21.333333 0 0 1 0-42.666667 341.073333 341.073333 0 0 1 341.333334 341.333334z" fill="#2c2c2c" p-id="4520"></path></svg><p>电话</p></i></a></li> <li><a class="opwx"><i><svg t="1638438138558" class="icon" viewBox="0 0 1025 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="10851" width="48" height="48"><path d="M498.816 345.056c26.336 0 43.936-17.632 43.936-43.904 0-26.56-17.568-43.744-43.936-43.744s-52.832 17.184-52.832 43.744C446.016 327.424 472.48 345.056 498.816 345.056zM253.088 257.408c-26.336 0-52.96 17.184-52.96 43.744 0 26.272 26.624 43.904 52.96 43.904 26.24 0 43.808-17.632 43.808-43.904C296.864 274.592 279.328 257.408 253.088 257.408zM1024 626.112c0-138.88-128.832-257.216-286.976-269.536 0.224-1.728 0.32-3.52-0.064-5.312-31.712-147.84-190.688-259.296-369.824-259.296C164.704 91.968 0 233.12 0 406.624c0 93.088 47.52 176.96 137.568 243.104l-31.392 94.368c-2.016 6.144-0.192 12.896 4.704 17.152 2.976 2.56 6.72 3.904 10.496 3.904 2.432 0 4.896-0.576 7.168-1.696L246.4 704.48l14.528 2.944c36.288 7.456 67.616 13.92 106.208 13.92 11.36 0 22.88-0.512 34.176-1.472 4.576-0.384 8.448-2.688 11.072-6.016 42.496 106.336 159.616 183.104 297.44 183.104 35.296 0 71.04-8.512 103.104-16.544l90.848 49.664c2.4 1.312 5.056 1.984 7.68 1.984 3.584 0 7.168-1.216 10.048-3.552 5.056-4.096 7.136-10.848 5.248-17.024l-23.2-77.152C981.344 772.864 1024 699.328 1024 626.112zM398.592 687.968c-10.4 0.896-20.96 1.344-31.424 1.344-35.328 0-65.216-6.112-99.776-13.248L247.296 672c-3.456-0.736-7.104-0.256-10.272 1.376l-88.288 44.192 22.944-68.928c2.24-6.752-0.224-14.112-6.016-18.176C76.96 568.64 32 493.312 32 406.624c0-155.84 150.336-282.656 335.136-282.656 163.36 0 308 99.392 337.856 231.584-171.296 2.24-309.888 122.656-309.888 270.56 0 21.504 3.264 42.336 8.768 62.432C402.208 688.128 400.448 687.808 398.592 687.968zM875.456 815.552c-5.344 4.032-7.616 10.976-5.696 17.376l15.136 50.336-62.112-33.984c-2.368-1.312-5.024-1.984-7.68-1.984-1.312 0-2.624 0.16-3.904 0.512-33.312 8.416-67.776 17.088-101.344 17.088-155.904 0-282.72-107.136-282.72-238.816 0-131.68 126.816-238.784 282.72-238.784 152.928 0 282.144 109.344 282.144 238.784C992 691.744 950.624 759.04 875.456 815.552zM612.992 511.968c-17.568 0-35.136 17.696-35.136 35.232 0 17.664 17.568 35.104 35.136 35.104 26.4 0 43.84-17.44 43.84-35.104C656.832 529.632 639.392 511.968 612.992 511.968zM806.016 511.968c-17.312 0-34.88 17.696-34.88 35.232 0 17.664 17.568 35.104 34.88 35.104 26.304 0 44.064-17.44 44.064-35.104C850.08 529.632 832.352 511.968 806.016 511.968z" p-id="10852" fill="#2c2c2c"></path></svg><p>微信</p></i></a></li> <li data-href="/about/"><a><i><svg t="1638438056011" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="9170" width="48" height="48"><path d="M896 405.333333v128c0 34.133333-29.866667 64-64 64S768 567.466667 768 533.333333v-128c0-17.066667 8.533333-34.133333 17.066667-42.666666C733.866667 251.733333 640 170.666667 516.266667 170.666667H512c-128 0-221.866667 81.066667-273.066667 192 8.533333 8.533333 17.066667 25.6 17.066667 42.666666v128c0 34.133333-29.866667 64-64 64S128 567.466667 128 533.333333v-128C128 371.2 157.866667 341.333333 192 341.333333h4.266667c51.2-123.733333 174.933333-213.333333 315.733333-213.333333s264.533333 89.6 315.733333 213.333333h4.266667c34.133333 0 64 29.866667 64 64zM896 896H128c0-98.133333 170.666667-213.333333 384-213.333333s384 115.2 384 213.333333z m-59.733333-42.666667c-42.666667-59.733333-170.666667-128-324.266667-128s-281.6 68.266667-324.266667 128h648.533334zM512 682.666667c-119.466667 0-213.333333-93.866667-213.333333-213.333334s93.866667-213.333333 213.333333-213.333333 213.333333 93.866667 213.333333 213.333333-93.866667 213.333333-213.333333 213.333334z m170.666667-213.333334c0-93.866667-76.8-170.666667-170.666667-170.666666s-170.666667 76.8-170.666667 170.666666 76.8 170.666667 170.666667 170.666667 170.666667-76.8 170.666667-170.666667z" fill="#2c2c2c" p-id="9171"></path></svg><p>联系</p></i></a></li> </ul> <div class="fbrbg"><img src="/Public/Home/img/fbarbg.png"></div> </div> </body> </html> <script src="/Public/Home/js/jquery.min.js"></script> <script src="/Public/Home/js/wow.min.js"></script> <script src="/Public/Home/js/common.js"></script> <script> $(".ny_con img").each(function(){ var src = $(this).attr("src"); //获取图片地址 var str=new RegExp("http"); var result=str.test(src); if(result==false){ var url = "https://www.cdcxhl.com"+src; //绝对路径 $(this).attr("src",url); } }); </script>