the new k-drama coming next fall
// global variables are "alive" and accessible during the life of the application
var bee_gees = 'stayin alive';
// create a new function scope
function childScope(){
  
  // variables declared in here only live as long as this function block statement is executing
  // or, as long as this function is on the stack
  var fruitFly = 1;
  // returns the value of fruitFly, not a reference to the local variable
  return fruitFly;
}
// childScope is has not yet run, it is not yet on the stack
// what variables are currently "alive"?
// now invoke childScope
childScope(); // is now on the stack until it returnsare created when variables are accessed
outside of the immediate scope
var money = 55.50;
var cupsDrunk = 0;
var coffeeCost = 8.50;
function drinkCoffee(){
  if( money >= coffeeCost ){
    money -= coffeeCost;
    cupsDrunk++;
  }
}
drinkCoffee();
drinkCoffee();
drinkCoffee();
console.log( cupsDrunk );The original author's intention is to
increment cupsDrunk by calling
the drinkCoffee() function, which also depletes money
var money = 55.50;
var cupsDrunk = 0;
var coffeeCost = 8.50;
function drinkCoffee(){
  if( money >= coffeeCost ){
    money -= coffeeCost;
    cupsDrunk++;
  }
}
drinkCoffee();
drinkCoffee();
drinkCoffee();
console.log( cupsDrunk ); // 3
console.log( money ); // 30what if helpful Jason accesses cupsDrunk directly?
var money = 55.50;
var cupsDrunk = 0;
var coffeeCost = 8.50;
function drinkCoffee(){
  if( money >= coffeeCost ){
    money -= coffeeCost;
    cupsDrunk++;
  }
}
cupsDrunk++;
cupsDrunk++;
cupsDrunk++;
// still 3, looks right
console.log( cupsDrunk );var money = 55.50;
var cupsDrunk = 0;
var coffeeCost = 8.95;
function drinkCoffee(){
  if( money >= coffeeCost ){
    money -= coffeeCost;
    cupsDrunk++;
  }
}
cupsDrunk++;
cupsDrunk++;
cupsDrunk++;
console.log( money ); // 55.50 !first, move cupsDrunk into the drinkCoffee() function
so that it's not globally accessible
var money = 55.50;
var coffeeCost = 8.50;
function drinkCoffee(){
  // move the variable into the inner scope
  var cupsDrunk = 0;
  if( money >= coffeeCost ){
    money -= coffeeCost;
    cupsDrunk++;
  }
}declare a new function inside the inner scope drinkCoffee()
that has access to all variables in that scope
var money = 55.50;
var coffeeCost = 8.50;
// function expression assigned to drinkCoffee
var drinkCoffee = (function(){
  var cupsDrunk = 0;
  // closure is created when function accesses cupsDrunk
  var checkFundsAndDrink = function(){
    if( money >= coffeeCost ){
      money -= coffeeCost;
      cupsDrunk++; // declared in immediate outer scope
    }
  }
  return checkFundsAndDrink;
})(); // invoke function, returns checkFundsAndDrink
drinkCoffee();
drinkCoffee();
drinkCoffee();
console.log( cupsDrunk ); // error! undefinedthe checkFundsAndDrink variable is not really needed
var money = 55.50;
var coffeeCost = 8.50;
var drinkCoffee = (function(){
  var cupsDrunk = 0;
  return function(){ // removed var checkFundsAndDrink
    if( money >= coffeeCost ){
      money -= coffeeCost;
      cupsDrunk++;
    }
  };
})();
drinkCoffee();
drinkCoffee();
drinkCoffee();
console.log( cupsDrunk ); // error! undefinedaccess the variables in the closure by returning the value
var money = 55.50;
var coffeeCost = 8.50;
var drinkCoffee = (function(){
  var cupsDrunk = 0;
  return function(){
    if( money >= coffeeCost ){
      money -= coffeeCost;
      cupsDrunk++;
    }
    return cupsDrunk; // return the value
  };
})();
var count;
count = drinkCoffee();  // 1
count = drinkCoffee();  // 2
count = drinkCoffee();  // 3
console.log( count );  // 3
console.log( money );  // 30