天天看點

JavaScript實作:如何寫出漂亮的條件表達式

摘要:就讓我們看看以下幾種常見的條件表達場景,如何寫的漂亮!

本文分享自華為雲社群《如何寫出漂亮的條件表達式 - JavaScript 實作篇》,原文作者:查爾斯。

條件表達式,是我們在coding過程中永遠躲不開的問題,也是我們騙代碼行數最友善的東西(狗頭.jpg),但作為一名程式員,我們也要追求自己的“信達雅”,下面就讓我們看看以下幾種常見的條件表達場景,如何寫的漂亮!

  • 多條件語句
  • 多屬性對象
  • 替換Switch語句
  • 預設參數與解構
  • 比對所有或部分條件
  • 使用可選鍊和 Nullish 合并

多條件語句使用Array.includes

舉個例子

function printAnimals(animal) {
  if (animal === "dog" || animal === "cat") {
    console.log(`I have a ${animal}`);
  }
}

console.log(printAnimals("dog")); // I have a dog      

這種寫法在條件比較少的情況下看起來沒有問題,此時我們隻有 2 種動物,但是如果我們有更多的條件需要判斷(更多的動物)呢?如果我們繼續拓展判斷的條件,那麼代碼将會變得難以維護,而且邏輯會不清晰。

解決方法

可以使用Array.includes來重寫條件語句

function printAnimals(animal) {
  const animals = ["dog", "cat", "hamster", "turtle"];

  if (animals.includes(animal)) {
    console.log(`I have a ${animal}`);
  }
}

console.log(printAnimals("hamster")); // I have a hamster      

在這裡,我們建立了一個動物數組,以便将條件與代碼的其餘部分分開提取。現在,如果我們想要檢查任何其他動物,我們需要做的就是添加一個新的數組項。

我們還可以在這個函數的範圍之外使用 animals 變量,以便在代碼的其他地方重用它。這是一種編寫更清晰、更容易了解和維護的代碼的方法。不是嗎?

這是一個非常好的技巧來壓縮你的代碼,使它看起來更簡潔。讓我們以前面的示例為例,添加更多的條件。如果這個動物不是一個簡單的字元串,而是一個具有某些屬性的對象呢?

是以現在的要求是:

  • 如果沒有動物,抛出一個錯誤
  • 列印動物的類型
  • 列印動物的名字
  • 列印動物的性别
const printAnimalDetails = (animal) => {
  let result; // declare a variable to store the final value

  // condition 1: check if animal has a value
  if (animal) {
    // condition 2: check if animal has a type property
    if (animal.type) {
      // condition 3: check if animal has a name property
      if (animal.name) {
        // condition 4: check if animal has a gender property
        if (animal.gender) {
          result = `${animal.name} is a ${animal.gender} ${animal.type};`;
        } else {
          result = "No animal gender";
        }
      } else {
        result = "No animal name";
      }
    } else {
      result = "No animal type";
    }
  } else {
    result = "No animal";
  }

  return result;
};

console.log(printAnimalDetails()); // 'No animal'

console.log(printAnimalDetails({ type: "dog", gender: "female" })); // 'No animal name'

console.log(printAnimalDetails({ type: "dog", name: "Lucy" })); // 'No animal gender'

console.log(
  printAnimalDetails({ type: "dog", name: "Lucy", gender: "female" })
); // 'Lucy is a female dog'      

上面的代碼它工作得很好,但是代碼很長,很難維護。如果不使用提示工具,可能會浪費一些時間來确定右括号的位置。想象将會發生什麼如果代碼更複雜的邏輯。很多if...else的語句!

我們可以使用三元操作符、&&條件等來重構上面的函數,但是讓我們使用多個傳回語句來編寫更精确的代碼。

const printAnimalDetails = ({ type, name, gender } = {}) => {
  if (!type) return "No animal type";
  if (!name) return "No animal name";
  if (!gender) return "No animal gender";

  // Now in this line of code, we're sure that we have an animal with all //the three properties here.

  return `${name} is a ${gender} ${type}`;
};

console.log(printAnimalDetails()); // 'No animal type'

console.log(printAnimalDetails({ type: dog })); // 'No animal name'

console.log(printAnimalDetails({ type: dog, gender: female })); // 'No animal name'

console.log(printAnimalDetails({ type: dog, name: "Lucy", gender: "female" })); // 'Lucy is a female dog'      

在重構版本中,還包括解構和預設參數。預設參數確定如果我們将 undefined 作為參數傳遞給方法,我們仍然有一個要解構的值,這裡是一個空對象 {}。

通常,代碼是在這兩種方法之間編寫的。

function printVegetablesWithQuantity(vegetable, quantity) {
  const vegetables = ["potato", "cabbage", "cauliflower", "asparagus"];

  // condition 1: vegetable should be present
  if (vegetable) {
    // condition 2: must be one of the item from the list
    if (vegetables.includes(vegetable)) {
      console.log(`I like ${vegetable}`);

      // condition 3: must be large quantity
      if (quantity >= 10) {
        console.log("I have bought a large quantity");
      }
    }
  } else {
    throw new Error("No vegetable from the list!");
  }
}

printVegetablesWithQuantity(null); //  No vegetable from the list!
printVegetablesWithQuantity("cabbage"); // I like cabbage
printVegetablesWithQuantity("cabbage", 20);
// 'I like cabbage`
// 'I have bought a large quantity'      

現在,我們有:

  • 過濾無效條件的 if/else 語句
  • 3 層嵌套的 if 語句(條件 1、2 和 3)
  • 一個通用的規則是當發現無效條件時盡早傳回。

一個通用的規則是發現無效的條件時盡早傳回

function printVegetablesWithQuantity(vegetable, quantity) {
  const vegetables = ["potato", "cabbage", "cauliflower", "asparagus"];

  // condition 1: throw error early
  if (!vegetable) throw new Error("No vegetable from the list!");

  // condition 2: must be in the list
  if (vegetables.includes(vegetable)) {
    console.log(`I like ${vegetable}`);

    // condition 3: must be a large quantity
    if (quantity >= 10) {
      console.log("I have bought a large quantity");
    }
  }
}      

通過這樣做,我們減少了一個嵌套語句的級别。這種編碼風格很好,特别是當使用長if語句時。通過反轉條件并提前傳回,我們可以進一步減少嵌套if。

請看下面的條件 2 是怎麼做的:

function printVegetablesWithQuantity(vegetable, quantity) {
  const vegetables = ["potato", "cabbage", "cauliflower", "asparagus"];

  if (!vegetable) throw new Error("No vegetable from the list!");
  // condition 1: throw error early

  if (!vegetables.includes(vegetable)) return;
  // condition 2: return from the function is the vegetable is not in
  //  the list

  console.log(`I like ${vegetable}`);

  // condition 3: must be a large quantity
  if (quantity >= 10) {
    console.log("I have bought a large quantity");
  }
}      

通過反轉條件 2 的條件,代碼不再具有嵌套語句。當我們有很多條件并且希望在任何特定條件不滿足時停止進一步的處理時,這種技術是有用的。

是以,總是以減少嵌套和盡早傳回為目标,但不要過度。

讓我們看一下下面的例子,我們想要根據顔色列印水果:

function printFruits(color) {
  // use switch case to find fruits by color
  switch (color) {
    case "red":
      return ["apple", "strawberry"];
    case "yellow":
      return ["banana", "pineapple"];
    case "purple":
      return ["grape", "plum"];
    default:
      return [];
  }
}

printFruits(null); // []
printFruits("yellow"); // ['banana', 'pineapple']      

上面的代碼實作沒有錯誤,但是很冗長,同樣的結果可以使用更簡潔的文法來實作。

// use object literal to find fruits by color
const fruitColor = {
  red: ["apple", "strawberry"],
  yellow: ["banana", "pineapple"],
  purple: ["grape", "plum"],
};

function printFruits(color) {
  return fruitColor[color] || [];
}      

同樣的,也可以使用 Map 來實作:

// use Map to find fruits by color
const fruitColor = new Map()
  .set("red", ["apple", "strawberry"])
  .set("yellow", ["banana", "pineapple"])
  .set("purple", ["grape", "plum"]);

function printFruits(color) {
  return fruitColor.get(color) || [];
}      

Map是 ES5 以來可用的對象類型,它允許存 key-value。

對于上面的示例,可以使用 Array.filter 實作相同的結果。

const fruits = [
  { name: "apple", color: "red" },
  { name: "strawberry", color: "red" },
  { name: "banana", color: "yellow" },
  { name: "pineapple", color: "yellow" },
  { name: "grape", color: "purple" },
  { name: "plum", color: "purple" },
];

function printFruits(color) {
  return fruits.filter((fruit) => fruit.color === color);
}      

在使用 JavaScript 時,我們總是需要檢查 null/undefined 并配置設定預設值或編譯中斷。

function printVegetablesWithQuantity(vegetable, quantity = 1) {
// if quantity has no value, assign 1

  if (!vegetable) return;
    console.log(`We have ${quantity} ${vegetable}!`);
  }
  //results
}

printVegetablesWithQuantity('cabbage'); // We have 1 cabbage!
printVegetablesWithQuantity('potato', 2); // We have 2 potato!      

如果蔬菜是一個對象呢?我們可以配置設定一個預設參數嗎?

function printVegetableName(vegetable) {
  if (vegetable && vegetable.name) {
    console.log(vegetable.name);
  } else {
    console.log("unknown");
  }
}

printVegetableName(undefined); // unknown
printVegetableName({}); // unknown
printVegetableName({ name: "cabbage", quantity: 2 }); // cabbage      

在上面的示例中,我們希望列印蔬菜名(如果它可用)或列印 unknown。

我們可以通過使用預設參數&解構來避免條件if (vegetable && vegetable.name){}。

// destructing - get name property only
// assign default empty object {}

function printVegetableName({ name } = {}) {
  console.log(name || "unknown");
}

printVegetableName(undefined); // unknown
printVegetableName({}); // unknown
printVegetableName({ name: "cabbage", quantity: 2 }); // cabbage      

因為我們隻需要屬性名,是以我們可以使用 {name} 來改變參數的結構,然後我們可以在代碼中使用 name 作為變量,而不是使用 vegetable.name。

我們還将一個空對象 {} 指派為預設值,否則在執行 printVegetableName(undefined) 時,它将給出一個錯誤—— Cannot destructure property name of undefined or null,因為在 undefined 中沒有 name 屬性。

我們可以通過使用這些Array方法來減少代碼行數。

下面的代碼,我們想要檢查是否所有的水果都是紅色的:

const fruits = [
  { name: "apple", color: "red" },
  { name: "banana", color: "yellow" },
  { name: "grape", color: "purple" },
];

function test() {
  let isAllRed = true;

  // condition: all fruits must be red
  for (let f of fruits) {
    if (!isAllRed) break;
    isAllRed = f.color == "red";
  }

  console.log(isAllRed); // false
}      

上面的代碼太過冗長,我們可以通過使用 Array.every 來減少代碼行:

const fruits = [
  { name: "apple", color: "red" },
  { name: "banana", color: "yellow" },
  { name: "grape", color: "purple" },
];

function test() {
  // condition: short way, all fruits must be red
  const isAllRed = fruits.every((f) => f.color == "red");

  console.log(isAllRed); // false
}      

同樣的,如果我們想要測試任何一個水果是否是紅色的,我們可以使用 Array.some:

const fruits = [
  { name: "apple", color: "red" },
  { name: "banana", color: "yellow" },
  { name: "grape", color: "purple" },
];

function test() {
  // condition: if any fruit is red
  const isAnyRed = fruits.some((f) => f.color == "red");

  console.log(isAnyRed); // true
}      

https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Operators/%E5%8F%AF%E9%80%89%E9%93%BE

這兩個功能對于 JavaScript 編寫更簡潔的條件非常有用。在編寫本文時,它們還沒有得到完全的支援,可能需要使用Babel進行編譯。

可選連結能夠處理類似樹的結構,而不需要顯式地檢查中間節點是否存在,并且Nullish與可選連結結合使用非常有效,可以確定不存在節點的預設值。

舉個例子:

const car = {
  model: "Fiesta",
  manufacturer: {
    name: "Ford",
    address: {
      street: "Some Street Name",
      number: "5555",
      state: "USA",
    },
  },
};

// to get the car model
const model = (car && car.model) || "default model";

// to get the manufacturer street
const street =
  (car &&
    car.manufacturer &&
    car.manufacturer.address &&
    car.manufacturer.address.street) ||
  "default street";

// request an un-existing property
const phoneNumber =
  car &&
  car.manufacturer &&
  car.manufacturer.address &&
  car.manufacturer.phoneNumber;

console.log(model); // 'Fiesta'
console.log(street); // 'Some Street Name'
console.log(phoneNumber); // undefined      

是以,如果我們想列印出來,如果汽車制造商來自美國,代碼應該是這樣的:

const isManufacturerFromUSA = () => {
  if (
    car &&
    car.manufacturer &&
    car.manufacturer.address &&
    car.manufacturer.address.state === "USA"
  ) {
    console.log("true");
  }
};

checkCarManufacturerState(); // 'true'      

可以清楚地看到,對于更複雜的對象結構,這會變得多麼混亂。有一些第三方庫,如 lodash 或idx,它們有自己的功能。例如 lodash 有 _.get 方法。但是,在 JavaScript 語言本身中引入這個特性。

以下是這些新功能的工作原理:

// to get the car model
const model = car?.model ?? "default model";

// to get the manufacturer street
const street = car?.manufacturer?.address?.street ?? "default street";

// to check if the car manufacturer is from the USA
const isManufacturerFromUSA = () => {
  if (car?.manufacturer?.address?.state === "USA") {
    console.log("true");
  }
};      

目前在 Stage 3 階段。

以上就是基于JavaScript實作條件表達式的一些分享,希望對你能有所幫助~

點選關注,第一時間了解華為雲新鮮技術~

繼續閱讀