面试最常被问的 Java 后端题目及参考答案

本文汇总了 Java 后端开发面试中常见的知识点,涵盖 Java 基础、JVM、Dubbo、MyBatis、MySQL、RabbitMQ、Redis、Spring Boot、Spring、ZooKeeper、并发编程、设计模式等多个方面。内容包括对象方法、对象创建、数据结构对比、线程安全、并发工具类、数据库操作、分布式服务、消息队列等,全面梳理了面试中可能遇到的技术点。
摘要由CSDN通过智能技术生成

一、Java 基础篇

1. Object 有哪些常用方法?大致说一下每个方法的含义

2. Java 创建对象有几种方式?

3. 获取一个类对象的方式有哪些?

4. ArrayList 和 LinkedList 的区别有哪些?

5. 用过 ArrayList 吗?说一下它有什么特点?

6. 有数组了为什么还要搞个 ArrayList 呢?

7. 说说什么是 fail-fast?

8. Hashtable 与 HashMap 的区别

9. HashMap 中的 key 我们可以使用任何类作为 key 吗?

10. HashMap 的长度为什么是 2 的 N 次方呢?

11. HashMap 与 ConcurrentHashMap 的异同

13. 红黑树有哪几个特征?

14. 说说你平时是怎么处理 Java 异常的

15. finally 模块执行了吗?是先执行 return 还是先执行 finally 模块?返回什么?

二、JVM 篇

16. Java 类加载器有几种?

17. 说一下有哪些类加载场景?

18. 说说 Java 类加载机制是什么?说说 new 创建一个普通对象的过程?

19. 说说类的生命周期?

20. 什么是双亲委派模型?

21. 如何破坏双亲委派模型?

22. 能不能自己也写一个 java.lang.String 类?

23. 说一下 JVM 运行时数据区有哪些?分别说一下它们的功能

24. 方法区和永久代有什么区别?

24. JVM 运行时数据区哪些地方会产生内存溢出?

25. 为什么要用 metaspace 替换 permspace 呢?

26. 熟悉哪些 JVM 调优参数?

27. Java 对象的引用类型有哪些?

28. JVM 垃圾回收算法有哪些?

29. 垃圾收集器有哪些?

30. 说说 JVM 中内存的分配与回收策略

三、Dubbo 篇

31. 说说一次 Dubbo 服务请求流程?

32. 说说 Dubbo 工作原理

33. Dubbo 支持哪些协议?

34. 注册中心挂了,consumer 还能不能调用 provider?

35. 怎么实现动态感知服务下线的呢?

36. Dubbo 负载均衡策略?

37. Dubbo 容错策略

38. Dubbo 动态代理策略有哪些?

39. 说说 Dubbo 与 Spring Cloud 的区别?

40. 说说 TCP 与 UDP 的区别,以及各自的优缺点

41. 说一下 HTTP 和 HTTPS 的区别

42. 说说 HTTP、TCP、Socket 的关系是什么?

43. 说一下 HTTP 的长连接与短连接的区别

四、MyBatis 篇

44. 说说 MyBatis 的缓存

45. JDBC 编程有哪些步骤?

46. 说一下 MyBatis 中使用的 #和 $ 有什么区别

47. MyBatis 中比如 UserMapper.java 是接口,为什么没有实现类还能调用?

48. MyBatis 中见过什么设计模式?

五、MySQL 篇

49. 简单说说在 MySQL 中执行依据查询 SQL 是如何执行的?

50. MySQL 有哪些存储引擎?

51. MySQL 中 varchar 与 char 的区别?varchar(30) 中的 30 代表的涵义?

52. int(11) 中的 11 代表什么涵义?

53. 为什么 SELECT COUNT(*) FROM table 在 InnoDB 比 MyISAM 慢?

54. 说说数据库的三范式和反模式

55. 在设计数据库表的时候,字段用于存储金额、余额时,选择什么类型比较好?

56. 大概说说 InnoDB 与 MyISAM 有什么区别?

57. 什么是索引?

58. 索引有什么优缺点?

59. MySQL 索引类型有哪些?

60. 什么时候不要使用索引?

61. 使用 MySQL 的索引应该注意些什么?

62. 怎么知道一条查询语句是否用到了索引,用了什么类型的索引?

63. 说说什么是 MVCC?

64. MVCC 可以为数据库解决什么问题?

65. 说说 MVCC 的实现原理

66. 什么是死锁?

67. MySQL 事务隔离级别?

69. 请说说 MySQL 数据库的锁?

70. 说说什么是锁升级?

71. 说说悲观锁和乐观锁

72. 怎样尽量避免死锁的出现?

六、RabbitMQ 篇

73. 看你简历上写了 RabbitMQ,通常会问:为什么要用 RabbitMQ?

74. 可能你讲了上面三个 RabbitMQ 的优点后,会继续问:使用 RabbitMQ 容易带来什么问题?

75. 那么多消息队列,为什么选 RabbitMQ 呢?

75. RabbitMQ 中什么是死信队列?

76. 如何处理死信队列?

77. 怎么保证消息不会被丢失?

78. RabbitMQ 怎么高可用呢?

79. RabbitMq 怎么保证消息的顺序性?

80. 如果有大量消息持续积压在队列了,怎么处理?

七、Redis 篇

81. 为什么要用缓存

82. 为什么 使用 Redis 而不是用 Memcached 呢?

83. 为什么 Redis 单线程模型效率也能那么高?

84. 说说 Redis 的线程模型

85. 说一下 Redis 有什么优点和缺点

86. Redis 缓存刷新策略有哪些?

87. Redis 持久化方式有哪些?以及有什么区别?

88. 持久化有两种,那应该怎么选择呢?

89. 怎么使用 Redis 实现消息队列?

90. 熟悉哪些 Redis 集群模式?

91. 缓存和数据库谁先更新呢?

八、Spring Boot 篇

92. Spring Boot 提供了哪些核心功能?

93. Spring Boot 核心注解是什么?

94. 说说 Spring Boot 的自动装配原理

95. Spring Boot 常用 starter 有哪些?

96. Spring 中的 starter 是什么?

97. Spring Boot 有什么优缺点?

98. 读取配置文件中配置项的有哪些方法?

九、Spring 篇

99. Spring 中 ApplicationContext 和 BeanFactory 的区别

100. 说一下你对 Spring IOC 的理解

101. Spring IOC 有什么优点?

102. Bean 的生命周期

103. Spring Bean 的作用域有哪些?

104. Spring 是怎么管理事务的?

105. 说说你对 Spring AOP 的理解

106. Spring 中用到了哪些设计模式?

107. Spring 框架中的单例 Bean 是线程安全的么?

108. Spring 是怎么解决循环依赖的?

十、ZooKeeper 篇

109. 说说 ZooKeeper 是什么?

110. ZooKeeper 有哪些应用场景?

111. ZooKeeper 有哪些节点类型?

112. 请描述一下 ZooKeeper 的通知机制是什么?

113. ZooKeeper 对节点的 watch 监听通知是永久的吗?

114. ZooKeeper 集群中有哪些角色?

115. ZooKeeper 集群中 Server 有哪些工作状态?

116. ZooKeeper 集群中是怎样选举 leader 的?

117. ZooKeeper 是如何保证事务的顺序一致性的呢?

118. ZooKeeper 集群中各服务器之间是怎样通信的?

119. ZooKeeper 分布式锁怎么实现的?

十一、并发编程篇

120. 通常创建线程有几种方式?

121. 说说线程的生命周期

122. 说说 synchronized 的使用和原理

123. synchronized 和 ReentrantLock 区别

124. 什么是线程安全?

125. 线程安全需要保证几个基本特征

126. 说一下线程之间是如何通信的?

127. 说说你对 volatile 的理解

128. 说一下 volatile 和 synchronized 的区别?

129. Thread 调用 run 方法和调 start 方法的区别?

130. 说一下 Java 创建线程池有哪些方式?

131. 说说 ThreadLocal 底层原理是什么,怎么避免内存泄漏?

132. 说说你对 JUC 下并发工具类

133. CyclicBarrier 和 CountdownLatch 有什么区别?

十二、设计模式篇

134. 你都熟悉哪些设计模式

十三、其他篇

135. 有 8 个球(大小颜色都一模一样),其中一个球比其他 7 个球中的任何一个都重,使用天平秤最多几次能找到最重的那个球?

136. 分布式幂等性如何设计?

137. 简单一次完整的 HTTP 请求所经历的步骤?

138. 说说分布式事务解决方案有哪些?

139. 说说常用的 JVM 调优命令和工具有哪些?

140. 说说你对 JVM 内存溢出和内存泄漏的理解

141. 说说 JVM 中有哪些常用参数?

一、Java 基础篇

1. Object 有哪些常用方法?大致说一下每个方法的含义

java.lang.Object

下面是对应方法的含义。

clone 方法

保护方法,实现对象的浅复制,只有实现了 Cloneable 接口才可以调用该方法,否则抛出
CloneNotSupportedException 异常,深拷贝也需要实现 Cloneable,同时其成员变量为引用类型的也需要实现 Cloneable,然后重写 clone 方法。

finalize 方法

该方法和垃圾收集器有关系,判断一个对象是否可以被回收的最后一步就是判断是否重写了此方法。

equals 方法

该方法使用频率非常高。一般 equals 和 == 是不一样的,但是在 Object 中两者是一样的。子类一般都要重写这个方法。

hashCode 方法

该方法用于哈希查找,重写了 equals 方法一般都要重写 hashCode 方法,这个方法在一些具有哈希功能的 Collection 中用到。

一般必须满足 obj1.equals(obj2)==true。可以推出 obj1.hashCode()==obj2.hashCode(),但是 hashCode 相等不一定就满足 equals。不过为了提高效率,应该尽量使上面两个条件接近等价。

  • JDK 1.6、1.7 默认是返回随机数;
  • JDK 1.8 默认是通过和当前线程有关的一个随机数 + 三个确定值,运用 Marsaglia’s xorshift scheme 随机数算法得到的一个随机数。

wait 方法

配合 synchronized 使用,wait 方法就是使当前线程等待该对象的锁,当前线程必须是该对象的拥有者,也就是具有该对象的锁。wait() 方法一直等待,直到获得锁或者被中断。wait(long timeout) 设定一个超时间隔,如果在规定时间内没有获得锁就返回。

调用该方法后当前线程进入睡眠状态,直到以下事件发生。

  1. 其他线程调用了该对象的 notify 方法;
  2. 其他线程调用了该对象的 notifyAll 方法;
  3. 其他线程调用了 interrupt 中断该线程;
  4. 时间间隔到了。

此时该线程就可以被调度了,如果是被中断的话就抛出一个 InterruptedException 异常。

notify 方法

配合 synchronized 使用,该方法唤醒在该对象上等待队列中的某个线程(同步队列中的线程是给抢占 CPU 的线程,等待队列中的线程指的是等待唤醒的线程)。

notifyAll 方法

配合 synchronized 使用,该方法唤醒在该对象上等待队列中的所有线程。

总结

只要把上面几个方法熟悉就可以了,toString 和 getClass 方法可以不用去讨论它们。该题目考察的是对 Object 的熟悉程度,平时用的很多方法并没看其定义但是也在用,比如说:wait() 方法,equals() 方法等。

Class Object is the root of the class hierarchy.Every class has Object as a superclass. All objects, including arrays, implement the methods of this class.

大致意思:Object 是所有类的根,是所有类的父类,所有对象包括数组都实现了 Object 的方法。

面试扩散

上面提到了 wait、notify、notifyAll 方法,或许面试官会问你为什么 sleep 方法不属于 Object 的方法呢?因为提到 wait 等方法,所以最好把 synchronized 都说清楚,把线程状态也都说清楚,尝试让面试官跟着你的节奏走。

2. Java 创建对象有几种方式?

这题目看似简单,要好好回答起来还是有点小复杂的,我们来看看,到底有哪些方式可以创建对象?

1. 使用 new 关键字,这也是我们平时使用的最多的创建对象的方式,示例:

User user=new User();

2. 反射方式创建对象,使用 newInstance(),但是得处理两个异常 InstantiationException、IllegalAccessException:

User user=User.class.newInstance();
Object object=(Object)Class.forName("java.lang.Object").newInstance()

3.使用 clone 方法,前面题目中 clone 是 Object 的方法,所以所有对象都有这个方法。

4.使用反序列化创建对象,调用 ObjectInputStream 类的 readObject() 方法。

我们反序列化一个对象,JVM 会给我们创建一个单独的对象。JVM 创建对象并不会调用任何构造函数。一个对象实现了 Serializable 接口,就可以把对象写入到文件中,并通过读取文件来创建对象。

总结

创建对象的方式关键字:new、反射、clone 拷贝、反序列化。

3. 获取一个类对象的方式有哪些?

搞清楚类对象和实例对象,但都是对象。

第一种:通过类对象的 getClass() 方法获取,细心点的都知道,这个 getClass 是 Object 类里面的方法。

User user=new User();
//clazz就是一个User的类对象
Class<?> clazz=user.getClass();

第二种:通过类的静态成员表示,每个类都有隐含的静态成员 class。

//clazz就是一个User的类对象
Class<?> clazz=User.class;

第三种:通过 Class 类的静态方法 forName() 方法获取。

Class<?> clazz = Class.forName("com.tian.User");    

面试扩散

可能面试官会问相关的题目,比如:

Class.forName 和 ClassLoader.loadClass 的区别是什么?

参考:

反射中 Class.forName() 和 ClassLoader.loadClass() 的区别

4. ArrayList 和 LinkedList 的区别有哪些?

ArrayList

  • 优点:ArrayList 是实现了基于动态数组的数据结构,因为地址连续,一旦数据存储好了,查询操作效率会比较高(在内存里是连着放的)。
  • 缺点:因为地址连续,ArrayList 要移动数据,所以插入和删除操作效率比较低。

LinkedList

  • 优点:LinkedList 基于链表的数据结构,地址是任意的,所以在开辟内存空间的时候不需要等一个连续的地址。对于新增和删除操作,LinkedList 比较占优势。LinkedList 适用于要头尾操作或插入指定位置的场景。
  • 缺点:因为 LinkedList 要移动指针,所以查询操作性能比较低。

适用场景分析

  • 当需要对数据进行对随机访问的时候,选用 ArrayList。
  • 当需要对数据进行多次增加删除修改时,采用 LinkedList。

如果容量固定,并且只会添加到尾部,不会引起扩容,优先采用 ArrayList。

当然,绝大数业务的场景下,使用 ArrayList 就够了,但需要注意避免 ArrayList 的扩容,以及非顺序的插入。

5. 用过 ArrayList 吗?说一下它有什么特点?

只要是搞 Java 的肯定都会回答“用过”。所以,回答题目的后半部分——ArrayList 的特点。可以从这几个方面去回答:

Java 集合框架中的一种存放相同类型的元素数据,是一种变长的集合类,基于定长数组实现,当加入数据达到一定程度后,会实行自动扩容,即扩大数组大小。

底层是使用数组实现,添加元素。

  • 如果 add(o),添加到的是数组的尾部,如果要增加的数据量很大,应该使用 ensureCapacity() 方法,该方法的作用是预先设置 ArrayList 的大小,这样可以大大提高初始化速度。
  • 如果使用 add(int,o),添加到某个位置,那么可能会挪动大量的数组元素,并且可能会触发扩容机制。

高并发的情况下,线程不安全。多个线程同时操作 ArrayList,会引发不可预知的异常或错误。

ArrayList 实现了 Cloneable 接口,标识着它可以被复制。注意:ArrayList 里面的 clone() 复制其实是浅复制。

6. 有数组了为什么还要搞个 ArrayList 呢?

通常我们在使用的时候,如果在不明确要插入多少数据的情况下,普通数组就很尴尬了,因为你不知道需要初始化数组大小为多少,而 ArrayList 可以使用默认的大小,当元素个数到达一定程度后,会自动扩容。

可以这么来理解:我们常说的数组是定死的数组,ArrayList 却是动态数组。

7. 说说什么是 fail-fast?

fail-fast 机制是 Java 集合(Collection)中的一种错误机制。当多个线程对同一个集合的内容进行操作时,就可能会产生 fail-fast 事件。

例如:当某一个线程 A 通过 iterator 去遍历某集合的过程中,若该集合的内容被其他线程所改变了,那么线程 A 访问集合时,就会抛出
ConcurrentModificationException 异常,产生 fail-fast 事件。这里的操作主要是指 add、remove 和 clear,对集合元素个数进行修改。

解决办法:建议使用“java.util.concurrent 包下的类”去取代“java.util 包下的类”。

可以这么理解:在遍历之前,把 modCount 记下来 expectModCount,后面 expectModCount 去和 modCount 进行比较,如果不相等了,证明已并发了,被修改了,于是抛出
ConcurrentModificationException 异常。

8. Hashtable 与 HashMap 的区别

本来不想这么写标题的,但是无奈,面试官都喜欢这么问 HashMap。

  1. 出生的版本不一样,Hashtable 出生于 Java 发布的第一版本 JDK 1.0,HashMap 出生于 JDK 1.2。
  2. 都实现了 Map、Cloneable、Serializable(当前 JDK 版本 1.8)。
  3. HashMap 继承的是 AbstractMap,并且 AbstractMap 也实现了 Map 接口。Hashtable 继承 Dictionary。
  4. Hashtable 中大部分 public 修饰普通方法都是 synchronized 字段修饰的,是线程安全的,HashMap 是非线程安全的。
  5. Hashtable 的 key 不能为 null,value 也不能为 null,这个可以从 Hashtable 源码中的 put 方法看到,判断如果 value 为 null 就直接抛出空指针异常,在 put 方法中计算 key 的 hash 值之前并没有判断 key 为 null 的情况,那说明,这时候如果 key 为空,照样会抛出空指针异常。
  6. HashMap 的 key 和 value 都可以为 null。在计算 hash 值的时候,有判断,如果 key==null,则其 hash=0;至于 value 是否为 null,根本没有判断过。
  7. Hashtable 直接使用对象的 hash 值。hash 值是 JDK 根据对象的地址或者字符串或者数字算出来的 int 类型的数值。然后再使用除留余数法来获得最终的位置。然而除法运算是非常耗费时间的,效率很低。HashMap 为了提高计算效率,将哈希表的大小固定为了 2 的幂,这样在取模预算时,不需要做除法,只需要做位运算。位运算比除法的效率要高很多。
  8. Hashtable、HashMap 都使用了 Iterator。而由于历史原因,Hashtable 还使用了 Enumeration 的方式。
  9. 默认情况下,初始容量不同,Hashtable 的初始长度是 11,之后每次扩充容量变为之前的 2n+1(n 为上一次的长度)而 HashMap 的初始长度为 16,之后每次扩充变为原来的两倍。

另外在 Hashtable 源码注释中有这么一句话:

Hashtable is synchronized.  If a thread-safe implementation is not needed, it is recommended to use HashMap in place of Hashtable . If a thread-safe highly-concurrent implementation is desired, then it is recommended to use ConcurrentHashMap in place of Hashtable.

大致意思:Hashtable 是线程安全,推荐使用 HashMap 代替 Hashtable;如果需要线程安全高并发的话,推荐使用 ConcurrentHashMap 代替 Hashtable。

这个回答完了,面试官可能会继续问:HashMap 是线程不安全的,那么在需要线程安全的情况下还要考虑性能,有什么解决方式?

这里最好的选择就是 ConcurrentHashMap 了,但面试官肯定会叫你继续说一下 ConcurrentHashMap 数据结构以及底层原理等。

9. HashMap 中的 key 我们可以使用任何类作为 key 吗?

平时可能大家使用的最多的就是使用 String 作为 HashMap 的 key,但是现在我们想使用某个自定义类作为 HashMap 的 key,那就需要注意以下几点:

  • 如果类重写了 equals 方法,它也应该重写 hashCode 方法。
  • 类的所有实例需要遵循与 equals 和 hashCode 相关的规则。
  • 如果一个类没有使用 equals,你不应该在 hashCode 中使用它。
  • 咱们自定义 key 类的最佳实践是使之为不可变的,这样,hashCode 值可以被缓存起来,拥有更好的性能。不可变的类也可以确保 hashCode 和 equals 在未来不会改变,这样就会解决与可变相关的问题了。

10. HashMap 的长度为什么是 2 的 N 次方呢?

为了能让 HashMap 存数据和取数据的效率高,尽可能地减少 hash 值的碰撞,也就是说尽量把数据能均匀的分配,每个链表或者红黑树长度尽量相等。

我们首先可能会想到 % 取模的操作来实现。

下面是回答的重点哟:

取余(%)操作中如果除数是 2 的幂次,则等价于与其除数减一的与(&)操作(也就是说 hash % length == hash &(length - 1) 的前提是 length 是 2 的 n 次方)。并且,采用二进制位操作 &,相对于 % 能够提高运算效率。

这就是为什么 HashMap 的长度需要 2 的 N 次方了。

11. HashMap 与 ConcurrentHashMap 的异同

  1. 都是 key-value 形式的存储数据;
  2. HashMap 是线程不安全的,ConcurrentHashMap 是 JUC 下的线程安全的;
  3. HashMap 底层数据结构是数组 + 链表(JDK 1.8 之前)。JDK 1.8 之后是数组 + 链表 + 红黑树。当链表中元素个数达到 8 的时候,链表的查询速度不如红黑树快,链表会转为红黑树,红黑树查询速度快;
  4. HashMap 初始数组大小为 16(默认),当出现扩容的时候,以 0.75 * 数组大小的方式进行扩容;
  5. ConcurrentHashMap 在 JDK 1.8 之前是采用分段锁来现实的 Segment + HashEntry,Segment 数组大小默认是 16,2 的 n 次方;JDK 1.8 之后,采用 Node + CAS + Synchronized 来保证并发安全进行实现。

13. 红黑树有哪几个特征?

紧接上个问题,面试官很有可能会问红黑树,下面把红黑树的几个特征列出来:

如果面试官还要继续问红黑树具体是怎么添加节点和删除节点的,推荐看:

30 张图带你彻底理解红黑树

14. 说说你平时是怎么处理 Java 异常的

try-catch-finally

  • try 块负责监控可能出现异常的代码
  • catch 块负责捕获可能出现的异常,并进行处理
  • finally 块负责清理各种资源,不管是否出现异常都会执行
  • 其中 try 块是必须的,catch 和 finally 至少存在一个标准异常处理流程

抛出异常→捕获异常→捕获成功(当 catch 的异常类型与抛出的异常类型匹配时,捕获成功)→异常被处理,程序继续运行 抛出异常→捕获异常→捕获失败(当 catch 的异常类型与抛出异常类型不匹配时,捕获失败)→异常未被处理,程序中断运行

在开发过程中会使用到自定义异常,在通常情况下,程序很少会自己抛出异常,因为异常的类名通常也包含了该异常的有用信息,所以在选择抛出异常的时候,应该选择合适的异常类,从而可以明确地描述该异常情况,所以这时候往往都是自定义异常。

自定义异常通常是通过继承 java.lang.Exception 类,如果想自定义 Runtime 异常的话,可以继承
java.lang.RuntimeException 类,实现一个无参构造和一个带字符串参数的有参构造方法。

在业务代码里,可以针对性的使用自定义异常。比如说:该用户不具备某某权限、余额不足等。

15. finally 模块执行了吗?是先执行 return 还是先执行 finally 模块?返回什么?

public class FinallyDemo {
    public String method111() {
        String ret = "hello";
        try {
            return ret;
        } finally {
            ret = "world";
        }
    }
}

把 FinallyDemo.java 编译成 class 文件后,找到该 class 文件的当前目录,执行 cmd 命令:

javap -verbose FinallyDemo.class >>test.txt

然后打开 test.txt,关键部分内容如下:

发现在字节码指令中,将 hello 保存在本地变量 2 中,然后直到把本地变量 2 加载到操作数栈中,然后就直接出栈,return 回去了,所以本题的返回去的是 hello,但是 finally 代码块也执行了,执行完 finally 模块后再返回一个临时变量 2。

二、JVM 篇

16. Java 类加载器有几种?

17. 说一下有哪些类加载场景?

18. 说说 Java 类加载机制是什么?说说 new 创建一个普通对象的过程?

类加载的过程包括了:

加载、验证、准备、解析、初始化。

new 创建一个普通对象的过程如下:

  1. 检测类是否被加载过
  2. 为对象分配内存
  3. 为分配的内存空间初始化为 0
  4. 对对象进行其他相关设置
  5. 执行 init 方法

下面用一张图来描述:

19. 说说类的生命周期?

注意类生命周期和对象声明周期,类生命周期主要有以下几个阶段:

20. 什么是双亲委派模型?

21. 如何破坏双亲委派模型?

22. 能不能自己也写一个 java.lang.String 类?

可以写,能编译,但是不能 run。禁止使用包名:java.开头的包名。

定义一个普通类:

package java.lang;

public class MyTest {

    public MyTest() {
    }
    public MyTest(String str, int a) {
    }

    public int length(){
        return 10;
  
  • 0
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值