玩转代码| Vue 中 JSX 的特性,这一篇讲的明明白白
目录
什么时候使用JSX
JSX在Vue2中的基本使用
配置
文本插值
条件与循环渲染
属性绑定
事件绑定
v-show与v-model
插槽
使用自定义组件
在method里返回JSX
JSX是一种Javascript的语法扩展,即具备了Javascript
的全部功能,同时又兼具html
的语义化和直观性。它可以让我们在JS中写模板语法:
const el = <div>Vue 2</div>;
上面这段代码既不是 HTML 也不是字符串,被称之为 JSX,是 JavaScript 的扩展语法。JSX 可能会使人联想到模板语法,但是它具备 Javascript 的完全编程能力。
什么时候使用JSX
当开始写一个只能通过 level
prop 动态生成标题 (heading) 的组件时,你可能很快想到这样实现:
<script type="text/x-template" id="anchored-heading-template">
<h1 v-if="level === 1"> <slot></slot>
</h1>
<h2 v-else-if="level === 2"> <slot></slot>
</h2>
<h3 v-else-if="level === 3"> <slot></slot>
</h3>
</script>
这里用template模板并不是最好的选择,在每一个级别的标题中重复书写了部分代码,不够简洁优雅。如果尝试用 JSX 来写,代码就会变得简单很多:
const App = {render() {const tag = `h${this.level}`return <tag>{this.$slots.default}</tag>}
}
或者如果你写了很多 render
函数,可能会觉得下面这样的代码写起来很痛苦:
createElement( 'anchored-heading', { props: { level: 1 } }, [ createElement('span', 'Hello'), ' world!' ]
)
特别是对应的模板如此简单的情况下:
<anchored-heading :level="1"> <span>Hello</span> world!
</anchored-heading>
这时候就可以在 Vue 中使用 JSX 语法,它可以让我们回到更接近于模板的语法上:
import AnchoredHeading from './AnchoredHeading.vue' new Vue({ el: '#demo', render: function (h) { return ( <AnchoredHeading level={1}> <span>Hello</span> world! </AnchoredHeading> ) }
})
在开发过程中,经常会用到消息提示组件Message,可能的一种写法是这样的:
Message.alert({messge: '确定要删除?',type: 'warning'
})
但是希望message
可以自定义一些样式,这时候你可能就需要让Message.alert
支持JSX
了(当然也可以使用插槽/html
等方式解决)
Message.alert({messge: <div>确定要删除<span style="color:red">xxx</span>的笔记?</div>,type: 'warning'
})
此外,一个 .vue
文件里面只能写一个组件,这个在一些场景下可能不太方便,很多时候写一个页面的时候其实可能会需要把一些小的节点片段拆分到小组件里面进行复用,这些小组件其实写个简单的函数组件就能搞定了。平时可能会由于SFC的限制让我们习惯于全部写在一个文件里,但不得不说可以尝试一下这种方式。
// 一个文件写多个组件
const Input = (props) => <input {...props} />
export const Textarea = (props) => <input {...props} />
export const Password = (props) => <input type="password" {...props} />export default Input
比如这里封装了一个 Input 组件,我们希望同时导出 Password 组件和 Textarea 组件来方便用户根据实际需求使用,而这两个组件本身内部就是用的 Input 组件,只是定制了一些 props。在 JSX 里面就很方便,写个简单的函数组件基本上就够用了,通过 interface 来声明 props 就好了。但是如果是用模板来写,可能就要给拆成三个文件,或许还要再加一个 index.js
的入口文件来导出三个组件。
由于 JSX 的本质就是 JavaScript,所以它具有 JavaScript 的完全编程能力。再举个例子,我们需要通过一段逻辑来对一组 DOM 节点做一次 reverse,如果在模板里面写,那估计要写两段代码。
虽然这个例子可能不太常见,但是不得不承认,在一些场景下,JSX 还是要比模板写起来更加顺手。
从 Vue 2 开始,template 在运行之前,会被编译成 JavaScript 的 render function
。
Vue 推荐在绝大多数情况下使用 template 来创建你的 HTML。然而在一些场景中,就需要使用 render 函数,它比 template 更加灵活。这些 render function
在运行时阶段,就是传说中的 Virtual DOM
。
JSX在Vue2中的基本使用
配置
在 Vue 2 中,JSX 的编译需要依赖 @vue/babel-preset-jsx
和 @vue/babel-helper-vue-jsx-merge-props
这两个包。前面这个包来负责编译 JSX 的语法,后面的包用来引入运行时的 mergeProps
函数。
npm install @vue/babel-preset-jsx @vue/babel-helper-vue-jsx-merge-props
并在babel.config.js中添加配置:
module.exports = {presets: ['@vue/babel-preset-jsx'],
}
文本插值
模板代码里文本插值默认是用双大括号:
<h1>{{ msg }}</h1>
在JSX中则需要使用单大括号:
const name = 'Vue'
const element = <h1>Hello, { name }</h1>
和模板语法中的文本插值一样,大括号内支持任何有效的JS表达式,比如:2 + 2
,user.firstName
,formatName(user)
等。
条件与循环渲染
在模板代码里面我们通过v-for
去遍历元素,通过v-if
去判断是否渲染元素,在JSX中,对于v-for
,可以使用for
循环或者array.map
来代替,对于v-if
,可以使用if-else
语句,三元表达式
等来代替
使用if-else
语句
const element = (name) => {if (name) {return <h1>Hello, { name }</h1>} else {return <h1>Hello, Stranger</h1>}
}
使用三元表达式
const element = icon ? <span class="icon"></span> : null;
使用数组的map方法
const list = ['java', 'c++', 'javascript', 'c#', 'php']
return (<ul>{list.map(item => {return <li>{item}</li>})}</ul>
)
属性绑定
在模板代码中,一般通过 v-bind:prop="value"
或:prop="value"
来给组件绑定属性,在JSX
里面就不能继续使用v-bind指令了,而是通过单大括号的形式进行绑定:
const href = 'https://xxx.com'
const element = <a href={href}>xxx</a>const properties = {a: 1, b: 2}
此外,模板代码中能通过<div v-bind="properties"></div>
批量绑定标签属性。
在JSX中也有相应的替换方案:<div {...properties}></div>
。
class绑定同样也是使用单大括号的形式
const element = <div className={`accordion-item-title ${ disabled ? 'disabled' : '' }`}></div>
const element = <div class={[ 'accordion-item-title', disabled && 'disabled' ]}
>Item</div>
style绑定需要使用双大括号
const width = '100px'
const element = <button style={{ width, fontSize: '16px' }}></button>
事件绑定
在模板代码中通过v-on指令监听事件,在JSX中通过on
+ 事件名称的大驼峰写法来监听,且绑定事件也是用大括号,比如click事件要写成onClick
,mouseenter事件要写成onMouseenter
const confirm = () => {// 确认提交
}
<button onClick={confirm}>确定</button>
有时候我们希望可以监听一个组件根元素上面的原生事件,这时候会用到.native
修饰符,但是在JSX中同样也不能使用,不过也有替代方案,监听原生事件的规则与普通事件是一样的,只需要将前面的on
替换为nativeOn
,如下
render() {// 监听下拉框根元素的click事件return <CustomSelect nativeOnClick={this.handleClick}></CustomSelect>}
除了上面的监听事件的方式之外,我们还可以使用对象的方式去监听事件
render() {return (<ElInputvalue={this.content}on={{focus: this.handleFocus,input: this.handleInput}}nativeOn={{click: this.handleClick}}></ElInput>)}
对于 .passive
、.capture
和 .once
这些事件修饰符,Vue 提供了相应的前缀可以用于 on
:
例如:
on: { '!click': this.doThisInCapturingMode, '~keyup': this.doThisOnce, '~!mouseover': this.doThisOnceInCapturingMode
}
对于所有其它的修饰符,私有前缀都不是必须的,因为你可以在事件处理函数中使用事件方法:
具体可查阅Vue规范文档。
v-show与v-model
大多数指令并不能在JSX中使用,对于原生指令,只有v-show
是支持的。
而v-model
是Vue
提供的一个语法糖,它本质上是由 value
属性(默认) + input
事件(默认)组成的,所以,在JSX
中,我们便可以回归本质,通过传递value
属性并监听input
事件来手动实现数据的双向绑定:
export default {data() {return {name: ''}},methods: {// 监听 onInput 事件进行赋值操作handleInput(e) {this.name = e.target.value}},render() {// 传递 value 属性 并监听 onInput事件return <input value={this.name} onInput={this.handleInput}></input>}
}
此外,在脚手架vue-cli4
中,已经默认集成了对v-model
的支持,可以直接使用<input v-model={this.value}>
,如果项目比较老,也可以安装插件babel-plugin-jsx-v-model
来进行支持。
同样的,在JSX
中,对于.sync
也需要用属性+事件来实现,如下代码所示:
export default {methods: {handleChangeVisible(value) {this.visible = value}},render() {return (<ElDialogtitle="测试.sync"visible={this.visible}on={{ 'update:visible': this.handleChangeVisible }}></ElDialog>)}
}
插槽
(1)默认插槽:
使用element-ui
的Dialog
时,弹框内容就使用了默认插槽,在JSX
中使用默认插槽的用法与普通插槽的用法基本是一致的,如下
render() {return (<ElDialog title="弹框标题" visible={this.visible}>{/*这里就是默认插槽*/}<div>这里是弹框内容</div></ElDialog>)}
自定义默认插槽:
在Vue
的实例this
上面有一个属性$slots
,这个上面就挂载了一个这个组件内部的所有插槽,使用this.$slots.default
就可以将默认插槽加入到组件内部
export default {props: {visible: {type: Boolean,default: false}},render() {return (<div class="custom-dialog" vShow={this.visible}>{/**通过this.$slots.default定义默认插槽*/}{this.$slots.default}</div>)}
}
(2)具名插槽
有时候我们一个组件需要多个插槽,这时候就需要为每一个插槽起一个名字,比如element-ui
的弹框可以定义底部按钮区的内容,就是用了名字为footer
的插槽
render() {return (<ElDialog title="弹框标题" visible={this.visible}><div>这里是弹框内容</div>{/** 具名插槽 */}<template slot="footer"><ElButton>确定</ElButton><ElButton>取消</ElButton></template></ElDialog>)}
自定义具名插槽: 在上节自定义默认插槽时提到了$slots
,对于默认插槽使用this.$slots.default
,而对于具名插槽,可以使用this.$slots.footer
进行自定义
render() {return (<div class="custom-dialog" vShow={this.visible}>{this.$slots.default}{/**自定义具名插槽*/}<div class="custom-dialog__foolter">{this.$slots.footer}</div></div>)}
(3)作用域插槽
有时让插槽内容能够访问子组件中才有的数据是很有用的,这时候就需要用到作用域插槽,在JSX
中,因为没有v-slot
指令,所以作用域插槽的使用方式就与模板代码里面的方式有所不同了。比如在element-ui
中,我们使用el-table
的时候可以自定义表格单元格的内容,这时候就需要用到作用域插槽
data() {return {data: [{name: 'xxx'}]}},render() {return ({/**scopedSlots即作用域插槽,default为默认插槽,如果是具名插槽,将default该为对应插槽名称即可*/}<ElTable data={this.data}><ElTableColumnlabel="姓名"scopedSlots={{default: ({ row }) => {return <div style="color:red;">{row.name}</div>}}}></ElTableColumn></ElTable>)}
自定义作用域插槽:
使用作用域插槽不同,定义作用域插槽也与模板代码里面有所不同。加入我们自定义了一个列表项组件,用户希望可以自定义列表项标题,这时候就需要将列表的数据通过作用域插槽传出来。
render() {const { data } = this// 获取标题作用域插槽const titleSlot = this.$scopedSlots.titlereturn (<div class="item">{/** 如果有标题插槽,则使用标题插槽,否则使用默认标题 */}{titleSlot ? titleSlot(data) : <span>{data.title}</span>}</div>)}
使用自定义组件
只需要导入进来,不用再在components属性声明了,直接写在jsx中:
import MyComponent from './my-component'export default {render() {return <MyComponent>hello</MyComponent>},
}
在method里返回JSX
我们可以定义method
,然后在method
里面返回JSX
,然后在render
函数里面调用这个方法,不仅如此,JSX
还可以直接赋值给变量,比如:
methods: {renderFooter() {return (<div><ElButton>确定</ElButton><ElButton>取消</ElButton></div>)}},render() {const buttons = this.renderFooter()return (<ElDialog visible={this.visible}><div>内容</div><template slot="footer">{buttons}</template></ElDialog>)}
相关文章:
玩转代码| Vue 中 JSX 的特性,这一篇讲的明明白白
目录 什么时候使用JSX JSX在Vue2中的基本使用 配置 文本插值 条件与循环渲染 属性绑定 事件绑定 v-show与v-model 插槽 使用自定义组件 在method里返回JSX JSX是一种Javascript的语法扩展,即具备了Javascript的全部功能,同时又兼具html的语义…...
(vue)el-descriptions 描述列表无效
(vue)el-descriptions 描述列表无效 原因:element 的版本不够 解决:运行下面两个命令 npm uninstall element-ui //卸载之前安装的版本 npm i element-ui -S //重新安装解决参考:https://blog.csdn.net/weixin_59769148/article/details/1…...
ios 苹果手机日期格式问题
目录 问题解决其他 问题 ios 无法识别的时间戳格式:2023-10-17 11:10:49 可识别的: 2023/10/17 11:10:49 解决 const startTime 2023/10/17 11:10:49 startTime.replace(/-/g, /)// 获取时间差值 export const useDateDiff (startTime , endTime …...
学习嵌入式系统的推荐步骤:
学习嵌入式系统的推荐步骤: 00001. 选择一款Linux发行版作为主要操作系统,如RedHat、Ubuntu、Fedora等。进入Linux后,使用终端进行任务操作。建议不要使用虚拟机,如有需要可考虑双系统安装。 00002. 00003. 学习C语言、数…...
勒索病毒LockBit2.0 数据库(mysql与sqlsever)解锁恢复思路分享
0.前言 今天公司服务器中招LockBit2.0勒索病毒,损失惨重,全体加班了一天基本解决了部分问题,首先是丢失的文件数据就没法恢复了,这一块没有理睬,主要恢复的是两个数据库,一个是16GB大小的SQLserver数据库&…...
超简单小白攻略:如何利用黑群晖虚拟机和内网穿透实现公网访问
文章目录 前言本教程解决的问题是:按照本教程方法操作后,达到的效果是前排提醒: 1. 搭建群晖虚拟机1.1 下载黑群晖文件vmvare虚拟机安装包1.2 安装VMware虚拟机:1.3 解压黑群晖虚拟机文件1.4 虚拟机初始化1.5 没有搜索到黑群晖的解…...
Ubuntu 16.04 LTS third maintenance update release
Ubuntu 16.04 LTS (Xenial Xerus)今天迎来的第三个维护版本更新中,已经基于Linux Kernel 4.10内核,而且Mesa图形栈已经升级至17.0版本。Adam Conrad表示:“像此前LTS系列相似,16.04.3对那些使用更新硬件的用户带来了硬件优化。该版…...
Java学习_day01_hello java
构成 JDK JDK是java开发者工具,由JRE和一些开发工具组成。JRE JRE是java运行环境,由JVM和java核心类库组成。JVM JVM是java虚拟机,主要用来运行字节码。 执行过程 由IDE或文本编辑器,编写源代码,并将文件保存为*.ja…...
UnitTesting 单元测试
1. 测试分为两种及详细介绍测试书籍: 1.1 Unit Test : 单元测试 - test the business logic in your app : 测试应用中的业务逻辑 1.2 UI Test : 界面测试 - test the UI of your app : 测试应用中的界面 1.3 测试书籍网址:《Testing Swift》 https://www.hackingwithswift.c…...
C++内存管理:其五、指针类型转换与嵌入式指针
一、内存池的缺陷 作者在上一版本里面介绍了链表实现内存池,其中有一个小缺陷:虽然较少了cookie的内存损耗,但是加入了一个额外的指针,仍然需要占用内存。我们仔细看内存池的设计思想,可以发现一个关键点:…...
常见锁的分类
入职体验: 今天运维岗位刚入职,但是目前还没有办理入职手续,但是领导发了一堆资料!看了一下,非常多的新东西,只能说努力一把!!! 一、锁的分类 1.1 可重入锁、不可重入锁…...
vue 鼠标划入划出多传一个参数
// item可以传递弹窗显示数据, $event相关参数可以用来做弹窗定位用 mouseover"handleMouseOver($event, item)" mouseleave"handleMouseLeave($event, item)"举个栗子: 做一个hover提示弹窗组件(用的vue3框架 less插件) 可以将组件…...
svn项目同步到gitLab
安装git 确保安装了git 新建一个文件夹svn-git 在文件夹中新建userinfo.txt文件,映射svn用户,这个文件主要是用于将SVN用户映射为Git用户(昵称及其邮箱)。 userinfo.txt具体格式如下: admin admin <admin163.com> lis…...
图解Dubbo,Dubbo 服务治理详解
目录 一、介绍1、介绍 Dubbo 服务治理的基本概念和重要性2、阐述 Dubbo 服务治理的实现方式和应用场景 二、Dubbo 服务治理的原理1、Dubbo 服务治理的架构设计2、Dubbo 服务治理的注册与发现机制3、Dubbo 服务治理的负载均衡算法 三、Dubbo 服务治理的实现方式1、基于 Docker 容…...
Css 如何取消a链接点击时的背景颜色
要取消 <a> 链接点击时的背景颜色,可以使用 CSS 的伪类 :active。你可以通过为 a:active 应用 background-color 属性设置为 transparent 或者 none,来取消点击时的背景色。下面是一个示例: a:active {background-color: transparent;…...
1.16.C++项目:仿muduo库实现并发服务器之HttpContext以及HttpServer模块的设计
文章目录 一、HttpContext模块二、HttpServer模块三、HttpContext模块实现思想(一)功能(二)意义(三)接口 四、HttpServer模块实现思想(一)功能(二)意义&#…...
ABAP 新增PO计划行时 新增行交货日期默认当前最大交期
ABAP 新增PO计划行时 新增行交货日期默认当前最大交期 DATA: ls_poitem TYPE mepoitem. DATA: ls_jhh TYPE meposchedule. DATA: ls_poitemc TYPE REF TO if_purchase_order_item_mm. DATA: is_persistent TYPE mmpur_bool. DATA: lt_eket TYPE TABLE OF eket. DATA: ls_e…...
VSCode怎么创建Java项目
首先安装好Java的开发环境:JDK在VSCode中安装适用于Java开发的插件。打开VSCode,点击左侧的扩展图标,搜索并安装Java Extension Pack插件。等待安装完成后,重启VSCode生效。创建一个新的Java项目,按下Ctrl Shift P&a…...
软件工程与计算(十四)详细设计中面向对象方法下的模块化
一.面向对象中的模块 1.类 模块化是消除软件复杂度的一个重要方法,每个代码片段相互独立,这样能够提高可维护性。在面向对象方法中,代码片段最重要的类,整个类的所有代码联合起来构成独立的代码片段。 模块化希望代码片段由两部…...
商城免费搭建之java商城 开源java电子商务Spring Cloud+Spring Boot+mybatis+MQ+VR全景+b2b2c
1. 涉及平台 平台管理、商家端(PC端、手机端)、买家平台(H5/公众号、小程序、APP端(IOS/Android)、微服务平台(业务服务) 2. 核心架构 Spring Cloud、Spring Boot、Mybatis、Redis 3. 前端框架…...
python教程:selenium WebDriver 中的几种等待
嗨喽,大家好呀~这里是爱看美女的茜茜呐 强制等待:sleep() import time sleep(5) #等待5秒设置固定休眠时间,单位为秒。 由python的time包提供, 导入 time 包后就可以使用。 缺点: 不智能,使用太多的sleep会影响脚本运行速度。…...
【MATLAB源码-第49期】基于蚁群算法(ACO)算法的栅格路径规划,输出最佳路径图和算法收敛曲线图。
操作环境: MATLAB 2022a 1、算法描述 蚁群算法是一种模拟自然界蚂蚁觅食行为的启发式优化算法。在蚁群系统中,通过模拟蚂蚁之间通过信息素沟通的方式来寻找最短路径。 在栅格路径规划中,蚁群算法的基本步骤如下: 1. 初始化: …...
LabVIEW生产者消费者架构
LabVIEW生产者消费者架构 生产者/消费者模式可以轻松地同时处理多个进程,同时还能以不同速率迭代。 缓冲通信 当多个进程以不同速度运行时,就适合采用进程间缓冲通信。有了足够大的缓冲区后,生产者循环可以以快于消费者循环的速度运行&…...
成都瀚网科技:如何有效运营抖店来客呢?
随着电子商务的快速发展和移动互联网的普及,越来越多的企业开始将目光转向线上销售渠道。其中,抖音成为备受关注的平台。作为中国最大的短视频社交平台之一,抖音每天吸引数亿用户,这也为企业提供了巨大的商机。那么,如…...
iMazing2.17.3免费苹果手机备份还原助手
强大的 iOS 设备管理软件不管是 iPhone、iPad 或 iPod Touch 设备,只要将 iOS 设备连接到计算机,就可以处理不同类型的数据。iMazing 功能强大、易于使用,可以帮助您安全备份任何 iPhone、iPad 或 iPod Touch,备受用户信赖。 现在…...
sql查询到了数据但是实体类个别字段为null(映射失败)
参考博客: sql查询到了数据但是实体类个别字段为null(映射失败)_sql语句查到值,加字段!null没值-CSDN博客 问题描述: 1.SQL查询到了数据,在idea调试output控制台窗口打印出SQL语句,字段有值; 2.接收实体对象字段无…...
配置VScode开发环境-CUDA编程
如果觉得本篇文章对您的学习起到帮助作用,请 点赞 关注 评论 ,留下您的足迹💪💪💪 本文主要介绍VScode下的CUDA编程配置,因此记录以备日后查看,同时,如果能够帮助到更多人…...
openGauss学习笔记-101 openGauss 数据库管理-管理数据库安全-客户端接入之用SSH隧道进行安全的TCP/IP连接
文章目录 openGauss学习笔记-101 openGauss 数据库管理-管理数据库安全-客户端接入之用SSH隧道进行安全的TCP/IP连接101.1 背景信息101.2 前提条件101.3 操作步骤 openGauss学习笔记-101 openGauss 数据库管理-管理数据库安全-客户端接入之用SSH隧道进行安全的TCP/IP连接 101.…...
STM32如何使用中断?
一:EXTI 简介 STM32F10x 外部中断/事件控制器(EXTI)包含多达 20 个用于产生事件/中断请求的边沿检测器。 EXTI 的每根输入线都可单独进行配置,以选择类型(中断或事件)和相应的触发事件(上升 沿触…...
用于物体识别和跟踪的下游任务自监督学习-2-(计算机视觉中的距离度量+损失函数)
2.4 计算机视觉中的距离度量 在深度学习和计算机视觉中,距离度量通常用于比较图像、视频或其他数据的特征或嵌入。根据具体任务和数据属性,可以使用不同类型的距离度量。下面介绍了深度学习和计算机视觉中使用的一些常见类型的距离度量。 余弦相似性距…...
怎么把电脑字体导入wordpress/做百度线上推广
在开始解读AQS的共享功能前,我们再重温一下CountDownLatch,CountDownLatch为java.util.concurrent包下的计数器工具类,常被用在多线程环境下,它在初始时需要指定一个计数器的大小,然后可被多个线程并发的实现减1操作&a…...
办公网站建设/搜索引擎优化seo多少钱
Java对象的内存布局及堆内存划分前言对象的指向Java内存模型Object objnew Object()占用字节对象的访问句柄访问和直接指针访问对比堆内存Young区Old区名词扫盲一个对象的人生轨迹图总结前言 上一篇我们分析了Java虚拟机方法执行流程及方法重载和方法重写原理,并分…...
网站关键词排名/今日热搜榜
折纸王子帮你找折纸图解你好,我是折纸王子,欢迎来到我的公众号。折纸王子教你折纸,讲解详细,演示清楚,容易听懂,一学就会。往期精彩:圣诞节剪纸折纸大全43款万圣节折纸大全专辑 29款折纸张口霸王…...
长沙做网站建设公司哪家好/新闻稿营销
还记得2010年的时候,那个时候移动互联网时代刚刚兴起,很多以前做java的,也就是做J2EE的人(当时J2EE是红海),抓住了这个机会进行的转型,然后得到红利,甚至实现了人生的转变࿰…...
如何做传奇私服网站/seo搜索优化排名
题目链接:https://loj.ac/problem/10013 分析: 对于题目中给定的二次函数是符合用三分求极值的,关键是如何在n个这样的函数中找到最大的那一个? 我们会的就是一个这样的函数在给定区间内求极值。 所以,在每次三分的过…...
国外专门做旅游攻略的网站/广州疫情最新消息今天封城了
6月29日,MWC2017大会在上海新国际博览中心拉开帷幕。本届MWC以“势在人为”为主题,邀请了多位来自全球顶尖的通信、运营商、终端等企业的重量级嘉宾进行主题演讲,就物联网、增强现实与虚拟现实等最新行业趋势展开了深入探讨。在智能革命加速到…...