React 處理事件

用 React 元素處理事件和在DOM元素里處理事件非常相似绳军。有一些語(yǔ)法差異:

  • React 事件使用駝峰式命名哄褒,而不是小寫
  • 用JSX依许,你能傳遞函數(shù)作為事件處理器,而不是一個(gè)字符串

HTML:

<button onclick="activateLasers()">
    Activate Lasers
</button>

React:

<button onClick={activateLasers}>
    Activate Lasers
</button>

另外一個(gè)區(qū)別刽沾,在 React 中本慕,不能用 return false 阻止默認(rèn)行為。必須直接調(diào)用 preventDefault 例如侧漓,用純HTML锅尘,為了防止打開新頁(yè)面的默認(rèn)鏈接行為,可以這樣:

<a href="#" onclick="console.log('The link was clicked.'); return false">
 Click me
</a>

在React中, 可能是這樣:

function ActionLink() {
  function handleClick(e) {
    e.preventDefault();
    console.log('The link was clicked.');
  }

  return (
    <a href="#" onClick={handleClick}>
      Click me
    </a>
  );
}

這里的 e 是一個(gè)合成事件布蔗,React 根據(jù) W3C spec 定義這些合成事件藤违,所以不需要擔(dān)心瀏覽器的兼容性。

備注:合成事件--是一個(gè)跨瀏覽器原生事件包裝器纵揍,它具有與瀏覽器原生事件相同的接口顿乒,包括 stopPropagation() 和 preventDefault() ,除了原生事件在所有瀏覽器中泽谨,他們工作方式都相同璧榄。

當(dāng)使用 React 時(shí),通常不需要調(diào)用 addEventListener 對(duì)創(chuàng)建后的DOM元素添加監(jiān)聽器吧雹。相反骨杂,當(dāng)元素最初被渲染時(shí),僅需提供一個(gè)監(jiān)聽器雄卷。

當(dāng)使用 ES6 class 定義一個(gè)組件搓蚪,一個(gè)常見的模式是一個(gè)事件處理器在類中是一個(gè)方法。例如丁鹉,toggle 組件渲染讓用戶在 ONOFF 之間切換的按鈕:

class Toggle extends React.Component {
  constructor(props) {
    super(props);
    this.state = {isToggleOn: true};
    
    // 此綁定是必需的妒潭,讓 `this` 在回調(diào)里生效
    this.handleClick = this.handleClick.bind(this);
  }

  handleClick() {
    this.setState(prevState => ({
      isToggleOn: !prevState.isToggleOn
    }));
  }

  render() {
    return (
      <button onClick={this.handleClick}>
        {this.state.isToggleOn ? 'ON' : 'OFF'}
      </button>
    );
  }
}

ReactDOM.render(
  <Toggle />,
  document.getElementById('root')
);

demo

在JSX回調(diào)中,你必須注意 this 的指向揣钦。在 Javascript中雳灾,類方法沒(méi)有被默認(rèn)綁定。如果你沒(méi)有綁定 this.handleClick 并把它傳遞到 onClick拂盯,則函數(shù)被實(shí)際調(diào)用時(shí)佑女,this 將會(huì)是 undefined

這不是 React 特有的行為谈竿;這是Javascript中函數(shù)如何工作的一部份团驱,通常,如果在它后面提供一個(gè)沒(méi)有 () 方法空凸,像onClick={this.handleClick}嚎花,你就應(yīng)該綁定( bind) 該方法。

如果你不想調(diào)用 bind 呀洲,有兩種方式能解決它紊选。如果使用試驗(yàn)性 public class fields syntax啼止,可以用類字段正確的綁定回調(diào):

class LoggingButton extends React.Component {
  // 這個(gè)語(yǔ)法確保 `this` 被綁定在handleClick中
  // 警告: 這是 *試驗(yàn)性* 語(yǔ)法
  handleClick = () => {
    console.log('this is:', this);
  }

  render() {
    return (
      <button onClick={this.handleClick}>
        Click me
      </button>
    );
  }
}

此語(yǔ)法在創(chuàng)建 React APP 中是默認(rèn)開啟的。

如果不用類字段語(yǔ)法兵罢,也可以在回調(diào)中庸一個(gè)箭頭函數(shù):

class LoggingButton extends React.Component {
  handleClick() {
    console.log('this is:', this);
  }

  render() {
    // 這個(gè)語(yǔ)法確保 `this` 被綁定在handleClick中
    return (
      <button onClick={(e) => this.handleClick(e)}>
        Click me
      </button>
    );
  }
}

這個(gè)語(yǔ)法的問(wèn)題是每次 LoggingButton 渲染時(shí)都會(huì)創(chuàng)建一個(gè)不同的回調(diào)献烦。大多數(shù)情況下,這沒(méi)問(wèn)題卖词。但是巩那,如果回調(diào)被當(dāng)作 prop 傳遞給子組件,這些組件可能做額外重復(fù)渲染此蜈。我們通常推薦綁定在構(gòu)造函數(shù)中或者使用類字段語(yǔ)法即横,避免這種性能問(wèn)題。


將實(shí)參傳遞給事件處理器

在循環(huán)內(nèi)部裆赵,通常想把額外的形參傳遞給事件處理器东囚。列如,如果 id 是行ID战授,以下二者都可以工作:

<button onClick={(e) => this.deleteRow(id, e)}>Delete Row</button>
<button onClick={this.deleteRow.bind(this, id)}>Delete Row</button>

上面二行是等價(jià)的页藻,使用 箭頭函數(shù) Function.prototype.bind

在這兩者情況下,e 實(shí)參表示 React 事件將作為ID后第二個(gè)實(shí)參傳遞陈醒。使用箭頭函數(shù)惕橙,我們必須傳遞它,但是用 bind 綁定的任何的參數(shù)會(huì)被自動(dòng)的轉(zhuǎn)發(fā)钉跷。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市肚逸,隨后出現(xiàn)的幾起案子爷辙,更是在濱河造成了極大的恐慌,老刑警劉巖朦促,帶你破解...
    沈念sama閱讀 206,126評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件膝晾,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡务冕,警方通過(guò)查閱死者的電腦和手機(jī)血当,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,254評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)禀忆,“玉大人臊旭,你說(shuō)我怎么就攤上這事÷嵬耍” “怎么了离熏?”我有些...
    開封第一講書人閱讀 152,445評(píng)論 0 341
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)戴涝。 經(jīng)常有香客問(wèn)我滋戳,道長(zhǎng)钻蔑,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,185評(píng)論 1 278
  • 正文 為了忘掉前任奸鸯,我火速辦了婚禮咪笑,結(jié)果婚禮上烦秩,老公的妹妹穿的比我還像新娘嘶伟。我一直安慰自己,他們只是感情好巧颈,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,178評(píng)論 5 371
  • 文/花漫 我一把揭開白布钝满。 她就那樣靜靜地躺著兜粘,像睡著了一般。 火紅的嫁衣襯著肌膚如雪弯蚜。 梳的紋絲不亂的頭發(fā)上孔轴,一...
    開封第一講書人閱讀 48,970評(píng)論 1 284
  • 那天,我揣著相機(jī)與錄音碎捺,去河邊找鬼路鹰。 笑死,一個(gè)胖子當(dāng)著我的面吹牛收厨,可吹牛的內(nèi)容都是我干的晋柱。 我是一名探鬼主播,決...
    沈念sama閱讀 38,276評(píng)論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼诵叁,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼雁竞!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起拧额,我...
    開封第一講書人閱讀 36,927評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤碑诉,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后侥锦,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體进栽,經(jīng)...
    沈念sama閱讀 43,400評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,883評(píng)論 2 323
  • 正文 我和宋清朗相戀三年恭垦,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了快毛。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 37,997評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡番挺,死狀恐怖唠帝,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情建芙,我是刑警寧澤没隘,帶...
    沈念sama閱讀 33,646評(píng)論 4 322
  • 正文 年R本政府宣布,位于F島的核電站禁荸,受9級(jí)特大地震影響右蒲,放射性物質(zhì)發(fā)生泄漏阀湿。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,213評(píng)論 3 307
  • 文/蒙蒙 一瑰妄、第九天 我趴在偏房一處隱蔽的房頂上張望陷嘴。 院中可真熱鬧,春花似錦间坐、人聲如沸灾挨。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,204評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)劳澄。三九已至,卻和暖如春蜈七,著一層夾襖步出監(jiān)牢的瞬間秒拔,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,423評(píng)論 1 260
  • 我被黑心中介騙來(lái)泰國(guó)打工飒硅, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留砂缩,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,423評(píng)論 2 352
  • 正文 我出身青樓三娩,卻偏偏與公主長(zhǎng)得像庵芭,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子雀监,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,722評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容