"Programmers have huge and fragile egos."
- Philip Greenspun, MIT Professor
From his article "Managing Software Engineers"
Tuesday, November 22, 2005
Quote of the Day
Friday, November 11, 2005
Veteran's Day Code
foreach (Soldier thisSoldier in US.Population)
{
ThankSoldier(thisSoldier, DateTime.Now());
}
Monday, November 07, 2005
Speeding Up MSDN Downloads
If you're an MSDN subscriber and have ever tried to download a huge set of files, I'm sure you know how long it can take when you're only getting about 25kbps speed from Microsoft's MSDN Subscriber Downloads. The problem is, you may be hitting a non-US mirror site. So, here's a way to speed things up a bit.
In the hosts file on the machine doing the downloading (%windir%\system32\drivers\etc\hosts) add an entry for the Microsoft server:
207.46.252.185 global.ds.microsoft.comThen, in Internet Explorer, add https://global.ds.microsoft.com in the list of trusted sites under Tools > Internet Options > Security > Trusted Sites. Also, under the privacy tab click the 'Sites' button and enter https://global.ds.microsoft.com so that microsoft.com is a trusted domain.
We did this at work and saw a speed increase from about 25kbps to ~150kbps.
Source: Jason Mauss' Blog Cabin
Saturday, November 05, 2005
Comments
The real reason for this post, however, is mostly to point out that I've turned the comment feature back on for this blog. As I pointed out in a post to my personal blog, Blog*Spot had some trouble with spam. It seems to be all better now, and I've had the commented turned on in my other blogs with no trouble for more than a week now.
Back to the relevant topic: comments. Developers have been arguing for years over what should be commented, and what shouldn't be. The rule of thumb I use is that the code itself should tell the what, not the comments. If you can't look at the code and easily determine what it's doing, you need to refactor it.
So then, what are comments for? The comments should tell the why. That's my rule of thumb: the code tells the what, the comments tell the why.
Maybe you wrote some code that deletes a file. The what will be fairly obvious. But why are you deleting this particular file? Here's an example:
In C#, you can also include XML comments. Visual Studio will automatically create an XML comment stub for you if you type 3 comment slashes (///) immediately preceeding a method, or other commentable element. (Note that there are several 3rd party add-ins that allow you to auto-create XML comments in Visual Basic).
When you're finished, it looks something like this:
XML comments are exceptionally useful, and I recommend them highly. First, they remove the need for those painful block comment headers that some companies require. You know the ones - they have such important information as the programmer's name and the date the method was written (because nothing will help me debug like knowing that something was written 3 months ago).
If you specify a file path in your project properties, Visual Studio will create an XML file of all the comments for you during compilation. To set this up, set the XML Documentation File in build section under Configuration Properties from the project properties window as in the following example:
If you create an XML, then a snazzy tool like NDoc will pick up the file, use a little reflection, and create an instanct CHM help file to document your code. Viola! Instant documentation. Few things make a developer happier than not having to waste time writing doc.
Even if you don't plan on creating documentation, there's another reason to use XML commenting. Intellisense looks for it when showing you a method signature. Look what happens when I use the UserHasPermission method from the example above:
This could be extremely helpful if you need to reuse someone else's code, or if somebody needs to use yours.
So the lesson in all this is:
- Code tells the what, comments tell the why
- Use the XML comments feature of visual studio
- I turned the comments back on for this blog, so you can leave some
Thursday, November 03, 2005
Programming Evolution
Fast forward 15 years...
Nowadays, the hardware has been largely abstracted away from most programmers (don't argue - I know there will always be a need for people to write device drivers, and I also understand that the Linux camp thinks they are 1337). It's no longer necessary to squeeze the most out of every byte because, quite frankly, there is plenty of memory and hard drive to spare.
We also have nice GUI development tools that provide slick WYSIWYG interfaces. What used to take hours of painstaking coding in a text editor can now be accomplished literally in minutes via drag-and-drop.
These things considered, is it easier to be a programmer now than it was 15 or so years ago? I don't think so. In fact, I think quite the opposite is true.
While we're not wasting time coding as much stuff by hand, and we don't need quiet the depth of knowledge about the hardware, the breadth of knowledge required to be a decent programmer has increased exponentially.
15 years ago we didn't have to worry about the Internet. Programmers these days not only need to know their core programming language (be it C#, C++, VB, Java, Perl, Python, Ruby, or something else), they also need to understand the Internet. That means having an understanding of the network infrastructure itself, as well as the relevant protocols (TCP, IP, HTTP, FTP, etc).
We need to understand interoperability better than ever before, and we also have to have specific knowledge relative to the challenges of distributed computing. I won't even get into the different database options, and what developers need to know. Clearly, the number of areas in which we all really need to have some level of knowledge, if not specialization, is staggering.
All things considered, I think it's actually a little more difficult to be a modern day Software Engineer. Our predecessors may complain of punch cards, command-line compiling, and text-based development tools, but at least they had enough time to do the work without the tools we enjoy today. It'll be interesting to see how the field changes in the next 15 years.
Wednesday, November 02, 2005
Making a Sortable Collection
The key to this approach is that part of the underlying plumbing of the CollectionBase is an ArrayList. Since ArrayLists are self-sorting, wiring this up is a breeze.
The sort method of an ArrayList have a few overloads. One takes no parameters, and the other takes an IComparer object. What does it mean? Well, if you don't specify a comparer object, the array list will be looking to the objects in the collection to perform comparisons against one another as part of the sort aglorithm.
Consider the following example (I like using the good old dog object). Let's say you have a simple "Dog" class that has two properties: Name, and BirthDate. You want to write a collection to store Dog objects, and you need the ability to quickly sort the Dogs in this collection by either name or Birthdate. (Maybe you're writing some kennel software or something - who knows?)
First, a look at the Dog object. It's pretty standard, but you'll notice that it implements IComparer (see my "Implementation and Inheritence Primer" post for more information about this).
|
You'll notice that as required by the IComparer class, Dog has a public method named "CompareTo". The ArrayList will execute this method when comparing two Dog objects during the sort algorithm.
What about non default sorts? We need a couple of IComparer objects to pass into the ArrayList. Here's some examples of objects that will compare Dog objects by Name or Birthdate:
|
|
Now on to that collection. We implement a pretty standard collection, including a couple of handy overloads for the Add method. We also add a "Sort" method to tell our collection to sort itself. Note that the Sort method makes the underlying ArrayList do the heavy lifting.
|
And finally, a little bit of enumeration to define the types of sorting allowed:
|
Viola! It's cake.
Watch for updates to this post (there are details I still need to add), but one thing I would point out is to ALWAYS CODE DEFENSIVELY. For example, you'll notice that the methods in these objects don't trust the data passed in - they all test for invalid values, and take appropriate action.
Happy coding!
Friday, October 14, 2005
Singleton Duplicity in 5 Minutes
Note that the Singleton is a design pattern, which means it is not language specific. I use C# in my code examples simply out of personal preference; however you can implement this pattern using just about any object-oriented language.
What is a Singleton?
In the simplest terms, a Singleton is a class for which there is only a single shared instance. Any code that accesses the class uses the same instance of the object. Consequently, Singletons must provide a global point of access to their instance.
Why use a Singleton?
Think about your computer. There are certain shared resources of which there can only be one. For example, your computer has a single keyboard that is used by all of your applications. You probably (but not necessarily) have only one mouse. From a software standpoint, your Operating System can reasonably have only one process manager.
Sometimes there are programmatic situations that require one and only one of something, too. A great example of this is a logging service. A logging service only needs to exist in memory one time, and that instance can be shared across all functionality that needs to log an event.
Quick and Dirty C# Singleton
Here's an example of an extremely basic Singleton class. In the real world, you'll want to make sure that any Singleton class is thread safe, but the intent here is to simply provide an overview of the basic Singleton framework.
public class Singleton
{
private static Singleton instance;
private Singleton()
{
}
public static Singleton GetInstance()
{
if (instance == null)
{
instance = new Singleton();
}
return instance;
}
public string SayHello()
{
return "Hello World!";
}
}
In the example above, you'll notice that the constructor is private rather than public. This prevents the class from being instanciated. Because the constructor uses the "private" access modifier, it is inaccessible to any code outside of the Singleton class. But, since a private method (and the constructor is simply a method) is accessible from within the same class, code within that class can create an instance of it. In this way, we force all access to the object to go through the GetInstance method, which will return the static (shared) instance.
To use the object above, you might have code that looks something like this:
Singleton mySingleton = Singleton.GetInstance();
string helloMessage = mySingleton.SayHello();
mySingleton = null;
Note that although the reference to the Singleton instance is destroyed when the local object is made null, the instance itself remains, and can be reused by subsequent calls to the GetInstance method. In the first example, note that the instance variable is private, so it is not implicitly accessible to code outside the Singleton class.
Other Considerations
There are multiple ways to implement the Singleton Pattern. The code provided here is merely an example of one implementation method to facilitate discussion of the pattern itself.
If you're planning on implementing the Singleton pattern, do so with care. There are some in the industry who refer to a Singleton as an anti-pattern because it is so often misused. Make sure any objects that interract with singletons are still loosely coupled, and that you're not using a Singleton simply as a global variable.
For more information on Singletons, visit these resources:
Tuesday, October 11, 2005
Sorting Directly in a Data Table
The standard approach to filtering and sorting data in .NET is to use a DataView. In a nutshell, the DataView acts as a layer between your bound object and the underlying data object. For example, to use a DataView to sort the data in a DataGrid, instead of binding the grid directly to a DataSet (just for the sake of argument), you would declare a DataView and set its source to the DataSet. Then you’d configure the sort options on the DataView, and bind the grid to it. Viola! Instant sorting under the hood without the need to write any code!
What if you have a situation where you need to sort data without binding to it? In that case, using the DataView isn’t a viable option. Well, take heart – the DataTable object provides a mechanism that allows you to sort and filter the data directly on the Data Table.
You can use the Select method of a Data Table to sort and filter the data. The syntax is:
datatable.Select(filterExp, sortExp, rowstatevalue)
The select method returns an array of rows that meet the filter criteria in the sort order specified. Furthermore, you can control what types of rows are returned. You say you only want the rows that have been added since you populated the Data Table? No problem – just set the RowStateValue parameter to DataViewRowState.Added.
What if you want to sort, but not filter the data? That’s not a problem either. You can pass in null to any parameter you don’t want to use. For example:
DataRow[] sortedRows = customerTable.Select(null, "LastName");
Monday, October 10, 2005
C# Ternary Operator
The C# language specification contains a really convoluted definition for what essentially amounts to a very simple operation. In my own words, the syntax for using the ternary operator is as follows:
test-condition ? true part : false part;
In other words, if “test-condition” evaluations to true, the true part code will execute. Otherwise, the false part code will execute. It’s like an if-then-else bundled into one statement. In fact, the statement above may be rewritten to:
if (test-condition)
{
//true part code here
}
else
{ //false part code here
}
Arguably, a programmer could get carried away with what amounts to a shortcut. One great use of the ternary operator that I’ve found, though, is assigning the value to a variable at the time of declaration based upon a specified condition. For example, reading the contents of a config file, and storing the value in a variable.
Note that when reading from a config file, if the specified key does not exist, the value returned will be null. If you then try to perform a string operation (like testing the value of the string), you’ll end up with a nasty little System.NullReferenceException. To avoid this, I use a technique with the ternary operator that allows me to grab my value, and account for the possibility of null all in one statement. Here’s an example:
//read a setting from the config file,
//and assign it to the mySetting string:
string mySetting = ConfigurationSettings.AppSettings["KeyName"] != null ?
ConfigurationSettings.AppSettings["KeyName"] : string.Empty;
Another great use is for the ternary operator is when retrieving values from a Data Reader or Data Table. You can use a similar approach to the example above to test for a condition of “== DBNull.Value” before performing any operations on the object that may potentially have a null value.
Tuesday, October 04, 2005
Implementation and Inheritance Primer
Lacking a more in-depth topic, I thought I’d take a few moments today to share a great way to explain the difference between inheritance, and implementation inheritance (that is, implementing or inheriting an Interface).
To put it as a one-liner, inheritance is like saying “is a…”, whereas implementation inheritance is like saying “acts like…”.
Consider the following C# code example. In the example, we have an abstract class named “Dog”. (Recall that an abstract class cannot be instantiated – it may only be inherited)
public abstract class Dog : IDog { private string _name; public Dog() { //Standard object constructor } public void Bark(int times) { //Actual code here to make the dog bark } public void Sit() { //Actual code here to make the dog sit } public string Name { get { return _name; } set { _name = value; } } } |
The Dog object presumably encapsulates all of the basic required functionality of a dog. We have a Name property and methods for Bark and Sit. Any object that class that inherits Dog will have access to all that functionality.
We also have an Interface, aptly named IDog. (Note to newbies: best practices are that Interface names are always prefixed with the letter I)
public interface IDog { void Bark(int times); void Sit(); string Name{get; set;} } |
In contrast to an abstract class, an interface does not contain any functional source code. Rather, it merely defines the way in which objects that implement it may be interacted with. In our example, any classes that implement the IDog interface must have their own Name property and Bark and Sit methods.
The following are examples of implementing the IDog interface, and inheriting the Dog class:
public class Poodle : Dog { public Poodle() { //Because this inherited the actual dog object, //we can make the base class do the bark base.Bark(3); } } |
public class Collie : IDog { public Collie() { //standard constructor, but we don't have //any functionality available from IDog - //it simply defines what methods and properties we //must have } public void Bark(int times) { //required by IDog interface throw new NotImplementedException(); } public void Sit() { //required by IDog interface throw new NotImplementedException(); } public string Name { //required by IDog interface get { throw new NotImplementedException(); } set { throw new NotImplementedException(); } } } |
Note that in our example, the Dog class itself implements the IDog interface. Strictly speaking, it doesn’t have to. However, if this were real world code, we could use the IDog interface as one way to address the Class Factory pattern as in the following example:
public class DogFactory { public DogFactory() { } public IDog GetDog(int height) { if (height <> { return new Poodle(); } else { return new Collie(); } } } |
In the real world, this approach is most often seen in functionality that interacts with databases. Taking this approach allows the programmer to write code that will interact with logic that is highly optimized for a specific database platform, while still affording the ability to support multiple platforms. I’ll discuss more about this topic in a future post.
Sunday, October 02, 2005
Finding Your Audience
My intent with this blog is evolving. Originally, I waxed poetically that perhaps I would be able to build something to “take it to the next level”. I’m beginning to see, however, that the best value for this blog may lie in helping out people who are perhaps not so technical.
My parents read my blog. There – I said it. Although I doubt Steve Jobs, Bill Gates, or Sergey Brin could boast such personal success, it does create some unique challenges. First, my mom continues to proof-read my work. I can get away with just about any technical error, but damn it, those words had better be spelled correctly, and word choice is up for evaluation (my mom's comments on my latest blog posts relative to my choice of the word “indeed”: You have to now retire your favorite word for at least a month or six. Indeed, it is inappropriate to use "indeed" in every blog. People who read you will think this is your personal crutch that replaces "uhhhh", or "ya know?"). Indeed, I see her logic.
Back to the topic at hand, I still need to decide who my audience is. While this is a work in progress and is apt to change over time, I’ve decided to write to the appropriate level of the topic I’m discussing (e.g. if I’m blogging on multi-threading, I’ll be writing for an audience I assume is somewhat familiar with the terms), but I’ll try to include more links to explain technical terms and concepts.
A few thoughts I have on upcoming blog posts:
- The programmer psyche and lack of architectural direction
- The hype around Service Orientation and contract-first development
Kids Programming Language
Morrison Schwartz seems to have tackled that issue with Kids Programming Language.
If you have kids and you want to get them interested in programming, or even if you don’t have kids but have a taste for nostalgia, I recommend you take a look.
Thursday, September 29, 2005
Update Time
Few people would argue that 7 months between blog posts is excessive. This is something I've been struggling with, but I've decided to make an honest effort to post more frequently.
It's difficult to step into the spotlight, largely because to do so is to risk ridicule, and indeed, public embarrassment. One thing I’ve struggled with is that I know dozens of people who know more about Software Development than I do. Notwithstanding my knowledge of this fact, I had an epiphany.
It’s true that I know far less than a lot of people. It’s equally true, however, that I know far more than a lot of people. So, while some will read this blog, uncover errors, and perhaps chuckle arrogantly at my posts, other might find it helpful. At the end of the day, isn’t that what techie blogging is all about? I know I’ve certainly benefited from blog posts!
Here’s a preview of some topics to come:
- Using .NET Directory Services with Active Directory and LDAP
- Service Oriented Architecture (SOA), or Service Orientation if You Prefer
- My Take on C# 3.0 and the Next Generation of .NET (a popular topic since the Microsoft PDC)
- Rants, Raves, and other Ramblings
- Assorted off-topic oddball posts
Sunday, February 13, 2005
Here We Go
In the world of the PhD and higher learning in general, the term “publish or perish” has been thrown about much to the chagrin of those forced to do the publishing. In software development, I’m noticing a trend that successful programmers (and indeed, other techies) share their knowledge with the world in the form of blogs. Not one to let a trend pass me by, I’ve decided to get with the proverbial program, and in the process get in my two cents on programming and other fascinating topics.
Another thing that comes to mind is that some skills are perishable. If you don’t continue to use exercise these skills, you’ll lose them. I contend that writing is one such skill, and having recently completed my Master’s Degree (in Internet Security, for those who are curious), I’m faced with the fact that my writing skills may wane without an active reason to write something for an audience each week. By having a blog, I hope to keep my writing skills honed.
These things considered, please be gentle with your comments because I am new to this. I hope you’ll enjoy reading my blog as much as I enjoy writing it. If you don't, there is always this site to amuse you.