最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

javascript - React - setState in componentWillMount() results in a UI lag - Stack Overflow

programmeradmin1浏览0评论

I have to fetch some data in ponentWillMount(), this takes some time (2 second), After this, I use setState to Update some values in the state, since this setState re-renders the UI, there is a delay between the ponent initial render and the render from setState, Is there any way to fix this UI lag?

Update: if I want to use a loading indicator, where should I put it? I use a promise to fetch my data like this:

ponentDidMount() {
  api.getData().then((response) => { ... }

I have to fetch some data in ponentWillMount(), this takes some time (2 second), After this, I use setState to Update some values in the state, since this setState re-renders the UI, there is a delay between the ponent initial render and the render from setState, Is there any way to fix this UI lag?

Update: if I want to use a loading indicator, where should I put it? I use a promise to fetch my data like this:

ponentDidMount() {
  api.getData().then((response) => { ... }
Share edited Apr 28, 2019 at 6:31 Sagiv b.g 31k10 gold badges72 silver badges104 bronze badges asked Sep 25, 2017 at 17:45 AdelAdel 3,7439 gold badges32 silver badges31 bronze badges 3
  • 1 if it takes 2 seconds to fetch data, there will always be 2 seconds delay between initial render and setState triggering a render. Are you asking how to prevent all rendering until after data is fetched? – Dan O Commented Sep 25, 2017 at 17:48
  • No, is it possible to prevent all rendering? – Adel Commented Sep 25, 2017 at 17:53
  • I mean isn't that the whole point of a SPA app? Instead of HTML ing down from the server with data, you load the skeleton first and load the data after your async returns. I'm not sure how you would avoid this situation, its either load the skeleton (have a loading indicator) then load the data or keep the page blank and load the data + skeleton after which i'm not sure why that would be better. – pk1m Commented Sep 25, 2017 at 21:35
Add a ment  | 

3 Answers 3

Reset to default 10

You should never use async operations in ponentWillMount or the constructor.
Instead do it in ponentDidMount.
You can read about it in the DOCS

setting state synchronously in this method will not trigger a re-rendering. Avoid introducing any side-effects or subscriptions in this method.

Edit
As a followup to your updated question

if I want to use a loading indicator, where should I put it? I use a promise to fetch my data

I've made a small example of fetching data and rendering a loader while data is being fetched.

In this example i'm using a free API tester called jsonplaceholder, I'm using this URL to fetch some random data of users.
You can see that i initialize the state in the contructor with an empty array of users, I'm fetching the users in the ponentDidMount life cycle method and updating the users array of the state inside the callback of the promise that has returned. Note that i did that inside a setTimeOut method in order to get a delay of 2 seconds.

Now, React won't really going to wait for our ajax request to get back with the results, it will invoke the render method no matter what, hence doing the ajax request in a life cycle method that runs before the render method (like ponentWillMount or the constructor) is not a best practice as mentioned above, so that's why we do it inside the ponentDidMount method.

You probably asking, Ok then! But HOW and WHAT should i render before the data is received and then render the data after it is received? I'm glad you asked :)

We can use the life cycle of react to work for us and take advantage of the powerful render options and state updates.
In this example I conditionally rendered a <Loader /> or the data <UserList/> with the help of the ternary operator.

return ({this.state.users.length > 0 ? <UserList /> : <Loader/>);

This way, when ever the users array of state is empty it will render the Loader ponent, after the state will updated (which will happen after the ajax request is finished) the render method will be invoked again but this time the condition will return true thus the UserList will render and not the Loader.

Here is the full running example:

const apiUrl = "https://jsonplaceholder.typicode./users";

const User = ({ name, username, email }) => (
  <div style={{ border: "1px solid #ccc", padding: "15px" }}>
    <div>Name: {name}</div>
    <div>User Name: {username}</div>
    <div>E-Mail: {email}</div>
  </div>
);

const UserList = ({ users }) =>(
  <div>
    {users.map(user => <User key={user.id} {...user} />)}
  </div>
);

const Loader = () => (
<div id="escapingBallG">
	<div id="escapingBall_1" className="escapingBallG"></div>
</div>
);

class App extends React.Component {
  constructor(props) {
    super(props);
    this.state = {
      users: []
    };
  }

  ponentDidMount() {
    // mimic 2 seconds delay
    setTimeout(() => {
      axios.get(apiUrl)
        .then(users => {
          this.setState({
            users: [...users.data]
          });
        })
        .catch(err => console.log(err));
    }, 2000);
  }
  render() {
    const { users } = this.state;
    return (
      <div>
        {
          users.length > 0 ? <UserList users={users} /> : <Loader />
        }
      </div>
    );
  }
}

ReactDOM.render(<App />, document.getElementById("root"));
#escapingBallG{
	position:relative;
	width:125px;
	height:43px;
	margin:auto;
}

.escapingBallG{
	background-color:rgb(0,0,0);
	position:absolute;
	top:0;
	left:0;
	width:43px;
	height:43px;
	border-radius:21px;
		-o-border-radius:21px;
		-ms-border-radius:21px;
		-webkit-border-radius:21px;
		-moz-border-radius:21px;
	animation-name:bounce_escapingBallG;
		-o-animation-name:bounce_escapingBallG;
		-ms-animation-name:bounce_escapingBallG;
		-webkit-animation-name:bounce_escapingBallG;
		-moz-animation-name:bounce_escapingBallG;
	animation-duration:1.5s;
		-o-animation-duration:1.5s;
		-ms-animation-duration:1.5s;
		-webkit-animation-duration:1.5s;
		-moz-animation-duration:1.5s;
	animation-iteration-count:infinite;
		-o-animation-iteration-count:infinite;
		-ms-animation-iteration-count:infinite;
		-webkit-animation-iteration-count:infinite;
		-moz-animation-iteration-count:infinite;
	animation-timing-function:linear;
		-o-animation-timing-function:linear;
		-ms-animation-timing-function:linear;
		-webkit-animation-timing-function:linear;
		-moz-animation-timing-function:linear;
	animation-delay:0s;
		-o-animation-delay:0s;
		-ms-animation-delay:0s;
		-webkit-animation-delay:0s;
		-moz-animation-delay:0s;
	transform:scale(0.5, 1);
		-o-transform:scale(0.5, 1);
		-ms-transform:scale(0.5, 1);
		-webkit-transform:scale(0.5, 1);
		-moz-transform:scale(0.5, 1);
}



@keyframes bounce_escapingBallG{
	0%{
		left:0px;
		transform:scale(0.5, 1);
	}

	25%{
		left:41px;
		transform:scale(1, 0.5);
	}

	50%{
		left:103px;
		transform:scale(0.5, 1);
	}

	75%{
		left:41px;
		transform:scale(1, 0.5);
	}

	100%{
		left:0px;
		transform:scale(0.5, 1);
	}
}

@-o-keyframes bounce_escapingBallG{
	0%{
		left:0px;
		-o-transform:scale(0.5, 1);
	}

	25%{
		left:41px;
		-o-transform:scale(1, 0.5);
	}

	50%{
		left:103px;
		-o-transform:scale(0.5, 1);
	}

	75%{
		left:41px;
		-o-transform:scale(1, 0.5);
	}

	100%{
		left:0px;
		-o-transform:scale(0.5, 1);
	}
}

@-ms-keyframes bounce_escapingBallG{
	0%{
		left:0px;
		-ms-transform:scale(0.5, 1);
	}

	25%{
		left:41px;
		-ms-transform:scale(1, 0.5);
	}

	50%{
		left:103px;
		-ms-transform:scale(0.5, 1);
	}

	75%{
		left:41px;
		-ms-transform:scale(1, 0.5);
	}

	100%{
		left:0px;
		-ms-transform:scale(0.5, 1);
	}
}

@-webkit-keyframes bounce_escapingBallG{
	0%{
		left:0px;
		-webkit-transform:scale(0.5, 1);
	}

	25%{
		left:41px;
		-webkit-transform:scale(1, 0.5);
	}

	50%{
		left:103px;
		-webkit-transform:scale(0.5, 1);
	}

	75%{
		left:41px;
		-webkit-transform:scale(1, 0.5);
	}

	100%{
		left:0px;
		-webkit-transform:scale(0.5, 1);
	}
}

@-moz-keyframes bounce_escapingBallG{
	0%{
		left:0px;
		-moz-transform:scale(0.5, 1);
	}

	25%{
		left:41px;
		-moz-transform:scale(1, 0.5);
	}

	50%{
		left:103px;
		-moz-transform:scale(0.5, 1);
	}

	75%{
		left:41px;
		-moz-transform:scale(1, 0.5);
	}

	100%{
		left:0px;
		-moz-transform:scale(0.5, 1);
	}
}
<script src="https://unpkg./axios/dist/axios.min.js"></script>
<script src="https://cdnjs.cloudflare./ajax/libs/react/15.1.0/react.min.js"></script>
<script src="https://cdnjs.cloudflare./ajax/libs/react/15.1.0/react-dom.min.js"></script>
<div id="root"></div>

Don't know if this will help, you can add loading state till you get all your data and setState. then make the loading sate to false and then render data

class Project extends Component {
constructor(props) {
    super(props);
    this.state = {
        isLoading: true,
        data : data
    };
}


ponentDidMount() {
    this
        .props
        .GetData(this.state)
        .then((res) => {
            this.setState.isLoading = false;
        }, (err) => this.setState({errors: err.response}));
}


render() {
     const { isLoading, data} = this.props;

    if(isLoading){
        return (
               <p>loading!!!</p>
        );

    return (
        <p>data</p>
    );
}

}

Depending on what you're doing with your fetched data (for instance rendering a long list of items), you might consider fetching the data in smaller chunks or pages. This will allow your ponent to render a small set of the data, and you can load in the rest behind the scenes, or as the user interacts with the view.

发布评论

评论列表(0)

  1. 暂无评论