从零开始学习React:react中事件处理与柯里化

从零开始学习React:了解当前React常用技术,编写第一个React组件
从零开始学习React:了解组件的三大属性
从零开始学习React:react中事件处理与柯里化

1. 事件处理

React 中元素也可接受、处理事件,但是在语法上有一点不同。

在React 中所有事件的命名采用的是小驼峰,而非原生 DOM 的纯小写,所有事件需要我们传入一个函数,而非字符串。

例如:

1
2
3
4
5
6
const Button = () => {
const handleClick = () => {
console.log('click')
}
return <button onClick={handleClick}>click button</button>
}

当事件的回调函数比较简单时,我们也可以简写箭头匿名函数,例如:

1
2
3
4
5
6
7
8
const Button = () => {
return (
<button
onClick={() => console.log('click')}
>
click button
</button>)
}

阻止默认行为

在React 中不能通过返回 false 来阻止默认行为,例如表单提交、a标签跳转。我们必须要通过显式调用 preventDefault 函数,来阻止这些默认行为。

1
2
3
4
5
6
7
8
const Link = () => {
return <a
href="https://www.baidu.com"
onClick={(e) => e.preventDefault()}
>
link
</a>
}

合成事件

在 React 中几乎所有的事件处理函数,都是一个 (event)=>void 函数,如果我们使用 typescript,可以清晰的看到每个事件对应的函数类型,React 自身也声明了很多的事件与事件处理函数类型,例如鼠标事件:MouseEvent<T = Element>MouseEventHandler<T = Element>,我们在使用时可以根据自己的喜欢,是定义函数类型还是定义参数类型,就像这样:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
const Link = () => {
const handleClick = (e: MouseEvent) => {
e.preventDefault()
console.log('click')
}
const handleMouseEnter:MouseEventHandler = (e) => {
console.log('mouse enter')
}
return <a
href="https://www.baidu.com"
onMouseEnter={handleMouseEnter}
onClick={handleClick}
>
link
</a>
}

在 React 中,所有事件都是 React 根据 W3C 规范定义的合成事件,所以我们完全不用担心兼容性问题,React 事件与原生事件不完全相同。

点击此处查看合成事件文档

2. 柯里化

柯里化这个名称对于 Android 开发可能有点陌生,因为我们一般使用 Java 开发,因为早期的 Java 不支持函数式编程(FP),而柯里化是一个函数式编程思想。

简而言之是将一个多参函数变成单参数函数,举个栗子:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
//柯里化后的单参数函数
function sumCurrying(a) {
return (b) => {
return (c) => {
return a + b + c;
};
};
}
//普通的多参数函数
function sumNormal(a, b, c) {
return a + b + c
}
console.log(sumCurrying(1)(2)(3));
console.log(sumNormal(1, 2, 3));

柯里化的本质,就是高阶函数的一个特性:函数的返回值可以是一个函数。

上面的例子,似乎有点脱裤子放屁,看似毫无意义。但实际工程中,柯里化是一个非常实用的小 trick。最常用在事件处理需要传入值的场景。

我们在上面说过了,React 中的事件回调函数是有固定的函数类型的,几乎都是 (event)=>void 函数。我们需要传入一些参数给这个事件处理函数呢?

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
const List = () => {
const list = [
{ id: 1, name: 'tom' },
{ id: 2, name: 'jerry' },
{ id: 3, name: 'jack' },
{ id: 4, name: 'lily' },
]
const handleClick = (id: number) => {
console.log(id)
}
return <ul>
{list.map(item => <li
onClick={() => handleClick(item.id)}
key={item.id}
>
{item.name}
</li>
)}
</ul>
}

这看起来似乎很不优雅,我们已经声明了 handle 函数,却又不得不在事件处理函数中写行内的箭头函数,如何才能更加优雅的处理呢?

其实很简单,我们只需要在原本的 handle 函数中,插入一个箭头即可,就像这样:

1
2
3
4
5
6
7
8
//before
const handleClick = (id: number) => {
console.log(id)
}
//after
const handleClick = (id: number) => (e:MouseEvent) => {
console.log(id)
}

然后我们的 onClick 事件回调函数就可以改成 onClick={handleClick(item.id)} ,这样看起来是不是就更加优雅了呢?

其实这种设计思想可以说是一说就透,只不过我现在告诉你,这种思想就叫做:柯里化

柯里化的目的

你可能会问我柯里化看起来只是让我们的代码优雅了一点,在目前看来似乎没有什么本质上的变化。

但其实柯里化帮助我们实现了函数的一变多,我们用一个日志输出的函数作为例子:

1
2
3
4
5
6
7
8
//原始函数
const log = (date, importance, message) => {
alert(`[${date.getHours()}:${date.getMinutes()}] [${importance}] ${message}`);
}
//柯里化
const logCurry = (date) => (importance) => (message) => {
alert(`[${date.getHours()}:${date.getMinutes()}] [${importance}] ${message}`);
}

柯里化后,函数变成这样调用:logCurry(new Date())("DEBUG")("some debug");

现在我们相当于拥有这些函数:

1
2
3
4
5
6
7
8
9
10
// logNow 会是带有固定第一个参数的日志的函数
let logNow = logCurry(new Date());

// 使用它
logNow("INFO", "message"); // [HH:mm] INFO message

// debugNow 会是带有固定第一个参数与第二个参数的函数
let debugNow = logNow("DEBUG");

debugNow("message"); // [HH:mm] DEBUG message

看起来只是增加了几个箭头,实际上我们函数的灵活性大为增加。通过固定不同的参数,我们从一个函数声明获得了多个函数。

一个简单的例子

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
const Form = () => {
const [form, setForm] = React.useState({});
const update = (name) => (event) => {
setForm({
...form,
[name]: event.target.value,
});
}
const handleSubmit = (event) => {
event.preventDefault();
alert(`${JSON.stringify(form)}`);
}
return (
<div>
<h1>柯里化表单</h1>
<FormItem label="用户名" name='username' update={update} />
<FormItem label="昵称" name='nickname' update={update} />
<FormItem label="邮箱" name='email' update={update} />
<button onClick={handleSubmit}>提交</button>
</div>
)
}

const FormItem = ({ label, name, update }) => {
return (
<div style={{ 'display': 'flex' }}>
<label>{label}</label>
<input onChange={update(name)} type="text" placeholder={`请输入${label}`} />
</div>
);
};