Vivek Agarwal’s Portal/Java Blog

An IBM Gold Consultant’s weblog about IBM, Lotus, WebSphere, J2EE, IT Processes, and other IT technologies

Want to test your object-oriented coding skills? Check out this code smelling exercise …

Posted by Vivek Agarwal on June 21, 2008


Sometime back, I wrote on Bad Smells in Code that were defined as – “A code smell identifies classic mistakes made in developing code. These mistakes typically result in code that is difficult to understand, maintain, debug, and extend.” In my previous blog entry, I captured some code smells that are commonly found in code written by new (and some experienced) programmers. Sample smells include “duplicated code”, “long method”, “inappropriate intimacy”, “large classes”, “inconsistent names”, and others. I had promised to be back later with my “code smelling exercise” that I used during my presentation to make my points about code smells. Well here it is …

The exercise has 3 Java classes – “Order”, “lineitemlist”, and “lineItem”. They were written by me ~8 years back with the express intent of embedding as many smells as possible. Attendees in the presentation were given ten minutes to mark up all the poor coding practices (in other words – smells) that they could find. Feel free to try your skills to identify the smells! 🙂

##########################################################################
Order.java
##########################################################################

package com.nq.util;

import java.io.*;
import java.util.*;
import java.lang.*;
import java.sql.*;

public class Order {

	private lineitemlist lineItemList;

	public Order(lineitemlist lis) {
		lineItemList = lis;
	}

	public boolean equals(Object aThat) {
    		if ( this == aThat ) return true;
		if ( !(aThat instanceof Order) ) return false;
    		Order that = (Order)aThat;
		return this.lineItemList.equals(that.lineItemList);
	}

	// writes this order object to the specified print writer
	public void writeOrder(Order order, PrintWriter pw) {
		// get a vector of line items
		Vector lineItems = order.getLineItemList().getLineItems();

		// ------------------------------------------------------
		// calculate total
		// ------------------------------------------------------
		// create an iterator for the vector
		Iterator iter = lineItems.iterator();
		lineItem item;
		// set total to zero
		int total = 0;
        	while (iter.hasNext()) {
	        	item = (lineItem)iter.next();

	        	// calculate total for line item
	        	int unitPrice = item.getUnitPrice();
	        	int qty = item.getQuantity();
	        	int lineitemtotal = unitPrice * qty;

	        	total += lineitemtotal;
	        }
		// ------------------------------------------------------
		// END calculate total
		// ------------------------------------------------------

		// ------------------------------------------------------
		// write order
		// ------------------------------------------------------
		// create an iterator for the vector
		iter = lineItems.iterator();
        	while (iter.hasNext()) {
	        	item = (lineItem)iter.next();

	        	// calculate total for line item
	        	int unitPrice = item.getUnitPrice();
	        	int qty = item.getQuantity();
	        	int productID = item.getProductID();
	        	int imageID = item.getImageId();
	        	int lineitemtotal = unitPrice * qty;

	        	pw.println("Begin Line Item");
	        	pw.println("Product = " + productID);
	        	pw.println("Image = " + imageID);
	        	pw.println("Quantity = " + qty);
	        	pw.println("Total = " + lineitemtotal);
	        	pw.println("End Line Item");
	        }
		pw.println("Order total = " + total);
	}

	public int getTotal() {
		// get a vector of line items
		Vector lineItems = lineItemList.getLineItems();
		// create an iterator for the vector
		Iterator iter = lineItems.iterator();
		lineItem item;
		// set total to zero
		int total = 0;
        	while (iter.hasNext()) {
	        	item = (lineItem)iterator.next();

	        	// calculate total for line item
	        	int unitPrice = item.getUnitPrice();
	        	int qty = item.getQuantity();
	        	int lineitemtotal = unitPrice * qty;

	        	total += lineitemtotal;
	        }
	        return total;
	}

	/** This method saves the order to the database */
	public void saveOrder()  throws SQLException
	{
		//create connection
		Connection conn = null;

		java.sql.Date date = new java.sql.Date((new java.util.Date())
		.getTime());
		PreparedStatement orderStatement = null;
		PreparedStatement getStatement = null;
		String sql = null;
		sql = new StringBuffer().append("INSERT INTO T_ORDER " )
			.append("(AUTHORIZATION_CODE, " )
			.append("SHIPMETHOD_ID, USER_ID, ADDRESS_ID) " )
			.append("VALUES ( ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?,
			?)" ).toString();
		conn = setConnection();
		orderStatement = conn.prepareStatement(sql);
		//set all parameters
		...
		//execute statement
		orderStatement.executeUpdate();
	}
}

##########################################################################
lineitemlist.java
##########################################################################

package com.nq.util;

import java.io.*;
import java.util.*;
import java.lang.*;

class lineitemlist {
	private Vector LIList;

	public void setLineItems(Vector lineItems) {
		LIList = lineItems;
	}

	Vector getLineItems() {
		return LIList;
	}
}

##########################################################################
lineItem.java
##########################################################################

package com.nq.util;

import java.io.*;
import java.util.*;
import java.lang.*;

class lineItem {
	protected int productId;
	private int ImageID;
	private int qty;
	private int Unitprice;

	public lineItem(int prodID, int ImageID, int inQty) {
		productId = prodID;
		this.ImageID = ImageID;
		qty = inQty;
	}

	public void setLineItems(Vector lineItems) {
		LineItems = lineItems;
	}

	Vector getLineItems() {
		return LineItems;
	}

	int getProductID() {
		return productId;
	}

	int getImageID() {
		return imageID;
	}

	int getQuantity() {
		return qty;
	}

	int getUnitPrice() {
		return Unitprice;
	}

	public void setProductID(int id) {
		productId = id;
	}

	public void setImageID(int ID) {
		imageID = ID;
	}

	public void setQty(int qty) {
		this.qty = qty;
	}

	public void setUnitPrice(int i) {
		Unitprice = i;
	}
}

Hope you had a fun time with identifying the smells! There are quite a few in here. I will look into writing another entry on another day that explains the smells that permeate this code sample.

Any comments on how reasonable this would be as an interview question for mid-to-senior level developers?


Check out this poll to submit your response on this code smelling exercise …

Sorry, the comment form is closed at this time.