React基礎(chǔ)(6) -- 事件處理

React 事件處理


React 元素的事件處理和 DOM 元素類似荡短。但是有一點(diǎn)語(yǔ)法上的不同:

  • React 事件綁定屬性的命名采用駝峰式寫法,而不是小寫类垫。
  • 如果采用 JSX 的語(yǔ)法你需要傳入一個(gè)函數(shù)作為事件處理函數(shù)培廓,而不是一個(gè)字符串(DOM 元素的寫法)
HTML 通常寫法是:
<button onclick="activateLasers()">
  激活按鈕
</button>
React 中寫法為:
<button onClick={activateLasers}>
  激活按鈕
</button>

在 React 中另一個(gè)不同是你不能使用返回 false 的方式阻止默認(rèn)行為, 你必須明確的使用 preventDefault误褪。

例如责鳍,通常我們?cè)?HTML 中阻止鏈接默認(rèn)打開(kāi)一個(gè)新頁(yè)面,可以這樣寫:

<a href="#" onclick="console.log('點(diǎn)擊鏈接'); return false">
  點(diǎn)我
</a>
在 React 的寫法為:
function ActionLink() {
  function handleClick(e) {
    e.preventDefault();
    console.log('鏈接被點(diǎn)擊');
  }
 
  return (
    <a href="#" onClick={handleClick}>
      click me
    </a>
  );
}

實(shí)例中 e 是一個(gè)合成事件兽间。

使用 React 的時(shí)候通常你不需要使用 addEventListener 為一個(gè)已創(chuàng)建的 DOM 元素添加監(jiān)聽(tīng)器历葛。你僅僅需要在這個(gè)元素初始渲染的時(shí)候提供一個(gè)監(jiān)聽(tīng)器。

當(dāng)你使用 ES6 class 語(yǔ)法來(lái)定義一個(gè)組件的時(shí)候嘀略,事件處理器會(huì)成為類的一個(gè)方法恤溶。例如乓诽,下面的 Toggle 組件渲染一個(gè)讓用戶切換開(kāi)關(guān)狀態(tài)的按鈕:

class Toggle extends React.Component {
  constructor(props) {
    super(props);
    this.state = {isToggleOn: true};
 
    // 這邊綁定是必要的,這樣 `this` 才能在回調(diào)函數(shù)中使用
    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('example')
);

你必須謹(jǐn)慎對(duì)待 JSX 回調(diào)函數(shù)中的 this咒程,類的方法默認(rèn)是不會(huì)綁定 this 的问裕。如果你忘記綁定 this.handleClick 并把它傳入 onClick, 當(dāng)你調(diào)用這個(gè)函數(shù)的時(shí)候 this 的值會(huì)是 undefined。

這并不是 React 的特殊行為孵坚;它是函數(shù)如何在 JavaScript 中運(yùn)行的一部分粮宛。通常情況下,如果你沒(méi)有在方法后面添加 () 卖宠,例如 onClick={this.handleClick}巍杈,你應(yīng)該為這個(gè)方法綁定 this。

如果使用 bind 讓你很煩扛伍,這里有兩種方式可以解決筷畦。如果你正在使用實(shí)驗(yàn)性的屬性初始化器語(yǔ)法,你可以使用屬性初始化器來(lái)正確的綁定回調(diào)函數(shù):

class LoggingButton extends React.Component {
  // 這個(gè)語(yǔ)法確保了 `this` 綁定在  handleClick 中
  // 這里只是一個(gè)測(cè)試
  handleClick = () => {
    console.log('this is:', this);
  }
 
  render() {
    return (
      <button onClick={this.handleClick}>
        Click me
      </button>
    );
  }
}

如果你沒(méi)有使用屬性初始化器語(yǔ)法刺洒,你可以在回調(diào)函數(shù)中使用 箭頭函數(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ǔ)法有個(gè)問(wèn)題就是每次 LoggingButton 渲染的時(shí)候都會(huì)創(chuàng)建一個(gè)不同的回調(diào)函數(shù)鳖宾。在大多數(shù)情況下,這沒(méi)有問(wèn)題逆航。然而如果這個(gè)回調(diào)函數(shù)作為一個(gè)屬性值傳入低階組件鼎文,這些組件可能會(huì)進(jìn)行額外的重新渲染。我們通常建議在構(gòu)造函數(shù)中綁定或使用屬性初始化器語(yǔ)法來(lái)避免這類性能問(wèn)題因俐。


向事件處理程序傳遞參數(shù)
通常我們會(huì)為事件處理程序傳遞額外的參數(shù)拇惋。例如,若是 id 是你要?jiǎng)h除那一行的 id抹剩,以下兩種方式都可以向事件處理程序傳遞參數(shù):

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

上述兩種方式是等價(jià)的撑帖。
上面兩個(gè)例子中,參數(shù) e 作為 React 事件對(duì)象將會(huì)被作為第二個(gè)參數(shù)進(jìn)行傳遞澳眷。通過(guò)箭頭函數(shù)的方式胡嘿,事件對(duì)象必須顯式的進(jìn)行傳遞,但是通過(guò) bind 的方式钳踊,事件對(duì)象以及更多的參數(shù)將會(huì)被隱式的進(jìn)行傳遞衷敌。

值得注意的是,通過(guò) bind 方式向監(jiān)聽(tīng)函數(shù)傳參箍土,在類組件中定義的監(jiān)聽(tīng)函數(shù)逢享,事件對(duì)象 e 要排在所傳遞參數(shù)的后面,例如:

class Popper extends React.Component{
    constructor(){
        super();
        this.state = {name:'Hello world!'};
    }
    
    preventPop(name, e){    //事件對(duì)象e要放在最后
        e.preventDefault();
        alert(name);
    }
    
    render(){
        return (
            <div>
                <p>hello</p>
                {/* 通過(guò) bind() 方法傳遞參數(shù)吴藻。 */}
                <a  onClick={this.preventPop.bind(this,this.state.name)}>Click</a>
            </div>
        );
    }
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末瞒爬,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌侧但,老刑警劉巖矢空,帶你破解...
    沈念sama閱讀 206,839評(píng)論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異禀横,居然都是意外死亡屁药,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,543評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門柏锄,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)酿箭,“玉大人,你說(shuō)我怎么就攤上這事趾娃$缘眨” “怎么了?”我有些...
    開(kāi)封第一講書人閱讀 153,116評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵抬闷,是天一觀的道長(zhǎng)妇蛀。 經(jīng)常有香客問(wèn)我,道長(zhǎng)笤成,這世上最難降的妖魔是什么评架? 我笑而不...
    開(kāi)封第一講書人閱讀 55,371評(píng)論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮炕泳,結(jié)果婚禮上纵诞,老公的妹妹穿的比我還像新娘。我一直安慰自己喊崖,他們只是感情好挣磨,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,384評(píng)論 5 374
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著荤懂,像睡著了一般。 火紅的嫁衣襯著肌膚如雪塘砸。 梳的紋絲不亂的頭發(fā)上节仿,一...
    開(kāi)封第一講書人閱讀 49,111評(píng)論 1 285
  • 那天,我揣著相機(jī)與錄音掉蔬,去河邊找鬼廊宪。 笑死,一個(gè)胖子當(dāng)著我的面吹牛女轿,可吹牛的內(nèi)容都是我干的箭启。 我是一名探鬼主播,決...
    沈念sama閱讀 38,416評(píng)論 3 400
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼蛉迹,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼傅寡!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書人閱讀 37,053評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤荐操,失蹤者是張志新(化名)和其女友劉穎芜抒,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體托启,經(jīng)...
    沈念sama閱讀 43,558評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡宅倒,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,007評(píng)論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了屯耸。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片拐迁。...
    茶點(diǎn)故事閱讀 38,117評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖疗绣,靈堂內(nèi)的尸體忽然破棺而出线召,到底是詐尸還是另有隱情,我是刑警寧澤持痰,帶...
    沈念sama閱讀 33,756評(píng)論 4 324
  • 正文 年R本政府宣布灶搜,位于F島的核電站,受9級(jí)特大地震影響工窍,放射性物質(zhì)發(fā)生泄漏割卖。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,324評(píng)論 3 307
  • 文/蒙蒙 一患雏、第九天 我趴在偏房一處隱蔽的房頂上張望鹏溯。 院中可真熱鬧,春花似錦淹仑、人聲如沸丙挽。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 30,315評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)颜阐。三九已至,卻和暖如春吓肋,著一層夾襖步出監(jiān)牢的瞬間凳怨,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 31,539評(píng)論 1 262
  • 我被黑心中介騙來(lái)泰國(guó)打工是鬼, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留肤舞,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,578評(píng)論 2 355
  • 正文 我出身青樓均蜜,卻偏偏與公主長(zhǎng)得像李剖,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子囤耳,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,877評(píng)論 2 345

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

  • 在這個(gè)城市我最愛(ài)的人 他在清晨的農(nóng)貿(mào)集市場(chǎng)上 經(jīng)由他篙顺,我覺(jué)得上帝仍然 關(guān)心我偶芍,不忍我痛苦 他的皮膚染著太陽(yáng)的顏色 ...
    milan_a07d閱讀 239評(píng)論 0 0
  • 1. 安卓系統(tǒng)的痛點(diǎn),慢慰安,卡腋寨,不容易用,死機(jī)化焕。因?yàn)樗旧聿皇菫橛脩粼O(shè)計(jì)的一個(gè)系統(tǒng)萄窜,設(shè)計(jì)思想有點(diǎn)像早期的window...
    晨宇閱讀 348評(píng)論 0 0
  • 今天查刻,還是一如既往的坐上地鐵,看著眼前這一大片一大片凤类,一群一群的人穗泵。 有帶著耳機(jī),背著公文包的文職者谜疤,有穿著有點(diǎn)泛...
    Lyndon7777閱讀 3,959評(píng)論 59 103
  • 每個(gè)女生可能都在青春期里喜歡過(guò)這樣一個(gè)人夷磕,他喜歡穿著白色的襯衫履肃,他笑起來(lái)明媚的像初升的太陽(yáng),他身上有好聞的橘...
    端木酸奶包閱讀 223評(píng)論 0 0