学习JavaScript进阶

news/2024/5/2 17:32:16/文章来源:https://blog.csdn.net/m0_57713054/article/details/128070599

JavaScript进阶

循环语句

  • for循环

    // 类似python中的for i in range(20)for(let i=0; i<20; i++){console.log(i)
    }
    
  • while循环

    const MAX_TIMES = 20;
    let cur = 0
    while (cur < MAX_TIMES){cur++;console.log(cur)
    }
    
  • do while

    do {cur ++;console.log(cur);
    }while (cur < MAX_TIMES)
    
    • while循环有什么区别?

      do while一定先执行一遍代码块的表达式;

  • for in

    遍历对象的属性

    let myObj = {a: 1, b:2, c:3, d:4}
    for (let e in myObj){console.log(e, myObj[e]);
    }
    
  • for of

    遍历可迭代对象的元素

    let myArray = [1, 2, 3, 4, 5]
    for (let e of myArray){console.log(e);
    }
    
    • 常见的可迭代对象有哪些?

      • Array

      • Set

      • Map

      • String

      • arguments

        function foo(a, b){console.log(arguments);
        }foo(1, 2)
        
  • forEach

    let myArray = [1, 2, 3, 4, 5]
    myArray.forEach(function (e){console.log(e * e);
    })
    

条件语句

  • 一个简单的判断语句

    for(let i=0; i<100; i++){if (i%2===0){console.log("偶数", i)}else if(i < 0){console.log("负数不判断")}else{console.log("奇数", i)}
    }
    
  • 逻辑运算符

    • ==

      对比操作数是否相同, 操作数会尝试进行类型转换后的比较, 不推荐做为比较符号.

      '1' == 1
      > truefalse == 0
      > true
      
    • ===

      严格等于

      '1' === 1
      > falsefalse === 0
      > false
      
    • !

      逻辑取反

      for(let i=0; i<100; i++){// 注意运算优先级的问题, 不能写成!i%2if (!(i%2)){console.log("偶数", i)}else if(i < 0){console.log("负数不判断")}else{console.log("奇数", i)}
      }
      
    • &&||

      • &&表示AND
      • ||表示OR

选择语句

  • switch case

    function foo(arg){switch (arg){case 'a':console.log(arg, 1);break;case 'b':console.log(arg, 2);break;case 'c':console.log(arg, 3);break;default:console.log('default')}
    }foo('e')
    

异常处理

try{表达式
}catch (e){表达式
}finally{表达式
}
  • 一个基本的异常捕获

    function foo(){try{throw TypeError('test');}catch (e){console.log('Error', e);}finally{console.log('Done!')}
    }foo()
  • 处理具体的异常

    处理具体的异常, 只能通过if条件语句来进行类型判断

    function foo(){try{throw TypeError('test');}catch (e){if (e instanceof TypeError){console.log("TypeError")}else{console.log('Error', e);}}finally{console.log('Done!')}
    }foo()
  • 抛出异常

    throw可以抛出任意对象, 让catch去捕获

    function foo(){try{throw {'a':1};}catch (e){if (e instanceof TypeError){console.log("TypeError")}else{console.log('Error', e);}}finally{console.log('Done!')}
    }foo()

对象和类

js当中其实没有明确的类的概念, js当中的类只是创建对象的模板. 它的本质还是一个特殊的函数

class关键字只是一层封装了原型链的语法糖, 但是方便我们理解.

  • 声明类

    class Rectangle {constructor(height, width) {this.name = 'ractangle';this.height = height;this.width = width;}
    }
    
  • 创建对象/实例

    let rectangle = new Rectangle(2, 5);
    console.log(rectangle)
    
  • 类方法

    • 构造方法

      constructor是一种用来创建和初始化class创建的对象;

    • 实例方法

      class Rectangle {constructor(height, width) {this.name = 'ractangle';this.height = height;this.width = width;}getArea(){return this.height * this.width;}
      }
      
    • 静态方法

          static staticMethod(){console.log("calling static method")}
      
    • gettersetter

          get area(){return this.getArea()}set area(value){this._value = value}
      
  • 类继承

     class Square extends Rectangle {constructor(a) {super(a, a);this.name = 'square'}
    }let suqare = new Square(10)
    console.log(suqare.area)
    
  • 私有方法和属性

    通过#来声明一个私有方法或者属性, 只允许类内部调用

    class Square extends Rectangle {// 私有属性需要事先进行声明#new_nameconstructor(a) {super(a, a);this.#new_name = 'square'}get new_name(){return this.#getName()}#getName(){return this.#new_name}
    }
    

构造函数和this

在JS中通过构造函数来创建对象的场景更多, 而不是通过class

  • 声明一个构造函数

    function Rectangle(height, width){this.name = 'rectangle';this.width = width;this.height = height;this.getArea = function (){return this.height * this.width}
    }
    
  • 创建对象/实例

    let rectangle = new Rectangle(10, 2)
    console.log(rectangle.getArea())  
    
  • 通过call方法继承

    第一个参数是要绑定的对象, 其他参数代表调用函数的参数.

    call方法将Rectangle下的属性和方法绑定到this上去

    这里的this指代的就是实例化的square对象

    function Square(a){Rectangle.call(this, a, a);this.name = 'square'
    }
    
    • apply

      call方法几乎没有区别, 只是传入的参数的方式不同

      function Square(a){Rectangle.apply(this, [a, a]);this.name = 'square'
      }
      
  • this

    不能简单地认为this指向的就是实例对象.

    可以简单地认为谁调用该函数, this就指向谁.

原型链式继承

在传统面向对象编程中, 我们继承的实现方式都是在创建实例时, 将类中定义的属性和方法都复制到实例中去.

但是JS中继承是在对象/实例和它的构造器之间创立一个链接, 这个链接就是__proto__

  • 原型链

    js中每个构造函数拥有一个原型对象prototype, 对象从原型继承方法和属性. 而当前对象继承的原型对象可能也有原型, 这样就形成了一条原型链

    square.__proto__.__proto__.__proto__
    
    • __proto__prototype的区别?

      __proto__是每个对象/实例都有的属性, 而prototype只是构造函数的属性.

  • 原型链式继承

    当前对象/实例square找不到getArea方法时, 会继续在原型链中寻找.

    function Square(a){this.height = a;this.width = a;this.name = 'square'
    }Square.prototype = new Rectangle()
    let square = new Square(10)
    console.log(square.getArea())
    

异步JS

JS引擎只是浏览器中的一个线程, 所以在JS当中没有线程和进程的概念.JS的高性能是通过一个基于事件循环的异步并发模型来完成.

  • 事件循环

    在页面环境中, 我们移动鼠标, 点击页面元素或者执行一段JS, 都可以认为当前是一个消息. 当前的消息会被丢进一个无限循环的消息队列当中, 每一个消息都关联着用来处理这个消息的回调函数

    document.addEventListener("click", function(e){console.log(e)})
    
    • 事件循环的伪代码

      while(queue.waitForMessage()){queue.processNextMessage();
      }
      
    • 消息的执行

      • JS在执行同步消息任务时, 会执行至完成.
      • JS在执行异步消息任务时, 遇到异步操作, 会将该消息丢回消息队列, 等待下一次调度执行.

callbacks(异步回调)

最常见于浏览器的监听事件, 本质就是将回调函数做为参数传递给后台执行的其他函数, 其他函数在执行完毕后自动调用回调函数

document.addEventListener("click", function(e){console.log(e)})
  • 回调地狱

    // 银行转账的伪代码
    transferAccount(cash,dealCash(cash, function(cash_status){authAccount(cash_status, function(order){transferStart(order, function(status){sendSMS(status)}, transferExceptionCallback)}, authExceptionCallback)}, cashExceptionCallback)
    )
    

Promise

Promise是一个对象, 它代表了一个异步操作的最终完成或者失败的结果.

img

  • 声明一个简单的promise对象

    let promiseOjb = new Promise((resolve, reject) => {setTimeout(() => {resolve("success");reject("failed");}, 3*1000)
    }).then(result => {console.log("result => ", result)})console.log(promiseOjb)
    
  • promise的状态

    • pending

      初始状态, 表示未接收到结果

    • fullfill

      已兑现, 表示操作成功完成

    • reject

      已拒绝, 表示操作失败

  • 在使用promise时, 需要注意以下约定

    • 回调函数时在当前事件循环结束后自动调用的;
    • 通过then添加的回调函数不管异步操作是否成功都会执行;
    • 通过调用多次then可以添加多个回调函数, 他们按插入顺序依次执行, 这个方式就叫做链式调用;
  • promise中处理异常

    • 声明一个简单的xhr请求

      function xhrRequest(url){return new Promise((resolve, reject) => {const xhr = new XMLHttpRequest();xhr.open('GET', url);xhr.onload = function(){resolve(xhr.responseText)};xhr.onerror = () => reject(xhr.status);xhr.send()}).then(result=>{console.log("SUCCESS", result)}).catch(error=>{console.log("FAILURE", error)})
      }xhrRequest("http://www.baidu.com")
      xhrRequest("https://www.baidu.com")
      xhrRequest("https://www.baidu.com/asdfasdf/")
      
      • 当前环境下, 客户端请求到达了服务端, 服务端也正常地返回了结果. 那么不管状态码是200还是404, 都算是一次成功的请求, 所以结果由then回调进行处理, 而不是catch. 如果有需要可以自己通过状态码判断后执行不同的流程.
    • 链式调用

      function xhrRequest(url){return new Promise((resolve, reject) => {const xhr = new XMLHttpRequest();xhr.open('GET', url);xhr.onload = function(){resolve(xhr.responseText)};xhr.onerror = () => reject(xhr.status);xhr.send()}).then(result=>{console.log("SUCCESS", result); return result}).then(result=>{console.log("SUCCES 2", result)}).catch(error=>{console.log("FAILURE", error)})
      }
      
      • 不管当前链式调用有多长, 异常都会进到catch回调函数当中. catch也可以进行链式调用, 但是一般一个函数只有一个catch回调函数.

async/await

async本质就是将函数转换为promise

  • 通过asyncawait等待完成结果

    function xhrRequest(url){return new Promise((resolve, reject) => {const xhr = new XMLHttpRequest();xhr.open('GET', url);xhr.onload = function(){resolve(xhr.responseText)};xhr.onerror = () => reject(xhr.status);xhr.send()})
    }async function requestBaidu(url){let result = await xhrRequest(url);console.log("DONE!", result)
    }
    
  • 异常处理

    function xhrRequest(url){return new Promise((resolve, reject) => {const xhr = new XMLHttpRequest();xhr.open('GET', url);xhr.onload = function(){resolve(xhr.responseText)};xhr.onerror = () => reject(xhr.status);xhr.send()})
    }async function requestBaidu(url){try{let result = await xhrRequest(url);console.log("DONE!", result)}catch (e){console.log("FAILURE", e)}
    }

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.luyixian.cn/news_show_37796.aspx

如若内容造成侵权/违法违规/事实不符,请联系dt猫网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

八、Gateway

文章目录一、Gateway网关1.网关的作用二、配置网关1.创建gateway模块2.引入依赖3.编写application.yml4.启动gateway模块&#xff0c;查看是否能访问user-service服务三、路由断言工厂Route Predicate Factory四、GatewayFilter&#xff08;过滤器&#xff09;1.添加过滤器方式…

毕业设计-机器学习图像卡通动漫化图像风格迁移

前言 &#x1f4c5;大四是整个大学期间最忙碌的时光,一边要忙着备考或实习为毕业后面临的就业升学做准备,一边要为毕业设计耗费大量精力。近几年各个学校要求的毕设项目越来越难,有不少课题是研究生级别难度的,对本科同学来说是充满挑战。为帮助大家顺利通过和节省时间与精力投…

餐厅食材采购信息管理系统的设计与实现

摘 要 网络的广泛应用给生活带来了十分的便利。所以把餐厅食材采购信息管理与现在网络相结合&#xff0c;利用JSP技术建设餐厅食材采购信息管理系统&#xff0c;实现餐厅食材采购的信息化。则对于进一步提高餐厅食材采购信息管理发展&#xff0c;丰富餐厅食材采购信息管理经验…

R语言对用电负荷时间序列数据进行K-medoids聚类建模和GAM回归

通过对用电负荷的消费者进行聚类&#xff0c;我们可以提取典型的负荷曲线&#xff0c;提高后续用电量预测的准确性&#xff0c;检测异常或监控整个智能电网&#xff08;Laurinec等人&#xff08;2016&#xff09;&#xff0c;Laurinec和Luck&#xff08; 2016&#xff09;&…

使用vue脚手架快速搭建vue 2项目

简单了解vue-cli 官网地址:https://cli.vuejs.org/zh/guide/browser-compatibility.html 前提 1.安装node(js代码的运行环境)、npm&#xff1b; 2.全局安装vue-cli; 命令创建项目 vue create hello-word 等待项目创建成功即可 开发工具打开刚刚创建的项目 项目结构如图…

TMS Sphinx Alexandria Full Source

TMS Sphinx Alexandria Full Source 用于身份访问管理的TMS Sphinx Delphi框架&#xff0c;包括授权和身份验证。 TMS Sphinx允许您为多个应用程序实现单点登录(SSO)&#xff1a;web、本机、移动或机器到机器API通信。它可用于通过登录表单、类似的用户界面和基于服务的身份验证…

Spring - BeanPostProcessors 扩展接口

文章目录PreBean的生成过程org.springframework.beans.factory.config.BeanPostProcessor 介绍ApplicationContext注册Bean PostProcessor源码解析AbstractApplicationContext#refreshAbstractApplicationContext#registerBeanPostProcessorsPostProcessorRegistrationDelegate…

[LeetCode周赛复盘] 第 92 场双周赛20221015

[LeetCode周赛复盘] 第 92 场双周赛20221015 一、本周周赛总结二、 [Easy] 6249. 分割圆的最少切割次数1. 题目描述2. 思路分析3. 代码实现三、[Medium] 6277. 行和列中一和零的差值1. 题目描述2. 思路分析3. 代码实现四、[Medium] 6250. 商店的最少代价1. 题目描述2. 思路分析…

77.【JavaWeb文件上传和邮件发送04】

JavaWeb(二十五)、文件上传1.准备工作2.实用类介绍3.思维导图:4.正戏开始5.完整代码(二十六)、邮箱发送1.邮箱发送的原理:2.服务器的原理3.下载两个jar包4.基本类:5.全部代码(二十七)、网站注册发送邮件实现(二十五)、文件上传 1.首先创建一个empty项目 2.配置project项目中的…

【配送路径规划】基于matlab遗传算法求解静态外卖骑手路径规划问题【含Matlab源码 2248期】

⛄一、遗传算法求解静态外卖骑手路径规划问题 1 模型假设 外卖配送的实际运行是一个复杂的过程, 受诸多因素影响, 为了建立调度模型, 本文做如下假设。 (1) 外卖配送更多的是服务特殊群体, 所以本文认为外卖配送是一种预约型配送, 即在进行调度安排前, 己经获取了所有顾客的地…

版本控制利器——changelog

问题描述 当前&#xff0c;我们项目需要进行版本的确定&#xff0c;人工审核代码已接近尾声&#xff0c;但为了防止后续继续出现该问题&#xff0c;我希望能够做到在每次push到master时&#xff0c;更新changelog 将每一个版本的commit记录下来&#xff0c;类似于下列 解决…

C++_串口编程_官方示例:监视通信事件

这是微软官方的一个例子&#xff0c;这个例子中&#xff0c;如果不做修改&#xff0c;那么他是可以异步运行的&#xff0c;会出现一个错误&#xff1a;官方也说了一下&#xff0c;但是不太好懂&#xff0c;我拷贝过来放在这里&#xff0c;作为参考。 如果无法立即完成重叠的操作…

【Canvas】js用Canvas绘制阴阳太极图动画效果

学习JavaScript是否兴趣缺缺&#xff0c;那就需要来一个兴趣学习&#xff0c;问一下有没有兴趣用Canvas画图呢&#xff0c;可以画很多有趣的事物&#xff0c;自由发挥想象&#xff0c;收获多多哦&#xff0c;这里有一个例子&#xff0c;如何用canvas画阴阳太极图动图效果&#…

【博客544】golang pprof性能调试:寻找memory瓶颈

golang pprof性能调试&#xff1a;寻找memory瓶颈 1、前置 pprof的使用与输出列解析看姐妹篇&#xff1a;golang pprof性能调试&#xff1a;寻找cpu瓶颈 2、引入pprof到程序中&#xff0c;以调试memory瓶颈 给程序加入&#xff1a; import _ "net/http/pprof"go…

【Android App】实现在线语音合成功能(使用云知声平台和WebSocket 超详细 附源码)

需要源码和Jar包请点赞关注收藏后评论区留下QQ~~~ 一、在线语音合成 虽然国产智能机大多集成了中文语音引擎&#xff0c;但是系统自带的语音工具无法满足商用要求&#xff0c;功能单一&#xff0c;所以势必引入第三方的语音引擎&#xff0c;依靠第三方提供的开发包统一支撑语音…

缓存穿透、缓存击穿、缓存雪崩及其解决方案

缓存&#xff08;cache&#xff09;&#xff0c;大家都非常熟悉&#xff0c;几乎每个系统乃至整个计算机体系中都会用到。在分布式系统架构中&#xff0c;主要用于减轻数据库的压力&#xff0c;提高系统的响应速度和并发吞吐&#xff0c;即空间(内存)换时间。当大量的读、写请求…

2023年天津财经大学珠江学院专升本经济学专业课考试大纲

天津财经大学珠江学院2023年高职升本科专业课考试《经济学》考试大纲一、本大纲系天津财经大学珠江学院2023年高职升本科《经济学》课程考试大纲。所列考试范围出自郑健壮、王培才主编的教材《经济学基础&#xff08;第二版&#xff09;》&#xff0c;清华大学出版社&#xff0…

React - Ant Design4.x版本安装使用,并按需引入和自定义主题

React - Ant Design4.x版本安装使用&#xff0c;并按需引入和自定义主题一. 安装使用 antd二&#xff0e;antd 高级配置安装 craco&#xff0c;对 create-react-app 的默认配置进行自定义自定义主题安装 babel-plugin-import &#xff0c;按需加载组件代码和样式Ant Design官网…

mycat-3-实战篇

1 总结&#xff1a; 1&#xff1a;用的表必须在mycat的配置文件中配置。 2&#xff1a;mycat默认分片策略中&#xff0c;都是针对表的主键&#xff0c;默认是id,如果主键不是id的&#xff0c;请去rule.xml自己复制一份修改 3&#xff1a; 2 注意细讲解 1&#xff1a;schem…

车辆大全和车牌识别系统毕业设计,车牌识别系统设计与实现,车牌AI识别系统论文毕设作品参考

功能清单 【后台管理员功能】 系统设置&#xff1a;设置网站简介、关于我们、联系我们、加入我们、法律声明 广告管理&#xff1a;设置小程序首页轮播图广告和链接 留言列表&#xff1a;所有用户留言信息列表&#xff0c;支持删除 会员列表&#xff1a;查看所有注册会员信息&a…