自学软件网站开发,网易企业邮箱域名,太湖云建站网站建设,wordpress文章对游客不显示原文自工程师Enmanuel Durn博客#xff0c;传送门 最近#xff08;或者不是最近#xff0c;这完全取决于您什么时候阅读这边文章#xff09;#xff0c;我正在跟我的团队伙伴讨论如何去处理这种需要根据不同的值去处理不同的情况的方法#xff0c;通常对于这种情况下传送门 最近或者不是最近这完全取决于您什么时候阅读这边文章我正在跟我的团队伙伴讨论如何去处理这种需要根据不同的值去处理不同的情况的方法通常对于这种情况下人们喜欢使用switch语句或者使用很多if搭配else if条件。在本文中我将重点介绍第三种方式(我更为喜欢的方法)即使用对象进行快速地查找。 switch 语句 switch语句允许我们根据传递的表达式的值来执行表达式并执行某些特定的操作通常当你学习编写代码和算法时你会发现可以将它专门用于多种值的情况你开始使用它它看起来很好你很快意识到它给了你很大的自由耶但是要小心自由度越大责任感也就越大。 让我们快速了解一下典型的switch语句是怎么样的 switch (expression) {case x: {/* Your code here */break;}case y: {/* Your code here */break;}default: {/* Your code here */}
} 很好现在有一些你可能不知道需要注意的事情 可选的关键字break break关键字允许我们在满足条件时停止执行块。如果不将break关键字添加到switch语句则不会抛出错误。如果我们不小心忘记break的话可能意味着在执行代码的时候你甚至不知道代码已经正在执行中了这还会在调试问题时增加实现结果的的不一致性、突变、内存泄漏和复杂度等问题。我们来看看这个问题的一种表示形式: switch (first) {case first: {console.log(first case);}case second: {console.log(second case);}case third: {console.log(third case);break;}default: {console.log(infinite);}
} 如果你在控制台中执行这段代码你会看到输出是 firt case
second case
third case switch语句在第二种和第三种情况下也会执行即使第一种情况已经是正确的然后它在第三种情况块中找到关键字break并停止执行控制台中没有警告或错误让你知道它这会让你认为这是预期的行为。 每种情况下的大括号都不是强制的 在javascript中大括号代表着代码块因为自ECMAscript 2015我们可以使用关键字声明块编译变量如const或let但对于switch来说并不是很好因为大括号不是强制性的重复声明会导致错误变量让我们看看当我们执行下面的代码时会发生什么 switch (second) {case first:let position first;console.log(position);break;case second:let position second;console.log(position);break;default:console.log(infinite);
} 我们会得到 Uncaught SyntaxError: Identifier position has already been declared 这里将会返回一个错误因为变量position已经在第一种情况下声明过了并且由于它没有大括号所以在第二种情况下尝试声明它它已经存在了。 现在想象使用带有不一致break关键字和大括号的switch语句时会发生什么事 switch (first) {case first:let position first;console.log(position);case second:console.log(second has access to ${position});position second;console.log(position);default:console.log(infinite);
} 控制台将输出以下内容 first
second has access to first
second
infinite 试想一下由此而引起的错误和突变是如此之多其可能性是无穷无尽的……不管怎样switch语句已经讲够了我们来这里是为了讨论一种不同的方法我们来这里是为了讨论对象。 更安全查找的对象 对象查找速度很快随着它们的大小增长它们也会更快它们也允许我们将数据表示为对于条件执行非常有用的键值对。 使用字符串 让我们从简单的switch示例开始让我们假设我们需要有条件地保存和返回一个字符串的情景并使用我们的对象: const getPosition position {const positions {first: first,second: second,third: third,default: infinite};return positions[position] || positions.default;
};const position getPosition(first); // Returns first
const otherValue getPosition(fourth); // Returns infinite 这可以做同样类型的工作如果你想进一步的压缩简化代码我们可以利用箭头函数 const getPosition position ({first: first,second: second,third: third}[position] || infinite);const positionValue getPosition(first); // Returns first
const otherValue getPosition(fourth); // Returns infinite 这与前面的实现完全相同我们在更少的代码行中实现了更紧凑的解决方案。 现在让我们更实际一点不是我们写的所有条件都会返回简单的字符串其中很多会返回布尔值执行函数等等。 使用布尔值 我喜欢创建返回类型一致的值的函数,但是,由于javascript是动态类型语言因此可能存在函数可能返回动态类型的情况因此我将在此示例中考虑这一点如果找不到键我将创建一个返回布尔值未定义或字符串的函数。 const isNotOpenSource language ({vscode: false,sublimetext: true,neovim: false,fakeEditor: undefined}[language] || unknown);const sublimeState isNotOpenSource(sublimetext); // Returns true 看起来不错对吧别急好像我们有一个问题......如果我们调用带有参数的函数会发生什么vscode或fakeEditor不是嗯让我们来看看 它会寻找对象中的键。它会看到vscode键的值是false。它会试图返回false但因为false || unknown是unknown我们最终会返回一个不正确的值。对于key为fakeEditor也会有同样的问题 Oh no, 好吧不要惊慌让我们来解决这个问题 const isNotOpenSource editor {const editors {vscode: false,sublimetext: true,neovim: false,fakeEditor: undefined,default: unknown};return editor in editors ? editors[editor] : editors.default;
};const codeState isNotOpenSource(vscode); // Returns false
const fakeEditorState isNotOpenSource(fakeEditor); // Returns undefined
const sublimeState isNotOpenSource(sublimetext); // Returns true
const webstormState isNotOpenSource(webstorm); // Returns unknown 这就解决了问题但是......我希望你们问自己一件事:这真的是问题所在吗?我认为我们应该更关心为什么我们需要一个返回布尔值未定义值或字符串的函数这里存在严重的不一致性无论如何对于这样一个非常棘手的情况这也只是一个可能的解决方案。 使用函数 我们继续讲函数通常我们会发现我们需要根据参数来执行一个函数假设我们需要根据输入的类型来解析一些输入值如果解析器没有注册我们只返回值: const getParsedInputValue type {const emailParser email email, ${email};const passwordParser password password, ${password};const birthdateParser date date , ${date};const parsers {email: emailParser,password: passwordParser,birthdate: birthdateParser,default: value value};return parsers[type] || parsers.default;
};// We select the parser with the type and then passed the dynamic value to parse
const parsedEmail getParsedInputValue(email)(myemailgmail.com); // Returns email, myemailgmail.com
const parsedName getParsedInputValue(name)(Enmanuel); // Returns Enmanuel 如果我们有一个类似的函数返回另一个函数但这次没有参数我们可以改进代码以便在调用第一个函数时直接返回如 const getValue type {const email () myemailgmail.com;const password () 12345;const parsers {email,password,default: () default};return (parsers[type] || parsers.default)(); // we immediately invoke the function here
};const emailValue getValue(email); // Returns myemailgmail.com
const passwordValue getValue(name); // Returns default 通用代码块 Switch语句允许我们为多个条件定义公共代码块。 switch (editor) {case atom:case sublime:case vscode:return It is a code editor;break;case webstorm:case pycharm:return It is an IDE;break;default:return unknown;
} 我们如何使用对象来处理它我们可以在下一个方面做到这一点 const getEditorType type {const itsCodeEditor () It is a code editor;const itsIDE () It is an IDE;const editors {atom: itsCodeEditor,sublime: itsCodeEditor,vscode: itsCodeEditor,webstorm: itsIDE,pycharm: itsIDE,default: () unknown};return (editors[type] || editors.default)();
};const vscodeType getEditorType(vscode); 现在我们有一种方法 更有条理更易拓展更容易维护更容易测试更安全并且副作用和风险更小注意事项 正如预期的那样所有的方法都有其缺点这一个也不例外。 由于我们正在使用对象所以我们将占用内存中的一些临时空间来存储它们当定义对象的作用域不再可访问时这个空间将被垃圾收集器释放。当没有太多情况需要处理时对象方法可能比switch语句的速度要慢这可能是因为我们正在创建一个数据结构然后接收一个键然而在switch中我们只是检查值并返回值。结论 本文不打算改变你的编码风格或让你停止使用switch语句它只是试图提高你对switch语句的认识以便它可以正确使用并开放你的思想探索新的替代方案在这种情况下我已经分享了我喜欢使用的方法但还有更多例如你可能想看一个称为模式匹配的ES6提案如果你不喜欢它你可以继续探索。 好的开发未来就是这样我希望你喜欢这篇文章如果你这样做你可能会喜欢这篇关于工厂模式的文章。此外不要忘记分享和点赞你可以在twitter上找到我或通过我的电子邮件duranenmanuelgmail.com联系我下一个见。 阅读EnmaScript.com上发布的原始文章 译者总结 本文介绍了一种使用对象去代替我们之前用switch和繁琐的if else语句的方法。其实很多情况下我们可以利用对象与其他组合搭配写出更为高效或可维护的代码。当然如何去灵活地使用对象去处理一些对应的情况还是靠我们自己。好的这篇就总结到这了不知道对你们有什么启发。相信会给到一些帮助给读者,我们可不是一个只会if else的工程师,哈哈~