【秋招冲刺-每日打卡】应届生JAVA岗-每日5道高频面试题【Day4】-基础篇(4)_age shorts

写在最后

可能有人会问我为什么愿意去花时间帮助大家实现求职梦想,因为我一直坚信时间是可以复制的。我牺牲了自己的大概十个小时写了这片文章,换来的是成千上万的求职者节约几天甚至几周时间浪费在无用的资源上。

复习一周,字节跳动三场技术面+HR面,不小心拿了offer

复习一周,字节跳动三场技术面+HR面,不小心拿了offer

上面的这些(算法与数据结构)+(Java多线程学习手册)+(计算机网络顶级教程)等学习资源

本文已被CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】收录

需要这份系统化的资料的朋友,可以点击这里获取


  

### (二) 深拷贝:


  **复制出来的对象拥有和原来对象相同的一套属性值,里面的属性和被复制的对象是相互独立的,修改任何一个对象都不会对另外一个对象产生影响**,  
  
   从上面可以知道Object提供的clone方法只能实现浅拷贝,**如果想实现深拷贝,可以采取以来两种方法:**  
  
   **1、每个引用类型内部都实现cloneable接口并重写clone方法即可。**  
  
   **2、使用序列化和反序列化(前提是类需要实现序列化接口Serializable)**  
  
   注意:序列化是将对象写到流中便于网络传输或者持久化到磁盘,而反序列化则是把对象从流/磁盘中读取出来。这里写到流中的对象则是原始对象的一个拷贝,因为原始对象还存在 JVM 中,所以我们通过对象的序列化产生克隆对象,然后通过反序列化获取这个对象。  
  
   **实现深拷贝方式一: 每个引用类型都实现Cloneable接口并重写clone方法**  
   
  
 ![在这里插入图片描述](https://img-blog.csdnimg.cn/20210610224050926.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzQwODkxMDA5,size_16,color_FFFFFF,t_70)  
   
  
 ![在这里插入图片描述](https://img-blog.csdnimg.cn/20210610225240481.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzQwODkxMDA5,size_16,color_FFFFFF,t_70)  
   




public class CloneDemo {
public static void main(String[] args) throws Exception{
Demo1 demo1 = new Demo1(“1”,2,new Demo2(“3”,4));
System.out.println(“原来的对象数据:” + demo1);

    // 深拷贝
    Demo1 cloneDemo = (Demo1) demo1.clone();
    System.out.println("拷贝出来的对象数据:" + cloneDemo);

    // 修改拷贝对象的属性
    cloneDemo.setAge(100);
    cloneDemo.setAgeCone(100);
    cloneDemo.setUserName("test");
    cloneDemo.setS1(Short.valueOf("200"));
    cloneDemo.getDemo2().setDemo2Age(1000);
    System.out.println();
    System.out.println("修改拷贝出来的对象引用类型属性:被拷贝对象的数据" + demo1);
    System.out.println("修改拷贝出来的对象引用类型属性:拷贝出来的对象数据" + cloneDemo);
}

}

@Data
class Demo1 implements Cloneable{
private String userName;
private Integer age;
private Short s1 = new Short(“100”);
private Integer ageCone = new Integer(900);
private Demo2 demo2;

public Demo1(String userName, Integer age, Demo2 demo2) {
    this.userName = userName;
    this.age = age;
    this.demo2 = demo2;
}

@Override
public String toString() {
    return "Demo1{" +
            "userName='" + userName + '\'' +
            ", age=" + age +
            ", s1=" + s1 +
            ", ageCone=" + ageCone +
            ", demo2=" + demo2 +
            '}';
}

@Override
protected Object clone() throws CloneNotSupportedException {
    Demo1 demo1 = (Demo1) super.clone();
    Demo2 demo2 = (Demo2) demo1.getDemo2().clone();
    demo1.setDemo2(demo2);
    return demo1;
}

}
@Data
class Demo2 implements Cloneable{
private String demo2Username;
private Integer demo2Age;

@Override
protected Object clone() throws CloneNotSupportedException {
    return super.clone();
}

public Demo2(String demo2Username, Integer demo2Age) {
    this.demo2Username = demo2Username;
    this.demo2Age = demo2Age;
}

}


  

  **实现深拷贝方式二: 使用序列化和反序列化方式达到深拷贝**  
 ![在这里插入图片描述](https://img-blog.csdnimg.cn/20210610230154429.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzQwODkxMDA5,size_16,color_FFFFFF,t_70)  
   




public class CloneDemo {
public static void main(String[] args) throws Exception{
Demo1 demo1 = new Demo1(“1”,2,new Demo2(“3”,4));
System.out.println(“原来的对象数据:” + demo1);
// 序列化
ByteArrayOutputStream bos = new ByteArrayOutputStream();
ObjectOutputStream oos = new ObjectOutputStream(bos);
oos.writeObject(demo1);
// 反序列化
ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray());
ObjectInputStream ois = new ObjectInputStream(bis);
// 深拷贝
Demo1 cloneDemo = (Demo1) ois.readObject();
System.out.println(“拷贝出来的对象数据:” + cloneDemo);

    // 修改拷贝对象的属性
    cloneDemo.setAge(100);
    cloneDemo.setAgeCone(100);
    cloneDemo.setUserName("test");
    cloneDemo.setS1(Short.valueOf("200"));
    cloneDemo.getDemo2().setDemo2Age(1000);
    System.out.println();
    System.out.println("修改拷贝出来的对象引用类型属性:被拷贝对象的数据" + demo1);
    System.out.println("修改拷贝出来的对象引用类型属性:拷贝出来的对象数据" + cloneDemo);
}

}

@Data
class Demo1 implements Cloneable, Serializable {
private String userName;
private Integer age;
private Short s1 = new Short(“100”);
private Integer ageCone = new Integer(900);
private Demo2 demo2;

public Demo1(String userName, Integer age, Demo2 demo2) {
    this.userName = userName;
    this.age = age;
    this.demo2 = demo2;
}

@Override
public String toString() {
    return "Demo1{" +
            "userName='" + userName + '\'' +
            ", age=" + age +
            ", s1=" + s1 +
            ", ageCone=" + ageCone +
            ", demo2=" + demo2 +
            '}';
}

@Override
protected Object clone() throws CloneNotSupportedException {
    Demo1 demo1 = (Demo1) super.clone();
    Demo2 demo2 = (Demo2) demo1.getDemo2().clone();
    demo1.setDemo2(demo2);
    return demo1;
}

}
@Data
class Demo2 implements Cloneable,Serializable{
private String demo2Username;
private Integer demo2Age;

@Override
protected Object clone() throws CloneNotSupportedException {
    return super.clone();
}

public Demo2(String demo2Username, Integer demo2Age) {
    this.demo2Username = demo2Username;
    this.demo2Age = demo2Age;
}

}


  

## 二: throw和throws的区别?


  

  **throw关键字用于主动抛出java.lang.Throwable类的一个实例化对象,当某些业务可能存在异常,但是你并不想在此处处理这个异常,可以使用throw关键字将异常抛出**。如:throw new Exception(“not need to deal″)。  
  
   **throws 的作用是作为方法声明和签名的一部分(放在方法声明处),可以接受多个异常,用逗号隔开,这个方法的调用者需要处理抛出的异常或者继续使用throws将异常网上抛出,最高可抛出到JVM进行处理**。Java 中,任何未处理的受检查异常强制在 throws 子句中声明。


  

![在这里插入图片描述](https://img-blog.csdnimg.cn/20210610232731602.png)


  

## 三: 受检查异常和运行时异常与有何区别?


  

  受检查异常: 在编译阶段被强制检查的异常称为"受检查的异常",这种异常JAVA编译器要求必须处理,如IO异常  
  
   运行时异常: 在编译阶段无法检测出来的,可能是由于开发者设计考虑不周全而引起的异常,这种异常只能在程序运行时才会发现,所以,处理这种异常要求开发者更加细致和有经验才能更好预测到。  
   
  
 ![在这里插入图片描述](https://img-blog.csdnimg.cn/20210610233745684.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzQwODkxMDA5,size_16,color_FFFFFF,t_70)


  

## 四: 列举一些工作中你常遇到的运行时异常


  

* NullPointerException (空指针异常)
* IndexOutOfBoundsException (下标越界异常)
* IllegalArgumentException (非法参数异常)
* ClassCastException (类转换异常)
* ArithmeticException(算术异常)


  

## 五: SimpleDateFormat是线程安全的吗?如果不是,怎么解决它线程不安全的问题?


  

  SimpleDateFormat是DateFormat 的一个实现,而DateFormat的实现都是线程不安全的,所以SimpleDateFormat 都不是线程安全的,在多线程环境下,会存在线程安全问题。


  解决: 可以将 SimpleDateFormat 存放在 ThreadLocal 中,因为ThreadLocal是线程变量,每个线程都有一个单独的ThreadLocal ,在多线程环境下也不会出现线程安全问题。


  JDK8推荐使用DateTimeFormatter来代替SimpleDateFormat ,因为它是线程安全的。


  

## 每日小结


  


# 面试准备+复习分享:

> 为了应付面试也刷了很多的面试题与资料,现在就分享给有需要的读者朋友,资料我只截取出来一部分哦

![秋招|美团java一面二面HR面面经,分享攒攒人品](https://img-blog.csdnimg.cn/img_convert/bf9ed1c490257635617beab229e31c52.webp?x-oss-process=image/format,png)



> **本文已被[CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】](https://bbs.csdn.net/forums/4f45ff00ff254613a03fab5e56a57acb)收录**

**[需要这份系统化的资料的朋友,可以点击这里获取](https://bbs.csdn.net/forums/4f45ff00ff254613a03fab5e56a57acb)**

需要的读者朋友,资料我只截取出来一部分哦

[外链图片转存中...(img-Xvlrq31X-1715486981107)]



> **本文已被[CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】](https://bbs.csdn.net/forums/4f45ff00ff254613a03fab5e56a57acb)收录**

**[需要这份系统化的资料的朋友,可以点击这里获取](https://bbs.csdn.net/forums/4f45ff00ff254613a03fab5e56a57acb)**

  • 5
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 包含风险分析的软件工程模型是瀑布模型。 瀑布模型是软件开发过程中的一种传统模型。它按照严格的阶段性流程进行软件开发,包括需求分析、系统设计、编码、测试和运维等多个阶段。其特点在于每个阶段的输出作为下一个阶段的输入,流程严格按照顺序进行。 在瀑布模型中,风险分析是一个重要的环节。在需求分析阶段,软件团队会对项目进行风险评估,识别出可能存在的风险因素。然后,在系统设计阶段,根据识别的风险因素,制定相应的风险应对措施。在编码和测试阶段,会根据风险评估结果,针对性地进行代码检查和测试,以降低风险产生的可能性。最后,在运维阶段,会持续监控和评估项目的风险,及时采取措施进行调整。 风险分析在瀑布模型中的应用,有助于提前发现和解决问题,减少项目风险。它可以帮助软件团队在项目开始前,全面了解可能存在的风险,并制定针对性的风险应对策略。这样可以有效地提高项目管理的可控性,降低项目失败的风险,确保软件开发过程的顺利进行。所以,包含风险分析的软件工程模型是瀑布模型。 ### 回答2: 包含风险分析的软件工程模型是敏捷开发模型。 敏捷开发模型是一种灵活的软件开发方法,强调快速反馈和快速适应变化。与传统的瀑布模型不同,敏捷开发模型将软件开发过程划分为短周期的迭代,每个迭代称为一个“冲刺”。在冲刺开始前,团队根据需求列表选择开发的功能点,并制定对应的开发计划。 在敏捷开发模型中,风险分析是非常重要的一部分。在每个冲刺的规划阶段,团队将识别和评估潜在的风险因素,并制定相应的风险管理策略。风险分析的目的是为了预测可能出现的问题,并采取相应的措施来降低风险,以确保软件开发过程能够按时、按质完成。 风险分析的过程包括以下几个步骤: 1. 风险识别:团队会通过讨论、头脑风暴等方式,识别可能存在的风险因素,包括技术风险、需求风险、资源风险等。 2. 风险评估:对于已经识别出的风险因素,团队会评估其可能性和影响程度,以确定哪些风险是最关键的。 3. 风险规划:根据风险评估的结果,团队会制定相应的风险管理策略和应对措施,包括风险的防范、缓解、转移等。 4. 风险监控:在软件开发过程中,团队会不断监控风险的变化和进展,并及时调整风险管理策略。 通过风险分析,敏捷开发模型可以帮助团队更好地理解项目中可能面临的挑战和风险,并提前采取相应的措施,从而降低项目失败的概率。同时,敏捷开发模型也强调团队的协作和迭代反馈,能够更好地适应需求的变化和客户的反馈,提高软件开发的成功率。 ### 回答3: 包含风险分析的软件工程模型是瀑布模型。 瀑布模型是一种线性顺序的软件开发模型,它将软件开发过程划分为不同的阶段,包括需求分析、系统设计、编码、测试和维护等。瀑布模型有明确的阶段顺序和阶段之间的输入输出关系,每个阶段的输出成果作为下一个阶段的输入。 瀑布模型的特点之一就是风险管理和分析。在每个阶段结束前,开发团队需要进行风险分析,并采取相应的措施来降低风险。风险可以包括技术风险、需求风险、进度风险等等。通过风险分析,开发团队可以及时发现问题并采取相应的措施,以避免项目进展受阻或者出现严重的质量问题。 在瀑布模型中,风险分析主要包括以下几个步骤:确定风险来源、评估风险的概率和影响、制定相应的风险应对策略。通过这些步骤,可以帮助开发团队及时识别并应对潜在的风险,保证项目顺利进行。 总之,瀑布模型是包含风险分析的软件工程模型。通过风险分析,可以帮助开发团队识别和应对潜在的风险,确保软件项目的成功实施。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值