zoukankan      html  css  js  c++  java
  • vue2.x学习笔记(四)

    接着前面的内容:https://www.cnblogs.com/yanggb/p/12563162.html

    模板语法

    vue使用了基于html的模板语法,允许开发者声明式地将dom绑定到底层vue实例的数据。所有vue的模板都是合法的html,因此能被遵循规范的浏览器或html解析器解析。在底层的实现上,vue将模板编译成虚拟dom渲染函数,并结合响应系统,智能地计算出最少需要重新渲染多少组件,来将dom操作的次数减到最少。

    插值-文本

    数据绑定最常见的形式就是使用mustache语法(双大括号)的文本插值。

    <span>i love {{ someone }}.</span>

    mustache标签(双大括号)中的内容将会被替代为对应数据对象上someone的值,比如如果【someone】的值是【yanggb】,那么页面上渲染的内容就会是【i love yanggb.】。无论任何时候,只要绑定的数据对象上someone的属性值发生了改变,插值处的内容也会被同步更新。另外,如果你想要插值中的数据只能被替换一次的话,可以使用【v-once】指令,这个指令的作用是限制插值的内容只会被更新一次,之后无论多少次改变someone的值,插值处的内容都一直是【i love yanggb forever.】。

    <span v-once>i love {{ someone }} forever.</span>

    使用【v-once】指令的时候要格外注意不要影响到该节点上的其他数据绑定。

    插值-原始html

    双大括号会将数据解释为普通文本,而非html代码。为了输出真正的html,你会需要使用【v-html】指令。

    <p>使用mustache语法: {{ rawHtml }}</p>
    <p>使用v-html指令: <span v-html="rawHtml"></span></p>

    这个时候,如果rawHtml的值是'<span>我是原始的html代码</span>'的话,使用mustache语法会将此字符串原样输出,而使用【v-html】指令,则会被解析成html渲染。要注意的是,不应该选择使用【v-html】指令来复合局部模板,因为vue不是基于字符串的模板引擎,这样做可能会造成意料之外的问题。再说了,vue还提供了组件这样的特性,组件在用户界面(UI)中更适合作为可重用和可组合的基本单位。

    你的站点上动态渲染的任意HTML可能会非常危险,因为它很容易导致XSS 攻击。请只对可信内容使用HTML插值,绝不要对用户提供的内容使用插值。

    官方文档上特别注明了使用【v-html】指令进行原始html插值是一项可能造成危险的行为,因此建议在开发中禁止使用这个语法,除非有非用不可的理由,将危险扼杀在摇篮里,只要知道有这个语法就可以了。

    插值-attribute(属性)

    因为mustache语法并不能作用在html的attribute上,因此vue提供了【v-bind】指令来满足这一场景。

    <div v-bind:id="dynamicId"></div>

    而对于布尔attribute来说,它们只要存在就意味着值为true,比如disabled属性,因此使用【v-bind】指令的时候会与常规的使用javascript操作原生html的方式有所不同。

    <button v-bind:disabled="isButtonDisabled">Button</button>

    在上面这个例子中,如果isButtonDisabled的值是null、undefined或false的话,disabled这个attribute是不会被包含在渲染出来的<button>元素中的。

    使用javascript表达式

    在前面的例子中,在模板中都是绑定的简单的属性键值。而实际上,对于所有的数据绑定,vue都提供了完全的javascript表达式支持。

    {{ number + 1 }}
    
    {{ ok ? 'yes' : 'no' }}
    
    {{ message.split('').reverse().join('') }}
    
    <div v-bind:id="'list-' + id"></div>

    这些表达式会在所属vue实例的数据作用域下作为javascript被解析。但是有个限制是,每个绑定都只能包含单个表达式,非单个表达式的话,数据绑定会失效,可能导致表达式被原样输出,甚至导致直接报错。

    <!-- 这是语句,不是表达式 -->
    {{ var a = 1 }}
    
    <!-- 流控制也不会生效,请使用三元表达式 -->
    {{ if (ok) { return message } }}

    此外,模板表达式都被放在沙盒中,只能访问全局变量的一个白名单,如Math和Date,开发者不应该在模板表达式中试图访问用户定义的全局变量。

    /* not type checking this file because flow doesn't play well with Proxy */
    
    import config from 'core/config'
    import { warn, makeMap, isNative } from '../util/index'
    
    let initProxy
    
    if (process.env.NODE_ENV !== 'production') {
      const allowedGlobals = makeMap(
        'Infinity,undefined,NaN,isFinite,isNaN,' +
        'parseFloat,parseInt,decodeURI,decodeURIComponent,encodeURI,encodeURIComponent,' +
        'Math,Number,Date,Array,Object,Boolean,String,RegExp,Map,Set,JSON,Intl,' +
        'require' // for Webpack/Browserify
      )
    
      const warnNonPresent = (target, key) => {
        warn(
          `Property or method "${key}" is not defined on the instance but ` +
          'referenced during render. Make sure that this property is reactive, ' +
          'either in the data option, or for class-based components, by ' +
          'initializing the property. ' +
          'See: https://vuejs.org/v2/guide/reactivity.html#Declaring-Reactive-Properties.',
          target
        )
      }
    
      const warnReservedPrefix = (target, key) => {
        warn(
          `Property "${key}" must be accessed with "$data.${key}" because ` +
          'properties starting with "$" or "_" are not proxied in the Vue instance to ' +
          'prevent conflicts with Vue internals' +
          'See: https://vuejs.org/v2/api/#data',
          target
        )
      }
    
      const hasProxy =
        typeof Proxy !== 'undefined' && isNative(Proxy)
    
      if (hasProxy) {
        const isBuiltInModifier = makeMap('stop,prevent,self,ctrl,shift,alt,meta,exact')
        config.keyCodes = new Proxy(config.keyCodes, {
          set (target, key, value) {
            if (isBuiltInModifier(key)) {
              warn(`Avoid overwriting built-in modifier in config.keyCodes: .${key}`)
              return false
            } else {
              target[key] = value
              return true
            }
          }
        })
      }
    
      const hasHandler = {
        has (target, key) {
          const has = key in target
          const isAllowed = allowedGlobals(key) ||
            (typeof key === 'string' && key.charAt(0) === '_' && !(key in target.$data))
          if (!has && !isAllowed) {
            if (key in target.$data) warnReservedPrefix(target, key)
            else warnNonPresent(target, key)
          }
          return has || !isAllowed
        }
      }
    
      const getHandler = {
        get (target, key) {
          if (typeof key === 'string' && !(key in target)) {
            if (key in target.$data) warnReservedPrefix(target, key)
            else warnNonPresent(target, key)
          }
          return target[key]
        }
      }
    
      initProxy = function initProxy (vm) {
        if (hasProxy) {
          // determine which proxy handler to use
          const options = vm.$options
          const handlers = options.render && options.render._withStripped
            ? getHandler
            : hasHandler
          vm._renderProxy = new Proxy(vm, handlers)
        } else {
          vm._renderProxy = vm
        }
      }
    }
    
    export { initProxy }

    上面这段代码是vue中定义全局变量的访问白名单,因为用户自定义的全局变量不在白名单内,在模板表达式中访问将会出现问题。

    指令

    指令(directives)是带有【v-】前缀的特殊attribute。指令attribute的值预期是单个javascript表达式(v-for是个例外)。指令的职责是,当表达式的值发生改变的时候,就将其产生的连带影响,响应式地作用到dom中。

    参数

    一些指令是能够接收一个参数的,这个参数跟在指令的后面,以冒号表示。

    比如,可以使用【v-bind】指令响应式地更新html的attribute。

    <a v-bind:href="url">...</a>

    在这里,href是参数,作用是告知【v-bind】指令将该元素的href属性与表达式url的值进行绑定。

    又比如,可以使用【v-on】指令监听dom事件。

    <a v-on:click="doSomething">...</a>

    在这个例子中,这样写就给这个<a>元素绑定了click点击事件。

    动态参数

    从2.6.0开始,可以用方括号括起来的javascript表达式作为一个指令的参数。

    <!-- 注意,参数表达式的写法存在一些约束,如之后的“对动态参数表达式的约束”章节所述。-->
    <a v-bind:[attributeName]="url"> ... </a>

    这里的attributeName会被作为一个javascript表达式进行动态求值,求得的值将会作为最终的参数来使用。例如,如果你的vue实例有一个data属性attributeName,其值为href的话,那么这个绑定将等价于【v-bind:href】。

    同样的,你可以使用动态参数为一个动态的事件名绑定处理函数。

    <a v-on:[eventName]="doSomething"> ... </a>

    在上面的这个例子中,当eventName的值为【focus】的时候,这个绑定将等价于【v-on:focus】。

    对动态参数值的约束

    动态参数预期会求出一个字符串,异常情况下值为null,这个特殊的null值可以被显性地用于移除绑定,任何其他非字符串类型的值都将会触发一个警告。

    <a v-on:[eventName]="doSomething"> ... </a>

    比如在上面这个例子中,如果eventName的值为null,那么这个<a>标签就不会被绑定任何dom事件了。

    对动态参数表达式的约束

    动态参数表达式有一些语法约束,因为某些字符,比如空格和引号,放在html的属性名里是无效的。

    <!-- 这会触发一个编译警告 -->
    <a v-bind:['foo' + bar]="value"> ... </a>

    变通的办法是使用没有空格或者引号的表达式,或使用计算属性替代这种复杂的表达式。

    另外,在dom中使用模板的时候(直接在一个html文件中编写模板),还需要避免使用大写字符来命名键名,因为浏览器会把attribute名全部强制转换为小写。

    <!-- 在 DOM 中使用模板时这段代码会被转换为 `v-bind:[someattr]`。
    除非在实例中有一个名为“someattr”的 property,否则代码不会工作。-->
    <a v-bind:[someAttr]="value"> ... </a>

    动态参数的使用场景在现有的开发场景中并没有使用到,大概只有特殊的场景才会用得上,有所了解就好了。

    修饰符

    修饰符(modifier)是以半角句号【.】指明的特殊后缀,用于指出一个指令应该以特殊的方式绑定。例如【.prevent】修饰符的作用是告诉【v-on】指令对于触发的事件调用【event.preventDefallut()】方法。

    <form v-on:submit.prevent="onSubmit">...</form>

    这个语法在一些特定的场景中十分有用,后面会详细了解。

    缩写

    【v-】前缀作为一种视觉提示,可以用于识别模板中vue特定的attribute。虽然在使用vue为现有的标签添加动态行为(dynamic behavior)的时候【v-】前缀会很有帮助,但是对于一些频繁使用到的指令来说,反复编写相同的代码就会让开发者觉得厌烦。同时,在构建由vue管理所有模板的单页面应用程序(SPA,single page application)的时候,【v-】前缀也变得没有那么重要了。因此,vue为【v-bind】和【v-on】这两个最常用的指令提供了特定的缩写,方便开发者编写vue应用程序。

    【v-bind】指令的缩写

    <!-- 完整语法 -->
    <a v-bind:href="url">...</a>
    
    <!-- 缩写 -->
    <a :href="url">...</a>
    
    <!-- 动态参数的缩写 (2.6.0+) -->
    <a :[key]="url"> ... </a>

    【v-on】指令的缩写

    <!-- 完整语法 -->
    <a v-on:click="doSomething">...</a>
    
    <!-- 缩写 -->
    <a @click="doSomething">...</a>
    
    <!-- 动态参数的缩写 (2.6.0+) -->
    <a @[event]="doSomething"> ... </a>

    虽然它们看起来可能和普通的html略有不同,但是【:】和【@】对于attribute名来说都是合法字符,在所有支持vue的浏览器中都能被正确解析,而且它们并不会出现在最终渲染的标记中。缩写语法是完全可选的,你在编写vue应用的时候,完全可以选择完整的语法或使用缩写的语法,但是随着你深入的使用缩写语法和了解它们的作用,你会像我一样喜欢上缩写的语法。

    "我还是很喜欢你,像日月轮回交替,不问朝夕。"

  • 相关阅读:
    测试
    pytest -- Windows fatal exception: code 1073807366
    cookie绕过验证码登录操作
    python接口自动化基本流程
    测试-pytest框架
    测试
    flask框架
    阿里巴巴集团面试
    字节跳动一面
    mysql 存储引擎
  • 原文地址:https://www.cnblogs.com/yanggb/p/12571062.html
Copyright © 2011-2022 走看看