JS實現——用3L和5L量出4L的水

把如下代碼保存成donglanguage.html文件,使用Google或360瀏覽器打開html

<!DOCTYPE html>
<html>

<head>
<meta charset="utf-8">
<title>donglanguage</title>
<style>
</style>
</head>

<body>

<div style="position:absolute;top:20px;left:10px;width:200px;height:550px;">
  <div style="position:relative;top:200px;z-index:999">a 3L</div>
  <div id="a" style="position:absolute;top:200px;width:100px;height:300px;background-color:white;border:1px solid blue">
    <div id="water" style="position:relative;top:300px;height:0%;background-color:blue"></div>
  </div>
  <button name="fill" style="position:relative;top:520px">裝滿水</button>
  <button name="empty"  style="position:relative;top:520px">倒空水</button>
  <button name="toFill"  style="position:relative;top:520px">倒滿水到另外一個桶</button>
</div>

<div style="position:absolute;top:20px;left:220px;width:200px;height:550px;">
  <div style="position:relative;top:0px;z-index:999">b 5L</div>
  <div id="b" style="position:absolute;top:0px;width:100px;height:500px;background-color:white;border:1px solid blue">
    <div style="position:relative;top:500px;height:0%;background-color:blue"></div>
  </div>
  <button name="fill" style="position:relative;top:520px">裝滿水</button>
  <button name="empty" style="position:relative;top:520px">倒空水</button>
  <button name="toFill" style="position:relative;top:520px">倒滿水到另外一個桶</button>
</div>

<div style="position:relative;top:10px;left:500px;">
  問題:一個3L的捅,還有一個5L的桶,水無限。怎麼操做才能準確地弄出4L的水?
  <button id="a_solution">我不會,看看解法</button>
</div>
<div style="position:relative;left:500px;top:40px;">
  <textarea id="program" style="font-size:20px;" rows="20" cols="40"></textarea>
  <button id="run">運行</button>
</div>

</body>


<script>

function $(selector) {
  var c = selector[0];
  if(c == '#') {
    var id = selector.substring(1);
    return document.getElementById(id);
  } else if(c == '@') {
    var name = selector.substring(1);
    return document.getElementsByName(name);
  } else {
    return document.getElementsByTagName(selector);
  }
};

//onload start
onload = function() {
$('#run').onclick = function() {
  var code = $('#program').value;
  run(code);
}

$('#a_solution').onclick = function() {
  var program = 'a.empty()\nb.empty()\n\n' +
  'b.fill()\nb.toFill(a)\na.empty()\nb.toFill(a)\nb.fill()\nb.toFill(a)';
  $('#program').value = program;
  run(program);
}

function fillClick() {
  run(this.parentNode.children[1].id + '.fill()');
}

function emptyClick() {
  run(this.parentNode.children[1].id + '.empty()');
}

function toFillClick() {
  var thisId = this.parentNode.children[1].id;
  var from, to;
  if(thisId == 'a') {
    from = 'a';
    to = 'b';
  } else if(thisId == 'b') {
    from = 'b';
    to = 'a';
  }
  run(from + '.toFill(' + to + ')');
}

window.opBtns = $('button');
window.disableAllOpBtns = function(b) {
  for(var i = 0; i < opBtns.length; i++)
    opBtns[i].disabled = b ? 'disabled' : '';
}

for(var i = 0; i < opBtns.length; i++) {
  var btn = opBtns[i];
  switch(btn.name) {
  case 'fill':
    btn.onclick = fillClick;
    break;
  case 'empty':
    btn.onclick = emptyClick;
    break;
  case 'toFill':
    btn.onclick = toFillClick;
    break;
  }
}

window.queue = false;
window.buckets = [$('#a'), $('#b')];

};//onload end

function isEmpty(bucket) {
  return parseInt(bucket.children[0].style.top) == parseInt(bucket.style.height);
}

function isFull(bucket) {
  return parseInt(bucket.children[0].style.top) == 0;
}

function addWater(bucket, litre, zeroH, fullH) {
  queue = false;
  disableAllOpBtns(true);
  var water = bucket.children[0];
  var timer = setInterval(function() {
    if(litre == 0) {
      clearInterval(timer);
      disableAllOpBtns(false);
      if(zeroH)
        zeroH();
      queue = true;
      return;
    }
    if(isFull(bucket)) {
      clearInterval(timer);
      disableAllOpBtns(false);
      if(fullH)
        fullH();
      queue = true;
    }

    var top = parseInt(water.style.top);
    water.style.top = (top - 100) + 'px';
    water.style.height = (parseInt(bucket.style.height) - (top - 100)) + 'px';
    --litre;
  }, 100);
}

function subWater(bucket, litre, zeroH, emptyH) {
  queue = false;
  disableAllOpBtns(true);
  var water = bucket.children[0];
  var timer = setInterval(function() {
    if(isEmpty(bucket)) {
      clearInterval(timer);
      disableAllOpBtns(false);
      if(emptyH)
        emptyH();
      queue = true;
      return;
    }
    if(litre == 0) {
      clearInterval(timer);
      disableAllOpBtns(false);
      if(zeroH)
        zeroH();
      queue = true;
      return;
    }
    var top = parseInt(water.style.top);
    water.style.top = (top + 100) + 'px';
    water.style.height = (parseInt(bucket.style.height) - (top + 100)) + 'px';
    --litre;
  }, 100);
}

function fill(bucket) {
  addWater(bucket, parseInt(bucket.children[0].style.top) / 100);
}

function empty(bucket) {
  subWater(bucket, parseInt(bucket.children[0].style.height) / 100);
}

function toFill(bucketFrom, bucketTo) {
  queue = false;
  disableAllOpBtns(true);
  var waterFrom = bucketFrom.children[0];
  var waterTo = bucketTo.children[0];
  var timer = setInterval(function() {
    var waterFromTop = parseInt(waterFrom.style.top);
    var waterFromHeight = parseInt(waterFrom.style.height);
    var waterToTop = parseInt(waterTo.style.top);
    var waterToHeight = parseInt(waterTo.style.height);

    if(isFull(bucketTo) || isEmpty(bucketFrom)) {
      clearInterval(timer);
      disableAllOpBtns(false);
      queue = true;
      return;
    }
    waterFrom.style.top = (waterFromTop + 100) + 'px';
    waterFrom.style.height = (parseInt(bucketFrom.style.height) - (waterFromTop + 100)) + 'px';
    waterTo.style.top = (waterToTop - 100) + 'px';
    waterTo.style.height = (parseInt(bucketTo.style.height) - (waterToTop - 100)) + 'px';
  }, 100);
}

var isOperator = function (c) { return /[+\-*\/\^%=(),.]/.test(c); },
    isDigit = function (c) { return /[0-9]/.test(c); },
    isWhiteSpace = function (c) { return /\s/.test(c); },
    isIdentifier = function (c) { return typeof c === "string" && !isOperator(c) && !isDigit(c) && !isWhiteSpace(c); };

function lex(input) {
  var tokens = [];
  var c, i = 0;
  var advance = function () { return c = input[++i]; };
  var addToken = function (type, value) {
    tokens.push({
      type: type,
      value: value
    });
  };
  while(i < input.length) {
    c = input[i];
    if(isWhiteSpace(c))
      advance();
    if(isIdentifier(c)) {
      var id = c;
      while(isIdentifier(advance())) id += c;
      addToken("identifier", id);
    } else if(isOperator(c)) {
      addToken(c);
      advance();
    }
  }
  addToken('(end)');

  return tokens;
}

function parse(tokens) {
  var expression = function() {
    var expression = {};
    expression.type = 'call';
    expression.args = [];
    var argExpression = {};
    expression.args.push(tokens[i]);
    advance();
    advance();
    expression.name = tokens[i].value;
    advance();
    advance();
    if(tokens[i].type == 'identifier') {
      expression.args.push(tokens[i]);
      advance();
    }
    advance();
    return expression;
  }

  var parseTree = [];
  var i = 0;
  var advance = function () { return tokens[++i]; };
  while(tokens[i].type != '(end)') {
    parseTree.push(expression());
  }

  return parseTree;

}

function evaluate(parseTree) {
  var functions = {
    'fill': fill,
    'empty': empty,
    'toFill': toFill
  };

  var variables = {
    'a': buckets[0],
    'b': buckets[1]
  };

  var parseNode = function(node) {
    if(node.type == 'identifier') {
      var value = variables[node.value];
      return value;
    }else if(node.type == 'call') {
      var args = node.args;
      for(var i = 0; i < args.length; i++)
        args[i] = parseNode(args[i]);
      return functions[node.name].apply(null, args);
    }
  }

  var i = 0;
  queue = true;
  var timer = setInterval(function() {
    if(queue) {
      parseNode(parseTree[i]);
      i++;
      if(i >= parseTree.length) {
        clearInterval(timer);
      }
    }
  }, 0);

}

function run(code) {
  try {
    var tokens = lex(code);
    var parseTree = parse(tokens);
    return evaluate(parseTree);
  } catch (e) {
    return e;
  }
}
</script>

</html>

出處:qq羣--編程算法&思想 459909287node

相關文章
相關標籤/搜索