Lodash debounce TypeError : 함수 반응이 필요합니다.
Nov 13 2020
debounce
데이터 테이블에서 무언가를 입력하고 검색 할 때 여러 호출을 피하기 위해 함수 를 사용하려고합니다 . 내가 지금하고있는 것은 입력에서
onChange={(e) => {
const delayedQuery = useCallback(debounce(this.handleSearch(e.target.value), 500));
return delayedQuery
}}
곳 handeSearch
이다
handleSearch(filter) {
service.getData(filter).subscribe((data) => {console.log(data)})
}
하지만이 오류가 TypeError: Expected a function
있습니다. 서비스는 작동하지만 디 바운스는 없습니다. 입력하는 동시에 문자별로 작성하는데 올바르지 않습니다.
답변
1 HMR Nov 13 2020 at 19:54
클래스 구성 요소 나 이벤트 핸들러 함수에서 후크를 사용할 수 없으며 lodash 디 바운스에 함수를 전달해야합니다.
또한; target.value
이벤트에서 비동기 적 으로 읽으 려고하면 이 합성 이벤트가 성능상의 이유로 재사용 된다는 오류가 발생합니다.
채팅에서 rxjs를 사용한다고 가정 from(promise)
하므로 사용자가 입력하고 비동기 요청을 실행할 때 다른 순서로 해결할 수 있다는 문제가 여전히 있습니다.
깨진 예는 아래를 참조하십시오 (빨리 hai를 입력하고 2 초간 기다리십시오. ha
)
//simple debounce implementation
const debounce = (fn, delay = 50) => {
let time;
return (...args) => {
clearTimeout(time);
time = setTimeout(() => fn(...args), delay);
};
};
//service where you can subscribe to
const service = (() => {
let listeners = [];
const later = (value, time = 50) =>
new Promise((r) =>
//if search is 2 characters long it'll resolve much later
setTimeout(
() => r(value),
value.length === 2 ? 2000 : time
)
);
//subscribe returns an unsubscribe function
const subScribe = (fn) => {
listeners.push(fn);
//return the Subscriber object
//https://rxjs-dev.firebaseapp.com/api/index/class/Subscriber
return {
unsubscribe: () =>
(listeners = listeners.filter(
(listener) => listener !== fn
)),
};
};
//notify all listeners
const notify = (value) =>
listeners.forEach((listener) => listener(value));
return {
getData: (value) => ({
subScribe: (fn) => {
const unsub = subScribe(fn);
later(value).then((value) => notify(value));
return unsub;
},
}),
};
})();
class App extends React.PureComponent {
constructor(props) {
super(props);
this.state = { asyncSearchResult: '' };
//use handleSearch to create a debounced version of it
this.handleSearchDebounced = debounce(
this.handleSearch
).bind(this); //you need to bind it to guearantee correct context
}
rendered = 0;
handleSearch = (value) =>
service
.getData(value)
.subScribe((value) =>
this.setState({ asyncSearchResult: value })
);
render() {
return (
<div>
<input
type="text"
// pass value to the handler, not the event
onChange={(e) =>
this.handleSearchDebounced(e.target.value)
}
/>
async result: {this.state.asyncSearchResult}
</div>
);
}
}
ReactDOM.render(<App />, document.getElementById('root'));
<script src="https://cdnjs.cloudflare.com/ajax/libs/react/16.8.4/umd/react.production.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/react-dom/16.8.4/umd/react-dom.production.min.js"></script>
<div id="root"></div>
해결책은 다음과 같습니다.
//simple debounce implementation
const debounce = (fn, delay = 50) => {
let time;
return (...args) => {
clearTimeout(time);
time = setTimeout(() => fn(...args), delay);
};
};
//service where you can subscribe to
const service = (() => {
let listeners = [];
const later = (value, time = 50) =>
new Promise((r) =>
//if search is 2 characters long it'll resolve much later
setTimeout(
() => r(value),
value.length === 2 ? 2000 : time
)
);
//subscribe returns an unsubscribe function
const subScribe = (fn) => {
listeners.push(fn);
//return the Subscriber object
//https://rxjs-dev.firebaseapp.com/api/index/class/Subscriber
return {
unsubscribe: () =>
(listeners = listeners.filter(
(listener) => listener !== fn
)),
};
};
//notify all listeners
const notify = (value) =>
listeners.forEach((listener) => listener(value));
return {
getData: (value) => ({
subScribe: (fn) => {
const unsub = subScribe(fn);
later(value).then((value) => notify(value));
return unsub;
},
}),
};
})();
const latestGetData = (value) => {
const shared = {};
return {
subScribe: (fn) => {
const current = {};
shared.current = current;
const subscriber = service
.getData(value)
.subScribe((result) => {
//only call subscriber of latest resolved
if (shared.current === current) {
fn(result);
}
subscriber.unsubscribe();
});
},
};
};
class App extends React.PureComponent {
constructor(props) {
super(props);
this.state = { asyncSearchResult: '' };
//use handleSearch to create a debounced version of it
this.handleSearchDebounced = debounce(
this.handleSearch
).bind(this); //you need to bind it to guearantee correct context
}
rendered = 0;
handleSearch = (value) =>
latestGetData(value).subScribe((value) =>
this.setState({ asyncSearchResult: value })
);
render() {
return (
<div>
<input
type="text"
// pass value to the handler, not the event
onChange={(e) =>
this.handleSearchDebounced(e.target.value)
}
/>
async result: {this.state.asyncSearchResult}
</div>
);
}
}
ReactDOM.render(<App />, document.getElementById('root'));
<script src="https://cdnjs.cloudflare.com/ajax/libs/react/16.8.4/umd/react.production.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/react-dom/16.8.4/umd/react-dom.production.min.js"></script>
<div id="root"></div>