微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

JavaScript 在页面上给出 NaN 错误,但变量实际上是一个数字

如何解决JavaScript 在页面上给出 NaN 错误,但变量实际上是一个数字

我正在尝试用 JS 制作一个基本的计算器。它正在为第一步工作。我的意思是它计算 1+3 = 4 但例如它不能计算 4+2。实际上它是在控制台上计算的,但我无法在网页上显示它,所以 DOM.那么你知道为什么会这样吗?

const operations = {
    '+': (a,b) => a + b,'-': (a,b) => a - b,'*': (a,b) => a * b,'/': (a,b) => a / b
}

const doOpr = function (sign) {
    labelUser.value += `${sign}`

    // const deleteFn = function () {
    //     labelUser.value = labelUser.value.slice(0,labelUser.value.length - 1)
    // }
    // btnDelete.addEventListener('click',deleteFn)

    const equalFn = function (sign) {
        let numbers = labelUser.value.split(`${sign}`)
        //console.log(numbers)
        const operation = operations[`${sign}`]
        let result = operation(+numbers[0],+numbers[1])
        labelUser.value = result

        //console.log(typeof result,numbers) //This line proves that result is a number. Not a NaN. 
        //Also it proves numbers array contains only two elements.

        return
    }
    btnEqual.addEventListener('click',equalFn.bind(null,sign))
}

btnPlus.addEventListener('click',doOpr.bind(null,'+'))

btnExtr.addEventListener('click','-'))

const btnPlus = document.querySelector('.btn-plus')
const btnExtr = document.querySelector('.btn-extr')
const btnEqual = document.querySelector('.btn-equal')
const btnDelete = document.querySelector('.btn-delete')
const labelUser = document.querySelector('.user__input')
const form = document.querySelector('.user__form')

form.reset()
const operations = {
    '+': (a,deleteFn)

    const equalFn = function (sign) {
        let numbers = labelUser.value.split(`${sign}`)
        const operation = operations[`${sign}`]
        let result = operation(+numbers[0],+numbers[1])
        labelUser.value = result
        console.log(typeof result,numbers)
        return
    }
    btnEqual.addEventListener('click','-'))
*,*::after,*::before {
  margin: 0;
  padding: 0;
  Box-sizing: inherit;
}

html {
  font-size: 62.5%;
  Box-sizing: border-Box;
}

.container {
  position: absolute;
  top: 50%;
  left: 50%;
  transform: translate(-50%,-50%);
  width: 25%;
  height: 85%;
  background-color: olive;
  display: grid;
  grid-template-columns: repeat(4,1fr);
  grid-template-rows: repeat(6,1fr);
}

.user__form {
  grid-row: 1/3;
  grid-column: 1/-1;
  width: 100%;
  height: 100%;
  display: flex;
  justify-content: center;
  align-items: center;
}

.user__input {
  width: 90%;
  height: 80%;
  font-size: 6rem;
  text-align: right;
  padding: 0 3rem;
}
<!DOCTYPE html>
<html lang="en">

<head>
    <Meta charset="UTF-8">
    <Meta name="viewport" content="width=device-width,initial-scale=1.0">
    <title>Basic Calculator</title>
    <link rel="stylesheet" href="./css/style.css">
</head>

<body>
    <div class="container">
        <form class="user__form">
            <input type="text" class="user__input" placeholder="Entry num">
        </form>

        <button class="btn-plus">
            addition
        </button>

        <button class="btn-extr">
            extraction
        </button>

        <button class="btn-equal">
            equal
        </button>

        <button class="btn-delete">
            delete
        </button>

    </div>
    <script src="./script.js"></script>
</body>

</html>

enter image description here

在这里你可以看到,我先输入了2+1,它计算并显示出来,但是我再次尝试了同样的操作,它计算了但无法显示页面上。

解决方法

如果您:

  1. 输入“2”。
  2. 按下“添加”按钮。
  3. 输入“3”。
  4. 按下“相等”按钮。
  5. 按下“添加”按钮。
  6. 输入“4”。
  7. 按下“相等”按钮。

第 2 步将添加一个事件侦听器到等号按钮,并在单击(第 4 步)时执行添加逻辑。但是,听众仍然存在。在第 5 步,添加另一个侦听器。在第 7 步,两个事件侦听器触发:第一个将执行加法 (5 + 4 = 9),第二个将尝试通过在 + 上拆分并对两个值求和来再次执行加法。然而,该值已经只是“9”,所以这会导致一个问题,因为没有两个操作数:

const operations = {
    '+': (a,b) => a + b,}

const value = "9";

const numbers = value.split("+");
const a = +numbers[0];
const b = +numbers[1];

console.log(numbers,a,b);

console.log(operations["+"](a,b))

由于问题是多个事件侦听器,您可以制作一个在触发后自毁的事件侦听器:

const btnPlus = document.querySelector('.btn-plus')
const btnExtr = document.querySelector('.btn-extr')
const btnEqual = document.querySelector('.btn-equal')
const btnDelete = document.querySelector('.btn-delete')
const labelUser = document.querySelector('.user__input')
const form = document.querySelector('.user__form')

form.reset()
const operations = {
    '+': (a,'-': (a,b) => a - b,'*': (a,b) => a * b,'/': (a,b) => a / b
}

const doOpr = function (sign) {
    labelUser.value += `${sign}`

    const equalFn = function(sign) { //equalFN should return a function 
      return function equal() { //give the function a name --+
        let numbers = labelUser.value.split(`${sign}`) //    |
        const operation = operations[`${sign}`] //           |
        let result = operation(+numbers[0],+numbers[1]) //  |
        console.log(typeof result,result,numbers) //       |
        labelUser.value = result //                          |
        btnEqual.removeEventListener("click",equal); //<----+ use the name to remove it
        return
      }
    }
    btnEqual.addEventListener('click',equalFn(sign))
}


btnPlus.addEventListener('click',doOpr.bind(null,'+'))

btnExtr.addEventListener('click','-'))
*,*::after,*::before {
  margin: 0;
  padding: 0;
  box-sizing: inherit;
}

html {
  font-size: 62.5%;
  box-sizing: border-box;
}

.container {
  position: absolute;
  top: 50%;
  left: 50%;
  transform: translate(-50%,-50%);
  width: 25%;
  height: 85%;
  background-color: olive;
  display: grid;
  grid-template-columns: repeat(4,1fr);
  grid-template-rows: repeat(6,1fr);
}

.user__form {
  grid-row: 1/3;
  grid-column: 1/-1;
  width: 100%;
  height: 100%;
  display: flex;
  justify-content: center;
  align-items: center;
}

.user__input {
  width: 90%;
  height: 80%;
  font-size: 6rem;
  text-align: right;
  padding: 0 3rem;
}
<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width,initial-scale=1.0">
    <title>Basic Calculator</title>
    <link rel="stylesheet" href="./css/style.css">
</head>

<body>
    <div class="container">
        <form class="user__form">
            <input type="text" class="user__input" placeholder="Entry num">
        </form>

        <button class="btn-plus">
            addition
        </button>

        <button class="btn-extr">
            extraction
        </button>

        <button class="btn-equal">
            equal
        </button>

        <button class="btn-delete">
            delete
        </button>

    </div>
    <script src="./script.js"></script>
</body>

</html>

然而,拥有一次性事件监听器通常是个坏主意。这意味着正确的操作取决于操作的顺序。实际上,目前无法输入“2+3”并进行计算,您必须按“addition”按钮来注册additiion handler。

相反,这可以解耦。如果您只需要一个运算符,那么只需稍加修改,您就可以让 equals 处理程序只检查 operations 中定义的操作数。如果有,就用普通逻辑对输入进行拆分处理。

这允许您将相等运算分开,而不是在评估结果之前依赖单击“加法”或“减法”。因此,这两个按钮变得非常简单,只需在输入字段中添加“+”或“-”。但是,仅使用键盘键入“2+3”的用户也可以使用,并将被评估为 5。

const btnPlus = document.querySelector('.btn-plus')
const btnExtr = document.querySelector('.btn-extr')
const btnEqual = document.querySelector('.btn-equal')
const btnDelete = document.querySelector('.btn-delete')
const labelUser = document.querySelector('.user__input')
const form = document.querySelector('.user__form')

form.reset()
const operations = {
    '+': (a,b) => a / b
}

const doOpr = function (sign) {
    labelUser.value += `${sign}`
}

//extract equalFn as a separate handler
const equalFn = function () {
    let sign;
    //search the user input for any supported operation
    //and set the sign to that operation
    for (const op of Object.keys(operations)) {
      if (labelUser.value.includes(op)) {
          sign = op;
          break;
        }
    }
    if (sign === undefined)
      return; //cannot be calculated
    
    //use the previous logic to split the input and process it
    let numbers = labelUser.value.split(sign)
    const operation = operations[sign]
    let result = operation(+numbers[0],+numbers[1])
    console.log(typeof result,numbers)
    labelUser.value = result
    return
}

btnPlus.addEventListener('click','+'))
btnExtr.addEventListener('click','-'))

//only add equalFn a single time as a handler
btnEqual.addEventListener('click',equalFn)
*,initial-scale=1.0">
    <title>Basic Calculator</title>
    <link rel="stylesheet" href="./css/style.css">
</head>

<body>
    <div class="container">
        <form class="user__form">
            <input type="text" class="user__input" placeholder="Entry num">
        </form>

        <button class="btn-plus">
            addition
        </button>

        <button class="btn-extr">
            extraction
        </button>

        <button class="btn-equal">
            equal
        </button>

        <button class="btn-delete">
            delete
        </button>

    </div>
    <script src="./script.js"></script>
</body>

</html>

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。