跳至主要內容

从1到n整数中1出现的次数


从1到n整数中1出现的次数

题目链接

题目描述

输入一个整数 n ,求 1~n 这 n 个整数的十进制表示中 1 出现的次数
例如, 1~13 中包含 1 的数字有 1 、 10 、 11 、 12 、 13 因此共出现 6 次

注意:11 这种情况算两次

数据范围:1≤n≤30000
进阶:空间复杂度O(1) ,时间复杂度O(lognn)

刷题思路

方案一:

  • 转化为字符串,再遍历计数,属于投机方法

方案二:

  • 数学方法,对进位、余数进行处理

代码实现

/*
 * @Description:第一个只出现一次的字符
 * @Version: Beta1.0
 * @Author: 微信公众号:储凡
 * @Date: 2021-04-28 22:23:51
 * @LastEditors: 微信公众号:储凡
 * @LastEditTime: 2021-04-28 22:24:20
 */

/**
 * 利用indexOf和lastIndexOf角标不一致
 */
function firstNotRepeatingCharOne(str) {
  const arr = str.split('')
  for (let index = 0; index < arr.length; index++) {
    if (arr.indexOf(arr[index]) === arr.lastIndexOf(arr[index])) {
      return index
    }
  }
  return -1
}

/**
 * 数组按字母查找
 */
function firstNotRepeatingCharTwo(str) {
  const len = str.length
  for (let index = 0; index < len - 1; index++) {
    const s = str.slice(index, index + 1)
    const remainStr = `${str.slice(0, index)}${str.slice(index + 1)}`
    if (!remainStr.includes(s)) {
      return index
    }
  }
  return -1
}

/**
 * 使用Map结构计数
 */
function firstNotRepeatingCharThree(str) {
  const resMap = new Map()
  const resArr = str.split('')

  // 计数操作
  resArr.forEach((r) => {
    if (resMap.has(r)) {
      resMap.set(r, resMap.get(r) + 1)
    }
    else {
      resMap.set(r, 1)
    }
  })

  for (const [key, value] of resMap) {
    if (value === 1) {
      return str.indexOf(key)
    }
  }
  return -1
}

console.log(firstNotRepeatingCharOne('google'))
console.log(firstNotRepeatingCharTwo('google'))
console.log(firstNotRepeatingCharThree('google'))

一些建议

  • 需要特别注意Math的一些操作方法
// 取整,丢弃小数部分,保留整数部分,输出2
Number.parseInt(5 / 2)

// 向上取整,有小数就整数部分加1,输出3
Math.ceil(5 / 2)

// 向下取整,丢弃小数部分,输出2
Math.floor(5 / 2)

// 四舍五入,输出3
Math.round(5 / 2)

// 取余,输出2
6 % 4