进程与程序的区别
1. 进程是程序的执行,属于动态,程序是完成某个功能的指令的集合,是静态的
2.进程的存在是暂时的,程序的存在是永久的。
3.进程=程序+数据+PCB (进程控制块,process control block),即进程是一个程序及其数据在处理机上顺序地执行时所发生的活动。
4.一个程序可以对应多个进程
进程的特征:
• 进程是一个能独立运行的单位,能与其它进程并行地活动。
• 进程是竞争计算机系统有限资源的基本单位,是一个可调度的实体。也是进行处理机调度的基本单位。
• 进程是操作系统进行资源分配的单位。进程是拥有自己资源的单元体。
线程的定义
线程是进程内一个相对独立的、可调度的执行单元。在一个进程内部可以有多个线程,即有多个执行流,这多个线程是共享进程的内存空间和系统资源。它们轮换使用这块内存空间。
线程的性质:
线程是进程内的一个相对独立的可执行单元
线程是操作系统中的基本调度单元,因此线程中应包含有调度所需的必要信息
由于线程是被调度的基本单元,而进程不是调度的单元。所以每个进程在创建时,至少需要同时为该进程创建一个线程。也就是说进程中至少要有一个或一个以上线程,否则该进程无法被调度执行。
需要时,线程可以创建其他线程
进程是被分给并拥有资源的基本单元,同一进程内的多个线程共享该进程的资源。但线程并不拥有资源,只是使用它们
由于共享资源(包括数据和文件),所以线程间需要通信和同步机制。
线程有生命期,有诞生和死亡。在生命期中有状态的变化。
多线程的优点:
1) 提高应用程序响应。这对图形界面的程序尤其有意义,当一个操作耗时很长时,整个系统都会等待这个操作,此时程序不会响应键盘、鼠标、菜单的操作,而使用多线程技术,将耗时长的操作(time consuming)置于一个新的线程,可以避免这种尴尬的情况。
2) 使多CPU系统更加有效。操作系统会保证当线程数不大于CPU数目时,不同的线程运行于不同的CPU上。
3) 改善程序结构。一个既长又复杂的进程可以考虑分为多个线程,成为几个独立或半独立的运行部分,这样的程序会利于理解和修改。
多进程的优点:
在同一个时间里,同一个计算机系统中如果允许两个或两个以上的进程处于运行状态,这便是多任务。现代的操作系统几乎都是多任务操作系统,能够同时管理多个进程的运行。
主线程
在Java程序启动时,一个线程立刻运行,该线程通常称为程序的主线程。
主线程的重要性体现在两个方面:
n 它是产生其他子线程的线程。
n 通常它必须最后完成执行,因为它执行各种关闭动作。
线程的创建
n 通过以下两种方法创建Thread 对象:
¨ 声明一个 Thread 类的子类,并覆盖 run() 方法。
class MyThread extends Thread {
public void run( ) {/* 覆盖该方法*/ } }
MyThread t = new MyThread();
¨ 声明一个实现 Runnable 接口的类,并实现 run() 方法(注:此种做法只实现的线程任务)。
class MyRunnable implementsRunnable{
publicvoid run( ) {/* 实现该方法*/ }
MyRunnable mr=new MyRunnable ();
Thread t=new Thread(mr);
n 要启动一个新线程,使用 start() 方法,如:
n MyThread t = new MyThread(); t.start();
n MyRunnable mr=new MyRunnable ();
Threadt=new Thread(mr); t.start();
n 在调用 start() 方法时,将创建一个新的控制线程,并加入线程队列,接着它将调用 run() 方法。
n run() 方法中的代码定义执行线程所需的功能。
线程同步
§ 有时两个或多个线程可能会试图同时访问一个资源
§ 例如,一个线程可能尝试从一个文件中读取数据,而另一个线程则尝试在同一文件中修改数据
§ 在此情况下,数据可能会变得不一致
§ 为了确保在任何时间点一个共享的资源只被一个线程使用,使用了“同步”
两种方式实现同步:
§ 使用同步方法
synchronized void methodA() { }
– 使用同步代码块
void methodA() {
synchronized(object) {
//要同步的语句
}}
§ 只要是多线程共享资源,都必须考虑同步。
多线程同步是什么?
在多线程程序下,同步能控制对共享资源的访问。如果没有同步,当一个 Java 线程在修改一个共享变量时,另外一个线程正在使用或者更新同一个变量,这样容易导致程序出现错误的结果。
当一个同步方法已经执行,线程能够调用对象上的非同步实例方法吗?
可以,一个非同步方法总是可以被调用而不会有任何问题。实际上,Java 没有为非同步方法做任何检查,锁对象仅仅在同步方法或者同步代码块中检查。如果一个方法没有声明为同步,即使你在使用共享数据 Java 照样会调用,而不会做检查是否安全,所以在这种情况下要特别小心。一个方法是否声明为同步取决于临界区访问(critial section access),如果方法不访问临界区(共享资源或者数据结构)就没必要声明为同步的。
在一个对象上两个线程可以调用两个不同的同步实例方法吗?
不能,因为一个对象已经同步了实例方法,线程获取了对象的对象锁。所以只有执行完该方法释放对象锁后才能执行其它同步方法
前台和后台线程 使用Thread建立的线程默认情况下是前台线程,在进程中,只要有一个前台线程未退出,进程就不会终止。主线程就是一个前台线程。而后台线程不管线程是否结束,只要所有的前台线程都退出(包括正常退出和异常退出)后,进程就会自动终止。一般后台线程用于处理时间较短的任务,如在一个Web服务器中可以利用后台线程来处理客户端发过来的请求信息。而前台线程一般用于处理需要长时间等待的任务,如在Web服务器中的监听客户端请求的程序,或是定时对某些系统资源进行扫描的程序。下面的代码演示了前台和后台线程的区别。
{
Thread.Sleep( 3000 );
}
Thread thread = new Thread(myStaticThreadMethod);
// thread.IsBackground = true;
thread.Start();
如果运行上面的代码,程序会等待3秒后退出,如果将注释去掉,将thread设成后台线程,则程序会立即退出。
要注意的是,必须在调用Start方法之前设置线程的类型,否则一但线程运行,将无法改变其类型。
通过BeginXXX方法运行的线程都是后台线程。