Multithreaded  Programming
Unlike most  other computer languages, Java provides built-in support for multithreaded  programming. A  multithreaded program contains two or more parts that can run concurrently. Each part of such a program is called a  thread, nd each thread defines a separate path of execution. Thus, multithreading  is a specialized form of multitasking.
You are almost certainly acquainted with multitasking, because it is  supported by virtually all modern operating systems. However, there are two  distinct types of multitasking: process-based and thread-based. It is important to  understand the difference between the two. For most readers, process-based multitasking  is the more familiar form. A process is, in essence, a program that is  executing. Thus, process-based multitasking is the feature that allows  your computer to run two or more programs concurrently.  
For example, process-based multitasking enables you to run the Java compiler at the same time that you are using a text editor. In process-based multitasking, a program is the smallest unit of code that can be dispatched by the scheduler.
In a thread-based multitasking environment, the thread is the  smallest unit of dispatchable code. This means that a single program can perform two or  more tasks simultaneously. For instance, a text editor can format text at the  same time that it is printing, as long as these two actions are being performed  by two separate threads. 
Thus, process-based multitasking deals with the "big picture," and  thread-based multitasking handles the details. Multitasking threads require less  overhead than multitasking processes. Processes are heavyweight tasks that  require their own separate address spaces. Interprocess communication is  expensive and limited. Context switching from one process to another is also  costly. 
Threads, on the other hand, are lightweight. They share the same address  space and cooperatively share the same heavyweight process. Interthread  communication is inexpensive, and context switching from one thread to the next  is low cost. While Java programs make use of process-based multitasking  environments, process-based multitasking is not under the control of Java.  However, multithreaded multitasking is.
Multithreading enables you to write very efficient programs that make  maximum use of the CPU, because idle time can be kept to a minimum. This is  especially important for the interactive, networked environment in which Java  operates, because idle time is common. 
For example, the transmission rate of data over a network is much slower  than the rate at which the computer can process it. Even local file system  resources are read and written at a much slower pace than they can be processed  by the CPU. And, of course, user input is much slower than the computer. In a  traditional, single-threaded environment, your program has to wait for each of  these tasks to finish before it can proceed to the next oneeven though the CPU  is sitting idle most of the time. Multithreading lets you gain access to this  idle time and put it to good use.
If you have programmed for operating systems such as Windows 98 or  Windows  2000, then you are already  familiar with multithreaded programming. However, the fact that Java manages  threads makes multithreading especially convenient, because many of the details  are handled for you.
The  Java Thread Model
The Java run-time system depends on threads for many things, and all the  class libraries are designed with multithreading in mind. In fact, Java uses  threads to enable the entire environment to be asynchronous. This helps reduce  inefficiency by preventing the waste of CPU cycles.
The value of a multithreaded environment is best understood in contrast  to its counterpart. Single-threaded systems use an approach called an event  loop with polling.
In this model, a single thread of control runs in an infinite loop,  polling a single event queue to decide what to do next. Once this polling  mechanism returns with, say, a signal that a network file is ready to be read,  then the event loop dispatches control to the appropriate event handler. Until  this event handler returns, nothing else can happen in the system. This wastes  CPU time. It can also result in one part of a program dominating the system and  preventing any other events from being processed. In general, in a  singled-threaded environment, when a thread blocks (that is, suspends  execution) because it is waiting for some resource, the entire program stops  running. 
The benefit of Java's multithreading is that the main loop/polling  mechanism is eliminated. One thread can pause without stopping other parts of your  program. 
For example, the idle time created when a thread reads data from a  network or waits for user input can be utilized elsewhere. Multithreading allows  animation loops to sleep for a second between each frame without causing the  whole system to pause. When a thread blocks in a Java program, only the single  thread that is blocked pauses. All other threads continue to  run.
Threads exist in several states. A thread can be running. It can  be ready to run as soon as it gets CPU time. A running thread can be suspended, which  temporarily suspends its activity. A suspended thread can then be resumed,  allowing it to pick up where it left off. A thread can be blocked  when waiting for a resource. At any time, a thread can be terminated, which  halts its execution immediately. Once terminated, a thread cannot be  resumed.
Thread  Priorities
Java assigns to each thread a priority that determines how that thread  should be treated with respect to the others. Thread priorities are integers that  specify the relative priority of one thread to another. As an absolute value, a  priority is meaningless; a higher-priority thread doesn't run any faster than a  lower-priority thread if it is the only thread running. Instead, a thread's  priority is used to decide when to switch from one running thread to the next.  This is called a context switch. 
The rules that determine when a context switch takes place are  simple:
Ø       A thread  can voluntarily relinquish control. This is  done by explicitly yielding, sleeping, or blocking on pending I/O. In this  scenario, all other threads are examined, and the highest-priority thread that  is ready to run is given the CPU.
Ø       A thread  can be preempted by a higher-priority thread. In this  case, a lower-priority thread that does not yield the processor is simply  preemptedno matter what it is doingby a higher-priority thread. Basically, as  soon as a higher-priority thread wants to run, it does. This is called  preemptive multitasking.
In cases where two threads with the same priority are competing for CPU  cycles, the situation is a bit complicated. For operating systems such as  Windows 98, threads of equal priority are time-sliced automatically in  round-robin fashion. For other types of operating systems, threads of equal  priority must voluntarily yield control to their peers.  If they don't, the other threads will  not run.
Problems can arise from the differences in the way that operating systems  context-switch threads of equal priority.
Synchronization
Because multithreading introduces an asynchronous behavior to your  programs, there must be a way for you to enforce synchronicity when you need it.  For example, if you want two threads to communicate and share a complicated data  structure, such as a linked list, you need some way to ensure that they don't  conflict with each other. That is, you must prevent one thread from writing data  while another thread is in the middle of reading it.  
For this purpose, Java implements an elegant twist on an age-old model of  interprocess synchronization: the monitor. The monitor is a control  mechanism first defined by C.A.R. Hoare. You can think of a monitor as a very  small box that can hold only one thread. Once a thread enters a monitor, all  other threads must wait until that thread exits the monitor.  
In this way, a monitor can be used to protect a shared asset from being  manipulated by more than one thread at a time. Most multithreaded systems expose  monitors as objects that your program must explicitly acquire and manipulate.  
Java provides a cleaner solution. There is no class "Monitor"; instead,  each object has its own implicit monitor that is automatically entered when one  of the object's synchronized methods is called. Once a thread is inside a  synchronized method, no other thread can call any other synchronized method on  the same object. This enables you to write very clear and concise multithreaded  code, because synchronization support is built in to the  language.
Messaging
After you divide your program into separate threads, you need to define  how they will communicate with each other. When programming with most other  languages, you must depend on the operating system to establish communication  between threads. This, of course, adds overhead. By contrast, Java provides a  clean, low-cost way for two or more threads to talk to each other, via calls to  predefined methods that all objects have. Java's messaging system allows a  thread to enter a synchronized method on an object, and then wait there until  some other thread explicitly notifies it to come  out.
The  Thread Class and the Runnable Interface
Java's multithreading system is built upon the Thread class, its  methods, and its companion interface, Runnable. Thread encapsulates a thread  of execution. Since you can't directly refer to the ethereal state of a running  thread, you will deal with it through its proxy, the Thread instance that  spawned it. To create a new thread, your program will either extend Thread  or implement the Runnable interface.
The Thread class defines several methods that help manage threads.  The ones that will be used in this Post are shown  here:
Method          Meaning
getName        Obtain  a thread's name.
getPriority      Obtain a thread's  priority.
isAlive            Determine  if a thread is still running.
join                 Wait  for a thread to terminate.
run                   Entry point for the thread.
sleep              Suspend  a thread for a period of time.
start                Start  a thread by calling its run method.
Thus far, all the examples in this book have used a single thread of  execution. The remainder of this Post explains how to use Thread and Runnable  to create and manage threads, beginning with the one thread that all Java programs  have: the main thread.
 
No comments:
Post a Comment