面试

怎么理解面向对象:

万物皆对象。现实生活中任何物体都属于一类事务,每一个个体都是一类事物的实例。
面向对象的特性:

  • 封装(把一类事物的属性和行为抽象为一类,属性私有,行为公开),
  • 继承(将一类事物共有的属性和行为抽象成父类,子类有自己特有的行为和属性,实现代码的复用),
  • 多态(实现接口的重用,is-a变成has-a,解除了父子类继承的耦合度)
    面向对象的原则:单一指责,开放封闭,里氏替换,接口隔离,依赖倒置
    重写和重载的区别:
    重载:
    定义:在同一个类或与它的子类中,方法名相同而参数列表不同。(参数列表不同指的是参数的数量,类型,类型的顺序这三种至少一种不同)
    注意:方法重载与返回值类型和访问修饰符无关
    重写:
    原因:父类的功能无法满足子类的需求
    前提:必须存在继承关系
    定义:在继承关系中,子类定义与父类相同的方法
    原则:
    “二同”:即方法名相同,形参列表相同;
    “二小”:子类方法返回值类型应比父类方法返回值类型更小或相等,子类方法声明抛出的异常比父类方法声明抛出的异常更小或者相等;
    “一大”:子类方法的访问修饰符应比父类方法更大或相等。
    注意:1.构造方法不能被重写(构造方法必须与当前类名相同)
    2.private修饰的方法不能被重写
    3.Satic修饰的方法不能被重写
    4.final修饰的方法不能被重写
    重载是编译时期的活动,重写是运行时期的活动

线程池和参数:

ThreadPoolExecutor的参数:corePoolSize(核心线程数量),maximumPoolSize(线程最大线程数),workQuene(阻塞队列)

如何理解线程安全:

确保接口堆共享变量的操作要具备原子性(同数据库的原子性),可见性(volatile保证可见性,当多个线程并发访问共享变量时,一个线程对变量进行修改,其他线程能够立即看到),顺序性

实现原子更新操作

常见的保证Java操作原子性的工具:锁,同步方法或代码块,循环CAS
使用锁,可以保证同一时间只有一个线程能拿到锁。

锁(synchronized和lock):

Synchronized修饰非静态同步方法时,锁住的是当前实例;synchronized修饰静态同步方法时,锁住的是类的class对象;synchronized修饰静态代码块时,锁住的是关键字后面括号里的对象

既然锁和synchronized可以保证原子性,为什么还需要AtomicInteger来保证原子操作

锁和synchronized需要操作系统判断谁来获得锁,开销大,而AtomicInteger是通过cpu级的cas操作来保证原子性,开销小,使用AtomicInteger可以提高性能。

a=a+b和a+=b的区别:

对于同样类型的a,b来说,执行结果相同,但a+=b效率高
对于不同类型的a,b来说,+=是运算符,会强制类型转换,不会编译出错

get和post区别:

get产生一个tcp数据包,浏览器把http header和data一起发送出去,浏览器响应200;post产生两个tcp数据包,浏览器先发送http header,服务器响应100 continue,浏览器再发送data,浏览器响应200

IoC原理:

反射与工厂模式
反射:通过获取某个类的class对象后反向的获取某个类或对象的属性及方法信息
反射实现的原因:每个类在加载的过程中都会生成一个代表这个类的java.lang.Class对象作为方法区数据访问的入口

Bean生命周期

  1. 实例化bean对象(通过构造方法或工厂方法)
  2. 设置对象属性(setter,依赖注入)
  3. 如果Bean实现了beanNameAware接口,工厂调用Bean的setBeanName()方法传递Bean的ID
  4. 如果Bean实现了BeanFactoryAware接口,工厂调用setBeanFactory()方法传入工厂自身
  5. 将Bean实例化传递给Bean的后置处理器的postProcessBeforeInitialization(Object bean,String beanName)方法
  6. 调用Bean的初始化方法
  7. 将Bean实例传递给Bean的后置处理器的postProcessAfterInitialization(Object bean,String beanName)方法
  8. 使用bean
  9. 容器关闭前,调用bean的销毁方法

二叉树的前序中序后序遍历:

前序:访问根节点,遍历左子树,遍历右子树
中序:遍历左子树,访问根节点,遍历右子树
后序:遍历左子树,遍历右子树,访问根节点

Collection类:

Collection是对象集合,有两个子接口List和Set:
List可以通过下标取值,值可以重复,Set只能通过游标取值,值不可以重复
ArrayList,Vector,LinkedList是List的实现类;
ArrayLsit线程不安全,Vector线程安全,这两个类由数组实现;
LinkedList线程不安全,底层由链表实现
Map是键值对集合:
HashTable和HashMap是Map的实现类
HashTable线程安全,不能存储null值
HashMap线程不安全,可以存储null值
线程安全的集合类:vector,stack,hashtable,enumeration

Tcp三次握手:

  1. 客户端创建tcb,向服务器发出连接请求报文,SYN=1,同时选择一个初始序列号seq=下,tcp客户端进程进入了SYN-SENT(同步已发送状态)
  2. Tcp服务器收到请求报文后,如果同意连接,则发出确认报文,ACK=1,SYN=1,确认号是ACK=x+1,同时也为自己初始一个序列号seq=y,tcp服务器进程进入了SYN-RCVD(同步收到)状态
  3. Tcp客户进程收到确认后,还要向服务器给出确认,确认的报文ACK=1,ACK=y+1,自己的序列号seq=x+1,tcp连接建立,客户端进入establish(已建立连接)状态
  4. 当服务器收到客户端的确认后也进入establish状态,然后开始通信

栈内存和堆内存的特点和区别,Java中怎么样分配:

栈内存中存放基本数据的变量和引用变量,堆内存中存放new的对象和数组
基本数据类型栈中的值就是实际存储的值,引用类型栈中的值就是指向堆中的地址

对象序列化:

序列化就是把Java对象转换为字节序列,写入输入流中。Java对象是在jvm中生成的,如果需要远程传输或保存在硬盘上,就需要将Java对象转换成可传输的文件流。
Java中实现序列化的两种方式:实现Serializable接口或Externalizable
使用transient关键字修饰的变量不会被序列化

PV操作

PV操作是对信号量进行的操作
信号量S是一个整数,S大于等于零时代表可供并发进程使用的资源实体数,但S小于0时则表示正在等待使用共享资源的进程数
P操作申请资源:
1. S-1
2. 若S-1后仍>0,则进程继续执行
3. 若S-1后<0,则该进程被阻塞后
V操作释放资源:

  1. S+1
  2. 若结果>0,则进程继续执行
  3. 若<0,则从该信号的等待队列中唤醒一个等待进程,然后返回原进程继续执行

线程池的优点:

使用线程池可以减少创建和销毁线程的次数,重复使用线程。可以根据系统的承受能力,调整线程池中工作线程的数量,防止因为消耗过多内存而导致服务器崩溃

你可能感兴趣的:(面试)