2026/2/5 21:15:02
网站建设
项目流程
辽宁住房和城乡建设部网站,东营市建设信息网站,整站seo定制,网站开发培训中心目录
第1关#xff1a;线程的状态与调度
第2关#xff1a;常用函数#xff08;一#xff09;
第3关#xff1a;常用函数#xff08;二#xff09;
第1关#xff1a;线程的状态与调度 相关知识 为了完成本关你需要掌握#xff1a; 1.线程的状态与调度#xff1b; …目录第1关线程的状态与调度第2关常用函数一第3关常用函数二第1关线程的状态与调度相关知识为了完成本关你需要掌握1.线程的状态与调度2.线程执行的优先级。线程的状态与调度如果看懂下图你对线程的了解就会更上一层楼。当我们使用new关键字新建一个线程这个时候线程就进入了新建状态New也就是图中未启动状态调用start方法启动线程这个时候就进入了可运行状态也就是就绪状态Runnable就绪状态获取了CPU资源开始执行run方法就进入了运行状态Running阻塞状态Blocked阻塞状态是线程因为某种原因放弃CPU使用权暂时停止运行。直到线程进入就绪状态才有机会转到运行状态。阻塞的情况分三种等待阻塞运行的线程执行wait()方法JVM会把该线程放入等待池中。(wait会释放持有的锁)同步阻塞运行的线程在获取对象的同步锁时若该同步锁被别的线程占用则JVM会把该线程放入锁池中其他阻塞运行的线程执行sleep()或join()方法或者发出了I/O请求时JVM会把该线程置为阻塞状态。当sleep()状态超时、join()等待线程终止或者超时、或者I/O处理完毕时线程重新转入就绪状态。注意,sleep是不会释放持有的锁死亡状态Dead线程执行完了或者因异常退出了run()方法该线程结束生命周期。线程优先级我们知道多个线程的执行是随机的在一个时段执行哪个线程是看哪一个线程在此时拥有CPU的执行权限。在Java中线程有优先级优先级高的线程会获得较多的运行机会。Java线程的优先级用整数表示取值范围是1~10Thread类有以下三个静态常量static int MAX_PRIORITY线程可以具有的最高优先级取值为10。static int MIN_PRIORITY线程可以具有的最低优先级取值为1。static int NORM_PRIORITY分配给线程的默认优先级取值为5。如果要设置和获取线程的优先级可以使用Thread类的setPriority()和getPriority()方法。每个线程都有默认的优先级。主线程的默认优先级为Thread.NORM_PRIORITY。线程的优先级有继承关系比如A线程中创建了B线程那么B将和A具有相同的优先级。JVM提供了10个线程优先级但与常见的操作系统都不能很好的映射。如果希望程序能移植到各个操作系统中应该仅仅使用Thread类有以下三个静态常量作为优先级这样能保证同样的优先级采用了同样的调度方式。关于线程调度与优先级你还需要了解线程睡眠Thread.sleep(long millis)方法使线程转到阻塞状态。millis参数设定睡眠的时间以毫秒为单位。当睡眠结束后就转为就绪Runnable状态。sleep()平台移植性好。线程等待Object类中的wait()方法导致当前的线程等待直到其他线程调用此对象的notify()方法或notifyAll()唤醒方法。这个两个唤醒方法也是Object类中的方法行为等价于调用wait(0)一样。线程让步Thread.yield()方法暂停当前正在执行的线程对象把执行机会让给相同或者更高优先级的线程。线程加入join()方法等待其他线程终止。在当前线程中调用另一个线程的join()方法则当前线程转入阻塞状态直到另一个进程运行结束当前线程再由阻塞转为就绪状态。线程唤醒Object类中的notify()方法唤醒在此对象监视器上等待的单个线程。如果所有线程都在此对象上等待则会选择唤醒其中一个线程。选择是任意性的并在对实现做出决定时发生。线程通过调用其中一个wait方法在对象的监视器上等待。 直到当前的线程放弃此对象上的锁定才能继续执行被唤醒的线程。被唤醒的线程将以常规方式与在该对象上主动同步的其他所有线程进行竞争例如唤醒的线程在作为锁定此对象的下一个线程方面没有可靠的特权或劣势。类似的方法还有一个notifyAll()唤醒在此对象监视器上等待的所有线程。第2关常用函数一相关知识本关你需要掌握sleep与join函数的用法。sleep()函数sleep(long millis): 在指定的毫秒数内让当前正在执行的线程休眠暂停执行。使用方式很简单在线程的内部使用Thread.sleep(millis)即可。sleep()使当前线程进入停滞状态阻塞当前线程让出CPU的使用目的是不让当前线程独自霸占该进程所获的CPU资源以留一定时间给其他线程执行的机会在sleep()休眠时间期满后该线程不一定会立即执行这是因为其它线程可能正在运行而且没有被调度为放弃执行除非此线程具有更高的优先级。join()函数join函数的定义是指等待线程终止。我们在运行线程的时候可能会遇到在主线程中运行子线程主线程需要获取子线程最终执行结果的情况。但是有很多时候子线程进行了很多耗时的操作主线程往往先于子线程结束这个时候主线程就获取不到子线程的最终执行结果了。使用join函数可以解决这一问题。我们通过两个示例来理解不使用join函数的版本class MyThread extends Thread { private String name; public MyThread(String name) { this.name name; } public void run() { System.out.println(子线程开始运行); for (int i 0; i 5; i) { System.out.println(子线程 name 运行 i); try { Thread.sleep(30); } catch (InterruptedException e) { e.printStackT\frace(); } } System.out.println(子线程结束); } } public class Test { public static void main(String[] args) { Thread t new MyThread(子线程A); t.start(); for (int i 0; i 5; i) { System.out.println(主线程执行 i); } System.out.println(主线程结束); } }输出结果每次都不一样主线程执行0主线程执行1主线程执行2主线程执行3主线程执行4主线程结束子线程开始运行子线程子线程A运行0子线程子线程A运行1子线程子线程A运行2子线程子线程A运行3子线程子线程A运行4子线程结束可以发现每次运行主线程都是先于子线程结束的。使用join函数package step1; class MyThread extends Thread { private String name; public MyThread(String name) { this.name name; } public void run() { System.out.println(子线程开始运行); for (int i 0; i 5; i) { System.out.println(子线程 name 运行 i); try { Thread.sleep(30); } catch (InterruptedException e) { e.printStackT\frace(); } } System.out.println(子线程结束); } } public class Test { public static void main(String[] args) { Thread t new MyThread(子线程A); t.start(); for (int i 0; i 5; i) { System.out.println(主线程执行 i); } try { t.join(); } catch (Exception e) { e.printStackT\frace(); } System.out.println(主线程结束); } }输出结果主线程执行0主线程执行1主线程执行2主线程执行3子线程开始运行主线程执行4子线程子线程A运行0子线程子线程A运行1子线程子线程A运行2子线程子线程A运行3子线程子线程A运行4子线程结束主线程结束可以发现无论运行多少次主线程都会等待子线程结束之后在结束。编程要求请仔细阅读右侧代码根据方法内的提示在Begin - End区域内进行代码补充具体任务如下创建自定义线程实现求第num项斐波那契数列的值num从0开始并且在main函数中获取子线程最终计算的结果。测试说明补充完代码后点击测评平台会对你编写的代码进行测试当你的结果与预期输出一致时即为通过。输入5输出子线程计算结果为5输入8输出子线程计算结果为21输入10输出子线程计算结果为55/*获取子线程执行的结果并输出*/ package step2; import java.util.Scanner; public class Task { public static void main(String[] args) { Scanner sc new Scanner(System.in); int num sc.nextInt(); //请在此添加实现代码 /********** Begin **********/ Thread tnew MyThread(子线程,num); t.start(); /********** End **********/ } } //请在此添加实现代码 /********** Begin **********/ class MyThread extends Thread { private int num;// 需要计算的斐波那契数列的项数 private String name; // 线程名称 public MyThread(String name,int num){ this.numnum; this.namename; } public void run(){ int[] arr new int[2]; arr[0] 1; arr[1] 1; for(int i2; inum; i){ int tmp arr[1]; arr[1]arr[0]arr[1]; arr[0]tmp; } // 输出结果 System.out.println(子线程计算结果为 arr[1]); } } /********** End **********/第3关常用函数二相关知识为了完成本关任务你需要掌握1.yield函数的用法2.wait函数的用法3.其他常用函数的使用。yield 函数yield函数可以理解为“让步”它的作用是暂停当前正在执行的线程对象并执行其他线程。yield()应该做的是让当前运行线程回到可运行状态以允许具有相同优先级的其他线程获得运行机会。因此使用yield()的目的是让相同优先级的线程之间能适当的轮转执行。但是实际中无法保证yield()达到让步目的因为让步的线程还有可能被线程调度程序再次选中。示例public class Task { public static void main(String[] args) { System.out.println(Thread.currentThread().getName()主线程运行开始!); MyThread mTh1new MyThread(A); MyThread mTh2new MyThread(B); mTh1.start(); mTh2.start(); System.out.println(Thread.currentThread().getName() 主线程运行结束!); } } class MyThread extends Thread { private String name; public MyThread(String name) { this.name name; } public void run() { for (int i 0; i 50; i) { System.out.println(name 线程 执行 i); if(i 10){ this.yield(); } } } }运行结果有两种情况第一种情况A线程当执行到10时会让掉CPU时间这时B线程抢到CPU时间并执行。第二种情况B线程当执行到10时会让掉CPU时间这时A线程抢到CPU时间并执行。我们还可以考虑在其中加入setPriority函数改变线程优先级从而改变线程的执行顺序。wait 函数要弄明白wait函数我们首先需要了解线程锁的概念。 线程锁其实就像我们日常生活中的锁如果一个房子上了锁别人就进不去在Java中也类似如果一段代码取得了锁那么其他地方就不能运行这段代码只能等待锁的释放。这里我们会涉及到两个函数1.wait()2.notify()。这两个函数都是Object类自带的函数。在下面的例子中我们会使用synchronized(Obj)来实现线程同步同步的概念后面会讲到这里不理解没关系不影响对于wait函数的理解。从功能上来说wait()就是说线程在获取对象锁后主动释放对象锁同时本线程休眠。直到有其它线程调用对象的notify()唤醒该线程才能继续获取对象锁并继续执行相应的notify()就是对对象锁的唤醒操作。但有一点需要注意的是notify()调用后并不是马上就释放对象锁的而是在相应的synchronized(){}语句块执行结束自动释放锁后JVM会在wait()对象锁的线程中随机选取一线程赋予其对象锁唤醒线程继续执行。这样就提供了在线程间同步、唤醒的操作。Thread.sleep()与Object.wait()二者都可以暂停当前线程释放CPU控制权主要的区别在于Object.wait()在释放CPU同时释放了对象锁的控制。纸上得来终觉浅欲知此事须躬行我们还是通过实例来学习这些函数的用法。问题建立两个线程A线程打印5次AB线程打印5次B要求线程同时运行交替打印5次AB。这个问题用Object的wait()notify()就可以很方便的解决。代码如下public class MyThread implements Runnable { private String name; private Object prev; private Object self; private MyThread(String name, Object prev, Object self) { this.name name; this.prev prev; this.self self; } public void run() { int count 5; while (count 0) { synchronized (prev) { synchronized (self) { System.out.print(name); count--; self.notify(); } try { prev.wait(); } catch (InterruptedException e) { e.printStackT\frace(); } } } System.exit(0);//退出jvm } public static void main(String[] args) throws Exception { Object a new Object(); Object b new Object(); MyThread ta new MyThread(A, b, a); MyThread tb new MyThread(B, a, b); new Thread(ta).start(); Thread.sleep(100); //确保按顺序A、B执行 new Thread(tb).start(); Thread.sleep(100); } }运行程序结果为ABABABABAB线程常用函数总结下表列出的是线程类常用的函数函数描述public void start()使该线程开始执行Java 虚拟机调用该线程的 run 方法。public void run()如果该线程是使用独立的 Runnable 运行对象构造的则调用该 Runnable 对象的 run 方法否则该方法不执行任何操作并返回。public final void setName(String name)改变线程名称使之与参数 name 相同。public final void setPriority(int priority)更改线程的优先级。public final void setDaemon(boolean on)将该线程标记为守护线程或用户线程。public final void join(long millisec)等待该线程终止的时间最长为 millis 毫秒public void interrupt()中断线程。public final boolean isAlive()测试线程是否处于活动状态。public static void yield()暂停当前正在执行的线程对象并执行其他线程。public static void sleep(long millisec)在指定的毫秒数内让当前正在执行的线程休眠暂停执行此操作受到系统计时器和调度程序精度和准确性的影响。public static boolean holdsLock(Object x)当且仅当当前线程在指定的对象上保持监视器锁时才返回 true。public static Thread currentThread()返回对当前正在执行的线程对象的引用。public static void dumpStack()将当前线程的堆栈跟踪打印至标准错误流。编程要求请仔细阅读右侧代码根据方法内的提示在Begin - End区域内进行代码补充具体任务如下建立三个线程A线程打印5次EB线程打印5次DC线程打印5次U要求线程同时运行交替打印5次EDU。/** 使用三个线程交替打印5次EDU。 */ package step3; public class MyThread implements Runnable { /********** Begin **********/ private String name; private Object prev; private Object self; //构造函数初始化线程名字、前一锁对象和自己的锁对象 private MyThread(String name,Object prev,Object self){ this.name name; this.prev prev; this.self self; } //实现Runnable接口的run方法 public void run(){ //先设定循环次数 int count 5; //循环输出线程名字 while(count0){ //获取前一锁对象的监视器锁 synchronized(prev){ //获取自己锁对象的监视器锁 synchronized(self){ //输出线程名字 System.out.print(name); //修改循环次数 count--; //唤醒在当前锁对象上等待的线程 self.notify(); } try { //释放前一锁对象的监视器锁使线程进入等待状态 prev.wait(); } catch (InterruptedException e) { e.printStackTrace(); } } } //退出程序 System.exit(0); } //主方法 public static void main(String[] args) throws Exception { //创建三个锁对象 Object a new Object(); Object b new Object(); Object c new Object(); //创建三个线程对象分别传入线程名字和前一个锁对象、自己的锁对象 MyThread ta new MyThread(E,c,a); MyThread tb new MyThread(D,a,b); MyThread tc new MyThread(U,b,c); //创建三个线程并启动 new Thread(ta).start(); Thread.sleep(100); new Thread(tb).start(); Thread.sleep(100); new Thread(tc).start(); Thread.sleep(100); } /********** End **********/ }Java开发的就业市场正在经历结构性调整竞争日益激烈传统纯业务开发岗位如仅完成增删改查业务的后端工程师的需求特别是入门级岗位正显著萎缩。随着企业技术需求升级市场对Java人才的要求已从通用技能转向了更深入的领域经验如云原生、微服务或前沿的AI集成能力。这也导致岗位竞争加剧在一、二线城市求职者不仅面临技术内卷还需应对学历与项目经验的高门槛。大模型为核心的AI领域正展现出前所未有的就业热度与人才红利2025年AI相关新发岗位数量同比激增543%单月增幅最高超过11倍大模型算法工程师位居热门岗位前列。行业顶尖人才的供需严重失衡议价能力极强跳槽薪资涨幅可达30%-50%。值得注意的是市场并非单纯青睐算法研究员而是急需能将大模型能力落地于复杂业务系统的工程人才。这使得具备企业级架构思维和复杂系统整合经验的Java工程师在向“Java大模型”复合人才转型时拥有独特优势成为企业竞相争夺的对象其薪资天花板也远高于传统Java岗位。说真的这两年看着身边一个个搞Java、C、前端、数据、架构的开始卷大模型挺唏嘘的。大家最开始都是写接口、搞Spring Boot、连数据库、配Redis稳稳当当过日子。结果GPT、DeepSeek火了之后整条线上的人都开始有点慌了大家都在想“我是不是要学大模型不然这饭碗还能保多久”先给出最直接的答案一定要把现有的技术和大模型结合起来而不是抛弃你们现有技术掌握AI能力的Java工程师比纯Java岗要吃香的多。即使现在裁员、降薪、团队解散的比比皆是……但后续的趋势一定是AI应用落地大模型方向才是实现职业升级、提升薪资待遇的绝佳机遇如何学习AGI大模型作为一名热心肠的互联网老兵我决定把宝贵的AI知识分享给大家。 至于能学习到多少就看你的学习毅力和能力了 。我已将重要的AI大模型资料包括AI大模型入门学习思维导图、精品AI大模型学习书籍手册、视频教程、实战学习等录播视频免费分享出来。因篇幅有限仅展示部分资料需要点击下方链接即可前往获取2025最新版CSDN大礼包《AGI大模型学习资源包》免费分享**一、2025最新大模型学习路线一个明确的学习路线可以帮助新人了解从哪里开始按照什么顺序学习以及需要掌握哪些知识点。大模型领域涉及的知识点非常广泛没有明确的学习路线可能会导致新人感到迷茫不知道应该专注于哪些内容。我们把学习路线分成L1到L4四个阶段一步步带你从入门到进阶从理论到实战。L1级别:AI大模型时代的华丽登场L1阶段我们会去了解大模型的基础知识以及大模型在各个行业的应用和分析学习理解大模型的核心原理关键技术以及大模型应用场景通过理论原理结合多个项目实战从提示工程基础到提示工程进阶掌握Prompt提示工程。L2级别AI大模型RAG应用开发工程L2阶段是我们的AI大模型RAG应用开发工程我们会去学习RAG检索增强生成包括Naive RAG、Advanced-RAG以及RAG性能评估还有GraphRAG在内的多个RAG热门项目的分析。L3级别大模型Agent应用架构进阶实践L3阶段大模型Agent应用架构进阶实现我们会去学习LangChain、 LIamaIndex框架也会学习到AutoGPT、 MetaGPT等多Agent系统打造我们自己的Agent智能体同时还可以学习到包括Coze、Dify在内的可视化工具的使用。L4级别大模型微调与私有化部署L4阶段大模型的微调和私有化部署我们会更加深入的探讨Transformer架构学习大模型的微调技术利用DeepSpeed、Lamam Factory等工具快速进行模型微调并通过Ollama、vLLM等推理部署框架实现模型的快速部署。整个大模型学习路线L1主要是对大模型的理论基础、生态以及提示词他的一个学习掌握而L3 L4更多的是通过项目实战来掌握大模型的应用开发针对以上大模型的学习路线我们也整理了对应的学习视频教程和配套的学习资料。二、大模型经典PDF书籍书籍和学习文档资料是学习大模型过程中必不可少的我们精选了一系列深入探讨大模型技术的书籍和学习文档它们由领域内的顶尖专家撰写内容全面、深入、详尽为你学习大模型提供坚实的理论基础。书籍含电子版PDF三、大模型视频教程对于很多自学或者没有基础的同学来说书籍这些纯文字类的学习教材会觉得比较晦涩难以理解因此我们提供了丰富的大模型视频教程以动态、形象的方式展示技术概念帮助你更快、更轻松地掌握核心知识。四、大模型项目实战学以致用当你的理论知识积累到一定程度就需要通过项目实战在实际操作中检验和巩固你所学到的知识同时为你找工作和职业发展打下坚实的基础。五、大模型面试题面试不仅是技术的较量更需要充分的准备。在你已经掌握了大模型技术之后就需要开始准备面试我们将提供精心整理的大模型面试题库涵盖当前面试中可能遇到的各种技术问题让你在面试中游刃有余。因篇幅有限仅展示部分资料需要点击下方链接即可前往获取2025最新版CSDN大礼包《AGI大模型学习资源包》免费分享