面试

作者: 冰与河豚鱼 | 来源:发表于2018-10-23 08:45 被阅读0次
    怎么理解面向对象:

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

    • 封装(把一类事物的属性和行为抽象为一类,属性私有,行为公开),
    • 继承(将一类事物共有的属性和行为抽象成父类,子类有自己特有的行为和属性,实现代码的复用),
    • 多态(实现接口的重用,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,则从该信号的等待队列中唤醒一个等待进程,然后返回原进程继续执行

    线程池的优点:

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

    相关文章

      网友评论

          本文标题:面试

          本文链接:https://www.haomeiwen.com/subject/xrlozftx.html