Springtime, baby. Sunshine, warmth, hiking, camping, grilling, gardening, and more outdoor activities are on the schedule. To be fair, the Bay area doesn't get much of a winter, just a couple of months of rainy weather, which frankly, are integral to the eco-system.
That said, the transition to more a real winter, moving from San Diego to the Bay, was not insignificant. Glad to be in the sunshine for the next few months.
Saturday, May 10, 2014
Saturday, May 3, 2014
Reversing C-Style Strings With Pointers
It's a common interview question to ask an applicant to reverse a c-style string, without using string utility functions. In practice, this is something you almost never do, as modern programming languages have nice, safe string classes and interfaces, most likely complete with their own string::reverse() methods.
However, the test has some value. Depending on how the applicant approaches the task, you can gauge his comfort level with pointers. You can get a feel for the applicant's efficiency with memory and operations. Do they reverse the string in place, or allocate new memory?
In particular, swapping the values of two variables, is an interesting sub-test. Can the applicant do this correctly? How do they choose to do it? They can do it in a separate function, with pointers, with references, etc.
Finally, testing. Does the applicant simply write an answer and say "tada"? Or do they try to walk through the code and test what it does, maybe catching mistakes or corner-cases?
As long as the applicant can correctly complete the task, there are no wrong answers. This test really elucidates their style and thought process.
Even though a person might scoff at such a "trivial" test that "doesn't prove anything," it turns out that the test is not so trivial for most programmers. Notably, most applicants cannot correctly complete this task the first time.
Now to the code:
Then, I swap the values of the lower and upper pointers, incrementing the lower and decrementing the upper, until they meet in the middle. And that's it.
If I wanted to be a little more elegant with my swapping, I might write a separate function to swap the values of the pointers. How might I do that?
In this case, I've chosen to swap using pointers. So, lines 10-12 in reverse_str() would change to:
One other option could be to swap using references. It's actually pretty important that an applicant understands references in C++, as this is a common and integral feature of the language. [Maybe I'll do a follow-up entry on this at some point]. Here's a swap with references:
In this case, lines 10-12 of reverse_str() would need the pointers to be dereferenced:
So yeah, there you go. There are myriads of other ways to approach this problem. For such a seemingly simple task, you get a lot of mileage and information about a software engineer.
However, the test has some value. Depending on how the applicant approaches the task, you can gauge his comfort level with pointers. You can get a feel for the applicant's efficiency with memory and operations. Do they reverse the string in place, or allocate new memory?
In particular, swapping the values of two variables, is an interesting sub-test. Can the applicant do this correctly? How do they choose to do it? They can do it in a separate function, with pointers, with references, etc.
Finally, testing. Does the applicant simply write an answer and say "tada"? Or do they try to walk through the code and test what it does, maybe catching mistakes or corner-cases?
As long as the applicant can correctly complete the task, there are no wrong answers. This test really elucidates their style and thought process.
Even though a person might scoff at such a "trivial" test that "doesn't prove anything," it turns out that the test is not so trivial for most programmers. Notably, most applicants cannot correctly complete this task the first time.
Now to the code:
1: void reverse_str(char* str){
2: //Receive a c-style string, meaning a pointer to an array of type char,
3: //Reverse in place
4: char* lower=str;
5: char* upper=lower;
6: while (*upper!='\0') //Searching for the end of a null-terminated string
7: upper++;
8: upper--; //Decrement it; we want the last non-null element
9: while (upper>lower){
10: char temp=*lower;
11: *lower=*upper;
12: *upper=temp;
13: upper--;
14: lower++;
15: }
16: }
In this example, I've used pointers. To find the length of the string, I iterate the pointer [upper] until I find the null-character. I then decrement this pointer, to the last non-null character.Then, I swap the values of the lower and upper pointers, incrementing the lower and decrementing the upper, until they meet in the middle. And that's it.
If I wanted to be a little more elegant with my swapping, I might write a separate function to swap the values of the pointers. How might I do that?
1: template<class T>
2: void swap(T* a, T* b){
3: //Swap the values of two pointers
4: T temp=*a;
5: *a=*b;
6: *b=temp;
7: }
In this case, I've chosen to swap using pointers. So, lines 10-12 in reverse_str() would change to:
1: swap(lower,upper);
One other option could be to swap using references. It's actually pretty important that an applicant understands references in C++, as this is a common and integral feature of the language. [Maybe I'll do a follow-up entry on this at some point]. Here's a swap with references:
1: template<class T>
2: void swap(T& a, T& b){
3: //Swap the values of two variables
4: T temp=a;
5: a=b;
6: b=temp;
7: }
In this case, lines 10-12 of reverse_str() would need the pointers to be dereferenced:
1: swap(*lower,*upper);
So yeah, there you go. There are myriads of other ways to approach this problem. For such a seemingly simple task, you get a lot of mileage and information about a software engineer.
The Only Democrat in CA Who Won't Blow the Budget
This year, amidst a tax increase and an economic recovery, after years of budget cuts and massive deficits, the state of California is turning a nice surplus. Most Democrats in Sacramento want to immediately establish new permanent programs.
Perhaps in some cases, they argue for reinstating some of the social welfare programs that were cut during the Great Recession. In other cases, they'd like to establish brand new programs, because if there's some extra money this year, why no establish new spending obligations with no guarantee of budget surpluses next year?
Need we remind them that the voter-approved tax increase was temporary, and that economic booms [Silicon Valley, Wall Street] are also temporary?
Jerry Brown is proposing an initiative that would set aside a rainy day fund, if the fiscal year's capital gains tax comprises beyond a certain percentage of overall tax revenue. He's the most responsible politician in Sacramento.
Perhaps in some cases, they argue for reinstating some of the social welfare programs that were cut during the Great Recession. In other cases, they'd like to establish brand new programs, because if there's some extra money this year, why no establish new spending obligations with no guarantee of budget surpluses next year?
Need we remind them that the voter-approved tax increase was temporary, and that economic booms [Silicon Valley, Wall Street] are also temporary?
Jerry Brown is proposing an initiative that would set aside a rainy day fund, if the fiscal year's capital gains tax comprises beyond a certain percentage of overall tax revenue. He's the most responsible politician in Sacramento.
Thursday, May 1, 2014
Binary Search With Pointers, C++
Today I'm posting my version of binary search, using pointers in C++. Binary search is a fundamental algorithm, which searches for an element in a sorted array. It is considered the mother of all divide-and-conquer algorithms, and executes in log(N) time, where N is the size of the array. The log is taken with respect to base 2 [binary].
Implementing Binary Search with pointers is a nice exercise that brings together a few concepts.
1. While binary search is an important algorithm from a conceptual standpoint, it's also highly practical and has come up in my research and work. Consider captured data that is indexed by its timestamp, e.g.
timestamp : value 1, value 2,....value N
The most efficient way to search for the values at a particular timestamp is binary search. In reality, we may want the nearest timestamp to a given reference, but that's a minor adjustment to the algorithm.
2. Implementing binary search with pointers helps to reinforce understanding of pointers, a concept which tends to trip up many C++ programmers.
3. Despite the fact that binary search is a fundamental algorithm, very few job applicants are able to code it correctly on a whiteboard. In short, it's good practice.
In our example, we initialize an array with random elements, then sort it. First we demonstrate how we find an element in the array. Second, we demonstrate how the function behaves when we search for an unknown element. Now, to the code:
Implementing Binary Search with pointers is a nice exercise that brings together a few concepts.
1. While binary search is an important algorithm from a conceptual standpoint, it's also highly practical and has come up in my research and work. Consider captured data that is indexed by its timestamp, e.g.
timestamp : value 1, value 2,....value N
The most efficient way to search for the values at a particular timestamp is binary search. In reality, we may want the nearest timestamp to a given reference, but that's a minor adjustment to the algorithm.
2. Implementing binary search with pointers helps to reinforce understanding of pointers, a concept which tends to trip up many C++ programmers.
3. Despite the fact that binary search is a fundamental algorithm, very few job applicants are able to code it correctly on a whiteboard. In short, it's good practice.
In our example, we initialize an array with random elements, then sort it. First we demonstrate how we find an element in the array. Second, we demonstrate how the function behaves when we search for an unknown element. Now, to the code:
#include <iostream> //For std::cout
#include <algorithm> //For std::sort
#include <stdlib.h> //For srand, rand
#include <time.h> //For time
template<class T> //template class means we can accept arguments of different classes: int, float, double, etc.
int binarysearch(T* arr /*pointer to beginning of array*/, T* low, T* high, T searchkey /*seeking this value in array*/){
int index=-1; //Default value, if we don't find searchkey
if (*low>*high) //Double-checking that the array's actually sorted
return index;
if (low>high) //At this point, we've searched the whole array
return index;
T* mid=(high-low)/2+low; //The array index midway between high and low
if (*mid==searchkey) //We've found the searchkey
return mid-arr;
if (searchkey<*mid)
return binarysearch(arr,low,mid-1,searchkey); //Recursive call to search the lower half of the array
else if (searchkey>*mid)
return binarysearch(arr,mid+1,high,searchkey); //Recursive call to search the upper half of the array
return index;
}
int main(int argc, char* argv[]){
//Using srand to initialize rand() with time()
srand (time(NULL));
//Size of our array will be 15
int size=15;
int* arr=new int [size]; //Note that arr is the array, and is also a pointer. It points to a memory block of 15 integers.
for (int i=0; i<size; i++){
arr[i]=rand()%100; //Randomly initialize the elements of the array, integers between 0 and 100
std::cout<<arr[i]<<",";
}
std::cout<<"\n";
std::sort(arr,arr+size); //Sort the array
for (int i=0; i<size; i++)
std::cout<<arr[i]<<",";
std::cout<<"\n";
//Finding element in array.
int to_find=arr[rand()%size]; //Choosing an element at random from the array to search using binary search
int found_index=binarysearch(arr,arr,arr+size-1,to_find);
if (found_index>-1)
std::cout<<"Found "<<to_find<<" at location "<<found_index<<"\n\n";
else
std::cout<<"Couldn't find "<<to_find<<" in array\n\n";
//Finding unknown element
to_find=100; //We initialized the array with elements strictly smaller than 100
found_index=binarysearch(arr,arr,arr+size-1,to_find);
if (found_index>-1)
std::cout<<"Found "<<to_find<<" at location "<<found_index<<"\n";
else
std::cout<<"Couldn't find "<<to_find<<" in array\n";
delete [] arr; //Clean up dynamically allocated memory
return 0;
}
Subscribe to:
Posts (Atom)