CSCI 1120 (Low-Level Computing), Fall 2021:
Homework 4

Credit:
20 points.

Reading

Be sure you have read, or at least skimmed, the assigned readings for classes through 09/15.

Programming Problems

Do the following programming problems. You will end up with at least one code file per problem. Submit your program source (and any other needed files) one of two ways:

You can develop your programs on any system that provides the needed functionality, but I will test them on one of the department's Linux machines, so you should probably make sure they work in that environment before turning them in.

  1. (10 points) NOTE that this problem is meant as an opportunity to get some practice with while loops in C, so you will only get full credit if you use at least one.

    Newton's method for computing the square root of a non-negative number $x$ starts with an initial guess $r_0$ and then repeatedly refines it using the formula

    $\displaystyle r_n = (r_{n-1} + (x / r_{n-1})) / 2
$

    Repetition continues until the absolute value of $(r_n)^2$ - $x$ is less than some specified threshold value. An easy if not necessarily optimal initial guess is just $x$. So for example the calculation starts like this for $x = 2$:
    $\displaystyle r_0$ $\displaystyle =$ $\displaystyle 2$ (1)
    $\displaystyle r_1$ $\displaystyle =$ $\displaystyle (r_0 + 2/r_0) / 2$ (2)
      $\displaystyle =$ $\displaystyle (2 + 2/2) / 2$ (3)
      $\displaystyle =$ $\displaystyle 1.5$ (4)
    $\displaystyle r_2$ $\displaystyle =$ $\displaystyle (r_1 + 2/r_1) / 2$ (5)
      $\displaystyle =$ $\displaystyle (1.5 + 2/1.5) / 2$ (6)
      $\displaystyle =$ $\displaystyle 1.1417 \;\; \mathrm{(approximately)}$ (7)

    Write a C program that implements this algorithm and compares its results to those obtained with the library function sqrt(). Have the program prompt for $x$, the threshold value, and a maximum number of iterations; do the above-described computation; and print the result, the actual number of iterations, the square root of $x$ as computed using library function sqrt(), and the difference between the value you compute and the one you get from sqrt(). Also have the program print an error message if the input is invalid (non-numeric or negative -- but note that zero is okay).

    Here are some sample executions (assuming you call your program newton and compile with make)

    [bmassing@diasw04]$ ./newton
    enter values for input, threshold, maximum iterations
    2 .0001 10
    square root of 2:
    with newton's method (threshold 0.0001):  1.41422 (3 iterations)
    using library function:  1.41421
    difference:  2.1239e-06
    
    [bmassing@diasw04]$ ./newton
    enter values for input, threshold, maximum iterations
    2 .000001 10
    square root of 2:
    with newton's method (threshold 1e-06):  1.41421 (4 iterations)
    using library function:  1.41421
    difference:  1.59472e-12
    

    Hints:

  2. (10 points) NOTE that this problem is meant as an opportunity to get some practice with for loops in C, so you will only get full credit if you use at least one.

    C, like many programming languages, has a library function (rand()) that can be used to generate a “random” sequence of numbers (quotes because it's not truly random -- more in video lecture). Many languages have a similar function that generates “random” numbers in some specified range, useful if for example you're trying to simulate rolling a 6-sided die. C doesn't have such a function, but you can get the same effect using rand() and a little additional code. rand() itself generates a number between 0 and the library-defined constant value RAND_MAX, so to get a value in a smaller range you have to somehow map the larger range to the smaller one. The somewhat obvious way to do this is by computing a remainder (e.g., to map to two possible values, assign even values to 0 and odd values to 1). (I'll call this the “remainder method”.) But with some implementations of rand() this gives results that aren't very good. The conventional wisdom is therefore to instead try to do a more-direct map (e.g., to map to two possible values, assign values from 0 through RAND_MAX/2 to 0 and the remaining values to 1). (I'll call this the “quotient method”.)

    Your mission for this problem is to complete a C program that, given a number of samples N and a number of “bins” B generates a sequence of N “random” numbers, uses both methods (remainder and quotient) to map each generated number to a number between 0 and B-1 inclusive, and counts for each method how many elements of the sequence fall into each bin (e.g., for each method bin 0 is how many elements of the sequence map to 0), and prints the result, as in the sample output below. (The fraction at the end is count divided by number of samples, and serves as a crude check of how even the distribution is.) To help you (I hope!) I'm providing a starter program, link below, which you should use as your starting point. Code at the bottom of the program shows how to apply both methods to something returned by rand(). The remainder method is straightforward; the quotient method is less so, but see the footnote1for an explanation. The starter code includes this formula, and I'd rather you just use it rather than figuring it out yourself -- not a bad thing but not the point of this assignment.

    One other thing to know about rand() is that by default it always starts with the same value (and produces the same sequence). To make it start with a different value, you can call srand() with an integer “seed”, so your program should prompt for one of those too.

    Sample execution (assuming you call your program rands and compile with make):

    seed?
    5
    how many samples?
    1000
    how many bins?
    6
    counts using remainder method:
    (0) 154 (0.15 of total) 
    (1) 188 (0.19 of total)
    (2) 171 (0.17 of total)
    (3) 161 (0.16 of total)
    (4) 155 (0.15 of total)
    (5) 171 (0.17 of total)
    counts using quotient method:
    (0) 172 (0.17 of total) 
    (1) 175 (0.17 of total) 
    (2) 183 (0.18 of total) 
    (3) 150 (0.15 of total) 
    (4) 168 (0.17 of total) 
    (5) 152 (0.15 of total) 
    

    Here is a starter program that prompts for the seed, generates a few “random” numbers, and illustrates the two methods of mapping to a specified range: rands.c.

    Of course, your program should check to make sure all the inputs are positive integers. (Yes, error checking is a pain, but it's an incentive to get better at copy-and-paste?)

    Hints:

Essay and pledge

Include with your assignment the following information.

For programming assignments, please put it a separate file. (I strongly prefer plain text, but if you insist you can put it in a PDF -- just no word-processor documents or Google Drive links please.) For written assignments, please put it in your main document.

Pledge

This should include the Honor Code pledge, or just the word “pledged”, plus at least one of the following about collaboration and help (as many as apply). Text in italics is explanatory or something for you to fill in; you don't need to repeat it!

Essay

This should be a brief essay (a sentence or two is fine, though you can write as much as you like) telling me what if anything you think you learned from the assignment, and what if anything you found interesting, difficult, or otherwise noteworthy.



Footnotes

... footnote1
If you're curious about the admittedly cryptic formula: Suppose n is the value returned by rand(). We could convert it to a bin number in two steps: First, we scale it to a floating-point number in the range from 0 up to but not including 1, thus:

$\displaystyle x = \mathit{n} / (\mathtt{RAND\_MAX} + 1)
$

(We divide by RAND_MAX+1 so the largest possible value maps to something still slightly less than 1. Note also that in the code we have to be sure this addition is done using floating point, since otherwise it could overflow; we can do this by writing the 1 as 1.0.) Then we scale this range of floating-point values to our desired range (still in floating point) by multiplying by B, which gives from 0 up to but not including B. Finally, we convert back to an int with a cast, which drops the fractional part, and leaves us with an index between 0 and B-1 inclusive.



2021-09-28