面试:Vue中使用了哪些设计模式?

1.单例模式

router/vuex只能vue.use一次,对应这些实例只能有一个。

2.观察者模式

Vue使用观察者模式来实现数据的响应式,当数据发生变化时,相关的视图会自动更新。

2.1 实现机制

  • Observer: Vue中的Observer对象用于监听数据变化,当数据变动时会通知订阅者。
  • Dep: 依赖收集器(Dep)用于管理所有的观察者。
  • Watcher: 观察者(Watcher)会在数据变化时执行相应的更新操作。

2.2 应用场景

数据绑定:
在Vue模板中使用双向数据绑定 v-model,当输入框中的内容发生变化时,绑定的数据也会相应地更新,这就是观察者模式的应用。

计算属性:
Vue中的计算属性会依赖于其所引用的数据,在相关数据发生变化时,计算属性会重新计算并更新,这也是观察者模式的体现。

computed: {
  reversedMessage() {
    return this.message.split('').reverse().join('');
  }
}

侦听器:
Vue中可以通过侦听器 watch 监听数据的变化,并执行相应的操作,这也是观察者模式的应用。

watch: {
  message(newValue, oldValue) {
    console.log('Message changed from', oldValue, 'to', newValue);
  }
}

3.发布-订阅模式

Vue 的事件系统基于发布-订阅模式,允许组件之间进行解耦,通过订阅和发布事件来通信。

假设有一个父组件和一个子组件,父组件中有一个按钮,点击按钮后触发一个事件,并传递数据给子组件,子组件接收到数据后进行相应的处理。

父组件模板:

<template>
  <div>
    <button @click="publishEvent">Click me</button>
  </div>
</template>

<script>
export default {
  methods: {
    publishEvent() {
      // 发布事件并传递数据
      this.$emit('custom-event', { message: 'Hello from parent' });
    }
  }
}
</script>

子组件模板:

<template>
  <div>
    <p>{{ receivedMessage }}</p>
  </div>
</template>

<script>
export default {
  data() {
    return {
      receivedMessage: ''
    };
  },
  mounted() {
    // 订阅事件
    this.$parent.$on('custom-event', this.handleEvent);
  },
  methods: {
    handleEvent(data) {
      // 接收并处理事件数据
      this.receivedMessage = data.message;
    }
  }
}
</script>

在这个例子中,父组件通过 $emit 方法发布一个名为 ustom-event 的事件,并传递了一个包含消息的对象。子组件通过 $on 方法订阅了 custom-event 事件,并在收到事件时执行 handleEvent 方法来更新接收到的消息。这样,父子组件之间就实现了通过发布-订阅模式进行通信。

4.代理模式

Vue 3中使用了 Proxy 对象来实现响应式数据,它可以拦截对对象的各种操作,从而实现数据的监听和更新。

const data = { message: 'Hello, Vue!' };

// 创建一个响应式代理对象
const reactiveData = new Proxy(data, {
  // 拦截属性的读取操作
  get(target, key) {
    console.log(`Getting ${key}: ${target[key]}`);
    return target[key];
  },
  // 拦截属性的写入操作
  set(target, key, value) {
    console.log(`Setting ${key}: ${value}`);
    target[key] = value;
    // 触发更新操作
    // 这里可以通知相关的视图进行更新
  }
});

// 通过代理对象访问数据
console.log(reactiveData.message); // 输出:Hello, Vue!

// 修改数据
reactiveData.message = 'Hello, Proxy!'; // 输出:Setting message: Hello, Proxy!
console.log(reactiveData.message); // 输出:Hello, Proxy!

在这个例子中,通过创建一个Proxy对象reactiveData,对原始数据data进行代理。在Proxygetset方法中,可以监听属性的读取和写入操作,并在需要时触发相应的更新操作,实现了数据的响应式。

5.组件模式

Vue 的核心思想之一是组件化开发,将页面拆分成独立的组件,每个组件负责自己的一部分功能,便于复用和维护。

首先定义一个简单的子组件MyButton,这个组件封装了一个按钮的基本行为和样式。

<!-- MyButton.vue -->
<template>
  <button @click="clickHandler">{{ buttonText }}</button>
</template>

<script>
export default {
  props: ['buttonText'],
  methods: {
    clickHandler() {
      this.$emit('click');
    }
  }
}
</script>

然后,在父组件中使用这个子组件:

<!-- App.vue -->
<template>
  <div>
    <my-button :button-text="buttonText" @click="handleButtonClick"></my-button>
  </div>
</template>

<script>
import MyButton from './MyButton.vue';

export default {
  components: {
    MyButton
  },
  data() {
    return {
      buttonText: 'Click Me'
    };
  },
  methods: {
    handleButtonClick() {
      alert('Button clicked!');
    }
  }
}
</script>

在这个例子中,我们定义了一个MyButton子组件,它接受一个buttonText属性来显示按钮上的文本,并且当按钮被点击时,它会触发一个click事件。父组件App.vue通过导入并注册MyButton组件,然后在模板中使用它。这里通过属性绑定(:button-text="buttonText")来传递按钮的文本,通过事件监听(@click="handleButtonClick")来处理按钮点击事件。

这个例子展示了Vue中组件模式的基本应用:通过属性传递数据,通过事件进行父子组件间的通信,以及通过组件化构建复用的界面元素。

6.策略模式

Vue 的指令系统和插件系统可以看作是策略模式的应用,通过不同的指令和插件来实现不同的功能扩展。

假设有一个自定义指令,根据不同的条件展示不同的提示信息。

// 自定义指令
Vue.directive('show-tip', {
  bind(el, binding) {
    // 根据条件展示不同的提示信息
    const condition = binding.value.condition;
    const message = binding.value.message;

    if (condition) {
      el.innerText = message;
    } else {
      el.innerText = 'No tip available';
    }
  }
});

// Vue实例
new Vue({
  el: '#app',
  data: {
    showTip: true,
    tipMessage: 'This is a tip message!'
  }
});

在这个例子中,自定义指令show-tipbind钩子函数根据传入的条件来展示不同的提示信息,这就是策略模式的应用。用户可以根据需要传递不同的条件和提示信息,指令会根据条件选择合适的策略来展示提示信息。

7.依赖注入模式

Vue的依赖注入主要体现在组件之间的数据传递和通信上,通过props属性传递数据或使用 provide/inject实现跨层级组件的数据传递。

假设有一个全局的用户认证服务,我们希望在整个应用中的任何组件都能够访问到这个认证服务。

// 创建一个全局的用户认证服务
const authService = {
  isAuthenticated: false,
  login() {
    this.isAuthenticated = true;
  },
  logout() {
    this.isAuthenticated = false;
  }
};

// 在根组件中通过provide提供全局服务
new Vue({
  provide: {
    authService: authService
  },
  el: '#app',
  // ...
});

现在,我们可以在任何子组件中通过inject来注入这个全局的用户认证服务,并使用它提供的方法和属性:

// 在任意子组件中注入全局的用户认证服务
export default {
  inject: ['authService'],
  methods: {
    handleLogin() {
      this.authService.login();
    },
    handleLogout() {
      this.authService.logout();
    }
  }
};

通过这种方式,我们可以在任何组件中访问全局的用户认证服务,而不需要显式地传递它。这就是依赖注入模式在Vue中的应用。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

太阳与星辰

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值