java 线程 操作系统线程_线程基础:线程(1)——操作系统和线程原理

1、概述

我在写“系统间通信技术专栏”的时候,收到很多读者的反馈。其中有一部分读者希望我抽空写一写自己关于对Java线程的使用经验和总结。巧的是,这个月我所在的技术团队也有很多同事跟我讨论关于JAVA中线程的操作。正好本月我工作也不是很忙,除了继续推进我的重点专栏“系统间通信技术”外,可以更多的空余时间跟各位读者分享自己对JAVA线程技术的理解和使用经验。

本人不才,应读者要求新开专栏,与各位读者分享自己对JAVA线程技术的理解和使用经验。这个专栏将分成两个部分:线程基础知识和锁知识。专栏的难度应该是我所开专栏中难度最低的一个,着重于线程基础知识的讲解,更适合JAVA初学者阅读,目的是希望能够帮助大家提高codeing水平和程序质量。如果您是经验老道的高手也欢迎和本人讨论相关问题,对本人文章的论点进行勘误,您的支持是我写作的关键动力。当然本人的更多精力还是放在继续完成“系统间通信技术”这个专栏。(本系列的博客文章将不会置顶)

2、操作系统和线程原理

线程是一个操作系统级别的概念。JAVA语言(包括其他编程语言)本身不创建线程;而是调用操作系统层提供的接口创建、控制、销毁线程实例。

df3e86a124c7247b1628b10f614b33d5.png

首先要说明的是,根据操作系统的不同(Windows/Unix/Linux/其他),他们所支持的线程底层实现和操作效果也是不尽相同的。不过一个操作系统支持的线程至少会有四种状态:就绪、执行、阻塞和终结。线程在四种状态下进行切换,都是要消耗不少的CPU计算能力的。

并且根据操作系统使用线程的进程的不一样,线程还分为用户线程和操作系统线程。操作系统线程(内核线程),是指操作系统内核为了完成硬件接口层操作,由操作系统内核创建的线程:例如I/O操作的内核线程,这些线程应用程序是不能干预的;用户线程,是指用户安装/管理的应用程序,为执行某一种操作,而由这个应用程序创建的线程。后文我们讨论的JAVA线程,都是用户级线程。

线程在创建时,操作系统不会为这个线程分配独立的资源(除了必要的数据支撑)。一个应用程序(进程)下的所有线程,都是共享这个应用程序(进程)中的资源,例如这个应用程序的CPU资源、I/O资源、内存资源。

现在基本上主流操作系统都支持多线程实现。即一个应用程序中(一个进程中),可以创建多个线程。一个应用程序下,各个线程间都可以进行通讯、可以进行状态互操作。且一个进程中,至少有一个线程存在。

3、JAVA中最简单的线程示例

JAVA中提供了丰富的操作系统接口实现,帮助我们进行线程操作。这些实现分布在java的java.lang基础包、java.io基础包和java.util.concurrent工具包当中;这个专栏所涉及到的代码示例也会从易到难向大家进行演示。我们先来看看JAVA中最基本的线程操作实现(高手请绕行)。

3-1、Thread父类

java.lang.Thread类是JAVA中用于实现线程操作的最基本的类之一。您可以创建一个集成Thread类的子类来定义您自己的线程实现:

1 packagetest.thread.base;2

3 importorg.apache.commons.logging.Log;4 importorg.apache.commons.logging.LogFactory;5 importorg.apache.log4j.BasicConfigurator;6

7 public class MyDefindThread extendsThread {8

9 static{10 BasicConfigurator.configure();11 }12

13 /**

14 * 日志。一定要使用Log4j才行。否则你就用System.out吧15 */

16 private static final Log LOGGER= LogFactory.getLog(MyDefindThread.class);17

18 /*(non-Javadoc)19 * @see java.lang.Thread#run()20 */

21 @Override22 public voidrun() {23 Long threadId = this.getId();24 MyDefindThread.LOGGER.info("线程(" + threadId + ")做了一些事情,然后结束了。");25 }26

27 public static void main(String[] args) throwsException {28 newMyDefindThread().start();29 }30 }

3-2、Runable接口

除了可以继承java.lang.Thread类来定义自己的线程外,您还可以实现java.lang.Runnable接口来定义一个线程(一般情况,我们优先使用这种方式):

1 packagetest.thread.base;2

3 importorg.apache.commons.logging.Log;4 importorg.apache.commons.logging.LogFactory;5 importorg.apache.log4j.BasicConfigurator;6

7 public class MyDefindRunnable implementsRunnable {8

9 static{10 BasicConfigurator.configure();11 }12

13 /**

14 * 日志。一定要使用Log4j才行。否则你就用System.out吧15 */

16 private static final Log LOGGER= LogFactory.getLog(MyDefindThread.class);17

18 /*(non-Javadoc)19 * @see java.lang.Runnable#run()20 */

21 @Override22 public voidrun() {23 //获取当前线程的ID

24 long threadId =Thread.currentThread().getId();25 MyDefindRunnable.LOGGER.info("线程(" + threadId + ")做了一些事情,然后结束了。");26 }27

28 public static void main(String[] args) throwsException {29 new Thread(newMyDefindRunnable()).start();30 }31 }

以上的两段代码都没有太多可讲解的。您可以在调试环境下观察到JAVA应用程序是如何运行线程的:

4ead0238549eb1ae69f1b39dc16f18dc.png

4、下文介绍

下一篇文章中,我们将继续介绍Java所支持的线程间基本互操作,包括:阻塞、唤醒、终止等操作;然后介绍Java原生线程池的工作原理和基本操作。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java线程和操作系统线程是有关系的,Java线程是建立在操作系统线程之上的,Java虚拟机(JVM)会把Java线程映射到操作系统线程上执行。每个Java线程都会创建一个对应的操作系统线程,Java线程的调度由JVM负责,而操作系统线程的调度则由操作系统负责。 Java线程和操作系统线程的区别主要有以下几个方面: 1. 调度方式不同:Java线程的调度由JVM负责,而操作系统线程的调度则由操作系统负责。 2. 创建和销毁的开销不同:Java线程的创建和销毁比操作系统线程开销要小,因为操作系统线程的创建和销毁需要切换内核态和用户态,这个开销比较大,而Java线程的创建和销毁只需要在用户态完成即可。 3. 并发能力不同:Java线程的并发能力比操作系统线程要弱,因为Java线程是建立在操作系统线程之上的,Java线程的调度会受到JVM的限制,而操作系统线程则可以充分利用操作系统的调度能力。 4. 内存消耗不同:Java线程的内存消耗比操作系统线程要小,因为Java线程的栈空间可以动态调整,而操作系统线程的栈空间是固定的。 总的来说,Java线程是一种轻量级的线程,它在操作系统线程之上增加了一层抽象,可以更方便地进行线程的管理和调度。但是,Java线程的并发能力比操作系统线程要弱,适合于一些轻量级的并发操作。而对于一些需要充分利用多核CPU的高并发场景,操作系统线程则更加适合。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值