本文作者:geek,一个聪明好学的同事

1. 简介

开发中我们常用@Commpont,@Service,@Resource等注解或者配置xml去声明一个类,使其成为spring容器中的bean,以下我将用从源码角度看以AnnotationConfigApplicationContext为例看spring如何把带有注解的类生成spring中bean。

2. 示例代码

  1. public class TestContext {
  2. public static void main(String[] args) {
  3. AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class);
  4. SingleBean singleBean = context.getBean(SingleBean.class);
  5. System.out.println("<=====>"+singleBean.getTestStr());
  6. }
  7. }
  1. @ComponentScan("com.geek")
  2. public class AppConfig {
  3. }
  1. @Component
  2. public class SingleBean {
  3. private String testStr = "testStr";
  4. public String getTestStr() {
  5. return testStr;
  6. }
  7. }

注意:以上代码仅需要引入spring-context依赖即可。

3. 源码分析

​ 上面的demo在调用AnnotationConfigApplicationContext构造函数的时候,AppConfig类会被注册到AnnotatedBeanDefinitionReader,由这个reader把AppConfig解释为beanDefination,从而被spring获取到要实例化的类信息,以下为bean生产的源码及其注释。(源码基于springFramework 5.1.X)

3.1 创建入口

​ 单例bean的创建的入口为DefaultListableBeanFactory.java#preInstantiateSingletons,下面源码可见创建的bean条件为非抽象,非@LazyInit注解,Scope为singleTon(默认为singleTon)。

  1. @Override
  2. public void preInstantiateSingletons() throws BeansException {
  3. if (logger.isTraceEnabled()) {
  4. logger.trace("Pre-instantiating singletons in " + this);
  5. }
  6. // Iterate over a copy to allow for init methods which in turn register new bean definitions.
  7. // While this may not be part of the regular factory bootstrap, it does otherwise work fine.
  8. //所有可能需要去实例化的class(lazy,scope)
  9. List<String> beanNames = new ArrayList<>(this.beanDefinitionNames);
  10. // Trigger initialization of all non-lazy singleton beans...
  11. for (String beanName : beanNames) {
  12. RootBeanDefinition bd = getMergedLocalBeanDefinition(beanName);
  13. /**
  14. * 非抽象,非懒初始化,单例bean
  15. */
  16. if (!bd.isAbstract() && bd.isSingleton() && !bd.isLazyInit()) {
  17. if (isFactoryBean(beanName)) {
  18. Object bean = getBean(FACTORY_BEAN_PREFIX + beanName);
  19. if (bean instanceof FactoryBean) {
  20. final FactoryBean<?> factory = (FactoryBean<?>) bean;
  21. boolean isEagerInit;
  22. if (System.getSecurityManager() != null && factory instanceof SmartFactoryBean) {
  23. isEagerInit = AccessController.doPrivileged((PrivilegedAction<Boolean>)
  24. ((SmartFactoryBean<?>) factory)::isEagerInit,
  25. getAccessControlContext());
  26. }
  27. else {
  28. isEagerInit = (factory instanceof SmartFactoryBean &&
  29. ((SmartFactoryBean<?>) factory).isEagerInit());
  30. }
  31. if (isEagerInit) {
  32. getBean(beanName);
  33. }
  34. }
  35. }
  36. //非工厂bean实例化
  37. else {
  38. getBean(beanName);
  39. }
  40. }
  41. }
  42. /**
  43. * 实例化完成后触发实现了SmartInitializingSingleton方法的bean
  44. * 的afterSingletonsInstantiated方法
  45. */
  46. // Trigger post-initialization callback for all applicable beans...
  47. for (String beanName : beanNames) {
  48. Object singletonInstance = getSingleton(beanName);
  49. if (singletonInstance instanceof SmartInitializingSingleton) {
  50. final SmartInitializingSingleton smartSingleton = (SmartInitializingSingleton) singletonInstance;
  51. if (System.getSecurityManager() != null) {
  52. AccessController.doPrivileged((PrivilegedAction<Object>) () -> {
  53. smartSingleton.afterSingletonsInstantiated();
  54. return null;
  55. }, getAccessControlContext());
  56. }
  57. else {
  58. smartSingleton.afterSingletonsInstantiated();
  59. }
  60. }
  61. }
  62. }

3.2 创建前doGetBean代码逻辑

  1. getBean方法进来后便是直接调用doGetBeandoGetBean执行的源码解释如下:
  1. protected <T> T doGetBean(final String name, @Nullable final Class<T> requiredType,
  2. @Nullable final Object[] args, boolean typeCheckOnly) throws BeansException {
  3. /**
  4. * 获取beanName
  5. * 1,去掉factortoryBean前缀&
  6. * 2,带有别名的bean转换为原来名字
  7. */
  8. final String beanName = transformedBeanName(name);
  9. Object bean;
  10. // Eagerly check singleton cache for manually registered singletons.
  11. /**
  12. * 从DefaultSingletonBeanRegistry的singletonObjects
  13. * (spring内部用来缓存单例bean的currentHashMap)检查是否存在该bean,不存在则创建
  14. * 涉及单例模式下的循环依赖解决
  15. */
  16. Object sharedInstance = getSingleton(beanName);
  17. if (sharedInstance != null && args == null) {
  18. if (logger.isTraceEnabled()) {
  19. if (isSingletonCurrentlyInCreation(beanName)) {
  20. logger.trace("Returning eagerly cached instance of singleton bean '" + beanName +
  21. "' that is not fully initialized yet - a consequence of a circular reference");
  22. }
  23. else {
  24. logger.trace("Returning cached instance of singleton bean '" + beanName + "'");
  25. }
  26. }
  27. /**
  28. *获取给定bean实例的对象,如果是Factory Bean,
  29. * 则可以是bean实例本身或其创建的对象。
  30. */
  31. bean = getObjectForBeanInstance(sharedInstance, name, beanName, null);
  32. }
  33. else {
  34. // Fail if we're already creating this bean instance:
  35. // We're assembly within a circular reference.
  36. /**
  37. * 原型模式下的bean存在循环依赖则会抛异常
  38. */
  39. if (isPrototypeCurrentlyInCreation(beanName)) {
  40. throw new BeanCurrentlyInCreationException(beanName);
  41. }
  42. // Check if bean definition exists in this factory.
  43. /**
  44. * 找不到则从父容器中查找
  45. */
  46. BeanFactory parentBeanFactory = getParentBeanFactory();
  47. if (parentBeanFactory != null && !containsBeanDefinition(beanName)) {
  48. // Not found -> check parent.
  49. String nameToLookup = originalBeanName(name);
  50. if (parentBeanFactory instanceof AbstractBeanFactory) {
  51. return ((AbstractBeanFactory) parentBeanFactory).doGetBean(
  52. nameToLookup, requiredType, args, typeCheckOnly);
  53. }
  54. else if (args != null) {
  55. // Delegation to parent with explicit args.
  56. return (T) parentBeanFactory.getBean(nameToLookup, args);
  57. }
  58. else if (requiredType != null) {
  59. // No args -> delegate to standard getBean method.
  60. return parentBeanFactory.getBean(nameToLookup, requiredType);
  61. }
  62. else {
  63. return (T) parentBeanFactory.getBean(nameToLookup);
  64. }
  65. }
  66. if (!typeCheckOnly) {
  67. markBeanAsCreated(beanName);
  68. }
  69. try {
  70. /**
  71. * 父容器中也找不到该bean,则需要重新实例化
  72. * 1,获取要实例化bean的beanDefinition,
  73. * 2,检查bean的实例化需要依赖的其他bean
  74. */
  75. final RootBeanDefinition mbd = getMergedLocalBeanDefinition(beanName);
  76. checkMergedBeanDefinition(mbd, beanName, args);
  77. // Guarantee initialization of beans that the current bean depends on.
  78. String[] dependsOn = mbd.getDependsOn();
  79. if (dependsOn != null) {
  80. for (String dep : dependsOn) {
  81. /**
  82. * 若给定的依赖 bean 已经注册为依赖给定的bean
  83. */
  84. if (isDependent(beanName, dep)) {
  85. throw new BeanCreationException(mbd.getResourceDescription(), beanName,
  86. "Circular depends-on relationship between '" + beanName + "' and '" + dep + "'");
  87. }
  88. registerDependentBean(dep, beanName);
  89. try {
  90. /**
  91. * 递归调用getBean,优先创建依赖的bean
  92. */
  93. getBean(dep);
  94. }
  95. catch (NoSuchBeanDefinitionException ex) {
  96. throw new BeanCreationException(mbd.getResourceDescription(), beanName,
  97. "'" + beanName + "' depends on missing bean '" + dep + "'", ex);
  98. }
  99. }
  100. }
  101. // Create bean instance.
  102. if (mbd.isSingleton()) {
  103. /**
  104. * 通过调用DefaultSingletonBeanRegistry的getSingleton,从而调用核心方法createBean创建
  105. */
  106. sharedInstance = getSingleton(beanName, () -> {
  107. try {
  108. return createBean(beanName, mbd, args);
  109. }
  110. catch (BeansException ex) {
  111. // Explicitly remove instance from singleton cache: It might have been put there
  112. // eagerly by the creation process, to allow for circular reference resolution.
  113. // Also remove any beans that received a temporary reference to the bean.
  114. destroySingleton(beanName);
  115. throw ex;
  116. }
  117. });
  118. bean = getObjectForBeanInstance(sharedInstance, name, beanName, mbd);
  119. }
  120. else if (mbd.isPrototype()) {
  121. // It's a prototype -> create a new instance.
  122. Object prototypeInstance = null;
  123. try {
  124. beforePrototypeCreation(beanName);
  125. prototypeInstance = createBean(beanName, mbd, args);
  126. }
  127. finally {
  128. afterPrototypeCreation(beanName);
  129. }
  130. bean = getObjectForBeanInstance(prototypeInstance, name, beanName, mbd);
  131. }
  132. else {
  133. String scopeName = mbd.getScope();
  134. final Scope scope = this.scopes.get(scopeName);
  135. if (scope == null) {
  136. throw new IllegalStateException("No Scope registered for scope name '" + scopeName + "'");
  137. }
  138. try {
  139. Object scopedInstance = scope.get(beanName, () -> {
  140. beforePrototypeCreation(beanName);
  141. try {
  142. return createBean(beanName, mbd, args);
  143. }
  144. finally {
  145. afterPrototypeCreation(beanName);
  146. }
  147. });
  148. bean = getObjectForBeanInstance(scopedInstance, name, beanName, mbd);
  149. }
  150. catch (IllegalStateException ex) {
  151. throw new BeanCreationException(beanName,
  152. "Scope '" + scopeName + "' is not active for the current thread; consider " +
  153. "defining a scoped proxy for this bean if you intend to refer to it from a singleton",
  154. ex);
  155. }
  156. }
  157. }
  158. catch (BeansException ex) {
  159. cleanupAfterBeanCreationFailure(beanName);
  160. throw ex;
  161. }
  162. }
  163. /**
  164. * 检查需要的bean类型是否符合
  165. */
  166. // Check if required type matches the type of the actual bean instance.
  167. if (requiredType != null && !requiredType.isInstance(bean)) {
  168. try {
  169. T convertedBean = getTypeConverter().convertIfNecessary(bean, requiredType);
  170. if (convertedBean == null) {
  171. throw new BeanNotOfRequiredTypeException(name, requiredType, bean.getClass());
  172. }
  173. return convertedBean;
  174. }
  175. catch (TypeMismatchException ex) {
  176. if (logger.isTraceEnabled()) {
  177. logger.trace("Failed to convert bean '" + name + "' to required type '" +
  178. ClassUtils.getQualifiedName(requiredType) + "'", ex);
  179. }
  180. throw new BeanNotOfRequiredTypeException(name, requiredType, bean.getClass());
  181. }
  182. }
  183. return (T) bean;
  184. }

doGetBean的操作流程如下:

1,执行transformedBeanName方法转换beanName

​ 传递的参数可能是bean的alias或者为FactoryBean,transformedBeanName执行的操作:(1)若传进来的FactoryBean(FactoryBean以&作为前缀标记),去掉&修饰符。(2)经过(1)的处理后,有alias的bean则从aliasMap中获取bean的原始beanName。

2,从容器的缓存中获取bean

​ getSingleton先从spring三级缓存中的第一级singletonObjects(Map结构)中获取,若不存在,则检查该bean是否正在创建isSingletonCurrentlyInCreation(beanName)) ,正在创建的bean会从二级缓存earlySingletonObjects(Map结构)获取。获取到缓存的bean后会调用AbstractBeanFactory#getObjectForBeanInstance转换bean的实例本身返回。因为从缓存中拿到的可能是factoryBean,所以getObjectForBeanInstance需要把是通过从缓存factoryBeanObjectCache获取或通过factory.getObject()获得相应的bean返回。

3,bean实例化前检查

​ (1)先检查是否原型模式下的bean是否存在循环依赖,是则会抛异常。

​ (2)检查父类工厂(parentBeanFactory)是否存在,存在则从parentBeanFactory中递归调用doGetBean。

​ (3)获取改bean的beanDefinition,检查该bean实例化过程中是否涉及依赖了其他的bean,若是则递归调用getBean,优先创建依赖的bean。(涉及单例下的循环以来解决,下篇文章详细介绍)。

​ (4)对创建bean代码加synchronized和执行beforeSingletonCreation(beanName)前置处理。

3.3 创建前createBean逻辑

​ 经过前面的doGetBean的一轮检查与准备后,便在AbstractAutowireCapableBeanFactory#createBean中开始bean的创建。

  1. /**
  2. * Central method of this class: creates a bean instance,
  3. * populates the bean instance, applies post-processors, etc.
  4. * 解析指定 BeanDefinition 的 class
  5. * 处理 override 属性
  6. * 实例化的前置处理
  7. * 创建 bean
  8. * @see #doCreateBean
  9. */
  10. @Override
  11. protected Object createBean(String beanName, RootBeanDefinition mbd, @Nullable Object[] args)
  12. throws BeanCreationException {
  13. if (logger.isTraceEnabled()) {
  14. logger.trace("Creating instance of bean '" + beanName + "'");
  15. }
  16. RootBeanDefinition mbdToUse = mbd;
  17. // Make sure bean class is actually resolved at this point, and
  18. // clone the bean definition in case of a dynamically resolved Class
  19. // which cannot be stored in the shared merged bean definition.
  20. /**
  21. * 解释bean的class,看beanDefinition是否有class,否则load class
  22. */
  23. Class<?> resolvedClass = resolveBeanClass(mbd, beanName);
  24. if (resolvedClass != null && !mbd.hasBeanClass() && mbd.getBeanClassName() != null) {
  25. mbdToUse = new RootBeanDefinition(mbd);
  26. mbdToUse.setBeanClass(resolvedClass);
  27. }
  28. /**
  29. * 对bean不存在lookup-method 和 replace-method
  30. * 标记其方法的overloaded为false
  31. */
  32. // Prepare method overrides.
  33. try {
  34. mbdToUse.prepareMethodOverrides();
  35. }
  36. catch (BeanDefinitionValidationException ex) {
  37. throw new BeanDefinitionStoreException(mbdToUse.getResourceDescription(),
  38. beanName, "Validation of method overrides failed", ex);
  39. }
  40. try {
  41. /**
  42. * 调用实现实现BeanPostProcessor的bean后置处理生成代理对象,
  43. * 有代理对象则直接返回代理对象
  44. */
  45. // Give BeanPostProcessors a chance to return a proxy instead of the target bean instance.
  46. Object bean = resolveBeforeInstantiation(beanName, mbdToUse);
  47. if (bean != null) {
  48. return bean;
  49. }
  50. }
  51. catch (Throwable ex) {
  52. throw new BeanCreationException(mbdToUse.getResourceDescription(), beanName,
  53. "BeanPostProcessor before instantiation of bean failed", ex);
  54. }
  55. try {
  56. /**
  57. * 无需代理的bean实例化
  58. */
  59. Object beanInstance = doCreateBean(beanName, mbdToUse, args);
  60. if (logger.isTraceEnabled()) {
  61. logger.trace("Finished creating instance of bean '" + beanName + "'");
  62. }
  63. return beanInstance;
  64. }
  65. catch (BeanCreationException | ImplicitlyAppearedSingletonException ex) {
  66. // A previously detected exception with proper bean creation context already,
  67. // or illegal singleton state to be communicated up to DefaultSingletonBeanRegistry.
  68. throw ex;
  69. }
  70. catch (Throwable ex) {
  71. throw new BeanCreationException(
  72. mbdToUse.getResourceDescription(), beanName, "Unexpected exception during bean creation", ex);
  73. }
  74. }

createBean的操作流程如下:

1,resolveBeanClass

​ 解释beanDefinition的class,并且保存在beanDefinition中。

2,prepareMethodOverrides

​ 处理bean中的lookup-method (在单例bean用 @Lookup注解标记的方法,注解的方法返回的对象是原型)和 replace-method( 标记的方法,标记bean中A的方法被实现被另外一个实现MethodReplacer接口的B方法替代)。

3,resolveBeforeInstantiation

​ 调用实现实现BeanPostProcessor的bean后置处理生成代理对象,有代理对象则直接返回代理对象。(spring AOP则是基于此处实现)

3.4 bean的真正实例化createBeanInstance

​ 在AbstractAutowireCapableBeanFactory#createBeanInstance中,真正创建bean,源码及注释如下:

  1. protected BeanWrapper createBeanInstance(String beanName, RootBeanDefinition mbd, @Nullable Object[] args) {
  2. // Make sure bean class is actually resolved at this point.
  3. Class<?> beanClass = resolveBeanClass(mbd, beanName);
  4. if (beanClass != null && !Modifier.isPublic(beanClass.getModifiers()) && !mbd.isNonPublicAccessAllowed()) {
  5. throw new BeanCreationException(mbd.getResourceDescription(), beanName,
  6. "Bean class isn't public, and non-public access not allowed: " + beanClass.getName());
  7. }
  8. /**
  9. * 通过提供supplier回调方法创建
  10. */
  11. Supplier<?> instanceSupplier = mbd.getInstanceSupplier();
  12. if (instanceSupplier != null) {
  13. return obtainFromSupplier(instanceSupplier, beanName);
  14. }
  15. /**
  16. * 通过工厂方法创建 bean 实例,可以是静态工厂方法或者实例工厂
  17. */
  18. if (mbd.getFactoryMethodName() != null) {
  19. return instantiateUsingFactoryMethod(beanName, mbd, args);
  20. }
  21. // Shortcut when re-creating the same bean...
  22. boolean resolved = false;
  23. boolean autowireNecessary = false;
  24. if (args == null) {
  25. synchronized (mbd.constructorArgumentLock) {
  26. /**
  27. * 查找已经bean已经缓存解析的构造函数或者工厂方法
  28. */
  29. if (mbd.resolvedConstructorOrFactoryMethod != null) {
  30. resolved = true;
  31. autowireNecessary = mbd.constructorArgumentsResolved;
  32. }
  33. }
  34. }
  35. /**
  36. * 已经有缓存的构造函数或者工厂方法,直接实例化
  37. */
  38. if (resolved) {
  39. if (autowireNecessary) {
  40. return autowireConstructor(beanName, mbd, null, null);
  41. }
  42. else {
  43. return instantiateBean(beanName, mbd);
  44. }
  45. }
  46. /**
  47. * 通过实现BeanPostProcessor的代理类autowired的构造函数实例化
  48. */
  49. // Candidate constructors for autowiring?
  50. Constructor<?>[] ctors = determineConstructorsFromBeanPostProcessors(beanClass, beanName);
  51. if (ctors != null || mbd.getResolvedAutowireMode() == AUTOWIRE_CONSTRUCTOR ||
  52. mbd.hasConstructorArgumentValues() || !ObjectUtils.isEmpty(args)) {
  53. return autowireConstructor(beanName, mbd, ctors, args);
  54. }
  55. /**
  56. * 通过本身带有autowired的构造函数实例化,通过调用反射newInstance实现
  57. */
  58. // Preferred constructors for default construction?
  59. ctors = mbd.getPreferredConstructors();
  60. if (ctors != null) {
  61. return autowireConstructor(beanName, mbd, ctors, null);
  62. }
  63. /**
  64. * 无参构造函数实例化,通过调用反射newInstance实现
  65. */
  66. // No special handling: simply use no-arg constructor.
  67. return instantiateBean(beanName, mbd);
  68. }

由上述源码可以看出,实例化bean操作流程如下:

1,如果存在 Supplier 回调,则通过提供supplier回调方法创建,如以下方式定义的bean:

  1. @RunWith(SpringRunner.class)
  2. @SpringBootTest(classes = Spring5Application.class)
  3. public class BeanRegistrationTest {
  4. @Autowired
  5. private GenericWebApplicationContext context;
  6. context.registerBean(A.class, () -> new A());
  7. }

2,如果存在工厂方法,则通过工厂方法创建 bean 实例,可以是静态工厂方法或者实例工厂,如以下方式定义的bean:

  1. public class AFactory implements FactoryBean<A> {
  2. @Override
  3. public A getObject() throws Exception {
  4. return new A();
  5. }
  6. @Override
  7. public Class<?> getObjectType() {
  8. return A.class;
  9. }
  10. }
  11. //或:
  12. @Configuration
  13. public class BeanConfigration {
  14. @Bean
  15. public A a(){
  16. return new A();
  17. }
  18. }

3,已经有缓存的构造函数或者工厂方法,直接实例化。

4,以上三点都不存在,则使用带参构造函数与无参构造函数实例化。如以下方式定义的bean:

  1. @Commponet
  2. public class A{}

4.总结

​ spring单例bean的实例化流程大概就是这样,很多细节地方,包括循环依赖处理,bean属性填充等细节点下一章介绍。

参考

查看更多文章关注公众号:好奇心森林

Spring 源码学习 - 单例bean的实例化过程的更多相关文章

  1. Spring 源码学习——加载 Bean

    继上次注册 bean 之后好久没更新,这两天有空查了查资料也自己看了看 spring BeanFactory 的 getBean(beanName); 这个方法.因时间有限不能像之前那样复制代码并一行 ...

  2. Spring源码学习笔记之bean标签属性介绍及作用

    传统的Spring项目, xml 配置bean在代码中是经常遇到, 那么在配置bean的时候,这些属性的作用是什么呢? 虽然说现在boot项目兴起,基于xml配置的少了很多, 但是如果能够了解这些标签 ...

  3. Spring源码学习笔记12——总结篇,IOC,Bean的生命周期,三大扩展点

    Spring源码学习笔记12--总结篇,IOC,Bean的生命周期,三大扩展点 参考了Spring 官网文档 https://docs.spring.io/spring-framework/docs/ ...

  4. spring源码学习之路---深入AOP(终)

    作者:zuoxiaolong8810(左潇龙),转载请注明出处,特别说明:本博文来自博主原博客,为保证新博客中博文的完整性,特复制到此留存,如需转载请注明新博客地址即可. 上一章和各位一起看了一下sp ...

  5. Spring源码学习-容器BeanFactory(四) BeanDefinition的创建-自定义标签的解析.md

    写在前面 上文Spring源码学习-容器BeanFactory(三) BeanDefinition的创建-解析Spring的默认标签对Spring默认标签的解析做了详解,在xml元素的解析中,Spri ...

  6. Spring源码学习-容器BeanFactory(三) BeanDefinition的创建-解析Spring的默认标签

    写在前面 上文Spring源码学习-容器BeanFactory(二) BeanDefinition的创建-解析前BeanDefinition的前置操作中Spring对XML解析后创建了对应的Docum ...

  7. Spring源码学习笔记9——构造器注入及其循环依赖

    Spring源码学习笔记9--构造器注入及其循环依赖 一丶前言 前面我们分析了spring基于字段的和基于set方法注入的原理,但是没有分析第二常用的注入方式(构造器注入)(第一常用字段注入),并且在 ...

  8. spring源码学习之路---IOC初探(二)

    作者:zuoxiaolong8810(左潇龙),转载请注明出处,特别说明:本博文来自博主原博客,为保证新博客中博文的完整性,特复制到此留存,如需转载请注明新博客地址即可. 上一章当中我没有提及具体的搭 ...

  9. Spring 源码学习 04:初始化容器与 DefaultListableBeanFactory

    前言 在前一篇文章:创建 IoC 容器的几种方式中,介绍了四种方式,这里以 AnnotationConfigApplicationContext 为例,跟进代码,看看 IoC 的启动流程. 入口 从 ...

随机推荐

  1. Gitlab 安装、升级、备份、恢复、汉化等

    一.Gitlab安装 1. 基于yum方式安装Gitlab 安装步骤如下 (1)配置yum源 # vim /etc/yum.repos.d/gitlab-ce.repo (2)复制如下内容并保存(注意 ...

  2. .NET 程序员的 Playground :LINQPad

    如果想执行一个简单的 C# 语句并获得运行结果,通常我们需要做几个步骤才能达成: 打开 Visual Studio 并新建一个控制台项目. 在 Program.cs 中编写代码并保存. 点击运行按钮或 ...

  3. MySQL银行ATM存取款机系统(需求分析)

    银行ATM需求文档 一.E-R图形文 二.开发步骤 1.明确需求--数据库设计--编码实现功能--测试 2.绘制e-r图--绘制数据库模型图--使用三大方式规范数据库结构 三.开发思路 1. 模型图综 ...

  4. 2.2 Go变量类型

    内置类型 值类型: bool 布尔类型 int(32 or 64), int8, int16, int32, int64 整数类型 uint(32 or 64), uint8(byte), uint1 ...

  5. Django之ORM执行原生sql语句

    django中的ORM提供的操作功能有限,在模型提供的查询API不能满足实际工作需要时,可以在ORM中直接执行原生sql语句. Django 提供两种方法使用原生SQL进行查询:一种是使用raw()方 ...

  6. 如何利用BeautifulSoup选择器抓取京东网商品信息

    昨天小编利用Python正则表达式爬取了京东网商品信息,看过代码的小伙伴们基本上都坐不住了,辣么多的规则和辣么长的代码,悲伤辣么大,实在是受不鸟了.不过小伙伴们不用担心,今天小编利用美丽的汤来为大家演 ...

  7. 消息队列RabbitMQ的安装配置与PHP中的使用

    一.RabbitMQ安装 windows安装 下载地址: https://github.com/rabbitmq/rabbitmq-server/releases/download/v3.8.3/ra ...

  8. #442-Find All Duplicates in an Array-数组中重复的数字

    一.题目 给定一个整数数组 a,其中1 ≤ a[i] ≤ n (n为数组长度), 其中有些元素出现两次而其他元素出现一次. 找到所有出现两次的元素. 你可以不用到任何额外空间并在O(n)时间复杂度内解 ...

  9. [原创]RTX使用printf输出后进入hardfault中断的处理方法 - 讨论

    今天我用到RTX里面使用printf ,发现程序死掉了 我发现很多人遇到了这样的问题 找了网上很多的文章,说是这个是RTX的一个先天不足的问题 我发现了正点原子的 原子哥的解决方案,如下所示: --- ...

  10. mysql 的大文本存储TEXT & BLOB

    TEXT & BLOB 一般在保存少量字符串的时候,我们会选择 CHAR 或者 VARCHAR:而在保存较大文本时,通常会选择使用 TEXT 或者 BLOB,二者之间的主要差别是 BLOB 能 ...