Callable, Future, and Thread Pools in Java Explained Simply ExecutorService becomes powerful when you start using Callable and Future. They let you run tasks in the background and get results when ready. Runnable vs Callable Runnable runs a task but returns nothing. Callable runs a task and returns a value. Example: ExecutorService executor = Executors.newFixedThreadPool(3); Callable<Integer> task = () -> { System.out.println("Running in " + Thread.currentThread().getName()); return 5 * 2; }; Future<Integer> result = executor.submit(task); System.out.println("Result: " + result.get()); executor.shutdown(); Output: Running in pool-1-thread-1 Result: 10 Future.get() waits until the result is ready. Why use Callable and Future You can get return values from background tasks. You can handle timeouts with get(timeout, TimeUnit.SECONDS). You can check if a task is done using isDone(). Thread Pools simplify everything ExecutorService pool = Executors.newFixedThreadPool(4); Controls how many threads run at once. Reuses threads to save memory. Perfect for web requests, DB operations, or scheduled jobs. Best practices Always shut down the executor (shutdown() or shutdownNow()). Don’t block the main thread unnecessarily. Use fixed pools for predictable workloads. ExecutorService + Callable + Future = powerful, efficient concurrency. What kind of background tasks do you usually handle in your backend systems? #Java #SpringBoot #Programming #SoftwareDevelopment #Cloud #AI #Coding #Learning #Tech #Technology #WebDevelopment #Microservices #API #Database #SpringFramework #Hibernate #MySQL #BackendDevelopment #CareerGrowth #ProfessionalDevelopment
How to Use Callable and Future in Java for Efficient Concurrency
More Relevant Posts
-
Mastering Java Streams: Write Cleaner and Faster Code Loops are fine. But Streams change how you process data. They help you write shorter, cleaner, and more functional code. Here’s a simple comparison: Without Streams List<String> names = List.of("Umar", "Ali", "Sara", "Rehan"); List<String> result = new ArrayList<>(); for (String name : names) { if (name.startsWith("A")) { result.add(name.toUpperCase()); } } With Streams List<String> result = names.stream() .filter(n -> n.startsWith("A")) .map(String::toUpperCase) .toList(); Same result. Half the code. Easier to read. Key Stream operations you should know filter() – Select elements that meet a condition. map() – Transform elements to a new form. sorted() – Sort data based on custom logic. collect() – Gather results into a list or map. reduce() – Combine all elements into one result (like sum or concatenation). Example of reduce: int sum = List.of(1, 2, 3, 4) .stream() .reduce(0, Integer::sum); Why it matters Streams make your code expressive and less error-prone. Once you get used to them, you’ll never go back to traditional loops. The best part? Streams work great with parallelism, giving you performance boosts with minimal effort. Do you prefer Streams or traditional loops in your daily work? Why #Java #SpringBoot #Programming #SoftwareDevelopment #Cloud #AI #Coding #Learning #Tech #Technology #WebDevelopment #Microservices #API #Database #SpringFramework #Hibernate #MySQL #BackendDevelopment #CareerGrowth #ProfessionalDevelopment
To view or add a comment, sign in
-
Proxy Pattern in Java Use the Proxy Pattern when you want to control access to an object. You do not modify the original class. You place a proxy in front of it. Common situations • Logging • Security checks • Lazy loading • Caching Example interface Service { void process(); } class RealService implements Service { public void process() { System.out.println("Processing data"); } } class ServiceProxy implements Service { private RealService realService; public void process() { if (realService == null) { realService = new RealService(); } System.out.println("Access check done"); realService.process(); } } Usage Service service = new ServiceProxy(); service.process(); What happened • Proxy controls access • Real object is created only when needed • You add logic without touching the real class Benefits • Cleaner code • Better control • Improved performance when used for lazy loading or caching Takeaway Proxy adds control without changing the original class. #Java #SpringBoot #Programming #SoftwareDevelopment #Cloud #AI #Coding #Learning #Tech #Technology #WebDevelopment #Microservices #API #Database #SpringFramework #Hibernate #MySQL #BackendDevelopment #CareerGrowth #ProfessionalDevelopment
To view or add a comment, sign in
-
Java Performance Tuning Basics Every Developer Should Know Fast code is not an accident. It comes from understanding how Java runs your program and how to remove slow parts early. Here are simple performance habits that make real impact. 1. Choose the right data structure ArrayList is faster for reading. LinkedList is slower for reading. HashMap gives constant time lookups. Picking the right one saves time across your application. 2. Avoid unnecessary object creation Objects cost memory. Frequent creation increases garbage collection work. Reuse objects when possible, especially in loops. 3. Use StringBuilder for concatenation StringBuilder sb = new StringBuilder(); sb.append("Hello"); Faster and memory efficient compared to repeated string concatenation. 4. Cache repeated results If you compute something often, store the result and reuse it. This avoids extra CPU work. 5. Use streams carefully Streams improve readability, but they can be slower for simple loops. Test performance before switching everything to Streams. 6. Avoid synchronization where not needed Locking slows down execution. Use synchronized blocks only for shared mutable data. 7. Profile before optimizing Use tools like VisualVM or JProfiler to find real bottlenecks. Do not guess. Measure. 8. Tune JVM only when needed Flags like -Xms, -Xmx, and GC settings help, but only after profiling. Do not tweak without data. Takeaway Small optimizations add up. Measure, adjust, and write code that performs predictably under load. #Java #SpringBoot #Programming #SoftwareDevelopment #Cloud #AI #Coding #Learning #Tech #Technology #WebDevelopment #Microservices #API #Database #SpringFramework #Hibernate #MySQL #BackendDevelopment #CareerGrowth #ProfessionalDevelopment
To view or add a comment, sign in
-
Abstract Factory Pattern in Java Use Abstract Factory when the system needs to create related objects. You keep object creation separate from business logic. Your code becomes flexible and easier to extend. Example. You need UI components for two platforms. Each platform has its own style. Step 1. Product interfaces interface Button { void render(); } interface Dropdown { void open(); } Step 2. Concrete products class WindowsButton implements Button { public void render() { System.out.println("Windows button"); } } class MacButton implements Button { public void render() { System.out.println("Mac button"); } } class WindowsDropdown implements Dropdown { public void open() { System.out.println("Windows dropdown"); } } class MacDropdown implements Dropdown { public void open() { System.out.println("Mac dropdown"); } } Step 3. Abstract factory interface UIFactory { Button createButton(); Dropdown createDropdown(); } Step 4. Concrete factories class WindowsFactory implements UIFactory { public Button createButton() { return new WindowsButton(); } public Dropdown createDropdown() { return new WindowsDropdown(); } } class MacFactory implements UIFactory { public Button createButton() { return new MacButton(); } public Dropdown createDropdown() { return new MacDropdown(); } } Client usage UIFactory factory = new WindowsFactory(); Button button = factory.createButton(); Dropdown dropdown = factory.createDropdown(); button.render(); dropdown.open(); Key points • You create related objects in one place. • Your client code depends on interfaces, not on concrete implementations. • You can switch object families without changing business logic. Takeaway Use Abstract Factory when object creation must stay consistent across a product family. #Java #SpringBoot #Programming #SoftwareDevelopment #Cloud #AI #Coding #Learning #Tech #Technology #WebDevelopment #Microservices #API #Database #SpringFramework #Hibernate #MySQL #BackendDevelopment #CareerGrowth #ProfessionalDevelopment
To view or add a comment, sign in
-
Adapter Pattern in Java Problem You have code that works with one type of interface. You get a new class that does similar work but exposes a different method. Your existing code cannot use it directly. Adapter Pattern solves this. It lets you connect two incompatible classes without touching existing code. Example You already have this interface: interface PaymentProcessor { void pay(int amount); } A new payment service arrives but uses a different method: class NewPaymentService { void makePayment(int amount) { System.out.println("Payment processed"); } } Create an adapter that matches your existing interface: class PaymentAdapter implements PaymentProcessor { private NewPaymentService service; public PaymentAdapter(NewPaymentService service) { this.service = service; } public void pay(int amount) { service.makePayment(amount); } } Use it like this: PaymentProcessor processor = new PaymentAdapter(new NewPaymentService()); processor.pay(500); Key points • Adapter converts one interface into another. • It avoids modifying existing working code. • It helps integrate new systems smoothly. When to use • When a new class does not match existing method signatures. • When you integrate legacy code with new APIs. Takeaway The Adapter Pattern protects your codebase. You add new functionality without breaking anything. #Java #SpringBoot #Programming #SoftwareDevelopment #Cloud #AI #Coding #Learning #Tech #Technology #WebDevelopment #Microservices #API #Database #SpringFramework #Hibernate #MySQL #BackendDevelopment #CareerGrowth #ProfessionalDevelopment
To view or add a comment, sign in
-
Prototype Pattern in Java: Cloning Objects the Smart Way Imagine this. You’ve created a heavy object. It takes time to load data, read files, or fetch configurations. Now you need 10 more copies of it. Would you rebuild each one from scratch? Of course not. That’s where the Prototype Pattern comes in. It lets you clone an existing object instead of creating a new one. It saves time and memory when object creation is expensive. Example: class Document implements Cloneable { private String name; private String content; public Document(String name, String content) { this.name = name; this.content = content; } public Document clone() throws CloneNotSupportedException { return (Document) super.clone(); } public String toString() { return name + ": " + content; } } public class Main { public static void main(String[] args) throws Exception { Document doc1 = new Document("Report", "Quarterly sales data"); Document doc2 = doc1.clone(); System.out.println(doc1); System.out.println(doc2); } } Output: Report: Quarterly sales data Report: Quarterly sales data Both objects are identical, but stored separately in memory. Why it matters Saves resources when creating large objects. Helps copy complex states or configurations easily. Works well in frameworks where objects are created in bulk. Where you’ll see it Game development (cloning entities). UI templates. Caching and object pooling systems. Simple rule: When object creation is costly, cloning is your friend. Have you used cloning in any of your projects? #Java #SpringBoot #Programming #SoftwareDevelopment #Cloud #AI #Coding #Learning #Tech #Technology #WebDevelopment #Microservices #API #Database #SpringFramework #Hibernate #MySQL #BackendDevelopment #CareerGrowth #ProfessionalDevelopment
To view or add a comment, sign in
-
Java Lambda Expressions, A Simple Way to Write Cleaner Code Lambdas help you remove unnecessary code. They replace anonymous classes with short, readable expressions. They make your logic easy to understand. Here is the idea. A lambda is a short block of code that you can pass around like data. Basic form (parameter) -> expression Example with threads Runnable task = () -> System.out.println("Task running"); new Thread(task).start(); Cleaner than the old style new Thread(new Runnable() { public void run() { System.out.println("Task running"); } }).start(); Filtering a list List<Integer> numbers = List.of(10, 15, 20, 25); List<Integer> result = numbers.stream() .filter(n -> n > 15) .toList(); Sorting data List<String> names = List.of("Umar", "Ali", "Sara"); names.stream() .sorted((a, b) -> a.compareTo(b)) .forEach(System.out::println); Why lambdas help • Less code • Clear intent • Better use of Streams • Easy to combine with functional interfaces Common use cases Filtering. Mapping. Sorting. Background tasks. Event handling. Takeaway Use lambda expressions when your logic is small and focused. They make Java feel cleaner and more modern. #Java #SpringBoot #Programming #SoftwareDevelopment #Cloud #AI #Coding #Learning #Tech #Technology #WebDevelopment #Microservices #API #Database #SpringFramework #Hibernate #MySQL #BackendDevelopment #CareerGrowth #ProfessionalDevelopment
To view or add a comment, sign in
-
How Java Manages Memory (And Why You Should Care) Good code isn’t just about logic. It’s about how efficiently your program uses memory. Java does a lot for you behind the scenes, but knowing how memory works helps you write faster, more stable applications. Java memory is divided into two main areas: 1. Stack Memory Stores method calls and local variables. Each thread has its own stack. Fast and automatically cleared when a method ends. Example: int a = 10; int b = 20; int sum = a + b; All of these live in the stack. 2. Heap Memory Stores objects and instance variables. Shared among all threads. Managed by the Garbage Collector (GC). Example: User user = new User("Umar"); user reference lives on the stack, but the User object lives on the heap. Garbage Collection (GC) Java automatically frees memory from unused objects. You don’t need to manually delete anything. But… you still need to write memory-friendly code. Pro tips for developers Avoid unnecessary object creation. Release large data structures when no longer needed. Use profiling tools like VisualVM or JConsole to monitor memory. Understanding memory helps you prevent leaks, optimize performance, and build scalable systems. How well do you understand what happens inside the JVM when your code runs? #Java #SpringBoot #Programming #SoftwareDevelopment #Cloud #AI #Coding #Learning #Tech #Technology #WebDevelopment #Microservices #API #Database #SpringFramework #Hibernate #MySQL #BackendDevelopment #CareerGrowth #ProfessionalDevelopment
To view or add a comment, sign in
-
🚀 Java Tip of the Day: HashMap vs LinkedHashMap — Which One Should You Use? If you've ever added elements to a HashMap and noticed the order just vanished — you’re not alone 😅 Let’s decode the mystery between these two powerful collection classes 👇 🔹 1️⃣ HashMap Stores data in key-value pairs using hashing. No ordering guarantee — elements can appear in any sequence. Allows one null key and multiple null values. Fast lookups — constant time O(1) for get() and put(). Lightweight and ideal for quick data access where order isn’t important. 💬 Example Use Case: Counting occurrences, caching data, storing config values. 🔹 2️⃣ LinkedHashMap Inherits from HashMap but maintains a linked list of entries internally. Preserves insertion order (or access order if you enable access-order mode). Slightly slower than HashMap, but the order consistency is worth it. Also allows one null key and multiple null values. Great choice when predictable iteration is needed. 💬 Example Use Case: Maintaining order of user input, building LRU cache, or when output order matters. ⚙️ 3️⃣ Performance Insight HashMap -- Faster, unordered, best for general lookups. LinkedHashMap --- Predictable order with a small performance trade-off. Both are non-synchronized → use Collections.synchronizedMap() or ConcurrentHashMap in multi-threaded environments. #Java #CollectionsFramework #HashMap #LinkedHashMap #CodingTips #JavaDeveloper #FullStackDeveloper #ProgrammingCommunity #LearningEveryday #TechTalk
To view or add a comment, sign in
-
How I Learned to Control Async Tasks in Java the Right Way When I started working with Java threads, I made one mistake repeatedly. I created new threads for every task — thinking more threads meant faster execution. It didn’t. It only made the system unstable. Then I discovered ExecutorService with Callable and Future. That’s when things clicked. The smarter way: Use Callable when you need a result from your thread. Use Future to track that result later. Example: ExecutorService executor = Executors.newFixedThreadPool(3); Callable<Integer> task = () -> { Thread.sleep(1000); return 42; }; Future<Integer> future = executor.submit(task); System.out.println("Result: " + future.get()); executor.shutdown(); Here’s what’s happening: submit() runs your task in a thread. get() waits for the result when needed. Threads are reused inside the pool, avoiding memory waste. Why this matters Thread pools make your system predictable. You decide how many threads exist, no more uncontrolled spawns. Pro tip Never block your main thread waiting on too many Future.get() calls. Use async patterns or CompletableFuture for large workloads. Once you start managing tasks with Callable and thread pools, you move from writing code to orchestrating execution. What’s your experience using thread pools in production systems? #Java #SpringBoot #Programming #SoftwareDevelopment #Cloud #AI #Coding #Learning #Tech #Technology #WebDevelopment #Microservices #API #Database #SpringFramework #Hibernate #MySQL #BackendDevelopment #CareerGrowth #ProfessionalDevelopment
To view or add a comment, sign in
Explore content categories
- Career
- Productivity
- Finance
- Soft Skills & Emotional Intelligence
- Project Management
- Education
- Technology
- Leadership
- Ecommerce
- User Experience
- Recruitment & HR
- Customer Experience
- Real Estate
- Marketing
- Sales
- Retail & Merchandising
- Science
- Supply Chain Management
- Future Of Work
- Consulting
- Writing
- Economics
- Artificial Intelligence
- Employee Experience
- Workplace Trends
- Fundraising
- Networking
- Corporate Social Responsibility
- Negotiation
- Communication
- Engineering
- Hospitality & Tourism
- Business Strategy
- Change Management
- Organizational Culture
- Design
- Innovation
- Event Planning
- Training & Development