程序员面试金典(二)

2023-12-08 04:32
文章标签 面试 程序员 金典

本文主要是介绍程序员面试金典(二),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

线程与锁

1.java线程

在java中,每个线程的创建和控制都是由 java.lang.Thread 类的独特对象对象实现。一个独立的应用运行时,会自动创建一个用户线程,执行 main() 方法,这个线程叫主线程。在java中,实现线程有两种方式:

  • 通过实现 java.lang.Runnable 接口;
  • 通过扩展 java.lang.Thread 类。

(1)实现Runnable接口
Runnable接口结构非常简单:

public interface Runnable{void run();
}

要用这个接口创建和使用线程,看一个具体的例子;

package com.czl.thread;public class RunnableExample implements Runnable{public int count = 0;@Overridepublic void run() {System.out.println("线程开始");try {while (count < 5) {Thread.sleep(2000);count ++ ;}} catch (InterruptedException e) {// TODO Auto-generated catch blocke.printStackTrace();System.out.println("线程中断");}System.out.println("线程终结"); }}
package com.czl.thread;
public class TestRunnable {public static void main(String[] args) {RunnableExample instance = new RunnableExample();Thread thread = new Thread(instance);thread.start();//线程开始while (instance.count != 5) {try {Thread.sleep(2000);} catch (InterruptedException e) {// TODO Auto-generated catch blocke.printStackTrace();}}}
}

(2)扩展Thread
使用这种方式,基本就意味着要重写 run() 方法并且在子类的构造函数里,还需要显式调用这个县城的构造函数。

public class ThreadExaple extends Thread{int count = 0;@Overridepublic void run() {super.run();System.out.println("Thread starting");try {while (count < 5) {Thread.sleep(500);System.out.println("in thread,count is :" + count);count ++;}} catch (InterruptedException e) {// TODO Auto-generated catch blockSystem.out.println("thread interrupted.");}System.out.println("thread terminating.");}}public static void main(String[] args) {ThreadExaple threadExaple = new ThreadExaple();threadExaple.start();while (threadExaple.count != 5) {try {Thread.sleep(300);} catch (InterruptedException e) {// TODO Auto-generated catch blocke.printStackTrace();}}}

Tips:
在创建线程时,相比扩展Thread类,实现Runnable接口可能更优,有两个理由:

  • Java不支持多重继承。因此,扩展Thread类也就代表这个子类不能扩展其他类。而实现Runnble接口的类还能扩展另一个类。
  • 类可能只要求执行即可,因此,继承整个Thread的开销过大。

2.同步和锁

给定一个进程内的所有线程,都共享同一存储空间,这样有利有弊。这些线程可以共享数据,非常有用。不过,在两个线程同时修改同一资源时,会出现问题。Java提供了同步机制,以控制对共享资源的访问。
关键字 synchronizedlock 构成了代码同步执行的基础。

  • 同步方法
    最常见的就是使用 synchronized 关键字实现同步功能。该关键字可以用在方法和代码块上,限制多个线程,使之不能同时执行同一个对象的代码。
public class MyTest extends Thread{public void run(){}
}public class MyObject{public static synchronized void fun(){}public static synchronized void foo(){}
}
  • 同步块

代码块也可以实现同步化,其实现原理跟同步方法相似。

public class MyTest extends Thread{public void run(){}
}
public class MyObject{public void fun(String name){synchronized(this){}}
}

  • 若要实现更细致的控制,使用锁(lock),用于对共享资源的同步访问,方法是用锁将共享资源关联在一起,线程必须先取得与资源关联的锁,才能访问共享资源。不管在任何时间点,最多只能有一个线程拿到锁,因此只有一个线程可以访问共享资源。
    锁的常见用法,从多个地方访问同一资源时,同一时刻只有一个线程可以访问。
public class TestLock {private Lock lock;private int balance = 100;public TestLock(){lock = new ReentrantLock();}public int withdraw(int value){lock.lock();int temp = balance;try {Thread.sleep(200);temp = temp - value;Thread.sleep(200);balance = temp;} catch (InterruptedException e) {// TODO Auto-generated catch blocke.printStackTrace();}lock.unlock();return temp;}public int deposit(int value){lock.lock();int temp = balance;try {Thread.sleep(200);temp = temp + value;Thread.sleep(500);balance = temp;} catch (InterruptedException e) {// TODO Auto-generated catch blocke.printStackTrace();}lock.unlock();return temp;}
}

3.死锁及死锁的预防

死锁(deadlock)是这样一种情况:第一个线程在等待第二个线程持有的某个对象锁,而第二个线程又在等待第一个线程持有的对象锁(或是由两个以上线程形成的类似情形)。由于每个线程都在等待其他线程释放锁,以致每个线程都会一直这么等下去。于是,这些线程就陷入了死锁状态。
死锁的出现必须同时满足以下四个条件
(1)互斥:某一时刻只有一个线程能访问某一资源。(或者,更准确的说,对某一资源的访问有限制。若有资源限制,也有可能出现死锁)
(2)持有并等待:已持有某一资源的进程不必释放当前拥有的资源,就能要求更多的资源。
(3)没有抢占:一个进程不能强制另一个进程释放资源。
(4)循环等待:两个或两个以上的进程形成循环链,每个进程都在等待循环链中另一进程持有的资源。

若要预防死锁,只需避免上述任一条件,比较困难。比如,想要避免条件1就很困难,因为许多资源同一时刻只能被一个进程使用(如打印机)。大部分预防死锁的算法都把重心放在循环等待上。

这篇关于程序员面试金典(二)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/468544

相关文章

字节面试 | 如何测试RocketMQ、RocketMQ?

字节面试:RocketMQ是怎么测试的呢? 答: 首先保证消息的消费正确、设计逆向用例,在验证消息内容为空等情况时的消费正确性; 推送大批量MQ,通过Admin控制台查看MQ消费的情况,是否出现消费假死、TPS是否正常等等问题。(上述都是临场发挥,但是RocketMQ真正的测试点,还真的需要探讨) 01 先了解RocketMQ 作为测试也是要简单了解RocketMQ。简单来说,就是一个分

秋招最新大模型算法面试,熬夜都要肝完它

💥大家在面试大模型LLM这个板块的时候,不知道面试完会不会复盘、总结,做笔记的习惯,这份大模型算法岗面试八股笔记也帮助不少人拿到过offer ✨对于面试大模型算法工程师会有一定的帮助,都附有完整答案,熬夜也要看完,祝大家一臂之力 这份《大模型算法工程师面试题》已经上传CSDN,还有完整版的大模型 AI 学习资料,朋友们如果需要可以微信扫描下方CSDN官方认证二维码免费领取【保证100%免费

java面试常见问题之Hibernate总结

1  Hibernate的检索方式 Ø  导航对象图检索(根据已经加载的对象,导航到其他对象。) Ø  OID检索(按照对象的OID来检索对象。) Ø  HQL检索(使用面向对象的HQL查询语言。) Ø  QBC检索(使用QBC(Qurey By Criteria)API来检索对象。 QBC/QBE离线/在线) Ø  本地SQL检索(使用本地数据库的SQL查询语句。) 包括Hibern

贝壳面试:什么是回表?什么是索引下推?

尼恩说在前面 在40岁老架构师 尼恩的读者交流群(50+)中,最近有小伙伴拿到了一线互联网企业如得物、阿里、滴滴、极兔、有赞、希音、百度、网易、美团的面试资格,遇到很多很重要的面试题: 1.谈谈你对MySQL 索引下推 的认识? 2.在MySQL中,索引下推 是如何实现的?请简述其工作原理。 3、说说什么是 回表,什么是 索引下推 ? 最近有小伙伴在面试 贝壳、soul,又遇到了相关的

毕业前第二次面试的感慨

距面试已经过去了有几天了,我现在想起来都有说多的恨感慨。 我一直都是想找刚刚起步的企业,因为这能让我学到更多的东西,然而正好有一家企业是刚起步的,而且他还有自己的产品专利,可以说这是一家,即是创业又是刚起步的公司,这家公司回复了我投给他的简历,这家企业想进一步了解我的情况,因为简历上我符合这家企业的基本要求,所以要进一步了解。 虽然面试的过程中,他给我的面试题,我做得并不是很理想,

LabVIEW程序员是怎样成长为大佬

成为一名LabVIEW编程领域的“大佬”需要时间、实践、学习和解决复杂问题的经验。尽管LabVIEW作为一种图形化编程语言在初期可能相对容易上手,但要真正成为精通者,需要在多个层面上深入理解。以下是LabVIEW程序员如何逐步成长为“大佬”的路径: 1. 打好基础 LabVIEW的大佬们通常在初期会打下非常坚实的基础,理解LabVIEW编程的核心概念,包括: 数据流编程模型:Lab

腾讯社招面试经历

前提:本人2011年毕业于一个普通本科,工作不到2年。   15号晚上7点多,正在炒菜做饭,腾讯忽然打电话来问我对他们的Linux C++的职位是否感兴趣,我表达了我感兴趣之后,就开始了一段简短的电话面试,电话面试主要内容:C++和TCP socket通信的一些基础知识。之后就问我一道算法题:10亿个整数,随机生成,可重复,求最大的前1万个。当时我一下子就蒙了,没反应过来,何况我还正在烧

完整的腾讯面试经过

从9月10号开始到现在快两个月了,两个多月中,我经历数次面试和笔试,在经历这些的同时积累了不少的经验,也学到了不少东西,在此把它记录下来,算是和一起找工作中的同学一起共勉吧。我是本校的学生,专业是机械制造及其自动化,找工作的主要目标是计算机软件类和机械制造方向的国内的企业,所以意向去外企的同学就不必浪费时间看这些面经啦,想去国内IT企业的同学可以继续看下去。本贴中我把最近的腾讯面试经过写下

仕考网:结构化面试流程介绍

(一)结构化面试 结构化面试,也叫做标准化面试,考官按照预先设定好的一套试题以问答方式与应试者当面交谈,根据应试者的言语、行为表现,对其相关能力和个性特征作出相应评价。 (二)考试流程 抵达考场——审核抽签——面试候考——进入考场——面试答题——考生退场——计分审核 (三)答题技巧 1.声音洪亮,音量可以比平时说话声音大一点。 2.语速不要过快,语速快容易卡顿,而且不便于考官听清答

嵌入式面试经典30问:二

1. 嵌入式系统中,如何选择合适的微控制器或微处理器? 在嵌入式系统中选择合适的微控制器(MCU)或微处理器(MPU)时,需要考虑多个因素以确保所选组件能够满足项目的具体需求。以下是一些关键步骤和考虑因素: 1.1 确定项目需求 性能要求:根据项目的复杂度、处理速度和数据吞吐量等要求,确定所需的处理器性能。功耗:评估系统的功耗需求,选择低功耗的MCU或MPU以延长电池寿命或减少能源消耗。成本