2023前端面试题整理

news2025/1/10 5:46:25

前端面试大全整理

算法

n维数组旋转 90 度算法

export const rotate = function (matrix: number[][]) {
    let n = matrix.length

    // matrix[x][y] => matrix[y][n - 1 - x]
    const changeItem = (num: number, x: number, y: number, rodateTime: number, isOnce?: boolean) => {
        // 终止条件
        rodateTime++
        if (num === matrix[x][y] && !isOnce && rodateTime > 4) return

        let space = matrix[y][n - 1 - x]

        matrix[y][n - 1 - x] = num
        changeItem(space, y, n - 1 - x, rodateTime)
    }

    for (let j = 0; j < Math.floor(n / 2); j++) {
        let rodateTime = 0;
        for (let i = j; i < n - 1 - j; i++) {
            changeItem(matrix[j][i], j, i, rodateTime, true)
            rodateTime = 0
        }
    }

    return matrix
};

((())) 暴力解法找到所有的夸号组合(以下有解 搜索生成括号)

  • 待定,貌似不行,应该用动态规划的方式
  • 可以实现的思路:通过 f(1,2,3) = f(1,2) 拼接 f(3)
export const findkhByQuery = (num: number = 3) => {
    /**
     * arr.push() 
     * 怎样组合呢? (())() 
     * (  + )()()
     * 思路:一个夸号去移动
     * 
     * ()()()  取出第1个 从 0 移动到 最后一位
     * let str = '(' 
     * )()() 发现 不管第一个如何移动 只有第一次才行 移动到右边后发现有首位是 右夸号结束移动
     * 
     * (()()  取出第2个 ')' 从 0 移动到 最后一位
     * let str = ')'
     * 
     * (()()  取出第3个 ')' 从 0 移动到 最后一位
     * let str = '('
     * 
     * 
     * value
     */
    let initStr = '';
    let arr = []
    for (let i = 0; i < num; i++) {
        initStr += "()"
    }
    let n = initStr.length
    let map: any = {}

    const isValid = (str: string): boolean => {
        if (!str || str[0] === ')') {
            return false
        }

        let stack = []
        for (let i = 0; i < str.length; i++) {
            if (str[i] === '(') {
                stack.push('(')
            }
            if (str[i] === ')') {
                let x = stack.pop()
                if (!x) {
                    return false
                }
            }
        }

        if (stack.length == 0) {
            return true
        }

        return false
    }

    // 可以再套一个循环 用于决定要 去几个字符串 
    for (let k = 0; k < num * 2; k++) {

        for (let i = 0; i < num * 2; i++) {
            // 从 0 移动到最后一位
            // let val = initStr[i]
            // 取出需要拿出来的字符串
            let val = initStr.slice(0, k + 1)
            console.log(val, 'val');

            // 取到切割剩下的
            // 'abcdef' abc cde 
            let newStr = initStr.slice(k + 1, n)

            for (let j = 0; j < num * 2; j++) {
                let moveRes = newStr.slice(0, j) + `${val}` + newStr.slice(j, n)
                // debugger

                // debugger
                if (!map[moveRes]
                    && isValid(moveRes)
                ) {
                    map[moveRes] = moveRes
                    arr.push(moveRes)
                }
                // arr.push(moveRes)
            }
        }
    }
    console.log(arr, 'arr');
}


// console.log(findkhByQuery(3));

回文字串

  • 思路:列出所有的字串
var partition = function (s: string) {
    let arr: string[][] = []
    arr.push(s.split(''))

    for (let i = 0; i < s.length; i++) {
        let subitem: string[] = []
        for (let j = i + 1; j < s.length; j++) {

            let val = s.substring(i, j + 1)

            if (val.split('').reverse().join('') === val) {
                /* 
                如何做到自动补全的功能? abcde bcd => a,bcd,e
                * abacd_aba => ,aba,c,d,_,aba,
                * abacd_abaDDC => ,aba,c,d,_,aba,D,D,C
                * 切掉空的
                */
                let ns = s.replace(val, `,${val},`)
                // 去掉首尾的,号
                if (ns.startsWith(',')) {
                    ns = ns.substring(1, ns.length)
                } else if (ns.endsWith(',')) {
                    ns = ns.substring(0, ns.length - 1)
                }
                let resItem: any[] = []

                // aba,cd_,aba => aba,c,d,_,aba
                ns.split(',').forEach(item => {
                    if (item !== val) {
                        resItem = [...resItem, ...item.split('')]
                    } else {
                        resItem.push(item)
                    }
                })
                subitem = resItem

            }
        }
        if (subitem.length) {
            arr.push(subitem)
        }
    }


    return arr
};

console.log(partition("fff"));

模板字符串替换

const template = `<div>{{ name }}<span>{{ age }}</span><div>`;
let obj = { name: 'glack', age: '23' }

const func = (template: string, obj: any) => {
    Object.keys(obj).forEach(item => {
        let key = '{{ ' + item + ' }}'
        template = template.replace(key, ' ' + obj[item] + ' ')
    })

    return template
}

console.log(func(template, obj), 'xxx');

实现全排列

/*
    [1]
    [12, 21]
    [312, 132, 123, 321, 231, 213]
    
    [0,-1,1]
    [1]

    f( [1, 2, 3] ) = f([1, 2]) 和 3 的排列
    f( [1, 2]) = f([1]) 和 2 的排列
    f( [1] ) = [1]
    
    思路:f(12345) = f(1234) 和 5 组合起来的数组
*/
const qpl = (nums: number[]): any[] => {
    let time = 0

    const dfs = (a: number[]): any[] => {
        let n = a.length
        if (n == 1) {
            return [a]
        }

        let res: any[] = []
        let fontArr = a.slice(0, n - 1) // [1, -1]
        let lastStr = a.slice(n - 1, n)[0] //  0

        dfs(fontArr)?.forEach(it1 => {
            // [1, -1] [-1, 1] 中 使用 0 来移动位置
            for (let i = 0; i < it1.length + 1; i++) {
                res.push([...it1.slice(0, i), lastStr, ...it1.slice(i, it1.length)])
            }
        })

        return res
    }

    return dfs(nums)
}

console.log(qpl([-1, 2, 3]));

有序数组中找大于等于 n 最左边的数

给一个数组:[1,2,3,4,4,4,5,6,7,7,8,11,345,567] 找到大于 4最左边的数字;

  • 使用二分法
export const erfen = (arr: number[], n: number) => {
    let len = arr.length
    if (!arr || len < 2 || typeof n !== 'number') return arr

    let curValue = 0, midIndex, midValue
    let leftNums = [1]
    let rightNums = []
    let handleArr = JSON.parse(JSON.stringify(arr))

    while (leftNums.length + rightNums.length > 0) {
        midIndex = Math.floor(handleArr.length / 2)
        midValue = handleArr[midIndex]
        leftNums = handleArr.slice(0, midIndex - 1)
        rightNums = handleArr.slice(midIndex + 1, handleArr.length)

        if (midValue >= n) {
            curValue = midValue
            handleArr = leftNums
        } else {
            handleArr =  
        }
    }
    return curValue
}

优化简洁版

思路:如果传递的 n 大于等于 midValue 那就将右边的下标往中间挪动。这样就可以开始从 0 到 midIndex 之前找最小的值。如果小于,midIndex 到 len - 1 找小的最左边的值。

// 有序数组中找大于等于 n 最左边的数
export const erfen = (arr: number[], n: number) => {
    let len = arr.length
    if (!arr || len < 2 || typeof n !== 'number') return -1

    let leftIndex = 0
    let rightIndex = len - 1
    let ans = -1

    while(leftIndex <= rightIndex){
        let midIndex = Math.floor((leftIndex + rightIndex)/ 2)

        if (arr[midIndex] >= n) {
            ans = midIndex
            rightIndex = midIndex - 1
        }else{
            leftIndex = midIndex + 1
        }
    }

    return arr[ans]
}
console.log(erfen([1, 2, 3, 3, 3, 3, 4, 5, 5, 7, 8, 10, 11, 13], 3));

反转链表

  1. 保存gnext数据
  2. gPrev赋值给当前 head 节点的 next 节点
  3. 移动 gPrev 到下一步 (head 赋值给 gPrev)
  4. gPrev 接收搭配报错的gNext数据
interface INode {
    value: number,
    next?: INode,
    prev?: INode
}
export const linkReverseFunc = (head: INode) => {
    let prev: INode | undefined = undefined
    let next: INode | undefined = undefined
    while (head.next) {
        // 保存next数据
        next = head.next
        // 将当前节点的下一个指向prev 如果是第一次 则为空 如果第二次 在第一步prev已经赋值给了当前head节点
        head.next = prev
        // 当前节点给下一次的prev使用
        prev = head
        // 下一个节点变成开始全局记录的next 
        head = next

        if (!prev.next) {
            delete prev.next
        }
    }

    if(!head.next){
        console.log(head,'head');
        head.next = prev
        prev = head
    }

    return prev
}
let nodeinit: INode = {
    value: 100,
    next: {
        value: 200,
        next: {
            value: 300,
            next: {
                value: 400,
                next: {
                    value: 500,
                }
            }
        }
    }
}
console.log(linkReverseFunc(nodeinit), 'test');

并查集(待定)

如果 多个用户 a 或者 b 或者 c 的字段值一样 就合并
返回合并之后的用户加上count字段

思路:声明多个集合,如果,没出现过,就set到map里面,如果出现过,就count+1
下次再来。。。太难了,有思路再来。

// users合并
/* [
        { a: 1, b: 2, c: 3},
        { a: 1, b: 3, c: 4},
        { a: 8, b: 9, c: 10},
        { a: 10, b: 9, c: 8}
    ]
    思路:三个集合 aMap bMap cMap
    默认为空 最后会变成 aMap:  { a: 
                                { a: 1, b: 2, c: 3, count: 2}
                                { a: 8, b: 9, c: 10} }
                                { a: 10, b: 9, c: 8}
                            }, 

                     bMap: { a: { a: 2, b: 3, c: 9, count: 3}}
                     cMap: { a: { a: 3, b: 4, c: 10, count: 4}}
*/
export const mergeUser = (users: IUserInfo[]) => {
    let aMap = new Map()
    let bMap = new Map()
    let cMap = new Map()
    users.forEach(item => {
        // 如果三个集合都没有这个item 那就放在
        if (!aMap.has(item.a) && !bMap.has(item.b) && !cMap.has(item.c)) {
            aMap.set(item.a, { ...item, count: 1 })
            bMap.set(item.b, { ...item, count: 1 })
            cMap.set(item.c, { ...item, count: 1 })
            // return
        }
        if (aMap.has(item.a)) {
            let aItem = aMap.get(item.a)
            aItem.count++
            // return
        }
        if (bMap.has(item.b)) {
            let bItem = bMap.get(item.b)
            bItem.count++
            // return
        }
        if (cMap.has(item.c)) {
            let cItem = cMap.get(item.c)
            cItem.count++
            // return
        }
    })
    console.log(aMap, bMap, cMap, 'abc');
}
mergeUser([{ a: 1, b: 2, c: 3 }, { a: 1, b: 3, c: 4 }, { a: 8, b: 9, c: 10 }, { a: 10, b: 9, c: 8 }])

覆盖绳子最多的点数

// 盖住最大值的绳子上的点
export const ropeCoveringPoint = (arr: number[], n: number) => {
    let left = 0, right = 0, max = 0, len = arr.length

    while (left < len) {
        while (left < len && arr[right] - arr[left] <= n) {
            right++
        }
        // debugger
        max = Math.max(max, right - (left++))

    }

    return max
}

console.log(ropeCoveringPoint([1, 3, 6, 8, 10, 12, 13, 17, 19, 24, 28, 31], 16));

字符串只有 ”G“和”B“ 如果要把G放最左边,B放最右边,求需要移动几部才能让所有g在左边,b在右边。

export const moveChar = (str: string) => {
    let len = str.length
    let count = 0
    let gCount = 0
    for (let i = 0; i < len; i++) {
        if (str[i] === 'G') {
            gCount ++
            let res = (i - gCount) < 0 ? 0 : (i - gCount)
            // debugger
            count = count + res + 1
        }
    }

    return count
}
console.log(moveChar('GGBBGGGBBBGGGGBBBB'));

生成括号

思路:f(n) 的值永远都是 f(n-1) 的结果 里面每一项push ()

export function generateParenthesis(initn: number): string[] {
    const func = (n: number): string[] => {
        if (n == 1) {
            return ['()']
        }

        let res = func(n - 1)
        
        let pjRes = []
        for (let i = 0; i < res.length; i++) {
            let str = res[i]
            let sL = str.length
            for (let j = 0; j < sL; j++) {
                pjRes.push(`${str.slice(0, j)}()${str.slice(j, sL)}`)
            }
        }
        return pjRes
    }

    return Array.from(new Set(func(initn)))
}

console.log(generateParenthesis(3), 'na');

实现深拷贝包括 map set 类型

  1. map
  2. set
  3. 函数
  4. 循环引用
  5. 深层数据拷贝
export function cloneDeep(obj: any, map = new WeakMap()): any {
    if (typeof obj !== 'object' || obj == null) return obj

    // 避免循环引用
    const objFromMap = map.get(obj)
    if (objFromMap) return objFromMap

    let target: any = {}
    map.set(obj, target)

    // Map
    if (obj instanceof Map) {
        target = new Map()
        obj.forEach((v, k) => {
            const v1 = cloneDeep(v, map)
            const k1 = cloneDeep(k, map)
            target.set(k1, v1)
        })
    }

    // Set
    if (obj instanceof Set) {
        target = new Set()
        obj.forEach(v => {
            const v1 = cloneDeep(v, map)
            target.add(v1)
        })
    }

    // Array
    if (obj instanceof Array) {
        target = obj.map(item => cloneDeep(item, map))
    }

    // Object
    for (const key in obj) {
        const val = obj[key]
        const val1 = cloneDeep(val, map)
        target[key] = val1
    }

    return target
}

const a: any = {
    set: new Set([1, 2, 3, 4]),
    map: new Map([['x', 10], ['y', 20]]),
    info: {
        city: '北京'
    },
    fn: () => { },
}

a.self = a
console.log(cloneDeep(a), 'cloneDeep');

[‘1’, ‘2’, ‘3’].map(parseInt)

上面代码相当于

['1', '2', '3'].map((item, index) => {
    parseInt(item, index)
    // parseInt 
    // 转换的数字
    // raIndex 第二个参数是 进制 单位 (2-32)
    /**
     * parInt('1', 0) // 0 的时候 说明没传 是2进制
     * parInt('2', 1) // 1 的时候 没有 1 进制 所以返回 NAN
     * parInt('3', 2) // 2 的时候 3 不属于二进制的内容(011101010101010这种只有 0 和 1 才属于二进制) 
     */
})

所以最终返回 [1, NaN, NaN]

函数的参数传递就是赋值传递

const fn = (x, y) => {
    // 这里就相当于 x = num; y = obj
}
let num = 'aaa'
let obj = { name: 'glack' }
fn(num, obj)

数组转为树节点(针对有序的二维数组排列树节点)

const arr1 = [
    { id: 1, name: '部门A', parent_id: 0 },
    { id: 2, name: '部门B', parent_id: 1 },
    { id: 3, name: '部门C', parent_id: 1 },
    { id: 4, name: '部门D', parent_id: 2 },
    { id: 5, name: '部门E', parent_id: 4 },
    { id: 6, name: '部门F', parent_id: 4 },
];
/**
 * 针对有序的二维数组排列树节点
 * @param arr arr
 */
interface ITreeNode {
    id: number,
    name: string,
    children?: ITreeNode[]
}
interface ITreeItem {
    id: number,
    name: string,
    parent_id: number
}
export const treeTransform = (arr: ITreeItem[]): ITreeNode | null => {
    // 用于id 和 treeNode 的映射
    let treeMap: Map<number, ITreeNode> = new Map()

    let root: ITreeNode | null = null

    // 定义tree node 并加入 map
    arr.forEach(({ id, name }) => {
        const treeNode: ITreeNode = { id, name }
        treeMap.set(id, treeNode)
    })

    arr.forEach(item => {
        const { id, parent_id } = item

        const treeNode = treeMap.get(id)

        // 找到 parentNode 并加入到它的 children
        const parentNode = treeMap.get(parent_id)
        if (parentNode && treeNode) {
            if (parentNode.children == null) parentNode.children = []
            parentNode.children.push(treeNode)
        }

        // console.log(parentNode, 'parentNode');
        // console.log(treeNode, 'treeNode');

        // 找到根节点
        if (parent_id === 0) {
            // @ts-ignore
            root = treeNode
        }
    })

    return root
}
let result = treeTransform(arr1)
console.log(result);

怎么证明 root 的子元素一直在变?

  • 在 map 外部修改map里面的children 元素 里面的元素会跟着修改
let map = new Map();
map.set('arr', []);
let arr = map.get('arr');
arr.push('a');
map.get('arr') // ['a']

树转为数组

let res: any[] = []
let treeTransformArray = (root: any[], parent_id: number = 0) => {
    let arr = Array.isArray(root) ? root : [root]

    // 遍历所有元素
    arr.forEach(item => {
        if (item.children) {
            treeTransformArray(item.children, item.id)
        }

        res.push({ id: item.id, name: item.name, parent_id })
    })
}
// @ts-ignore 这里的result 是上面的结果
treeTransformArray(result)
console.log(res, 'zz');

原型链面试题

function Foo(){
    Foo.a = function name() {
        console.log(1);
    }
    this.a = function () {
        console.log(2);
    }
}

Foo.prototype.a = function(){ console.log(3); }

Foo.a = function(){ console.log(4) }

Foo.a()
let obj = new Foo()
obj.a()
Foo.a()

// 4 2 1

promise和事件循环的算法

Promise.resolve().then(() => {
    console.log(1)
}).then(() => {
    console.log(2)
}).then(() => {
    console.log(3)
}).then(() => {
    console.log(4)
}).then(() => {
    console.log(5)
}).then(() => {
    console.log(6)
})

Promise.resolve().then(() => {
    console.log(10)
}).then(() => {
    console.log(20)
}).then(() => {
    console.log(30)
}).then(() => {
    console.log(40)
}).then(() => {
    console.log(50)
}).then(() => {
    console.log(60)
})
// 1 10 2 20 3 30 4 40 5 50 6 60

多个promise 已完成的状态同时执行 .then 的链式调用
所以 .then会交替执行

Promise.resolve().then(() => {
    console.log(1)
    return Promise.resolve(100)
}).then((res) => {
    console.log(res)
}).then(() => {
    console.log(3)
}).then(() => {
    console.log(4)
}).then(() => {
    console.log(5)
}).then(() => {
    console.log(6)
})

Promise.resolve().then(() => {
    console.log(10)
}).then(() => {
    console.log(20)
}).then(() => {
    console.log(30)
}).then(() => {
    console.log(40)
}).then(() => {
    console.log(50)
}).then(() => {
    console.log(60)
})
// 1 10 20 30 100 40 3 50 4 60 5 6

.then 中返回promise 实例会慢两拍(三步) 因为需要时间把自己 penging 变成 fulfilled状态

setState 同步异步问题

didMount(){
    // 假设默认值是 0 
    this.setState({
        val: this.state.val + 1
    })
    console.log(this.state.val) // 异步更新 0 
    this.setState({
        val: this.state.val + 1
    })
    console.log(this.state.val) // 异步更新 0 

    setTimeout(() => {
        this.setState({
            val: this.state.val + 1
        })
        console.log(this.state.val) // 同步更新 2
        this.setState({
            val: this.state.val + 1
        })
        console.log(this.state.val) // 同步更新 3
    })
}

setState 会合并

如果同时多个setState 都要修改val 那么会优先使用 下面的那个setState语句

不合并的情况

传入函数: this.setState((prev, props) => { return …})

同步更新

setState 同步更新:在setTimeout setInterval promise.then
自定义的 dom 事件
ajax 回调
例子:

document.getElementById('id').addEventlistener('click', () => {
    this.setState({
        val: this.state.val + 1
    })
    console.log(this.state.val) // 同步更新
})

setState是同步 不过只不过是当做异步处理

React 18 中不一样了

可以在setTimeout 中 异步更新了
Auto Batch
需要将 ReactDOM.render 替换为 ReactDOM.createRoot

a.x 比赋值的优先级高

let a = { n: 1 }
let b = a
a.x = a = { n: 2 }

console.log(a.x) // undefined
console.log(b.x) //  n: 2

字符串对象的key 只能是字符串或者 symbol 类型

let a = {}, b = '123', c = 123
a[b] = 'b'
a[c] = 'c'

// log => a[b] c


let a = {}, b = Symbol('123'), c = Symbol('123')
a[b] = 'b'
a[c] = 'c'

// log => a[b] b

设计一个前端统计 SDK

  • 统计的范围:性能 错误 pv 自定义事件
  • 发送数据使用 img 的 src 来发送get请求
  • 报错统计结合 Vue React 报错

sourcemap 的作用

  • 线上代码压缩报错,sourcemap 可解决这个问题
  • 可以给压缩后的文件识别报错的行数,找到错误的地方

如何设置?

  • webpack 通过 devtool 配置sourcemap
    • eval - js 在 eval(…) 中 不生产sourcemap
    • source-map 生成单独的 map 文件, 并在js 最后指定
    • eval-source-map,js在eval(…) 中,sourcemap内嵌
    • inline-source-map - sourcemap 内嵌到 js 中
    • cheap-source-map - sourcemap 中只有行信息,没有列
    • eval-cheap-source-map 同上,只有行没有列
      非开源项目:不要泄漏sourcemap ! 否则容易被反解

SPA 和 MPA 怎么选择

  • SPA 特点

    • 功能较多,一个页面展示不完
    • 以操作为主,非展示为主
    • 适合一个综合Web应用
  • MPA

    • 功能较少,展示为主,例如:分享页,新闻详情页,微信公众号发出的页面

多种排序算法

快速排序

// slice
export const quickSort1 = (arr: number[]): number[] => {
    let length = arr.length

    if (length === 0) return []

    let left = [], right = []
    let midIndex = Math.floor(arr.length / 2)
    let midValue = arr.slice(midIndex, midIndex + 1)[0]
    
    for (let i = 0; i < arr.length; i++) {
        if (arr[i] < midValue) {
            left.push(arr[i])
        } else if (arr[i] > midValue) {
            right.push(arr[i])
        }
    }
    
    return quickSort1(left).concat([midValue], quickSort1(right))
}
// splice
export const quickSort2 = (arr: number[]): number[] => {
    let length = arr.length

    if (length === 0) return []

    let left = [], right = []
    let midIndex = Math.floor(arr.length / 2)
    let midValue = arr.splice(midIndex, 1)[0]
    
    for (let i = 0; i < arr.length; i++) {
        if (arr[i] < midValue) {
            left.push(arr[i])
        }
        if (arr[i] > midValue) {
            right.push(arr[i])
        }
    }

    return quickSort2(left).concat([midValue], quickSort2(right))
}

const arr = [4,5,7,1,2,3,6,3,15,5,123,8,9,1,9,8]
console.log(quickSort2(arr), 'arr')

选择排序

// 选择排序一
export function sort1(arr: number[]): number[] {
    let len = arr.length
    if (arr == null || arr.length < 2) return []

    let ti = 0

    while (ti < len) {
        // 找出最小值
        let num: number | null = null
        let idx: number = 0
        for (let i = ti; i < len; i++) {
            if (num == null || arr[i] < num) {
                num = arr[i]
                idx = i
            }
        }

        // 最小下标和 第 ti 项换位置
        let n = arr[ti]
        arr[ti] = arr[idx]
        arr[idx] = n

        ti++
    }

    return arr
}
// 选择排序二
export function sort2(arr: number[]): number[] {
    let length = arr.length

    for (let i = 0; i < length; i++) {
        let minValueIndex = i
        for (let j = i + 1; j < length; j++) {
            minValueIndex = arr[j] < arr[minValueIndex] ? j : minValueIndex
        }
        // 交换位置
        let n = arr[i]
        arr[i] = arr[minValueIndex]
        arr[minValueIndex] = n
    }

    return arr
}

冒泡排序

思路:
[ 3, 2, 1, 6, 5, 3, 7]
0 1 2 3 4 5 6
0 1 对比 大的往右
1 2 对比 大的往右
2 3

以此类推
最大的在最右边

export function sort3(arr: number[]): number[] {
    let length = arr.length
    if (arr.length < 2) return []
    let time = 0

    for (let i = 0; i < length; i++) {
        for (let j = 0; j < length - i; j++) {
            time++
            console.log(time);

            // 如果 i 大于 j 交换
            if (arr[j] > arr[j + 1]) {
                let n = arr[j + 1]
                arr[j + 1] = arr[j]
                arr[j] = n
            }
        }
    }

    return arr
}

插入排序

 /**
 * 
 * [2, 4, 1, 3, 8, 6, 7]
 * 思路:
 * 0 - 1 arr: [2, 4, 1, 3, 8, 6, 7] 排序
 * 0 - 2 arr: [1, 2, 4, 3, 8, 6, 7] 排序
 * 0 - 3 arr: [1, 2, 3, 4, 8, 6, 7] 排序
 * 0 - len - 1 排序 ...
 */
// 真·插入排序
function sort6(arr: number[]): number[] {
    let length = arr.length
    if (!arr || length < 2) return arr

    for (let i = 1; i < length; i++) {
        let endIndex = i
        while (endIndex - 1 >= 0 && arr[endIndex] < arr[endIndex - 1]) {
            // 交换位置
            let n = arr[endIndex]
            arr[endIndex] = arr[endIndex - 1]
            arr[endIndex - 1] = n
            endIndex --
        }
    }

    return arr
}
// 真·插入排序优化
function sort7(arr: number[]): number[] {
    let length = arr.length
    if (arr == null || length < 2) return arr

    for (let i = 1; i < length; i++) {
        for (let endI = i; endI - 1 >= 0 && arr[endI] < arr[endI - 1]; endI--) {
            let n = arr[endI]
            arr[endI] = arr[endI - 1]
            arr[endI - 1] = n
        }
    }

    return arr
}

数组扁平化

  • 扁平一层数组
    • 效果 [1, 2, [3, 4, [5]]] -> [1, 2, [3, 4, 5]]
export const flatten1 = (arr: any[]): any[] => {
    let res: any[] = []

    for (let i = 0; i < arr.length; i++) {
        let item = arr[i]
        if (Array.isArray(item)) {
            for (let j = 0; j < item.length; j++) {
                res.push(item[j])
            }
        } else {
            res.push(item)
        }
    }

    return res
}

export const flatten2 = (arr: any[]): any[] => {
    let res: any[] = []
    for (let i = 0; i < arr.length; i++) {
        res = res.concat(arr[i])
    }

    return res
}

const initarr = [1, [2, [3], 4], 5]
console.log(flatten1(initarr), '1');
console.log(flatten2(initarr), '2');
  • 扁平多层数组
    concat的方式
export const flatten3 = (arr: any[]): any[] => {
    let res: any[] = []

    for (let i = 0; i < arr.length; i++) {
        let item = arr[i]
        
        if (Array.isArray(item)) {
            let nArr = flatten3(item)            
            res = res.concat(nArr)
        } else {
            res.push(item)
        }
    }

    return res
}

push的方式

export const flatten3 = (arr: any[]): any[] => {
    let res: any[] = []

    for (let i = 0; i < arr.length; i++) {
        let item = arr[i]
        
        if (Array.isArray(item)) {
            let nArr = flatten3(item)
            // 解释:不管如何走,flatten3 返回的永远是拍平的数组,因为他不是数组的话会再次走进里面去。
            // 至于为什么需要在contact 或者 forEach 因为返回的是一层的数组。
            res = nArr.forEach(n => res.push(n))
        } else {
            res.push(item)
        }
    }

    return res
}

toString方式

const arr = [1, [2, [3], 4], 5]
arr.toString() // 1,2,3,4,5
// 这样的方法不健壮

常见的类型判断 引用数据类型和基本数据类型的区别

引用类型 (Object,Array) 基本类型 (number, string, boolean, symbol, null, undefined)

export const getType = (x: any) => {
    let type = Object.prototype.toString.call(x)
    let spaceIndex = type.indexOf(' ')
    let res = type.slice(spaceIndex + 1, -1)
    return res.toLowerCase()
}

new 一个对象发生了什么

class Foo {
    this.name = 'glack'
    this.age = 'age'

    getName () {
        return 'xxx'
    }
}

上面转换成了es5 的语法

function Foo {
    this.name = 'glack'
    this.age = 'age'
}
Foo.prototype.getName = function() { 
    return 'xxx'
}

手写一个new 方法

/**
 * 
 * @param consturctor 
 * @param args 
 */
export function customNew<T>(consturctor: Function, ...args: any[]): T {
    // 1. 创建一个空对象,集成 constructor 的原型
    const obj = Object.create(consturctor.prototype)
    // 2. 将 obj 作为this,执行 constructor ,传入参数
    consturctor.apply(obj, args)
    // 3. 返回 obj
    return obj
}

class Foo {
    // 属性
    name: string
    age: number

    constructor(name: string, age: number) {
        this.name = name
        this.age = age
    }

    getName() {
        return this.name
    }
}

// const f = new Foo('glack', 23)
const f = customNew<Foo>(Foo, 'glack', 23)
console.log(f, 'f');
console.log(f.age);
console.log(f.getName());

遍历一个 dom 树

// 遍历子节点
export function visitNode(n: Node) {
    if (n instanceof Comment) {
        console.log(n, 'n');
    }
    if (n instanceof Text) {
        console.log('Text node ---', n.textContent?.trim());
    }
    if (n instanceof HTMLElement) {
        console.log('HTML node ---', `<${n.tagName.toLowerCase()}>`);
    }
}

// 深度优先遍历
function deepFindNode(root: Node) {
    visitNode(root)

    const childNodes = root.childNodes
    if (childNodes.length) {
        childNodes.forEach(item => {
            deepFindNode(item)
        })
    }
}

// 广度优先遍历(队列)
function breadFirstTraverse(root: Node) {
    const quene: Node[] = []

    // 根节点入队
    quene.unshift(root)

    while(quene.length > 0){
        const curNode = quene.pop()
        if(curNode == null) break

        visitNode(curNode)

        // 子节点入队
        const childNodes = curNode.childNodes
        if (childNodes.length > 0) {
            childNodes.forEach(n => quene.unshift(n))
        }
    }
}

setTimeout(() => {
    const box = document.getElementById('box')
    if (!box) throw Error('box is null')

    breadFirstTraverse(box)
}, 1000);

递归

问题:计算 1 - 100 的和。

  1. 一般是先假装该函数是存在的,通过数学中的找规律的方法,找到方法的递归体
func(n) = func(n-1) + n
  1. 找到临界值,比如,1 - 100 的和,就是 func(n-1) + 100, 临界值是 n = 1 reurn 1
if (n === 1) return 1
  1. 写入函数体
function comput1(num: number): number {
    if (num === 1) return 1
    return comput1(num - 1) + num
}

一些基础题:

// 求 2,4,6,8,10... 第n项与前n项之和
function sum1(num: number): number {
    if (num === 1) return 2
    return sum1(num - 1) + 2 * num
}

// 求1 3 5 7 9 ... 第n项与第n项之和
function sum2(num: number): number {
    if (num === 1) return 1
    return sum2(num - 1) + (2 * num - 1)
}
console.log(sum2(5));

模板字符串替换

export function render(temp: string, person: { name: string, age: number, sex: boolean }): string {
    let reg = /\{\{(\w+)\}\}/
    
    if (reg.test(temp)) {
        // 找到模板字符串 
        // @ts-ignore
        const name = reg.exec(temp)[1]
        
        // @ts-ignore
        temp = temp.replace(reg, person[name])
        return render(temp, person)
    }
    return temp
}


let template = '我是{{name}},年龄{{age}},性别{{sex}}';
let person = {
    name: '布兰',
    age: 12,
    sex: true
}
console.log(render(template, person));

reduce 实现 map 和 filter 方法

export function reduceCustom(arr: any[]): any[] {
    return arr.reduce((filterRes, item) => {
        if (item > 20) {
            filterRes.push(item * 2)
        }
        return filterRes
    }, [])
}
const arr = reduceCustom([10, 20, 30, 40, 50])
console.log(arr, 'arr');

reduce 实现 map 方法

export function zMap(arr: any[], fn: Function) {
    return arr.reduce((initArr: any[], item: number) => {
        return initArr.concat(fn(item))
    }, [])
}
const arr = zMap([10, 20, 30, 40, 50], (x: any) => x * 2)
console.log(arr, 'arr'); // 20 40 60 80 100

二维数组累加

const arr = [
    { id: 1, name: 'glack1', count: 1 },
    { id: 2, name: 'glack2', count: 2 },
    { id: 3, name: 'glack3', count: 3 },
    { id: 4, name: 'glack4', count: 4 },
    { id: 1, name: 'glack5', count: 2 },
    { id: 3, name: 'glack6', count: 0 },
    { id: 2, name: 'glack7', count: 1 },
    { id: 4, name: 'glack8', count: 2 },
]

export function arrayAdd(arr: any[]) {
    let map: any = {}
    return arr.reduce((initArr, item) => {
        let key = item.id
        if (map[key]) {
            let num = map[key].count + item.count
            map[key].count = num
            initArr = initArr.map((it1: any) => {
                if (it1.id == key) {
                    return {
                        ...it1,
                        count: num
                    }
                }
                return it1
            })
        } else {
            map[key] = {
                count: item.count
            }
            initArr.push(item)
        }
        return initArr
    }, [])
}

console.log(arrayAdd(arr), 'xxx');

一维数组递归变成树节点。

interface Node {
    id: number,
    parent_id: number | string,
    children?: any[]
}

export function treeTransform(arr1: Node[], id: number | string = ''): any {
    let nArr: Node[] = arr1.filter(item => item.parent_id == id)
    return nArr.map(item => ({
        ...item,
        children: treeTransform(arr1, item.id)
    }))
}

const arr1 = [
    { id: 1, parent_id: '' },
    { id: 2, parent_id: 1 },
    { id: 3, parent_id: 1 },
    { id: 4, parent_id: 2 },
    { id: 5, parent_id: 4 }
];
console.log(treeTransform(arr1));

手写 LazyMan

  • 支持sleep
  • 支持链式调用

思路:

  1. 首先初始化一个任务队列
  2. 每碰到eat 或者 sleep 都先注册一个函数,把函数放进任务队列中
  3. 返回this,让它支持链式调用
  4. 最后等到 eat 和 sleep 创建完成了任务队列之后
  5. 再调用第一个next方法(shift)
export class LazyMan {
    private tasks: Function[] = []
    private name: string

    constructor (name: string){
        this.name = name

        setTimeout(() => {
            // 等到初始化执行完毕再执行next
            console.log(this.tasks.length);
            
            // 也就是等到 eat(a) eat(b) eat(c) sleep(xx) 才会触发next()
            this.next()
        });
    }

    private next(){
        const task = this.tasks.shift()
        if (task) task()
    }

    eat(food: string){
        const task = () => {
            console.log(`${this.name} eat ${food}`);
            this.next()
        }
        this.tasks.push(task)
        return this
    }

    sleep(seconds: number){
        const task = () => {
            setTimeout(() => {
                console.log(`${this.name}睡了${seconds}s 开始执行下一个任务`);
                this.next()
            }, seconds * 1000);
        }
        this.tasks.push(task)

        return this
    }
}

const me = new LazyMan('glack')
me.eat('apple').eat('bannan').sleep(3).eat('pear')

手写一个 curry 函数 把其他函数柯里化(函数式编程,不常用,但是会考)

思路:
curry 最终返回一个函数
执行fn 中间状态返回函数,add(1) 或者 add(1)(2)
最后情况执行函数 add(1)(2)(3) 执行

export function curry(fn: Function) {
    // 这里的fnArgsLength 就是指 fn 的参数的长度
    const fnArgsLength = fn.length
    let args: any[] = []

    function calc(this: any, ...newArgs: any[]) {
        args = [
            ...args,
            ...newArgs
        ]
        if (args.length < fnArgsLength) {
            // 如果当前拼接的长度小于整个方法应传递的长度 返回当前函数 给继续调用 add(20)
            return calc
        }else{
            // 如果长度是拼接的三个参数 那就直接执行传递的方法
            return fn.apply(this, args.slice(0, fnArgsLength))
        }
    }
    return calc
}

function add(a: number, b: number, c:number): number{
    return a + b + c
}

const curryAdd = curry(add)
console.log(curryAdd(10)(20)(30));
// 这里如果传递方式有误的话 curryAdd(10)(20)(30) 会导致length 判断出错 没有返回函数 导致fn()() 报错

intanceof 原理是什么 用代码表示

export const myIntanceof = (intance: any, origin: any) => {
    if (intance == null) return false

    let type = typeof intance

    if (type !== 'object' && type !== 'function') {
        // 值类型 直接返回false
        return false
    }
    
    let tempIntance = intance
    while(tempIntance){
        if (tempIntance.__proto__ == origin.prototype) {
            return true
        }
        // 向上查找 如果为 null 就不进入循环
        tempIntance = tempIntance.__proto__
    }

    return false
}

console.log(myIntanceof({}, Object)); // true
console.log(myIntanceof([], Object)); // true
console.log(myIntanceof([], Array)); // true
console.log(myIntanceof({}, Array)); // false
console.log(myIntanceof('123', String)); // false
console.log(myIntanceof(123, Number)); // false
console.log(myIntanceof(true, Boolean));  // false
console.log(myIntanceof(()=>{}, Function)); // true

手写bind

分析:

  1. 返回function
  2. 重命名传递的this
  3. 定义调用者的this
  4. 拼接参数
// @ts-ignore
Function.prototype.customBind = function (context: any, ...bindArgs: any[]) {
    // context 是 bind 传入的this,bindArgs 是bind 传入的各个参数
    const self = this

    return function (...args: any[]) {
        const newArgs = bindArgs.concat(args)
        return self.apply(context, newArgs)
    }
}

function fn(this: any, ...args: any[]) {
    console.log(this, ...args);
}

// @ts-ignore
const fn1 = fn.customBind({ x: 100 }, 10, 20, 30)
fn1(40, 50)

bind call apply 区别

不同点:
bind 返回一个函数(不执行)
call 和 apply 会立即执行
传递方式不同:

call(this, 10,20,30)
apply(this, [10,20,30])

相同:
都是用于绑定this
都可以传入执行参数

手写call

// @ts-ignore
Function.prototype.customCall = function (context: any, ...args: any[]) {
    if (context == null) context = globalThis
    // 值类型 需要转换成对象类型
    if (typeof context !== 'object') context = new Object(context)

    const fnKey = Symbol() // 防止出现属性名称覆盖
    context[fnKey] = this // this 就是当前函数

    const res = context[fnKey](...args) // 绑定 this

    delete context[fnKey] // 清理fn,防止污染

    return res
}

function fn(this: any, a: any, b: any, c: any) {
    console.log(JSON.stringify(this), a, b, c);

}

// @ts-ignore
fn.customCall({ x: 100 }, 10, 20, 30)
// @ts-ignore
fn.customCall(100, 10, 20, 30)

手写apply

// @ts-ignore
Function.prototype.customApply = function (context: any, args: any[] = []) {
    if (context == null) context = globalThis
    // 值类型 需要转换成对象类型
    if (typeof context !== 'object') context = new Object(context)

    const fnKey = Symbol() // 防止出现属性名称覆盖
    context[fnKey] = this // this 就是当前函数

    const res = context[fnKey](...args) // 绑定 this

    delete context[fnKey] // 清理fn,防止污染

    return res
}


function fn(this: any, a: any, b: any, c: any) {
    console.log(JSON.stringify(this), a, b, c);

}

// @ts-ignore
fn.customApply({ x: 100 }, [10, 20, 30])
// @ts-ignore
fn.customApply(100, [10, 20, 30])

手写EventBus 自定义事件

实现on once emit方法

方法一:

// 绑定成一个类似这种的数组
{
   'key1': [
    {fn1, isOnce: false},
    {fn2, isOnce: false},
    {fn3, isOnce: true},
   ]
   'key2': ...
}

具体代码:

export class EventBus {
    private events: {
        [key: string]: Array<{ fn: Function, isOnce: boolean }>
    }

    constructor() {
        this.events = {}
    }

    on(type: string, fn: Function, isOnce: boolean = false) {
        const events = this.events
        if (events[type] == null) {
            events[type] = [] // 初始化key 的 fn 数组
        }

        events[type].push({ fn, isOnce })
    }

    once(type: string, fn: Function) {
        this.on(type, fn, true)
    }

    off(type: string, fn?: Function) {
        if (!fn) {
            // 删除所有type的函数
            this.events[type] = []
        } else {
            // 解绑单个事件
            const fnList = this.events[type]
            if (fnList) {
                this.events[type] = fnList.filter(item => item.fn !== fn)
            }
        }
    }

    emit(type: string, ...args: any[]) {
        const fnList = this.events[type]
        if (fnList == null) return

        // filter 可以做到return 删除元素,并且还能遍历
        this.events[type] = fnList.filter(item => {
            const { fn, isOnce } = item
            fn(...args)

            if(!isOnce) return true

            return false
        })
    }
}

方法二:

export class EventBus {
    private events: {
        [key: string]: Array<Function>
    }
    private onceEvents: {
        [key: string]: Array<Function>
    }

    constructor() {
        this.events = {}
        this.onceEvents = {}
    }

    on(type: string, fn: Function) {
        const events = this.events
        if (events[type] == null) events[type] = [] // 初始化key 的 fn 数组

        events[type].push(fn)
    }

    once(type: string, fn: Function) {
        const onceEvents = this.onceEvents
        if (onceEvents[type] == null) onceEvents[type] = [] // 初始化key 的 fn 数组

        onceEvents[type].push(fn)
    }

    off(type: string, fn?: Function) {
        if (!fn) {
            // 删除所有type的函数
            this.events[type] = []
            this.onceEvents[type] = []
        } else {
            // 解绑单个事件
            const fnList = this.events[type]
            const onceFnList = this.onceEvents[type]

            if (fnList) {
                this.events[type] = fnList.filter(curFn => curFn !== fn)
            }
            if (onceFnList) {
                this.onceEvents[type] = onceFnList.filter(curFn => curFn !== fn)
            }
        }
    }

    emit(type: string, ...args: any[]) {
        const fnList = this.events[type]
        const onceFnList = this.onceEvents[type]

        if (fnList) {
            fnList.forEach(f => f(...args))
        }
        if(onceFnList){
            onceFnList.forEach(f => f(...args))

            // once 执行一次就删除
            this.onceEvents[type] = []
        }
    }
}

效果演示代码

const event = new EventBus()
event.on('key1', function (...args: any[]) {
    console.log('fn1', ...args);
})
event.on('key1', function () {
    console.log('fn2');
})
event.emit('key1', 10, 20, 30)

event.once('key2', () => {
    console.log('zzz');
})
event.emit('key2', () => {
    console.log('yyy');
})
event.emit('key2', () => {
    console.log('vvv');
})

LRUCache

Map 实现一个LRU 缓存类

/**
 * LRU cache
 */
export class LRUCache {
    private length: number
    private data: Map<any, any> = new Map()

    constructor(length: number) {
        if (length < 1) throw new Error('请输入大于 1 的数字')
        this.length = length
    }

    set(key: string, value: any) {
        const data = this.data
        if (data.has(key)) {
            data.delete(key)
        }
        data.set(key, value)

        // 如果长度大于传递的长度,需要删掉最后一个
        if (data.size > this.length) {
            let delKey = data.keys().next().value
            data.delete(delKey)
        }
    }

    get(key: string) {
        const data = this.data

        if(!data.has(key)) return null

        let value = data.get(key)
        data.delete(key)

        data.set(key, value)
        return value
    }

    getList(){
        return this.data
    }
}

const lruCache = new LRUCache(2)

lruCache.set('xx', 'xx')
lruCache.set('yy', 'yy')
lruCache.set('zz', 'zz')

console.log(lruCache.get('xx'));
console.log(lruCache.getList());
console.log('----------------');
console.log(lruCache.get('yy'));
console.log(lruCache.getList());
console.log('----------------');

生成符合条件的括号

思路:f(n) 的值永远都是 f(n-1) 的结果 里面每一项push ()

export function generateParenthesis(initn: number): string[] {
    const func = (n: number): string[] => {
        if (n == 1) {
            return ['()']
        }

        let res = func(n - 1)
        
        let pjRes = []
        for (let i = 0; i < res.length; i++) {
            let str = res[i]
            let sL = str.length
            for (let j = 0; j < sL; j++) {
                pjRes.push(`${str.slice(0, j)}()${str.slice(j, sL)}`)
            }
        }
        return pjRes
    }

    return Array.from(new Set(func(initn)))
}

console.log(generateParenthesis(3), 'na');

H5页面如何做首屏渲染

  • 路由懒加载
  • 服务端渲染 ssr
  • App 预取
  • 分页
  • 图片懒加载
  • hybrid 混合

后端返回 10w 条数据,前端如何优化

  • node 中间层里面去改
  • 虚拟列表
  • 插件
    • vue-virtual-scroll-list
    • React-virtualiszed

观察者模式和发布订阅者模式的区别

按钮点击事件
发布订阅是一个订阅事件 一个触发事件
emit on

观察者模式 发布者和订阅事件可以直接调用,比如addEventListener click
发布订阅模式 发布和监听者互相隔离,需要中间媒介来触发。

使用 Vue 的时候遇到那些坑

  • 内存泄漏
  • Vue 新增属性需要用 Vue.Set
  • Vue 删除属性需要用 Vue.delete
  • 无法直接修改数据 arr[index]= value
  • 从列表页切换至详情页,再返回列表页,会自动scroll到顶部。
    解决方案1:
    1. 在列表页数据缓存下来。 在返回的时候,列表数据渲染上去。
    2. 通过 MPA(多屏渲染) 的方式。eg:多个 WebView

实际工作中,React 做过什么优化

  • 修改css 模拟 v-show
  • 使用fragment 减少层级
  • render函数中减少定义,由于render 会频繁执行,所以尽量较少在dom节点新建逻辑。
  • SCU shouldComponentUpdate 来优化渲染
  • pureComponent React.memo

react的坑

js 关键字冲突
setState 异步更新的 需要在回调中用
对于深层数据的state遍历不是很友好,得通过浅拷贝的方式来手动赋值。

错误监听

  • window.error 监听全局组件报错,可以监听异步报错
  • 事件报错 try catch 报错 catch 抛出异常,让上级组件监听到
  • js 中 Promise 的 catch 报错,可以使用 onUnHandledrejection 监听。
  • 异步错误没法监听,需要结合window.error
  • 报错统计(埋点,上报,统计)
  • Vue
    • errorHandle 捕获下级组件的报错 但是没法捕获异步的报错
  • React
    • ErrorBoundary 组件 (核心就是componentDidCatch)

      • 统一监听下级组件内部报错
      • 降级展示UI(可以有友好提示)
      • 监听渲染报错
      • 不监听DOM事件,异步错误
      • Prod环境生效,dev环境抛出错误
      • 代码演示:
      import React from 'react';
      import ErrorTemplate from './template'
      
      class ErrorBoundary extends React.Component {
          constructor(props) {
              super(props);
              this.state = { error: null, errorInfo: null };
          }
      
          componentDidCatch(error, errorInfo) {
              this.setState({
                  error: error,
                  errorInfo: errorInfo
              })
          }
      
          render() {
              if (this.state.errorInfo) {
                  return <ErrorTemplate title="error" tip="糟糕,页面出错了!"/>
              }
              return this.props.children;
          }
      }
      export default ErrorBoundary
      
      • 生产环境验证
        yarn build
        http-server -p 8881 通过本地路径打开文件

如果一个H5 很慢,你该如何排查性能问题?

是哪里慢,怎么体现?
可以从很多个方面沟通。重点:沟通!沟通!沟通!

性能指标:

  • FP(First Paint)第一次加载的情况

  • FCP(First Contentful Paint)有内容的情况

  • FMP(First Meaningful Paint)第一次有意义的渲染(没有一个标准,已弃用)

  • DCL(DomContentLoaded)原生dom内容下载完成

  • LCP(Largest ContentFul Paint)重要的数据已经渲染完

  • L (Load)加载完毕

  • devtool 中的 【性能监控】 开启。可以监控加载慢的请求;network 看监控时间;

  • Lighthouse 测试报告

    • 第三方性能评测工具
    • 支持移动端和pc端
    • 优化建议:webp avif 格式的图片;有限的压缩方法,图片尺寸,http2;等等
    • 安装:npm i lighthouse -g
    • 使用:lighthouse https://www.imooc.com/–view --preset=desktop
  • 网页加载慢?

    • 优化服务器硬件配置,使用CDN
    • 路由懒加载,大组件异步加载 - 减少主包的体积
    • 优化HTTP 换成策略
  • 网页渲染慢?

    • 优化服务端接口(ajax返回时间优化)
    • 优化组件内部业务逻辑
    • 服务端渲染 SSR
  • 持续跟进

    • 性能优化是持续的过程
    • 持续跟进,统计结果,
    • 第三方统计服务:阿里云ARMS、百度统计
  • 二分法减少问题范围逐步更近

工作经历中,遇到什么项目难点,如何解决?

  • redux中深浅拷贝,深浅比较问题。层次比较深的时候,每一层都需要解构出来,代码容器不好看,而且要是碰到层级非常深导致set的代码很长。
this.state = {fourValue: { a: 'a', b: { c: { d: 'd', e: 'ee' } } }}
// 要想修改e = 'ff'
必须层层解构
let b = this.state.threeValue.b
this.setState({
  threeValue: { 
    ...this.state.threeValue,
    b: { 
      ...b, 
      c: 'xxx' 
    } 
  }
})

数组里面这么做:

import React, { Component } from 'react'

export default class About extends Component {
  state = {
    isSubmitting: false,
    inputs: {
      username: {
        touched: false,
        value: 'some_initial_value'
      },
      users: [
        { id: 1, name: 'glack1', count: 1 },
        { id: 2, name: 'glack2', count: 2 },
        { id: 3, name: 'glack3', count: 3 },
        { id: 4, name: 'glack4', count: 4 },
        { id: 5, name: 'glack5', count: 5 },
      ]
    }
  }

  twochange() {
    let s = this.state
    s.inputs.users[0].count = 100
    this.setState({
      inputs: {
        ...s.inputs,
        users: [...s.inputs.users]
      }
    })
  }

  render() {
    return (
      <>
        <button onClick={() => this.twochange()}>改变array中的count</button>
        {this.state.inputs.users.map(item => {
          return <>
            <div>name:{item.name}</div>
            <div>id:{item.id}</div>
            <div>count:{item.count}</div>
          </>
        })}
      </>
    )
  }
}

  • 歌词播放器时间播放和歌词滚动配合。
  • 改造之前小程序的代码
    • 群聊模块轮询的方式改造成wxWebscoket
    • 优化首次加载是app.js 还未执行完毕的情况,通过app中的一个回调。
  • 模板:描述问题:背景 + 现象 + 造成的影响
  • 解决:分析 + 解决
  • 学到了什么?如何避免?
    可以整个文档,遇到的难点都归纳起来。

时间复杂度

  • O(1) 一次就够(数量级)
    ex: 只走了一次处理逻辑

    const func = (a) => {
    	return a
    }
    
  • O(n) 和传输的数据量一样
    ex:时间复杂度和当前arr的长度一致

    const func = (arr) => {
    	for(let i = 0; i < arr.length; i++){
    		console.log(arr[i])
    	}
    }
    
  • O(n^2) 数据量的平方
    ex

    const func = (arr) => {
    	for(let i = 0; i < arr.length; i++){
    		for(let j = 0; j < arr.length; j++){
    			console.log(arr[j])
    		}
    	}
    }
    
  • O(logn) 数据量的对数(数量级) 100 => 10
    ex: 二分算法,每次把数据砍掉二分之一。
    [1,2,3,4,5,6,7,8,…] 找到6,可以通过砍掉中间的来比对

  • O(nlogn)

空间复杂度

  • O(1) 一次就够(数量级)
    ex: 像这样,不管数组的长度是多少,数组的内存大小是相对固定的
    const func = (arr: []) => {
    	arr[1] = 1
    	arr[2] = 2
    	arr[3] = 3
    	arr[4] = 4
    	return a
    }
    
  • O(n)
    ex:这里定义了一个新的数组,并且对相关的数组进行了赋值
    const func = (arr) => {
    	let arr2 = []
    	for(let i = 0; i < arr.length; i++){
    		arr2[i] = arr[i]
    	}
    }
    

单元测试

jest
判断数组是否相等 toEquel
判断bool是否相等 toBe

二分查找

/**
 * desc: 给一个有序的数组,[10, 20, 50, 70, 90] 查到50下标
 * 算法:二分法查找
 */

// 循环的方式
export const binarySearch1 = (arr: number[], target: number) => {
    let startIndex: number = 0
    let endIndex: number = arr.length - 1

    while (startIndex < endIndex) {
        let mdIndex = Math.floor((endIndex + startIndex) / 2)
        let value = arr[mdIndex]

        // 如果正好是中间的数字
        if (value === target) return mdIndex

        // 在左边
        if (target < value) {
            endIndex = mdIndex - 1
        }

        // 在右边
        if (target > value) {
            startIndex = mdIndex - 1
        }

    }

    return -1
}

// 递归的方式
const binarySearch2 = (arr: number[], target: number, startIndex?: number, endIndex?: number): number => {
    if(arr.length <= 0) return -1
    if(startIndex == null) startIndex = 0
    if(endIndex == null) endIndex = arr.length - 1
    
    if(startIndex > endIndex) return -1

    // 找到中间下标和value
    let midIndex = Math.floor((startIndex + endIndex) / 2)
    let midValue = arr[midIndex]

    if (midValue > target) {
        return binarySearch2(arr, target, startIndex, midIndex - 1)
    }else if(midValue < target){
        return binarySearch2(arr, target, midIndex + 1, endIndex)
    }else{
        return midIndex
    }
}

const arr = [10, 20, 40, 50, 70, 90, 120]
let res = binarySearch2(arr, 90)
console.log(res, 'res');

这里写了两种方式,递归和循环都可以完成问题,时间复杂度为O(logn),但是如果一定得使用最好的方式,循环更好,因为递归调用方法也会耗时。

将一个数组旋转k步

方案一 unshift pop
时间复杂度:O(n^2)
unshift 也是一个O(n) 的结构

方案二 contact
时间复杂度:O(1)

/**
 * descripbe: 
 * 1. 完成两个算法,k位数往数组的前面加
 * 2. 测试时间 log time endTime
 */


// 通过unshift的方法 [1,2,3,4,5,6,7] 3 => [5,6,7,1,2,3,4]
export const rodate1 = (arr: number[], len: number): number[] => {
    // 去k的绝对值
    let k = Math.abs(len % arr.length)
    for (let i = 0; i < k; i++) {
        let chu = arr.pop()
        if (chu) {
            arr.unshift(chu)
        }
    }

    return arr
}

// 通过contact的方法
export const rodate2 = (arr: number[], len: number): number[] => {
    let k = Math.abs(len % arr.length)
    len = k
    let l = arr.length

    let left = arr.slice(0, l - len)
    let right = arr.slice(l - len, l)

    return [...right, ...left]
}

let arrHuge = []
for (let i = 0; i < 100000; i++) {
    arrHuge.push(i)
}
let step = 8 * 10000

// 这个过程执行了 4000ms 也就是4s 时间复杂度 O(n^2)
console.time()
rodate1(arrHuge, step)
console.timeEnd()

// 这个过程仅仅2.7毫秒 时间复杂度 O(1)
console.time()
rodate2(arrHuge, step)
console.timeEnd()

数据结构

栈的应用

判断是否匹配 夸号类型

/**
 * descripbe: 算法 {a[b(c)d]e}
 * 判断是否匹配 夸号类型
 */
const isMatch = (top: string, s: string): boolean => {
    if (top === "{" && s === '}') return true
    if (top === "(" && s === ')') return true
    if (top === "[" && s === ']') return true
    return false
}

export const matchFunc = (str: string) => {
    let length = str.length
    if (length <= 0) return false
    let leftSyb = '{[('
    let rightSyb = '}])'
    let stack = []

    for (let i = 0; i < length; i++) {
        let s = str[i]
        if (leftSyb.includes(s)) {
        	// 碰到左边的字符串先入栈
            stack.push(s)
        } else if (rightSyb.includes(s)) {
        	// 碰到右边的字符串先判断是否和顶层的匹配
            let top = stack[stack.length - 1]

            if (isMatch(top, s)) {      
            	// 匹配的话,就删除顶层
                stack.pop()
            } else {
                return false
            }
        }
    }

    return stack.length === 0
}

let str1 = '{1[2(3)4]5}' // 匹配
let str2 = '{1[2(34]5}19)' // 顺序不一样
console.log(matchFunc(str1),matchFunc(str2), // true, false
找数组中两个字符串相加为10的值

O(n^2)的方式两个循环如下:

/**
 * desc: 找数组中两个字符串相加为10的一个值
 */
export const twoNumberAdd = (arr: number[], total: number): number[] => {
    if(arr.length === 0) return []
    if(total === 0) return []
    let length = arr.length
    const res: number[] = []
    let flag = false

    for (let i = 0; i < length - 1; i++) {
        for (let j = i; j < length - 1; j++) {
            if (arr[i] + arr[j] == total) {
                res.push(arr[i])
                res.push(arr[j])
                flag = true
                break;
            }
        }
        if (flag) break // for循环中,break可以停止循环。
    }
    return res
}

const arrlog = [1,2,3,4,5,6,8]
const res = twoNumberAdd(arrlog, 10)
console.log(res, 'res');

二分理念去查,时间复杂度 O(N)

export const twoNumberAdd2 = (arr: number[], total: number): number[] => {
    if(arr.length === 0) return []
    if(total === 0) return []
    const res: number[] = []
    let i = 0
    let j = arr.length - 1

    while (i < j) {
        let a = arr[i]
        let b = arr[j]
        let t = a + b

        if (t > total) {
            j--
        } else if (t < total) {
            i++
        } else {
            res.push(a)
            res.push(b)
            break
        }
    }

    return res
}


const arrlog = [1, 2, 3, 4, 5, 6, 8]
const res = twoNumberAdd2(arrlog, 10)
console.log(res, 'res');

时间复杂度对比,可以看出,二分法的时间复杂度更低。选第二个方法。

const arrlog = [1, 2, 1, 2, 1, 2, 1, 2, 1, 2, 1, 2, 1, 2, 1, 2, 1, 2, 1, 2, 1, 2, 1, 2, 3, 4, 5, 6, 8]
console.time()
for (let i = 0; i < 100 * 10000; i++) {
    twoNumberAdd(arrlog, 10)
}
console.timeEnd()


console.time()
for (let i = 0; i < 100 * 10000; i++) {
    twoNumberAdd2(arrlog, 10)
}
console.timeEnd()

// default: 288.071044921875 ms
// default: 14.84619140625 ms

链表

js实现单向链表结构


// 实现一个单项链表功能
// 变成 {value: '100', next: { value: "200", next: { value: '300', next: { value: '400' } } }}
const arrInit = [100, 200, 300, 400, 500]
const func = (arr) => {
  let length = arr.length
  if (length <= 0) return [] 
  
  let curNode = {
    value: arr[length-1],
  }
  for (let i = length-2; i >= 0; i--) {
    console.log(i, 'i');
    curNode = {
      value: arr[i],
      next: curNode
    }
  }
  return curNode
}

console.log(func(arrInit), 'arr')

实现反转单向链表算法

/**
 * {value: 100, next: { value: 200, next: { value: 300, next: { value: 400, next: { value: 500 } } } }}
 * 处理成为下面结构
 * {value: 500, next: { value: 400, next: { value: 300, next: { value: 200, next: { value: 100 } } } }}
 * @param arr 反转单向链表结构结构
 */
interface INode {
    value: number,
    next?: INode
}
export const reverseLinkNode = (linkNode: INode) => {
    // 定义三个指针用于接收
    let prevNode: INode | undefined = undefined
    let curNode: INode | undefined = undefined
    let nextNode: INode | undefined = linkNode
    
    // 循环赋值引用
    while(nextNode){
        console.log('1');
        
        // 第一种情况
        if (curNode && !prevNode) {
            delete curNode.next
        }

        // 第二种和倒数第二种情况
        if (curNode && prevNode) {
            curNode.next = prevNode
        }

        prevNode = curNode
        curNode = nextNode
        // @ts-ignore
        nextNode = nextNode.next
    }

    
    curNode!.next = prevNode

    
    return curNode
}

const list = {value: 100, next: { value: 200, next: { value: 300, next: { value: 400, next: { value: 500 } } } }}
let list2 = reverseLinkNode(list)
console.log(list2, 'list2');

链表中,查询慢删除快
数组中,查询快删除慢。

队列

用两个栈实现队列(数组方式)

/**
 * 两个栈实现队列
 */
export class MyQuene {
    constructor(stack1: number[]){
        this.stack1 = stack1
    }
    stack1: number[] = []
    stack2: number[] = []

    add(n: number) {
        this.stack1.push(n)
    }
    delete(): number | null {
        let stack1 = this.stack1
        let stack2 = this.stack2

        // 1. 压栈处理成stack2
        while(stack1.length){
            let n = stack1.pop()
            if (n) stack2.push(n)
        }
        
        // 2. stack2.pop
        let deleteItem = stack2.pop()
        // 3. stack2 压栈成stack1
        while(stack2.length){
            let n = stack2.pop()
            if(n) stack1.push(n)
        }
        
        return deleteItem as number
    }
    get length() {
        return this.stack1.length
    }
}

const q = new MyQuene([1,2,3,4,5])
let dItem = q.delete()
console.log(dItem, q.stack1); // 1, [2,3,4,5]
q.add(6)						
console.log('add', q.stack1); // add, [2,3,4,5,6]
console.log(q.length);        // 5

大致思路如下:
首先有两个栈,一个栈用于存初始值。一个用于转换的时候用,stack1压栈成stack2,顺序改变,A在首部变成尾部,直接pop删除最后一个元素,再压栈成stack1 得到的就是 删掉了A的 BCDE。

一些数据量庞大的场景中,由于数组unshiftshift消耗的性能比较大时间复杂度(O(n^2)),使用栈的概念,可以通过 两个栈+队列 的方式完成数组的unshift操作。

用链表实现队列

/**
 * 链表实现队列
 */
interface INode {
    value: number,
    next: INode | null
}
export class MyQuene {
    head: INode | null = null
    tail: INode | null = null
    len: number = 0

    add(n: number) {
        let newNode: INode = {
            value: n,
            next: null
        }

        // 处理head
        if (this.head == null) {
            this.head = newNode
        }

        // // 处理tail
        let tailNode = this.tail
        if (tailNode) {
            // 第二次进入已经新add了一个newNode,可以指定next为新节点
            tailNode.next = newNode
        }
        this.tail = newNode

        this.len++
    }

    delete(): number | null {
        const headNode = this.head
        if (this.len == 0) return null
        if (this.head == null) return null

        let value = headNode!.value
        
        // 处理head
        this.head = headNode!.next

        this.len--
        
        return value

    }
    get length(): number {
        return this.len
    }
}
数组和链表哪个更快?
  • 数组是连续存储,push很快,shift很慢
  • 链表是非连续存储,add和delete都很快(查找很慢)
  • 所以链表实现队列更快

二叉树结构

题目:给一个有序的二叉树结构,找到第k位的树
/**
 * 二叉搜索树中的第k位
 */
interface INode {
    value: number,
    left: INode | null,
    right: INode | null
}
const bst: INode = {
    value: 5,
    left: {
        left: {
            value: 2,
            left: null,
            right: null
        },
        right: {
            value: 4,
            left: null,
            right: null
        },
        value: 3
    },
    right: {
        left: {
            value: 6,
            left: null,
            right: null
        },
        right: {
            value: 8,
            left: null,
            right: null
        },
        value: 7
    },
}


let arr: number[] = []

/**
 * 前序遍历
 * Preorder traversal
 */
const preorderTraversal = (node: INode | null) => {
    if (!node) return
    if (!node) return
    arr.push(node.value)
    preorderTraversal(node.left)
    preorderTraversal(node.right)
}

/**
 * 中序遍历
 * Preorder traversal
 */
const inorderTraversal = (node: INode | null) => {
    if (node === null) return
    inorderTraversal(node.left)
    arr.push(node.value)
    inorderTraversal(node.right)
}

/**
 * 后序遍历
 * Preorder traversal
 */
const postorderTraversal = (node: INode | null) => {
    if (!node) return
    if (!node) return
    postorderTraversal(node.left)
    postorderTraversal(node.right)
    arr.push(node.value)
}


export const biranyTreeSearch1 = (node: INode, k: number): number | null=> {
    inorderTraversal(node)

    return arr[k - 1] || null
}

console.log(biranyTreeSearch1(bst, 3));

单元测试
一般react-create-app有集成jest包,可以直接通过 yarn test 运行
describe:描述内容
it: 提示内容
expect: 期望的方法
toBe: 期望得到某个数字
toBeNull: 期望得到null

import { INode, biranyTreeSearch1, bst } from './index'

describe('二叉搜索树求key位的值', () => {
    it('普通情况', () => {
        expect(biranyTreeSearch1(bst, 3)).toBe(4)
    });
    it('为0的情况', () => {
        expect(biranyTreeSearch1(bst, 0)).toBeNull()
    });
})

// Test Suites: 1 passed, 1 total
// Tests:       2 passed, 2 total

三种遍历

  • 前序遍历
  • 中序遍历
  • 后序遍历
重点:
  • 二叉树和三种遍历
  • 二叉搜索树的特点:left <= root;right >= root
  • 二分搜索树的价值:可使用二分法进行快速查找

算法三大规则

  • 贪心
  • 二分
  • 动态规划
斐波拉切数列 找第 N 位

递归方式
这种方式 时间复杂度(O(n^2))

export function fiboracheSequence1(n: number): number {
    if (n <= 0) return 0
    if (n === 1) return 1

    return fiboracheSequence1(n - 1) + fiboracheSequence1(n - 2)
}

// 测试
console.log(fiboracheSequence1(8)) // 21

循环的方式

export function fiboracheSequence2(n: number): number {
    if (n <= 0) return 0
    if (n === 1) return 1

    let n1 = 1
    let n2 = 0
    let res = 0

    for (let i = 2; i < n; i++) {
        res = n1 + n2

        n2 = n1
        n1 = res
    }

    return res 
}
移动数组中的0到末尾

O(n^2) 不好用

export const moveNumber1 = (arr: number[]) => {
    if (arr.length === 0) return []

    let length = arr.length
    let blLen = 0
    for (let i = 0; i < length - blLen; i++) {
        if (arr[i] === 0) {
            
            arr.push(0)
            arr.splice(i, 1)
            i--
            blLen ++
        }
    }

    return arr
}

let initArr = [1, 0, 1, 0, 1, 0, 234, 1, 34521, 0, 1, 90, 0]
moveNumber1(initArr)
console.log(initArr, 'arr')

O(n)时间复杂度

// O(n)的时间复杂度
export const moveNumber2 = (arr: number[]) => {
    let i, j = -1, length = arr.length

    for (i = 0; i < length; i++){
        if (arr[i] === 0) {
            if (j < 0) {
                j = i
            }
        }
        if(arr[i] !== 0 && arr[j] >= 0){
            // 交换
            const n = arr[i]
            arr[i] = arr[j]
            arr[j] = n
            
            j++
        }
    }
}

let initArr = [1, 0, 1, 0, 1, 0, 234, 1, 34521, 0, 1, 90, 0]
moveNumber2(initArr)
console.log(initArr, 'arr')
字符串中连续最多的字符,以及次数

O(n)

// abcdddddfffg
interface IRes {
    char: string,
    length: number
}
export const findStr = (str: string) => {
    let length = str.length
    let res: IRes = {
        char: '',
        length: 0
    }
    let temLength

    for (let i = 0; i < length; i++) { 
        console.log('---------------');
        
        temLength = 0
        for (let j = i; j < length; j++) {
            console.log(`i:${i}\nj:${j}\ntemp:${temLength}\nres.length:${res.length}\nres.char:${res.char}`);
            if (str[i] === str[j]) {
                temLength++
            }
            if (str[i] !== str[j] || j === length - 1) {
                if (temLength > res.length) {
                    res.length = temLength
                    res.char = str[i]
                }
                if (i < length - 1) {
                    i = j - 1
                }

                break;
            }
        }
    }

    return res

}

let str = 'abcdddefg'
console.log(findStr(str));

O(n)的时间复杂度

思路: 通过判断当前元素和上一个元素是否相等,来决定要不要累加。


export const findStr2 = (str: string): IRes => {
    let length = str.length
    let res: IRes = {
        char: '',
        length: 0
    }
    if (length === 0) return res

    let tempLength = 1
    for (let i = 0; i < length - 1; i++){
        if (str[i] === str[i+1]) {
            tempLength++
        }else if(str[i] !== str[i+1]){
            if (tempLength > res.length) {
                res = {
                    char: str[i],
                    length: tempLength
                }
                tempLength = 1
            }
        }
    }

    return res
}

let str = 'abcdddeeeeeeefffg'
console.log(findStr2(str)); // char: e, length: 7

双指针


export const findStr3 = (str: string): IRes => {
    let length = str.length
    let res: IRes = {
        char: '',
        length: 0
    }
    if (length === 0) return res

    let tempLength = 0
    let i = 0,j = 0
    for (; i < length; i++){
        if (str[i] === str[j]) {
            tempLength++
        }else if(str[i] !== str[j] || i === length - 1){
            if (tempLength > res.length) {
                res = {
                    char: str[j],
                    length: tempLength
                }
            }
            tempLength = 0
            if (i < length - 1){
                j = i
                i --
            }
        }
    }
    

    return res
}

单元测试

import { findStr1, findStr2 } from './index'

describe('寻找重复的字串第一种方法:跳步', () => {
    it('普通情况', () => {
        expect(findStr1('abcdddedff')).toEqual({char: 'd', length: 3})
    })
    it('都是连续字符', () => {
        expect(findStr1('dddeeeeddd')).toEqual({char: 'e', length: 4})
    })
    it('字符串为空', () => {
        expect(findStr1('')).toEqual({char: '', length: 0})
    })
    it('无连续字符', () => {
        expect(findStr1('abcdefghijk')).toEqual({char: 'a', length: 1})
    })
})


describe('寻找重复的字串第二种方法:一次循环判断是否为相同字符', () => {
    it('普通情况', () => {
        expect(findStr2('abcdddedff')).toEqual({char: 'd', length: 3})
    })
    it('都是连续字符', () => {
        expect(findStr2('dddeeeeddd')).toEqual({char: 'e', length: 4})
    })
    it('字符串为空', () => {
        expect(findStr2('')).toEqual({char: '', length: 0})
    })
    it('无连续字符', () => {
        expect(findStr2('abcdefghijk')).toEqual({char: 'a', length: 1})
    })
})


求回文数(aba 121 454)
// 求一个范围内的回文数 转字符串 转数字判断是否相等
export const getPalindromeNumberFunc1 = (max: number): number[] => {
    let res: number[] = []
    if (max <= 0) return []

    for (let i = 1; i <= max; i++) {
        let n = i.toString()
        if (n === n.split('').reverse().join('')) {
            res.push(i)
        }
    }

    return res
}

// 求一个范围内的回文数 依次判断首位和末尾是否一直相等
export const getPalindromeNumberFunc2 = (max: number): number[] => {
    let res: number[] = []
    if (max <= 0) return []

    for (let i = 1; i <= max; i++) {
        let n = i.toString()
        let startIndex = 0
        let endIndex = n.length - 1
        let flag = true

        while (startIndex < endIndex) {
            if (n[startIndex] === n[endIndex]) {
                startIndex++
                endIndex--
            } else {
                flag = false
                break
            }
        }

        if (flag) {
            res.push(i)
        }
    }

    return res
}

// 求一个范围内的回文数 反转数字
export const getPalindromeNumberFunc3 = (max: number): number[] => {
    let res: number[] = []
    if (max <= 0) return []

    for (let i = 1; i <= max; i++) {
        let n = i
        let rev = 0

        while (n > 0) {
            rev = rev * 10 + n % 10
            n = Math.floor(n / 10)
        }

        if (rev === i) res.push(i)
    }

    return res
}

console.log(getPalindromeNumberFunc2(500));

测试效果:方法1:400ms;方法二:50ms;方法三:42ms
时间复杂度分析:方法一最慢,因为数组转换需要时间

数字转成千分位的字符串

export const numberToStr = (num: number): string => {
    
    let res = ''
    let str = num.toString()
    let length = str.length
    let times = 0

    for (let i = length - 1; i >= 0; i--) {
        times ++
        
        if (times == 3 && i !== 0) {
            res = ',' + str[i] + res
            times = 0
        }else{
            res = str[i] + res
        }
    }
    return res
}

console.log(numberToStr(13880000000));

大小写转换
// 正则表达式大小写转换
export const toggleCase1 = (str: string): string => {
    let res = ''
    let length = str.length

    for (let i = 0;i < length; i++){
        
        let s = str[i]
        let reg1 = /[a-z]/
        let reg2 = /[A-Z]/
        
        if (reg1.test(s)) {
            res = res + s.toUpperCase()
        }else if(reg2.test(s)){
            res = res + s.toLowerCase()
        }else {
            res = res + s
        }
    }
    return res
}

// 通过ASCII编码
export const toggleCase2 = (str: string): string => {
    let res = ''
    let length = str.length

    for (let i = 0;i < length; i++){
        let s = str[i]
        let code = s.charCodeAt(0)

        if (code >= 65 && code <= 90) {
            res = res + s.toLowerCase()
        }else if(code >= 97 && code <= 122){
            res = res + s.toUpperCase()
        }else {
            res = res + s
        }
    }
    
    return res
}

console.log(toggleCase2('avheDF!DsadSFEWF'));
0.1 + 0.2 !== 0.3

整数转换二进制没有误差
有些小数可能是无法用二进制精准转化
各个计算机语言的通病
通过math.js 进行准确的计算

跨域问题的解决

  • 前端解决
    • Vue中可以修改proxy代理
    • React
      • 通过package.json修改proxy属性;
      • 通过下载http-proxy-middleware ,修改setUp.js文件
      • 通过npm run eject 暴露配置修改webpack配置,不推荐。
      • 普通的html页面使用jsonp也可以实现,需要和后端规定一些标准。
    • 后端通过修改跨域头解决
    • 小程序可以通过添加白名单解决

This指向问题

- 在function中,this的指向是指调用该方法的指针本身。
- 在箭头函数中,this指向是当前的作用域
- class的class component 指向一般都是指当前组件,有一些情况,class组件的方法没有bind this 指针,需要手动指定。如果不指定,可能会导致方法调用不到的问题。

原型作用域的相关问题

  • 学习链接戳我:
    • 原型
      • 在我们创建函数的时候,会自动生成一个prototype属性,我们函数需要的所有属性和方法都绑定在这个prototype上,这个属性指向一个对象,也就是原型对象。
      • 原型对象里面拥有constructor,这个对象指向的是他的构造函数。比如用new 出来的对象下的prototype下面的consturctor会与函数相等,他们都是一个构造函数。他们指向的都是该构造函数本身
    • 原型特点:
      • 实例中的属性和方法会共享。
      • new出来的实例。如果set了同名属性,将会覆盖原本原型链上的属性。如果在实例外获取不到,会在原型链上面查找。
      • 如果碰到需求,需要重构原型函数,需要重写该函数时,需要将constructor再次赋值给原型琏上的实例方法。否则会导致方法没法完全相等的问题。
      • 举例:
         function Person(){}
         var p = new Person();
         // 这里的重写就是通过修改propotype来重写
         Person.prototype = {
          name: 'tt',
           age: 18
         }
         Person.prototype.constructor === Person // false 
         p.name // undefined 
        
        优化:
        Person.prototype = {
           constructor: Person,
           name: 'tt',
           age: 18
        }
        
    • 原型链
      • js中的继承是通过原型链的方式来的,例如声明了一个新的对象或者函数,他们的原型prototype也是一个原型对象,在他之上也有他的原型对象,于是层层链入,形成了一个类似链表的结构,就被称之为原型链。
      • 所有原型对象的终点都是Object的prototype对象。
      • Object的原型对象是null,null之上没有原型。
        原型链图示
    • 原型链的问题
      • 引用问题
      • 原型链上面如果是一个引用类型(Object Array),多个实例同时修改原型属性的值,所有的实例都会收到影响二发生改变。
      • 例子;这里不管如何修改arr的数据,其他引用arr的地方都会改变。
      function Person(){}
      Person.prototype.arr = [1, 2, 3, 4];
      var person1 = new Person();
      var person2 = new Person();
      person1.arr.push(5) 
      person2.arr // [1, 2, 3, 4, 5]
      
    • class和原型链的关系:
      • 本质上class语法只是es6的语法糖,构建工具将class语法构建成了原型对象的写法。例如constructor中的代码就是函数内部方法,class中的方法也就是挂载到了prototype上去。
      • JavaScript是基于原型的
      • 图解
      • 在这里插入图片描述

宏任务和微任务

  • 干货
    • http://events.jianshu.io/p/fd15db94a034
      总结:
      常见的宏任务:script语句 setTimeout setInterval setImmedate promise
      微任务:Promise.[ then/catch/finally ] node中的process.nextTick()
    • 一个任务开始的时候先执行srcipt 第一个宏任务,其中碰到了其他宏任务(setTimeout),先放入宏任务队列。碰到了微任务(.then),放入微任务队列,以此类推。
    • 接着执行完宏任务的结尾,微任务开始按顺序执行,
    • 再回来执行第二个宏任务(setTimeout的回调)以此类推。

函数的节流和防抖

  • https://gitee.com/HeiGes/net-case-must/blob/master/clientNetCaseMusic/README.md#%E9%98%B2%E6%8A%96%E5%92%8C%E8%8A%82%E6%B5%81
    • 为了解决一些场景连续触发一个函数,或者请求,需要通过防抖节流将方法减少运行。

class生命周期

  • https://blog.csdn.net/qq_43013884/article/details/125421659
  • 初始化
    • constructor
    • getDeriviedStateFormProps
    • render
    • componentDidMount
  • 更新时
    • getDeriviedStateFormProps
    • shouComponentUpdate
    • render
    • getSnapShotBeforeUpdate
    • componentDidUpdate
  • 卸载时
    • componentWillUnMount

一个前端大佬的文章

https://blog.csdn.net/qq_34998786/article/details/118887109?spm=1001.2101.3001.6650.1&utm_medium=distribute.pc_relevant.none-task-blog-2%7Edefault%7ECTRLIST%7Edefault-1-118887109-blog-122952429.pc_relevant_multi_platform_whitelistv3&depth_1-utm_source=distribute.pc_relevant.none-task-blog-2%7Edefault%7ECTRLIST%7Edefault-1-118887109-blog-122952429.pc_relevant_multi_platform_whitelistv3&utm_relevant_index=2

浏览器对于小数单位是怎么计算的?

interface和type的区别是什么?你日常工作中是用interface还是type?

快速理解 TypeScript 的逆变、协变、双向协变、不变

https://zhuanlan.zhihu.com/p/500762226

能不能讲讲小程序的原理?

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

微信小程序裁剪图片成圆形

概述 分装的图片剪裁组件&#xff0c;可以把图片剪裁成圆形&#xff0c;主要思路就是使用canvas绘图&#xff0c;把剪裁的图片绘制成圆形&#xff0c;另外剪裁图片的窗口还可以移动放大缩小 详细 前言 最近在开发小程序&#xff0c;产品经理提了一个需求&#xff0c;要求微…

Java Web项目中spring.xml或springmvc.xml配置文件中出现名称空间爆红

在做项目的过程中&#xff0c;通常需要配置spring.xml或者springmvc.xml等配置文件&#xff0c;但是会出现名称空间爆红的情况&#xff0c;如下所示&#xff1a; 解决方法&#xff1a;打开Settings ——> Schemas and DTDs 然后添加爆红的语句即可

PSP - 蛋白质结构预测 AlphaFold2 的结构模版 (Template) 搜索与特征逻辑

欢迎关注我的CSDN&#xff1a;https://spike.blog.csdn.net/ 本文地址&#xff1a;https://spike.blog.csdn.net/article/details/132427617 结构模版 (Template) 是一种已知的蛋白质结构&#xff0c;可以作为 AlphaFold2 蛋白质结构预测的参考&#xff0c;AlphaFold2 可以从多…

构建系统自动化-autoreconf

autoreconf简介 autoreconf是一个GNU Autotools工具集中的一个命令&#xff0c;用于自动重新生成构建系统的配置脚本和相关文件。 Autotools是一组用于自动化构建系统的工具&#xff0c;包括Autoconf、Automake和Libtool。它们通常用于跨平台的软件项目&#xff0c;以便在不同…

AIR001开箱测试

最近&#xff0c;合宙的动作还是挺大的&#xff0c;又出了两款AIR001和RP2040&#xff0c;而且前段时间还出了AIR32F103系列&#xff0c;记的21年要采购STM32F1103的时候&#xff0c;1片的价格从开发时的5块涨到了生产阶段的100多&#xff0c;即使最后无奈采用了别的芯片&#…

容器化微服务:用Kubernetes实现弹性部署

随着云计算的迅猛发展&#xff0c;容器化和微服务架构成为了构建现代应用的重要方式。而在这个过程中&#xff0c;Kubernetes&#xff08;常简称为K8s&#xff09;作为一个开源的容器编排平台&#xff0c;正在引领着容器化微服务的部署和管理革命。本文将深入探讨容器化微服务的…

LeetCode150道面试经典题-- 二叉树的最大深度(简单)

1.题目 给定一个二叉树 root &#xff0c;返回其最大深度。 二叉树的 最大深度 是指从根节点到最远叶子节点的最长路径上的节点数。 2.示例 3.思路 深度优先遍历 一个二叉树要查询到最大深度&#xff0c;可以将问题转为从根节点出发&#xff0c;查看左右子树的最大深度&am…

php 系列题目,包含查看后端源代码

一、弱类型比较问题 原则&#xff1a; 1.字符串和数字比较&#xff0c;字符串回被转换成数字。 "admin" 0&#xff08;true) admin被转换成数字&#xff0c;由于admin是字符串&#xff0c;转换失败&#xff0c;变成0 int(admin)0,所以比较结果是ture 2.混合字符串转…

opencv 进阶17-使用K最近邻和比率检验过滤匹配(图像匹配)

K最近邻&#xff08;K-Nearest Neighbors&#xff0c;简称KNN&#xff09;和比率检验&#xff08;Ratio Test&#xff09;是在计算机视觉中用于特征匹配的常见技术。它们通常与特征描述子&#xff08;例如SIFT、SURF、ORB等&#xff09;一起使用&#xff0c;以在图像中找到相似…

【C#学习笔记】匿名函数和lambda表达式

文章目录 匿名函数匿名函数的定义匿名函数作为参数传递匿名函数的缺点 lambda表达式什么是lambda表达式闭包 匿名函数 为什么我们要使用匿名函数&#xff1f;匿名函数存在的意义是为了简化一些函数的定义&#xff0c;特别是那些定义了之后只会被调用一次的函数&#xff0c;与其…

【Unity3D赛车游戏制作】初步导入,资源很哇塞【一】

&#x1f468;‍&#x1f4bb;个人主页&#xff1a;元宇宙-秩沅 &#x1f468;‍&#x1f4bb; hallo 欢迎 点赞&#x1f44d; 收藏⭐ 留言&#x1f4dd; 加关注✅! &#x1f468;‍&#x1f4bb; 本文由 秩沅 原创 &#x1f468;‍&#x1f4bb; 收录于专栏&#xff1a;Uni…

位运算相关总结

371. 两整数之和 给你两个整数 a 和 b &#xff0c;不使用 运算符 和 - ​​​​​​​&#xff0c;计算并返回两整数之和。 class Solution { public:int getSum(int a, int b) {while (b ! 0) {// 计算进位&#xff1a;将 a 和 b 进行位与操作&#xff0c;然后左移 1 位。u…

ExoPlayer如何使用MediaExtractor的思路

本文主要针对于&#xff0c;自己的设备解码能力&#xff08;比如底层集成ffmpeg 、qti、android 、需要付费的格式等等&#xff09;大于ExoPlayer自己封装的固有Extractor&#xff0c;基于现在Android架构通俗的来说&#xff0c;就是MediaPlayer可以播&#xff0c;但是ExoPlaye…

解码客厅:知名设计师带你探索其历史与设计风格

会客厅又称接待室&#xff0c;&#xff0c;它们是宾客和家人享受下午或晚上娱乐时光的天然聚会场所。由于会客厅反映了每个家庭的个性&#xff0c;因此在家具和设计上花费了很多心思。装饰品、复古艺术品、三角钢琴以及雕塑和花瓶等其他装饰元素在今天的会客厅中已司空见惯。 下…

UE4 地形编辑基础知识 学习笔记

之前自己写过这样的功能&#xff0c;今天看到一个UE现成的 点击地形&#xff0c;选择样条 按住CTRL键点击屏幕中某一个点会在场景内生成一个这样的图标 再点两次&#xff0c;会生成B样条的绿线条 点击号再选择一个模型&#xff0c;会生成对应的链条状的mesh 拉高最远处的一个图…

2023国赛数学建模思路 - 案例:粒子群算法

文章目录 1 什么是粒子群算法&#xff1f;2 举个例子3 还是一个例子算法流程算法实现建模资料 # 0 赛题思路 &#xff08;赛题出来以后第一时间在CSDN分享&#xff09; https://blog.csdn.net/dc_sinor?typeblog 1 什么是粒子群算法&#xff1f; 粒子群算法&#xff08;Pa…

无涯教程-PHP - preg_split()函数

preg_split() - 语法 array preg_split (string pattern, string string [, int limit [, int flags]]); preg_split()函数的操作与split()完全相同&#xff0c;只不过正则表达式被接受为pattern的输入参数。 如果指定了可选的输入参数limit&#xff0c;则仅返回子字符串的限…

测试框架pytest教程(11)-pytestAPI

常量 pytest.__version__ #输出pytest版本 pytest.version_tuple #输出版本的元组形式 功能 pytest.approx pytest.approx 是一个用于进行数值近似比较的 pytest 断言工具。 在测试中&#xff0c;有时候需要对浮点数或其他具有小数部分的数值进行比较。然而&#xff0c;由于…

正中优配:散户也算股东吗?能不能参加股东大会?

股市上&#xff0c;常有散户自称韭菜&#xff0c;长一波被股市收割一波&#xff0c;不可谓不惨。但要较真的话&#xff0c;散户仍是有必定身份的&#xff0c;最少他是所持股的股东。有人会好奇&#xff0c;原来散户也算股东吗&#xff1f;那他能不能参与公司的股东大会&#xf…

一篇文章教你自动化测试如何解析excel文件?

前言 自动化测试中我们存放数据无非是使用文件或者数据库&#xff0c;那么文件可以是csv&#xff0c;xlsx&#xff0c;xml&#xff0c;甚至是txt文件&#xff0c;通常excel文件往往是我们的首选&#xff0c;无论是编写测试用例还是存放测试数据&#xff0c;excel都是很方便的。…