面经题目及答案整理
1、聚簇索引与主键的关系?
聚簇索引默认是主键,如果表中没有定义主键,InnoDB会选择一个唯一的非空索引代替(“唯一的非空索引”是指列不能出现null值的唯一索引,跟主键性质一样)。如果没有这样的索引,InnoDB会隐式地定义一个主键来作为聚簇索引。
注意:严格来说,聚簇索引不一定是唯一索引,也就是聚簇索引那一列的值可以是重复的,唯一聚簇索引才是值唯一的。
参考:
通俗易懂 索引、单列索引、复合索引、主键、唯一索引、聚簇索引、非聚簇索引、唯一聚簇索引 的区别与联系 - pickle pee的文章 - 知乎
2、Tcp和udp能共用一台主机的同一个端口吗,为什么?
可以。因为数据接收时时根据五元组{传输协议,源IP,目的IP,源端口,目的端口}判断接受者的,既然传输协议不一样,自然是可以的,tcp和udp的端口互相不会影响。
3、Redis rdb子进程在持久化过程中又产生了新的数据怎么保证数据完整性?
知识点补充:首先,rdb持久化命令为:save和bgsave。
1)、save命令会阻塞服务器进程直到持久化结束,也就是此时服务器不能处理任何请求
2)、bgsave会派生出一个子进程,由子进程进行持久化,服务器父进程仍然可以处理请求。
3)、在bgsave的过程中,save和bgsave命令都会被拒绝,防止产生竞争条件;
但是对于bgrewriteaof命令则不同,不会拒绝该命令,但是需要延迟到bgsave完成后才能进行。
(而如果bgrewriteaof命令正在执行,则会拒绝bgsave)
回到问题本身
redis是通过dirty计数器和lastsave属性来保证数据的完整性。
dirty计数器记录距离上一次持久化后,产生了多少次修改操作。
而lastsave属性是一个unix时间戳,记录了服务器上一次持久化命令的时间。
4、jvm安全点是什么?
jvm在进行垃圾回收时,各个线程需要“跑”到一个安全点上,才能停下来进行gc,可以作为安全点的地方有:
1、循环的末尾
2、方法临返回前
3、调用方法之后
4、抛异常的位置
各个线程怎么知道需要停下来开始gc?
jvm采用主动式中断,在GC开始时,设置一个标志,让线程去不断轮询该标志,如果标志为真,则在安全点中断挂起。
5、java序列化和反序列化原理
**序列化:**将java对象转换为字节码,用来持久化保存在本地或者进行网络传输(比如rpc框架中)
反序列化:将字节码恢复成java对象。
序列化和反序列化的对象通过一个serialId来验证是否对应。
以上都是比较清楚的,那么具体是怎么进行序列化和反序列化的?
首先待序列化的类需要实现serializable接口
1、序列化的过程如下:
步骤一:创建一个对象输出流,它可以包装一个其它类型的目标输出流,如文件输出流:
ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("D:\\object.out"));
步骤二:通过对象输出流的**writeObject()**方法写对象:
oos.writeObject(new User("xuliugen", "123456", "male"));
2、反序列化过程为:
步骤一:创建一个对象输入流,它可以包装一个其它类型输入流,如文件输入流:
ObjectInputStream ois= new ObjectInputStream(new FileInputStream("object.out"));
步骤二:通过对象输出流的readObject()方法读取对象:
User user = (User) ois.readObject();
一个实际例子如下:
public static void main(String[] args) throws IOException, ClassNotFoundException {
//序列化
FileOutputStream fos = new FileOutputStream("object.out");
ObjectOutputStream oos = new ObjectOutputStream(fos);
User user1 = new User("xuliugen", "123456", "male");
oos.writeObject(user1);
oos.flush();
oos.close();
//反序列化
FileInputStream fis = new FileInputStream("object.out");
ObjectInputStream ois = new ObjectInputStream(fis);
User user2 = (User) ois.readObject();
System.out.println(user2.getUserName()+ " " +
user2.getPassword() + " " + user2.getSex());
//反序列化的输出结果为:xuliugen 123456 male
}
}
一些注意点:
1、序列化时,只对对象的状态进行保存,而不管对象的方法;
2、当一个父类实现序列化,子类自动实现序列化,不需要显式实现Serializable接口;
3、当一个对象的实例变量引用其他对象,序列化该对象时也把引用对象进行序列化;
4、并非所有的对象都可以序列化,至于为什么不可以,有很多原因了,比如:
安全方面的原因,比如一个对象拥有private,public等field,对于一个要传输的对象,比如写到文件,或者进行RMI传输等等,在序列化进行传输的过程中,这个对象的private等域是不受保护的;
资源分配方面的原因,比如socket,thread类,如果可以序列化,进行传输或者保存,也无法对他们进行重新的资源分配,而且,也是没有必要这样实现;
5、声明为static和transient类型的成员数据不能被序列化。因为static代表类的状态,transient代表对象的临时数据。
6、序列化运行时使用一个称为 serialVersionUID 的版本号与每个可序列化类相关联,该序列号在反序列化过程中用于验证序列化对象的发送者和接收者是否为该对象加载了与序列化兼容的类。为它赋予明确的值。显式地定义serialVersionUID有两种用途:
在某些场合,希望类的不同版本对序列化兼容,因此需要确保类的不同版本具有相同的serialVersionUID;
在某些场合,不希望类的不同版本对序列化兼容,因此需要确保类的不同版本具有不同的serialVersionUID。
7、Java有很多基础类已经实现了serializable接口,比如String,Vector等。但是也有一些没有实现serializable接口的;
8、如果一个对象的成员变量是一个对象,那么这个对象的数据成员也会被保存!这是能用序列化解决深拷贝的重要原因;
未完待续…
另外:关于我的个人博客网站.,欢迎访问