I would like these posts to be knowledge dense, so let’s cut to the chase without further ado. When you know what project you would like to do - what tool to create, what application to make - you would probably like to get your hands into the code as fast as possible. While it may be tempting, it would be wise to do a few things first.
When creating an enterprise scale application with complicated business logic and lots of technical demands, Java seems like a good choice. But what if you wanted to create a simple or even more complex, reactive page? Or a simple command line tool? You could actually do all these things in Java (though I’ve seen only bad Vaadin usage, unfortunately; also - welcome to the future), but other languages and tools such as plain HTML+CSS, React or Python probably would be more suitable for the aforementioned purposes.
Similarly, in just Java world there is a ton of tools and libraries and choosing the right ones for the project is an important thing to do at the very start and during the development. Lots of code has been written and there are so many open source and free to use projects that there is no point in writing everything from scratch. Of course, it’s worth knowing how they work or even write some similar tools to learn, but in the end using appropriate tools and frameworks can save us a lot of time and trouble. On the other hand, multiplicity of such projects often makes it hard to come up with a good choice at the very beginning. It all comes with experience, so if you can use your IT colleagues’ knowledge at the university or at work - do it! You can find a vast list of Java frameworks here. Usually the best idea is to just look up the Internet - blog posts and articles - to find the best tool for your purpose - here’s an example. StackOverflow pages are also a great source with often thorough analysis and discussion.
Plain Java projects are hard to maintain and hard to build for others. Always use Maven or Gradle. These tools allow you to set up a complete build with proper packaging, manage all the project’s dependencies and utilize plugins: run tests during build process, generate JavaDoc and do many more things. Thanks to that, a properly configured Maven/Gradle project can take up only a few seconds from downloading it to getting it to run or to being able to modify it easily.
An important thing when creating a tool that others might find useful is to choose the right Java version. Big companies are often reluctant to change the JDK version every few months and to choose the newest ones without them being properly tested and bugfixed. Current Java Support Roadmap makes it more tempting to choose newer and newer Java versions as most of them provide some new features, API enrichment and optimization. The other side of the coin is that already existing projects using older Java version will not be able to use your library until they migrate to new version, and it may not happen as often as it could be expected. Right now, in my opinion, starting with current LTS (long term support) Java would be the best option. All information about release dates and LTS-marked versions can be found in the link above.
The last thing I want to mention is maybe less important for a client or a tool user at first sight, but cannot be overlooked by developers. Packaging. Modules. Code division overall.
A usual way to divide a project is by logical components. Let’s say we have 3 logical layers and the data model in a web application:
com.example.dao
(Data Access Object;
repository is sometimes used instead of dao)com.example.service
(logic and data manipulation)com.example.controller
(serving the actual web contents)com.example.model
(data holding classes -
could be simple objects only transferring data or objects incorporating logic)If you want to get into more details here’s a nice place to start (logical components as defined in Spring Framework).
Now let’s imagine we would like to create an e-commerce application (web shop). A few exemplary data model entities that could be included: Product, Customer, Cart, Order, Payment, Delivery. When using the approach described above, we could end up with a catalog structure like so:
com.example.dao
com.example.service
com.example.controller
com.example.model
…and that’s without interfaces and their implementations. That’s package by layer. Often used, has its advantages, but now that I worked in environments with such approach more, I prefer package by feature.
Package-by-feature’s main goal is to separate the files by their business and not technical context. First let’s see how would the structure above look in this approach:
com.example.product
com.example.customer
com.example.cart
com.example.order
com.example.payment
com.example.delivery
More usual situation is to have to “fix something related to order” than “fix all controllers”,
so this structure makes it easier to find yourself in. Testing modules with concise
and strictly separated logic is easier and more consistent as well.
Another huge advantage is the possibility to actually use the default
access level modifier,
which is not public
on the contrary to popular belief, and it was made like this for a reason.
API (application programming interface) is a commonly used term. Maybe even used to the point that the abbreviation loses its actual meaning, becoming anything the person saying it wanted to say. Emphasis on the word interface is not coincidental. API is what you’re actually exposing in your application. If someone is to use your code, they will do it through the API. If you make everything public, it may turn out that the tool or framework you’ve written is not used as you intended. You create a new version of your library, and you want to change some names, modify method signatures - if public classes are modified, the version will have to increase it’s major part. That’s why most libraries don’t change this part so often - such changes are “breaking”, require changes in client code or even migration. This is the reason not to start coding without analyzing the problem, thinking it over, designing an API and encapsulating it.
Note: “REST API” is commonly used altogether, and it also fits the definition above - this is an externally exposed web interface that has a strict contract, hence the existence of versioning mechanism.
So - having public
modifier only on the classes we want to be used externally of the package
makes it easier to expose an API, view and analyze the code, test it and refactor. In the shop example
above we should probably make only Service
classes public, and the rest could stay hidden.
The last advantage of the package by feature approach is its modularity. With strict boundaries between the packages and loose coupling it is pretty easy to extract some parts of the application to a separate one, maybe even deployed on a separate server. Boom! Microservices.
In the next part we will finally see some code - based on everything said above.
Go to main page Next post