渲染函数&JSX

大多数情况使用vue模板语法创建应用,在某些场景下,需要用到JavaScript完全的编程能力,这时就用到渲染函数了。

基本用法

创建Vnodes

Vue提供了一个h()函数用于创建vnodes:

import {h} from 'vue'

const vnode = h(
    'div', // type
    { id:'foo', class:'bar' }, //props
    [] // children
)

h()是hyperscript的简称,代表“能生成HTML的JavaScript”.

// h()函数除了类型必填以外,其他的参数都是可选的,
h('div')
h('div',{id:'foo'})

// attribute(用户手动赋值的属性,后天属性值)和property(html自带的属性,原生属性值) 
// 都能在prop中书写,vue会将它们分配到正确的位置
h('div', {class:'bar',innerHTML:'hello' })

//类与样式可以像在模板中一样  用数组或对象的形式书写
h('div',{class:[foo,{bar}],style:{color:'red'} })

// 事件监听器应以onXxx的形式书写
h('div',{onClick:()=>{} })

// children可以是一个字符串
h('div',{id:'foo'},'hello')

// 没有props时可以省略不写
h('div','hello')
h('div', [h('span','hello')] )

// children数组可以同时包含vnodes与字符串
h('div',['hello',h('span','hello')])

声明渲染函数

当组合式API与模板一起使用时,setup()钩子的返回值是用于暴露数据给模板。然而当我们使用渲染函数时,可以直接把渲染函数返回:

import {ref,h} from 'vue'

export default{
    props:{

    },
    setup(props){
        const count = ref(1)
        return () => h('div',props.msg +count.value)
    }
}

Vnodes 必须唯一

function render(){
    return h(
        'div',
        p,
        p, // 重复的vnodes是无效的,可以使用一个工厂函数来渲染多个相同的段落
        Array.from({length:20}).map(()=>{
            return h('p','hi')
        })
    )

}

JSX/TSX

JSX是JavaScript的一个类似XML的扩展,可以用以下方式来写代码

cosnt vnode = <div>hello</div>

// 在JSX表达式中,使用大括号来嵌入动态值
const vnode2 = <div id={dynamicId}> hello,{username} </div>

注意Vue的JSX转换方式与React中JSX的转换方式不同

看一下v-if在模板、渲染函数和JSX中的不同写法:

<div>
  <div v-if="ok">yes</div>
  <span v-else>no</span>
</div>

等价于使用如下渲染函数/JSX语法:

h('div', [ok.value? h('div','yes') : h('span','no') ] )

<div> {ok.value ? <div>yes</div> : <span>no</span>  } </div>

v-for

<ul>
  <li v-for="{ id, text } in items" :key="id">
    {{ text }}
  </li>
</ul>

等价于

h(
'ul',
 items.value.map((item)=>{
     return  h('li', {key:item.id} , item.text)
 }) 
)
<ul>
 { 
    items.value.map(({id,text})=>{
         return <li key={id}>{text}<li>   
     }) 
 }
 </ul>

v-on

h('button',{onClick: ()=>{} },'click me')
<button
 onClick={ ()=>{
    /*...*/
    }}>
click me
</button>

在 React 中,`render` 函数渲染 JSX 是最常见、最核心的用法之一。JSXJavaScript XML)是一种语法扩展,允许你在 JavaScript 中直接写 HTML-like 的结构,它最终会被 Babel 编译成 `React.createElement()` 调用。 --- ## ✅ 示例:基本 render 函数渲染 JSX ### 1. 创建一个组件并渲染 JSX: ```jsx import React from 'react'; import ReactDOM from 'react-dom/client'; function App() { return ( <div> <h1>Hello, JSX!</h1> <p>这是使用 JSX 渲染的内容。</p> </div> ); } const root = ReactDOM.createRoot(document.getElementById('root')); root.render(<App />); ``` 上面代码中,`App` 组件返回了一个 JSX 结构,并通过 `ReactDOM.createRoot().render()` 方法进行渲染。 --- ## ✅ JSX 是如何工作的? 你写的 JSX: ```jsx <h1>Hello, JSX!</h1> ``` Babel 会将其编译为: ```js React.createElement('h1', null, 'Hello, JSX!') ``` 所以本质上,所有 JSX 最终都会被转译为 `React.createElement(...)` 调用。 --- ## ✅ 使用函数组件和条件渲染 JSX 你可以根据状态或 props 来动态返回不同的 JSX: ```jsx function Greeting({ isLoggedIn }) { return ( <div> {isLoggedIn ? ( <p>欢迎回来!</p> ) : ( <p>请先登录。</p> )} </div> ); } ``` --- ## ✅ 在变量中存储 JSX JSX 可以赋值给变量,也可以作为函数返回值: ```jsx function StatusMessage({ status }) { let message; if (status === 'loading') { message = <p>加载中...</p>; } else if (status === 'success') { message = <p>操作成功!</p>; } else { message = <p>发生错误。</p>; } return <div>{message}</div>; } ``` --- ## ✅ 在 JSX 中嵌入表达式 你可以使用 `{}` 插入任何有效的 JS 表达式: ```jsx function User({ name, age }) { return ( <div> <p>姓名: {name}</p> <p>年龄: {age}</p> <p>出生年份: {new Date().getFullYear() - age}</p> </div> ); } ``` --- ## ✅ 使用 map 渲染列表 ```jsx function TodoList({ items }) { return ( <ul> {items.map((item, index) => ( <li key={index}>{item}</li> ))} </ul> ); } ``` --- ## ✅ 注意事项 - 所有 JSX 必须包裹在一个根元素中(不能有多个同级标签)。 - JSX 支持所有 HTML 属性,但部分属性名需要使用驼峰命名(如 `className`, `htmlFor`)。 - 不要在 JSX 中使用 `if/else`,可以使用三元运算符或者提前赋值。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值