Tuesday, March 19, 2013

Book Review of JavaScript the Good Parts by Douglas Crockford and a Dijkstra implementation by George Borden

JavaScript: The Good Parts is informative about the bad parts too.

Leaders need to key into the big things happening in their field, so with HTML5 making waves, it seemed the right time to learn JavaScript (JS). JavaScript: The Good Parts is a great reference to JS. It is in the style of great classics such as C Programming Language (2nd Edition) and The C++ Programming Language (3rd Edition). The chapters cover JS grammar, how objects work, functions, inheritance and describe the available methods. Crockford describes the areas of JS to avoid such as global variables or JS's handling of Unicode.

In most of the chapters the book goes into a good level of detail with great code examples. In a couple of chapters it is a bit vague. For example, the grammer gives good diagrams as well as code examples, but the chapters covering the poorer aspects of JS sometimes just give a description of the problem without workarounds and there are a couple of chapters that feel like filler. Lastly, since the book does not cover the entire language, I also recommend getting JavaScript: The Definitive Guide: Activate Your Web Pages (Definitive Guides).

I've used JavaScript: The Good Parts to help me understand the language better, and I really appreciate its brevity and ease of access. If you want to have a solid reference to the language as well as some good practical advice on best practices and issues to avoid, then this is the book for you.

Just for fun. Here's a bit of code implementing Dijkstra's shortest path.




/* Dijkstra's algorithm */

/* This function is JavaScript's way of creating an object. This object
   encapsulates Linked List functionality.
    Pass: nothing
    Returns: nothing
*/
function gbLinkedList() {
    this.head = null;
    this.value = null;

    function llItem(value) {
        this.next = null;
        this.value = value;
    }

    /* This function adds an item to the linked list.
        Pass: String, value to be inserted in the list.
        Returns: nothing
    */
    function addItem(value) {
        if (this.head === null) {
            this.head = new llItem(value); /* exception case for first item */
            return;
        }
        var currentItem = this.head;
        while (currentItem.next !== null) {
            currentItem = currentItem.next;
        }
        currentItem.next = new llItem(value);
    }
    /* This function finds an item to the linked list.
        Pass: String, value to be inserted in the list.
        Returns: true if found, false if not found.
    */
    function findItem(value) {
        var currentItem = this.head;
        while (currentItem !== null) {
            if (currentItem.value == value) {
                return true;
            }
            else {
                currentItem = currentItem.next;
            }
        }
        return false;
    }
    this.addItem = addItem; // This is required to add the member to the object
    this.findItem = findItem;
    this.llItem = llItem;
}
/* This function is JavaScript's way of creating an object. This object
   encapsulates min heap functionality. 
    Pass: boolean, if true max heap, if false min heap
    Returns: nothing
*/
function gbMinHeap() {
    var gbHeapMaxSize = 101; // remember we are wasting entry 0 to save math
    var gbNumberOfEntries = 0; // remember we are wasting entry 0 to save math
    // This heap does not use the 0th element to avoid -1 to every operation.
    var gbHeapArray = new Array(gbHeapMaxSize + 1);
    var i;
    for (i = 0; i < gbHeapArray.length; i++) { // initialize the heap array.
        gbHeapArray[i] = null;
    }

    function gbPrintHeap() {
        var i;
        var values = "";
        for (i = 1; i < gbNumberOfEntries + 1; i++) {
            values = values + gbHeapArray[i].value + ", ";
        }
        console.log(values);
    }

    /* This function extracts changes the value of a node
    Pass: item, the item whose value is to be changed
          value, the new value
    Returns: nothing
    */
    function reduceValue(item, value) {
        item.value = value;
        gbMinHeapify(1);
    }


    /* This function extracts the min from the heap
    Pass: nothing
    Returns: min, the item with the minimum .value property
    */
    function gbExtractMin() {
        var min = null;

        min = gbHeapArray[1];
        gbHeapArray[1] = gbHeapArray[gbNumberOfEntries];
        gbHeapArray[gbNumberOfEntries] = null;
        gbNumberOfEntries--;
        gbMinHeapify(1);
        gbPrintHeap();
        return min;
    }

    /* This function inserts an item into the heap
    Pass: item, an object with a .value property
    Returns: nothing
    */
    function gbMinInsert(item) {
        var i;
        if (item.hasOwnProperty("value")) {
            gbNumberOfEntries++;
            i = gbNumberOfEntries;
            while (i > 1 && gbHeapArray[Math.floor(i / 2)].value > item.value) {
                gbHeapArray[i] = gbHeapArray[Math.floor(i / 2)];
                i = Math.floor(i / 2);
            }
        }
        gbHeapArray[i] = item;
        gbPrintHeap();
    }

    /* This repairs the property of the min heap at the current node
    Pass: index, the parent to fix
    Returns: nothing
    */
    function gbMinHeapify(index) {
        var left = 2 * index;
        var right = 2 * index + 1;
        var smallestIndex = 0;
        var temp = 0;

        if (gbHeapArray[left] !== null && gbHeapArray[index] !== null) { // left and index have values
            if (gbHeapArray[left].value < gbHeapArray[index].value) { // Choose the smallest
                smallestIndex = left;
            }
            else {
                smallestIndex = index;
            }
            if (gbHeapArray[right] !== null && gbHeapArray[right].value < gbHeapArray[smallestIndex].value) {
                smallestIndex = right;
            }
            if (smallestIndex !== index) {
                temp = gbHeapArray[index];
                gbHeapArray[index] = gbHeapArray[smallestIndex];
                gbHeapArray[smallestIndex] = temp;
                gbMinHeapify(smallestIndex);
            }
        }
    }

    this.gbExtractMin = gbExtractMin;
    this.gbMinInsert = gbMinInsert;
    this.reduceValue = reduceValue;
}
/* This function is JavaScript's way of creating an object. This object
   encapsulates a heap node for the test.
    Pass: item, an object with a .value property
    Returns: nothing
*/
function gbHeapNode(value) {
    this.value = value;
}

function gbEdge(vertice, value) {
    this.edgeVertice = vertice;
    this.value = value;
}

function gbVertice(index) {
    this.value = 0; // this is dist.
    this.index = index;
    this.neighbors = null;
}
/* This function is JavaScript's way of creating an object. This object
   encapsulates a Dijstra's algorithm for shortest path.
    Pass: graph, an array of objects with linked lists for neighbors
          s, index of start point.
    Returns: nothing
*/
function gbDijkstra() {
    var priorityQueue = null;
    var dist = null;
    var prev = null;

    function gbFindPath(graph, s) {
        priorityQueue = new gbMinHeap();
        dist = new Array(graph.length);
        prev = new Array(graph.length);
        var i;
        var currentNeighbor = null;
        var currentVertice = null;
        var currentWeight = 0;
        var newLength = 0;
        for (i = 0; i < dist.length; i++) {
            dist[i] = graph[i].value = 1000000000;
            prev[i] = null;
        }
        dist[s] = graph[s].value = 0;
        for (i = 0; i < dist.length; i++) {
            priorityQueue.gbMinInsert(graph[i]); // insert the items in priority order
        }
        currentVertice = priorityQueue.gbExtractMin(); // get the vertice with the smallest dist
        while (currentVertice !== null) {
            currentNeighbor = currentVertice.neighbors.head; // start at the first neighbor
            while (currentNeighbor !== null) {
                currentWeight = currentNeighbor.value.value; // get the weight of the edge
                newLength = dist[currentVertice.index] + currentWeight; // Add it to the current shortest distance stored for that vertice
                if (newLength < dist[currentNeighbor.value.edgeVertice.index]) {
                    priorityQueue.reduceValue(currentNeighbor.value.edgeVertice, newLength);
                    dist[currentNeighbor.value.edgeVertice.index] = graph[currentNeighbor.value.edgeVertice.index].value = newLength;
                    prev[currentNeighbor.value.edgeVertice.index] = currentVertice;
                }
                currentNeighbor = currentNeighbor.next;
            }
            gbPrintGraph();
            currentVertice = priorityQueue.gbExtractMin(); // get the next vertice
        }
    }
    function gbPrintGraph() {
        var tempS = "";
        for(i=0;i<dist.length;i++) {
            tempS = tempS + i + ":" + dist[i] + ", ";
        }
        console.log("Dist: "+tempS);
        tempS = "";
        for(i=0;i<prev.length;i++) {
            if(prev[i] !== null) {
                tempS = tempS + i + ":" + prev[i].index + ", ";
            }
        }
        console.log("Prev: "+tempS);   
    }
    this.gbFindPath = gbFindPath;
}


// Test
var myGraph = Array(5);
var i;
for (i = 0; i < myGraph.length; i++) {
    myGraph[i] = new gbVertice(i); // label the vertices
}

var edge = null;
var ll = null;
// create edges to be inserted into gbVertice 0
ll = new gbLinkedList();
edge = new gbEdge(myGraph[1], 2);
ll.addItem(edge);
edge = new gbEdge(myGraph[4], 4);
ll.addItem(edge);
myGraph[0].neighbors = ll;
// create edges to be inserted into gbVertice 1
ll = new gbLinkedList();
edge = new gbEdge(myGraph[2], 3);
ll.addItem(edge);
myGraph[1].neighbors = ll;
// create edges to be inserted into gbVertice 2
ll = new gbLinkedList();
edge = new gbEdge(myGraph[4], 1);
ll.addItem(edge);
edge = new gbEdge(myGraph[3], 5);
ll.addItem(edge);
myGraph[2].neighbors = ll;
// create edges to be inserted into gbVertice 3
ll = new gbLinkedList();
edge = new gbEdge(myGraph[0], 8);
ll.addItem(edge);
myGraph[3].neighbors = ll;
// create edges to be inserted into gbVertice 4
ll = new gbLinkedList();
edge = new gbEdge(myGraph[3], 7);
ll.addItem(edge);
myGraph[4].neighbors = ll;

var myDijkstra = new gbDijkstra();
myDijkstra.gbFindPath(myGraph, 0);

1 comment:

Your thoughts?