As you likely already know, React is a library to create UI components that can be used as the basis of web and mobile applications. What distinguishes React from some of its competitors is that its code is written entirely with JavaScript. Even the HTML-like templates are written in JS using JSX, which is an extension of the JS language to structure UI components.
The goal of this article is to help aspiring React developers get started by highlighting the JavaScript they ought to master before really diving into React. A complete introduction to JavaScript would go beyond the scope of this article, but React builds on the use of modern JavaScript features that were mainly introduced with ES2015.
Below, I give an overview of the common JavaScript patterns and language features that are heavily used in virtually every React application. For each concept, I provide external links. If you are interested, you can learn more about using it in React context.
if
statements, the ternary operator, and logical operatorsThese operators have been part of JavaScript for a very long time. In React, they are especially useful for conditional rendering of components.
The ternary operator looks like this:
const buttonLabel = playback === "stop" ? "play ▶️" : "stop ⏹️";
If the variable playback
has the value stop
, then the operator assigns to buttonLabel
the string value play ▶️
and, in all other cases, the string value stop ⏹️
. It is basically the same as the following code:
let buttonLabel; if (playback === "stop") { buttonLabel = "play ▶️"; } else { buttonLabel = "stop ⏹️" }
Of course, you can use such an if...else
statement, but the ternary operator is often the instrument of choice if you need to use a single line expression for conditionally rendering elements inline.
Otherwise, you have to call a function where you put your code for conditional rendering. In React, you can also use more complex condition logic (e.g., an if...else
cascade) and store values in variables that can be used for conditional rendering in JSX code.
Logical operators &&
or ||
are very handy for building React components.
const isLoggedIn = true; const userComponent = isLoggedIn && getUserComponent();
In our example, the left operand (isLoggedIn
) of the &&
operator evaluates to true
. Therefore, the result of the right operand (the function call getUserComponent()
) gets assigned to the userComponent
variable.
This concept is also very useful for conditional rendering in React because true && jsxComponent
returns jsxComponent
, and false && jsxComponent
returns false
. If you return false
, React ignores it and simply renders nothing.
It is also possible to combine multiple conditions. In the next example, the result of getOtherUsersComponent()
is returned when both conditions are met.
const otherUsers = isLoggedIn && users?.length > 0 && getOtherUsersComponent();
Notice the ?
in the second operand users?.length > 0
. This is optional chaining, which is not uncommon in React projects.
If you return null
, React doesn’t render anything. In contrast to undefined
, null
represents the intentional absence of any value.
if (shouldRenderComponent()) { return getComponent(); } else { return null; }
This is useful to prevent components from rendering.
There are multiple ways to create objects. Initializing objects with the literal notation looks like this:
const foo = { bar: 3, hello: "world" };
This notation is frequently used in React projects to create objects inline without assigning them to a variable, e.g., for the initial state of useReducer
.
// 2nd function argument uses inline object literal foo("bar", { hello: "world" })
With ES2015 syntax, you can also use shorthand properties and method names.
// instead of function foo(id) { return { name: "dummy", id: id, bar: function() { console.log("bar"); } } } // you can do function foo(id) { return { name: "dummy", id, // shorthand property name bar() { // shorthand method name console.log("bar"); } } }
Shorthand properties especially are used all over the place in React development since they eliminate redundant code.
That said, you have to be aware of the subtle difference between an inline object literal and a variable pointing to an object (created by an object literal). In some cases, for React performance optimization purposes, you should avoid passing object literals to React components because a new object is created every time, causing unnecessary re-renders.
The same principle applies to anonymous functions (i.e., inline functions), which should be avoided in some React performance use cases.
// inline function foo(() => {console.log("bar")}); // passing variable pointing to function const barFunc = () => console.log("bar"); foo(barFunc);
Template literals, or template strings, were introduced with ES2015 and allow for creating strings with embedded JavaScript expressions. Within backticks, you can combine “hardcoded” strings with JavaScript expressions inside of ${}
.
const name = "doppelmutzi"; console.log(`Hello, my name is ${name}`); // Hello, my name is doppelmutzi
Expressions can, of course, be more complex, like inline calculations or function calls.
const name = "doppelmutzi"; const getRandomIndex = max => Math.floor(Math.random() * Math.floor(max)) const food = ["🥞", "🧇", "🍔", "🍟", "🍕"]; const getFood = index => food[index] console.log(`Hello, my name is ${name} and I'm hungry for ${getFood(getRandomIndex(food.length))}`);
The latter example also uses the multiline feature so that the output has a line break after the expression interpolation (${name}
).
In medium- and large-sized React applications, you’ll most likely be confronted with the switch
statement for managing state across components. Tools like the useReducer
Hook or Redux are often used for such tasks.
The following example shows a so-called reducer function using a switch
statement for state management. You don’t necessarily have to use switch
statements with a reducer, but it is a well-established pattern.
export default (state, action) => { switch (action.type) { case "TOGGLE_DARK_MODE": return { ...state, darkMode: action.darkMode, }; case "UPDATE_PLAYBACK": { return { ...state, currentSound: action.currentSound, }; } default: return state; } };
The example above checks the value of action.type
and executes the code of a case
statement. If it evaluates to a string, TOGGLE_DARK_MODE
, then the code of the first case statement is executed.
It is good practice to have an optional default clause. It gets executed if the switch
expression matches none of the case clauses. Using the spread operator (e.g., ...state
) is a common practice.
In the above example, every case (and default) clause returns a new object, representing the new React state. This brings us to an important topic of React development.
The principle of object destructuring is pretty simple. With the elegant syntax below, we can extract properties into variables.
const creatures = { human: ["👨🏿💼", "👩🏼💼", "🧑🏻💼", "👩🏾💻", "🧑💻"], supernatural: ["👾", "🤖", "👽", "👹", "👺"] }; const { human, supernatural } = creatures; console.log(human); // ["👨🏿💼", "👩🏼💼", "🧑🏻💼", "👩🏾💻", "🧑💻"] console.log(supernatural); // ["👾", "🤖", "👽", "👹", "👺"]
If you use assignment without variable declaration, you need to use parentheses.
const creatures = { human: ["👨🏿💼", "👩🏼💼", "🧑🏻💼", "👩🏾💻", "🧑💻"], supernatural: ["👾", "🤖", "👽", "👹", "👺"] }; let human, supernatural; ({human, supernatural} = creatures); console.log(human); // ["👨🏿💼", "👩🏼💼", "🧑🏻💼", "👩🏾💻", "🧑💻"] console.log(supernatural); // ["👾", "🤖", "👽", "👹", "👺"]
Object destructuring offers you syntactical sugar to save extra lines of code.
// you can do this const { human, supernatural } = creatures; // instead of const human = creatures.human; const supernatural = creatures.supernatural;
In the context of React, object destructuring is frequently used with function parameters.
const infos = {name: "doppelmutzi", hobby: "React" }; function printInfos({name, hobby}) { console.log(name, hobby); } printInfos(infos); const printName = ({name}) => console.log(name); printName(infos);
For cleaner code, React developers use this pattern with props, which are the input for React components.
function MyReactComponent({name, age}) { // ... }
Assigning in combination with renaming variables might be useful to increase the readability of your code.
const creatures = { human: ["👨🏿💼", "👩🏼💼", "🧑🏻💼"] }; const { human: people } = creatures; console.log(people); // ["👨🏿💼", "👩🏼💼", "🧑🏻💼"]
You can also define default values while unpacking fields from the assigned object. The following example combines multiple techniques.
const { human: people = ["👨🏿💼"], supernatural = ["👾", "👽"] } = { human: ["👨🏿💼", "👩🏼💼", "🧑🏻💼"] }; console.log(people); // ["👨🏿💼", "👩🏼💼", "🧑🏻💼"] console.log(supernatural); // ["👾", "👽"]
Nesting is also possible, but I wouldn’t recommend overdoing it; otherwise, understandability decreases.
const creatures = { animals: { wildlife: ["🦂", "🐍"], pet: ["🐕", "🐈"] }, human: ["👨🏿💼", "👩🏼💼", "🧑🏻💼"] }; const { animals: { pet }} = creatures; console.log(pet); // ["🐕", "🐈"]
Ronald Chen provides some more insights on destructuring nested objects.
With the help of the destructuring assignment, an array can be unpacked in such a way that its values are extracted into distinct variables, like this:
const array = [1, 2]; const [varForVal1, varForVal2] = array; console.log(varForVal1); // 1 console.log(varForVal2); // 2
As you can see, variables are assigned from the left to the right of the array, so order is maintained.
const [fruit, veggie] = ["🍓", "🥦", "🍕", "🌮", "🥪", "🍔"]; console.log(fruit); // 🍓 console.log(veggie); // 🥦
You can also skip values under consideration of the order.
const [fruit,,pizza,,,burger] = ["🍓", "🥦", "🍕", "🌮", "🥪", "🍔"]; console.log(fruit); // 🍓 console.log(pizza); // 🍕 console.log(burger); // 🍔
The following “chatty code” might help to understand what exactly happened.
const [ fruit, /* skip entry 2 (🥦) */, pizza, /* skip entry 4 (🌮) */, /* skip entry 5 (🥪) */, burger] = ["🍓", "🥦", "🍕", "🌮", "🥪", "🍔"];
You can also assign multiple values at once with the rest pattern.
const [fruit, veggie, ...junkfood] = ["🍓", "🥦", "🍕", "🌮", "🥪", "🍔"]; console.log(fruit); // 🍓 console.log(veggie); // 🥦 console.log(junkfood); // ["🍕", "🌮", "🥪", "🍔"]
Array destructuring allows for default values. Of course, you can combine this pattern with function calls, too.
const getFood = () => ["🍓", "🥦"]; const [fruit, veggie, junkfood = "🍕"] = getFood(); console.log(fruit); // 🍓 console.log(veggie); // 🥦 console.log(junkfood); // 🍕
The elegant concept of array destructuring is used frequently with React Hooks because you can come up with a few lines of semantic code. To create a state variable along with an updater function for a React component, you can use React’s useState
Hook.
const initialValue = false; // instead of const stateWithUpdater = useState(initialValue); const darkMode = stateWithUpdater[0]; const darkModeUpdater = stateWithUpdater[1]; // you can do const [darkMode, setDarkMode] = useState(initialValue);
The following example demonstrates that you can implement generic functions for concrete use cases. Array destructuring allows the function caller to use semantic variable names.
const getFood = type => { let food = []; let error = false; if (type === "fruits") { food = ["🍓", "🥝", "🍌"]; } else if (type === "junk") { food = ["🍕", "🌮", "🥪"]; } else { error = true; } const addFood = newFood => food.push(newFood); return [food, error, addFood]; }; const [healthyFood, noFruitsAvailable, addFruitFunc] = getFood("fruits"); console.log(healthyFood); // ["🍓", "🥝", "🍌"] console.log(noFruitsAvailable); // false console.log(addFruitFunc("🍒")); console.log(healthyFood); // ["🍓", "🥝", "🍌", "🍒"]
For this use case, in my opinion, returning an array with the getFood
function leads to more concise code than with object destructuring. Array destructuring allows for custom variable names. In contrast, with object destructuring, you need to rename the variables.
const getFood = type => { // same function body as above, only different return statement return {food, error, addFood}; }; const {food: healthyFood, error: noFruitsAvailable, addFood: addFruitFunc} = getFood("fruits"); console.log(noFruitsAvailable); // false console.log(addFruitFunc("🍒")); console.log(healthyFood); // ["🍓", "🥝", "🍌", "🍒"]
That’s probably why useState
returns an array and not an object — to be more generic.
The spread operator (...
) allows an iterable item (e.g., an array) to be extracted into its parts and plugged into places that expect individual elements. With this syntax, you can split up object properties or array elements.
In the next use case below, we pull out elements of an array and pass each element as individual function argument.
const numbers = [11, 5, 3, 1, 26]; // Math.max expects to be called like Math.max(11,5,3,1,26) console.log(Math.max(...numbers)); // 26
Another use case is to copy object properties and, thus, create a new object.
const food = { breakfast: ["🥞", "🧇"], lunch: ["🍔", "🍟", "🍕"] }; const foodAndDrinks = { ...food, drinks: ["🍷", "🍹", "🍺", "🥃"], }; console.log(foodAndDrinks); /* { breakfast: ["🥞", "🧇"], lunch: ["🍔", "🍟", "🍕"], drinks: ["🍷", "🍹", "🍺", "🥃"], } */
With this succinct syntax, you can conveniently create a copy of an array.
const food = ["🥞", "🧇", "🍔", "🍟", "🍕"]; const copy = [...food]; console.log(copy); // ["🥞", "🧇", "🍔", "🍟", "🍕"] console.log(food === copy); // false
Recalling the last paragraph of our subsection on the switch statement, the spread syntax is frequently used in the context of React state. With React, you should not manipulate state objects directly. Instead, you need to create a brand-new state object whenever you want to update the state. The following concept is considered good practice.
const restaurantState = { drinks: ["🍷", "🍹", "🍺", "🥃"], food: ["🥞", "🧇", "🍔", "🍟", "🍕"], lastOrder: null } // the customer ordered a 🍔 const stateAfterOrder = { drinks: [...restaurantState.drinks], // copy drinks food: [...restaurantState.food], // copy food lastOrder: "🍔" // override lastOrder }
With the help of the rest operator (...
), you can merge a list of function arguments into an array. While the syntax of the rest operator is the same as the spread operator, their place of use makes all the difference.
If you use the operator inside of {}
or []
, you use object or array spreading, respectively. On the other hand, if you use the operator with the last argument in a function signature, that is the rest parameter.
Its purpose is to merge a list of values into an array. In the next example, various food is provided as function arguments. With the rest operators, they are put into an array called food
. The findIndex
method is operating on an array and tests whether broccoli is included.
const cleanFoodIncluded = (...food) => food.findIndex(f => f === "🥦") !== -1; console.log(cleanFoodIncluded("🍕", "🌮", "🥪", "🥦", "🍔")); // true
It is a common practice to use the rest operator in combination with the spread operator. This combines multiple arguments into an array to distribute the entries again in another place inside of a React component.
function renderProduct( // these are considered as component-specific {id, name}, /* all other arguments are relevant for container component. Therefore, consolidate them into an array with rest operator */ ...containerProps) { // output 0815, choco, [{margin: "10px"}, {padding: "5px"}] console.log(id, name, containerProps); /* unpack array again with spread operator to provide them as individual args */ renderContainer(...containerProps); } function renderContainer(margin, padding) { // output {margin: "10px"}, {padding: "5px"} console.log(margin, padding); } const product = { id: "0815", name: "choco" } renderProduct(product, {margin: "10px"}, {padding: "5px"} );
A function declaration looks like this:
function getFood(index) { const food = ["🥞", "🧇", "🍔", "🍟", "🍕"]; return food[index]; }
In contrast, this is a function expression:
const getFood = function(index) { const food = ["🥞", "🧇", "🍔", "🍟", "🍕"]; return food[index]; }
An arrow function expression constitutes an alternative to the former two. Its advantage is in its syntactical sugar, which allows you to write functions in a more concise manner.
const getFood = (index) => { const food = ["🥞", "🧇", "🍔", "🍟", "🍕"]; return food[index]; }
If you have only one parameter, you can skip the parentheses.
const getFood = index => { const food = ["🥞", "🧇", "🍔", "🍟", "🍕"]; return food[index]; }
If your function does not accept any parameter, you need to use parentheses.
const getFood = () => { return ["🥞", "🧇", "🍔", "🍟", "🍕"]; }
If the function body consists of a single return statement, you can replace this explicit return with an implicit return, like so:
const getFood = index => \["🥞", "🧇", "🍔", "🍟", "🍕"\][index];
Only with function declarations can you invoke functions before they have even been defined. This is because function declarations are hoisted, i.e., they are moved to the top of their scope before execution.
console.log(getFoodDeclaration()); // "🍟" console.log(getFoodExp()); // ReferenceError: Cannot access 'getFoodExp' before initialization console.log(getFoodArrow()); // ReferenceError: Cannot access 'getFoodArrow' before initialization function getFoodDeclaration() { return "🍟"; } const getFoodExp = () => { return "🍔"; } const getFoodArrow = () => "🍕";
Another difference between function declarations/expressions and arrow function expressions is the this
keyword, which I’ll discuss in the context of classes.
A class represents a blueprint for new objects. Variables and functions can be attached to a class and are called properties and methods, respectively. In the context of a class, the this
keyword refers to the current instance. If you are coming from an object-oriented background, you most likely have some incorrect assumptions about it; this article helps to demystify the this
keyword.
A class can have a constructor method, which represents a special kind of function to initialize new objects of the blueprint. You can instantiate the class with the new
keyword. With this, the constructor is invoked (or the default constructor, if you do not provide any), and a new object is created.
class Fruit { // class body constructor() { // property this.popular = "🥝" } whatsPopular() { // method body console.log(this.popular) "🥝" } } // instantiate an object from the class const fruit = new Fruit(); // call the method on the instance fruit.whatsPopular();
Another crucial concept is inheritance with the class syntax. With the super
keyword, you can access the parent.
class Food { constructor() { this.popular = "🍕" } } class Fruit extends Food { constructor() { // required to call constructor of parent class // needs to be first statement in child constructor super(); // override this.popular = "🥝" } whatsPopular() { console.log(this.popular) // "🥝" } } const fruit = new Fruit(); fruit.whatsPopular();
With ES2017, the syntax for using class properties and methods is a little bit more concise. You can use arrow functions as methods, too.
class Food { popular = "🍕"; } class Fruit extends Food { popular = "🥝"; // method by arrow function expression whatsPopular = () => { console.log(this.popular) }; } const fruit = new Fruit(); fruit.whatsPopular();
Often, when you come across classes in React development, you will find the ES2017 syntax.
// class-based React components must extend React.Component (or a subclass like React.PureComponent) class Food extends React.Component { // define default React state with ES2017 property syntax state = { popular = "🍕" } render() { // required method by every class-based React component } }
Note that this section is by no means a complete explanation of JavaScript classes. In my opinion, you do not need to devote too much time to learning classes if your plan is to learn React. My recommended learning path for React beginners is to understand the basics of classes as presented here to be able to read and understand class-based React components.
I think understanding classes extensively for new React developers is not necessary because the importance of classes has decreased drastically since last year. This is related to the introduction of React Hooks.
Before then, it was only possible to have sophisticated React components with the class-based approach. Only with classes was it possible to define component state and use lifecycle methods. Hooks allow similar things with functional components, too. The whole React community is strongly pushing towards only using function components.
However, if you work on legacy projects with existing class-based components or if you come across some of the few use cases that require you to use classes, or even if you just like to use classes (e.g., use a specific lifecycle method), then you need to understand the React-related class foundations.
Luckily, for React development, the relevant aspects of JavaScript classes are not very complicated. I like to use the ES2017 syntax for class components and arrow functions for methods because they do not require use of the bind()
method. The code gets more understandable.
Consider the first example, which requires a bind
; otherwise, the invocation of this.setState
causes an error.
class Button extends React.Component { constructor(props) { super(props); this.state = { clicked: false }; // this.onClick = this.onClick.bind(this); } onClick() { this.setState({ clicked: true }); // ERROR this.setState is not a function } render() { return ( <div> <p>{this.state.clicked && "clicked"}</p> <button onClick={this.onClick}>click</button> </div> ); } }
With the ES2017 syntax, you can write more understandable class-based components. The reason is because arrow functions have a lexical this
, and its value within the arrow function is determined by the surrounding scope (in our case, the render
method that has access to state with this
).
class Button extends React.Component { state = { clicked: false } onClick = () => { this.setState({ clicked: true }); } render() { return ( <div> <p>{this.state.clicked && "clicked"}</p> <button onClick={this.onClick}>click</button> </div> ); } }
Mastering array functions is an important skill for React developers. map()
is used in basically every React application, e.g., to loop over a list of food objects and render every single entry within an li
tag.
The map()
function produces a new array with the same number of elements. However, for every entry of the original entry, an operation was applied to produce new entries. The following example creates a new array with duplicated fruits for every entry.
const fruits = ["🍓", "🥝", "🍌", "🍒"];
const moreFruits = fruits.map(f => `${f}${f}`);
console.log(moreFruits); // ["🍓🍓", "🥝🥝", "🍌🍌", "🍒🍒"]
filter()
is often used with state management because it returns a brand-new array containing only those elements of the original array that pass a provided condition.
const people = [ { name: "Max", sex: "male" }, { name: "Jacky", sex: "female" }, { name: "Stephanie", sex: "female" } ]; const women = people.filter(person => person.sex === "female"); console.log(women); /* [{ name: "Jacky", sex: "female" }, { name: "Stephanie", sex: "female"}] */..
findIndex()
returns the index of the first element that passes the test; otherwise, it returns -1
.
const fruits = ["🍓", "🥝", "🍒", "🍌", "🍒"]; console.log(fruits.findIndex(fruit => fruit === "🥝")); // 1 console.log(fruits.findIndex(fruit => fruit === "🍌🍌")); // -1 console.log(fruits.findIndex(fruit => fruit === "🍒")); // 2 (first match)
find()
returns the first entry that passes the test. This is useful in the context of React state management. As an example, we have a list of users. We clicked on a particular user in a list and want to display a modal dialog showing this user’s information.
const users = [ { id: 1, name: "Max", sex: "male" }, { id: 2, name: "Jacky", sex: "female" }, { id: 3, name: "Stephanie", sex: "female" } ]; function logUserInfo(id) { console.log(users.filter(user => user.id === id)); } logUserInfo(2); // { id: 2, name: "Jacky", sex: "female" }
This concept is important to understand. Immutable values and objects cannot be changed afterwards, so the original remains untouched. Primitive values like strings or numbers are immutable by nature. On the other hand, objects are mutable by default. Let’s take a look what this means.
// immutable use case // Change strings won't work. Throws error in strict mode "use strict"; const hello = "world"; hello[0] = "W"; // try to upper case the first char console.log(hello); // world (in none-strict mode)
A misguided mindset about mutability can lead to bugs.
// mutability use case const meal = { kind: "🍕", origin: { country: "Italy" } } const fruit = { kind: "🍇", origin: meal.origin }; console.log(`${fruit.kind} from ${fruit.origin.country}`); // ✅ "🍇 from Italy" console.log(`${meal.kind} from ${meal.origin.country}`); // ✅ "🍕 from Italy" // we bought new grapes from Germany fruit.origin.country = "Germany"; console.log(`${fruit.kind} from ${fruit.origin.country}`); // ✅ "🍇 from Germany" // we have caused an unwanted side effect console.log(`${meal.kind} from ${meal.origin.country}`); // ❌ "🍕 from Germany"
Objects are mutable, but you can use Object.freeze()
or third-party libraries like Immutable.js to make them immutable.
The React team recommends you use immutable objects in multiple areas of your application design, e.g., with component-based and global state. This is because immutability typically leads to multiple architectural benefits. And that’s why most of the developer community suggests pursuing a coding mindset of immutablity.
I need to emphasize that mutation is not bad in itself. One problem with undisciplined mutation, however, is that it can lead to unexpected side effects, which are often the cause of unexpected bugs.
In React development, do not try to mutate state variables directly; rather, use the preferred method of the state management library. As an example, the following code snippet shows how you should and how you should not update the local state of a class-based React component.
class Button extends React.Component { state = { clicked: false } onClick = () => { // ❌ don't do this this.state.clicked = true; // ✅ instead do this: pass a new object to setState this.setState({ clicked: true }); } render() { return ( <div> <p>{this.state.clicked && "clicked"}</p> <button onClick={this.onClick}>click</button> </div> ); } }
If you use global state management tools like useReducer
or Redux, you should update state like this.
const newState = { ...state, // creates a copy of the current state darkMode: action.darkMode, // just override to reflect the changes };
A function passed as an argument to another function is called a callback if the function invokes the argument at a later time.
setTimeout
gets a callback function as the first argument that is called after the timer has expired (second argument).
window.setTimeout(() => console.log("I'm a callback function"), 1000); // after 1s: "I'm a callback function"
In the context of React, callback functions are often passed as props to React components. Thereby, child components can execute passed callbacks at a later time in a way that parent components can react to it (e.g., update state and, thus, update the UI).
Callback functions are also important in the context of React Hooks, e.g., to trigger side effects with useEffect
.
Since React development consists mainly of writing vanilla JavaScript code, I recommend acquiring a good understanding of JavaScript fundamentals before learning React. With this learning path, you will have a much smoother start — I promise you.
Install LogRocket via npm or script tag. LogRocket.init()
must be called client-side, not
server-side
$ npm i --save logrocket // Code: import LogRocket from 'logrocket'; LogRocket.init('app/id');
// Add to your HTML: <script src="https://cdn.lr-ingest.com/LogRocket.min.js"></script> <script>window.LogRocket && window.LogRocket.init('app/id');</script>
Hey there, want to help make our blog better?
Join LogRocket’s Content Advisory Board. You’ll help inform the type of content we create and get access to exclusive meetups, social accreditation, and swag.
Sign up nowuseState
useState
can effectively replace ref
in many scenarios and prevent Nuxt hydration mismatches that can lead to unexpected behavior and errors.
Explore the evolution of list components in React Native, from `ScrollView`, `FlatList`, `SectionList`, to the recent `FlashList`.
Explore the benefits of building your own AI agent from scratch using Langbase, BaseUI, and Open AI, in a demo Next.js project.
Demand for faster UI development is skyrocketing. Explore how to use Shadcn and Framer AI to quickly create UI components.
16 Replies to "JavaScript concepts to master before learning React"
Interesting post. Clear explanation on new JavaScript syntax.
Good overview. Thanks.
This is really a fantastic article. It’s so complete and covers so many concepts. I’m actually from a Vuejs background but there is a huge amount of overlap with regards to these JavaScript concepts which applies to both React and Vuejs. I really admire people who can write such lengthy concise and to the point articles. Thank you so much, I’ll use it as a blueprint for my learning journey going forward.
No matter how I learn JavaScript fundamentals, once I start using React, I get confused.
Thanks mate
hey Eugène, that’s an awesome comment. I really appreciate that you like my article. I wish you all the best for your learning journey!
Hi Uzochukwu, I hope this article will help to reduce your confusion. Do you have a particular question or an example for me? Maybe, I can help.
I’m glad you like it!
Thank you very much for your feedback!
Hi Alan, I’m glad you like it. Thank you very much for your feedback!
What is the first thing you recommend to learn for a none IT person to learn coding in React?
Hi Ray, I would recommend to learn Javascript from scratch. Dan Abramov – by the way one of the core members of the React team – has created an awesome course on Javascript. It’s called “Just Javascript”: https://justjavascript.com/
The good thing is, you learn a lot of key programming concepts that will help you to learn the things I described in this article, too. Then you are ready to master React.
Let me know what do you think.
Clear and point to point, Amazingly Written (Y)
Hi Nitin, I’m thrilled you like it.
Nice article!! Thank you so much.
Well written, easy to follow (even for a Javascript newb like me), and a well-balanced amount of concepts. Great job!