Spring源码解析(二)BeanDefinition的Resource定位

   IOC容器的初始化过程主要包括BeanDefinition的Resource定位、载入和注册。在实际项目中我们基本上操作的都是ApplicationContex的实现,我们比较熟悉的ClassPathXmlApplicationContext、FileSystemXmlApplicationContext、XmlWebapplicationContext等。ApplicationContext的具体继承体系如下图所示:

  

   其实,不管是XmlWebApplicationContext还是ClasspathXmlApplicationContext 他们的区别只是Bean的资源信息来源不一样而已,最终都会解析为统一数据结构BeanDefinition。

下面我们源码的解析就从高富帅的ClassPathXmlApplicationContext开始。

        ClassPathXmlApplicationContext context=new ClassPathXmlApplicationContext("classpath*:test.xml");

构造方法:

/*** Create a new ClassPathXmlApplicationContext, loading the definitions* from the given XML file and automatically refreshing the context.* @param configLocation resource location* @throws BeansException if context creation failed*/public ClassPathXmlApplicationContext(String configLocation) throws BeansException {this(new String[] {configLocation}, true, null);}

最终调用构造方法:

/*** Create a new ClassPathXmlApplicationContext with the given parent,* loading the definitions from the given XML files.* @param configLocations array of resource locations* @param refresh whether to automatically refresh the context,* loading all bean definitions and creating all singletons.* Alternatively, call refresh manually after further configuring the context.* @param parent the parent context* @throws BeansException if context creation failed* @see #refresh()*/public ClassPathXmlApplicationContext(String[] configLocations, boolean refresh, ApplicationContext parent)throws BeansException {super(parent);setConfigLocations(configLocations);if (refresh) {refresh();}}

1.设置父级上下文,最终是给AbstractApplicationContext的parent属性赋值,AbstractApplicationContext是ApplicationContext最顶层的实现类。

2.设置XML文件的位置,调用了AbstractRefreshableConfigApplicationContext的setConfigLocations方法

/*** Set the config locations for this application context.* <p>If not set, the implementation may use a default as appropriate.*/public void setConfigLocations(String[] locations) {if (locations != null) {Assert.noNullElements(locations, "Config locations must not be null");this.configLocations = new String[locations.length];for (int i = 0; i < locations.length; i++) {this.configLocations[i] = resolvePath(locations[i]).trim();}}else {this.configLocations = null;}}

3.刷新容器,调用了AbstractApplicationContext的refresh方法,这是一个模板方法,具体的操作都是有子类去实现的。

public void refresh() throws BeansException, IllegalStateException {synchronized (this.startupShutdownMonitor) {// Prepare this context for refreshing.//刷新容器前的准备工作
            prepareRefresh();// Tell the subclass to refresh the internal bean factory.//由子类实现容器的刷新(重启)ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();// Prepare the bean factory for use in this context./*容器使用前的准备工作*/prepareBeanFactory(beanFactory);try {// Allows post-processing of the bean factory in context subclasses.//甚至beanFacotry的后置处理
                postProcessBeanFactory(beanFactory);// Invoke factory processors registered as beans in the context.//调用BeanFactory的后置处理器,这些后置处理是在Bean定义中想容器注册的
                invokeBeanFactoryPostProcessors(beanFactory);// Register bean processors that intercept bean creation.//注册Bean的后置处理器,在Bean的创建过程中调用
                registerBeanPostProcessors(beanFactory);// Initialize message source for this context.//对上下文中的消息源进行初始化
                initMessageSource();// Initialize event multicaster for this context.//初始化上下文的事件
                initApplicationEventMulticaster();// Initialize other special beans in specific context subclasses.//初始化其他特殊的Bean
                onRefresh();// Check for listener beans and register them.//向容器注册监听Bean
                registerListeners();// Instantiate all remaining (non-lazy-init) singletons.//实例化所有非延迟加载的Bean
                finishBeanFactoryInitialization(beanFactory);// Last step: publish corresponding event.//发布容器事件,结束refresh过程
                finishRefresh();}catch (BeansException ex) {logger.warn("Exception encountered during context initialization - cancelling refresh attempt", ex);// Destroy already created singletons to avoid dangling resources.
                destroyBeans();// Reset 'active' flag.
                cancelRefresh(ex);// Propagate exception to caller.throw ex;}}}
prepareRefresh():主要是设置启动时间、状态等等;
我们着重看一下刷新容器的obtainFreshBeanFactory()方法:
/*** Tell the subclass to refresh the internal bean factory.* @return the fresh BeanFactory instance* @see #refreshBeanFactory()* @see #getBeanFactory()*/protected ConfigurableListableBeanFactory obtainFreshBeanFactory() {//销毁已有容器,重新创建容器并加载Bean
        refreshBeanFactory();ConfigurableListableBeanFactory beanFactory = getBeanFactory();if (logger.isDebugEnabled()) {logger.debug("Bean factory for " + getDisplayName() + ": " + beanFactory);}return beanFactory;}
AbstractApplicationConetxt的refreshBeanFactory()方法是一个抽象方法,是由它的子类AbstractRefreshableApplicationContext实现的,从类的命名上可以看出这个类主要就是进行容器Refresh用的。
/*** This implementation performs an actual refresh of this context's underlying* bean factory, shutting down the previous bean factory (if any) and* initializing a fresh bean factory for the next phase of the context's lifecycle.*/@Overrideprotected final void refreshBeanFactory() throws BeansException {//如果容器已经存在则销毁容器中的bean并关闭容器if (hasBeanFactory()) {destroyBeans();closeBeanFactory();}try {//创建beanFacotryDefaultListableBeanFactory beanFactory = createBeanFactory();beanFactory.setSerializationId(getId());customizeBeanFactory(beanFactory);//根据bean定义的方式(XML、注解等)不同,由子类选择相应的BeanDefinitionReader去解析
            loadBeanDefinitions(beanFactory);synchronized (this.beanFactoryMonitor) {this.beanFactory = beanFactory;}}catch (IOException ex) {throw new ApplicationContextException("I/O error parsing bean definition source for " + getDisplayName(), ex);}}

第一步: 这个方法会判断如果已存在容器,则先销毁所有的Bean并且关闭容器,这也是为了保证容器的唯一性。

第二步:createBeanFactory()创建了一个DefaultListableBeanFactory,这个类是BeanFacotry最高级的实现,有了它就有个容器最基本的功能了。

/*** Create an internal bean factory for this context.* Called for each {@link #refresh()} attempt.* <p>The default implementation creates a* {@link org.springframework.beans.factory.support.DefaultListableBeanFactory}* with the {@linkplain #getInternalParentBeanFactory() internal bean factory} of this* context's parent as parent bean factory. Can be overridden in subclasses,* for example to customize DefaultListableBeanFactory's settings.* @return the bean factory for this context* @see org.springframework.beans.factory.support.DefaultListableBeanFactory#setAllowBeanDefinitionOverriding* @see org.springframework.beans.factory.support.DefaultListableBeanFactory#setAllowEagerClassLoading* @see org.springframework.beans.factory.support.DefaultListableBeanFactory#setAllowCircularReferences* @see org.springframework.beans.factory.support.DefaultListableBeanFactory#setAllowRawInjectionDespiteWrapping*/protected DefaultListableBeanFactory createBeanFactory() {//新建一个DefaultListableBeanFactoryreturn new DefaultListableBeanFactory(getInternalParentBeanFactory());}

 DefaultListableBeanFactory的继承关系:

我们看到DefaultListableBeanFactory实现了BeanDefinitionRegistry接口,也就是说最终BeanDefinition的注册工作是由它和它的子类来完成的。

第三步:loadBeanDefinitions(beanFactory),这个方法也是一个抽象方法。因为Bean定义方式不同(XML、注解等),会有多个子类分别去实现具体的解析。

此处,调用的是AbstractXmlApplicationContext的实现:

/*** Loads the bean definitions via an XmlBeanDefinitionReader.* @see org.springframework.beans.factory.xml.XmlBeanDefinitionReader* @see #initBeanDefinitionReader* @see #loadBeanDefinitions*/@Overrideprotected void loadBeanDefinitions(DefaultListableBeanFactory beanFactory) throws BeansException, IOException {// Create a new XmlBeanDefinitionReader for the given BeanFactory.XmlBeanDefinitionReader beanDefinitionReader = new XmlBeanDefinitionReader(beanFactory);// Configure the bean definition reader with this context's// resource loading environment.beanDefinitionReader.setEnvironment(this.getEnvironment());//ApplicationContext继承了ResourceLoader接口,所以this是可以直接使用的beanDefinitionReader.setResourceLoader(this);beanDefinitionReader.setEntityResolver(new ResourceEntityResolver(this));// Allow a subclass to provide custom initialization of the reader,// then proceed with actually loading the bean definitions.
        initBeanDefinitionReader(beanDefinitionReader);//委派模式,具体事情委派给beanDefinitionReader去做
        loadBeanDefinitions(beanDefinitionReader);}

 进入loadBeanDefinitions(XmlBeanDefinitionReader reader):

/*** Load the bean definitions with the given XmlBeanDefinitionReader.* <p>The lifecycle of the bean factory is handled by the {@link #refreshBeanFactory}* method; hence this method is just supposed to load and/or register bean definitions.* @param reader the XmlBeanDefinitionReader to use* @throws BeansException in case of bean registration errors* @throws IOException if the required XML document isn't found* @see #refreshBeanFactory* @see #getConfigLocations* @see #getResources* @see #getResourcePatternResolver*/protected void loadBeanDefinitions(XmlBeanDefinitionReader reader) throws BeansException, IOException {Resource[] configResources = getConfigResources();if (configResources != null) {reader.loadBeanDefinitions(configResources);}String[] configLocations = getConfigLocations();if (configLocations != null) {reader.loadBeanDefinitions(configLocations);}}

 

该方法调用了XmlBeanDefinitionReader父类AbstractBeanDefinitionReader的loadBeanDefinitions方法:

public int loadBeanDefinitions(String... locations) throws BeanDefinitionStoreException {Assert.notNull(locations, "Location array must not be null");int counter = 0;for (String location : locations) {counter += loadBeanDefinitions(location);}return counter;}

 

l循环加载location并返回加载个数,最终调用了本类的loadBeanDefinitions(String location, Set<Resource> actualResources)方法,actualResources为null:

/*** Load bean definitions from the specified resource location.* <p>The location can also be a location pattern, provided that the* ResourceLoader of this bean definition reader is a ResourcePatternResolver.* @param location the resource location, to be loaded with the ResourceLoader* (or ResourcePatternResolver) of this bean definition reader* @param actualResources a Set to be filled with the actual Resource objects* that have been resolved during the loading process. May be {@code null}* to indicate that the caller is not interested in those Resource objects.* @return the number of bean definitions found* @throws BeanDefinitionStoreException in case of loading or parsing errors* @see #getResourceLoader()* @see #loadBeanDefinitions(org.springframework.core.io.Resource)* @see #loadBeanDefinitions(org.springframework.core.io.Resource[])*/public int loadBeanDefinitions(String location, Set<Resource> actualResources) throws BeanDefinitionStoreException {ResourceLoader resourceLoader = getResourceLoader();if (resourceLoader == null) {throw new BeanDefinitionStoreException("Cannot import bean definitions from location [" + location + "]: no ResourceLoader available");}//resourceLoader是ClasspathXmlApplicationContext,ApplicationContext接口本身继承了ResourcePatternResolver接口if (resourceLoader instanceof ResourcePatternResolver) {// Resource pattern matching available.try {//location转为Resource完成定位工作Resource[] resources = ((ResourcePatternResolver) resourceLoader).getResources(location);int loadCount = loadBeanDefinitions(resources);if (actualResources != null) {for (Resource resource : resources) {actualResources.add(resource);}}if (logger.isDebugEnabled()) {logger.debug("Loaded " + loadCount + " bean definitions from location pattern [" + location + "]");}return loadCount;}catch (IOException ex) {throw new BeanDefinitionStoreException("Could not resolve bean definition resource pattern [" + location + "]", ex);}}else {// Can only load single resources by absolute URL.Resource resource = resourceLoader.getResource(location);int loadCount = loadBeanDefinitions(resource);if (actualResources != null) {actualResources.add(resource);}if (logger.isDebugEnabled()) {logger.debug("Loaded " + loadCount + " bean definitions from location [" + location + "]");}return loadCount;}}
Resource[] resources = ((ResourcePatternResolver) resourceLoader).getResources(location)将location转成了Resource[],这一步完成了资源的定位工作。
它调用了PathMatchingResourcePatternResolver的getResources方法:
 1 public Resource[] getResources(String locationPattern) throws IOException {
 2         Assert.notNull(locationPattern, "Location pattern must not be null");
 3         //是否以classpath*:开头
 4         if (locationPattern.startsWith(CLASSPATH_ALL_URL_PREFIX)) {
 5             // a class path resource (multiple resources for same name possible)
 6             //是否为Ant-style路径
 7             //? 匹配任何单字符
 8             //* 匹配0或者任意数量的字符
 9             //** 匹配0或者更多的目录
10             if (getPathMatcher().isPattern(locationPattern.substring(CLASSPATH_ALL_URL_PREFIX.length()))) {
11                 // a class path resource pattern
12                 return findPathMatchingResources(locationPattern);
13             }
14             else {
15                 // all class path resources with the given name
16                 return findAllClassPathResources(locationPattern.substring(CLASSPATH_ALL_URL_PREFIX.length()));
17             }
18         }
19         else {
20             // Only look for a pattern after a prefix here
21             // (to not get fooled by a pattern symbol in a strange prefix).
22             int prefixEnd = locationPattern.indexOf(":") + 1;
23             if (getPathMatcher().isPattern(locationPattern.substring(prefixEnd))) {
24                 // a file pattern
25                 return findPathMatchingResources(locationPattern);
26             }
27             else {
28                 // a single resource with the given name
29                 return new Resource[] {getResourceLoader().getResource(locationPattern)};
30             }
31         }
32     }

 

 根据location写法,解析方式也不同:

1、前缀为classpath*

  1)文件路径路径中包含*和?

    调用findPathMatchingResources方法

  2)文件路径中不含*和?

    调用findAllClassPathResources方法

2.、前缀为classpath

  1)文件路径路径中包含*和?

  调用findPathMatchingResources方法
 2)文件路径中不含*和?
  调用DefaultResourceLoader的getResource方法new一个ClasspathResource并返回,如果资源文件根本就不存在,此处也不会校验。
findPathMatchingResources和findAllClassPathResources具体都干了什么呢?
先看一下findAllClassPathResources:
/*** Find all class location resources with the given location via the ClassLoader.* @param location the absolute path within the classpath* @return the result as Resource array* @throws IOException in case of I/O errors* @see java.lang.ClassLoader#getResources* @see #convertClassLoaderURL*/protected Resource[] findAllClassPathResources(String location) throws IOException {String path = location;if (path.startsWith("/")) {path = path.substring(1);}ClassLoader cl = getClassLoader();Enumeration<URL> resourceUrls = (cl != null ? cl.getResources(path) : ClassLoader.getSystemResources(path));Set<Resource> result = new LinkedHashSet<Resource>(16);while (resourceUrls.hasMoreElements()) {URL url = resourceUrls.nextElement();result.add(convertClassLoaderURL(url));}return result.toArray(new Resource[result.size()]);}

   protected Resource convertClassLoaderURL(URL url) {
return new UrlResource(url);
}
这个方法很简单,根据具体的location通过classLoader的getResources方法返回RUL集合,根据URL创建UrlResource并返回UrlResource的集合。
再来看一下findPathMatchingResources方法:
/*** Find all resources that match the given location pattern via the* Ant-style PathMatcher. Supports resources in jar files and zip files* and in the file system.* @param locationPattern the location pattern to match* @return the result as Resource array* @throws IOException in case of I/O errors* @see #doFindPathMatchingJarResources* @see #doFindPathMatchingFileResources* @see org.springframework.util.PathMatcher*/protected Resource[] findPathMatchingResources(String locationPattern) throws IOException {String rootDirPath = determineRootDir(locationPattern);String subPattern = locationPattern.substring(rootDirPath.length());Resource[] rootDirResources = getResources(rootDirPath);Set<Resource> result = new LinkedHashSet<Resource>(16);for (Resource rootDirResource : rootDirResources) {rootDirResource = resolveRootDirResource(rootDirResource);if (rootDirResource.getURL().getProtocol().startsWith(ResourceUtils.URL_PROTOCOL_VFS)) {result.addAll(VfsResourceMatchingDelegate.findMatchingResources(rootDirResource, subPattern, getPathMatcher()));}else if (isJarResource(rootDirResource)) {result.addAll(doFindPathMatchingJarResources(rootDirResource, subPattern));}else {result.addAll(doFindPathMatchingFileResources(rootDirResource, subPattern));}}if (logger.isDebugEnabled()) {logger.debug("Resolved location pattern [" + locationPattern + "] to resources " + result);}return result.toArray(new Resource[result.size()]);}
1.String rootDirPath = determineRootDir(locationPattern),获取location前缀classpath*/classpath
2.Resource[] rootDirResources = getResources(rootDirPath),调用的上面讲到的getResources方法,返回classpath根路径的Resource[],如果是classpath会返回一个Resource,
如果是classpath*会放回所有的classpath路径。
3.遍历根路径Resource[],doFindPathMatchingFileResources方法就是获取给定路径下的所有文件,根据指定的文件名test*.xml去模糊匹配,返回的是FileSystemResource。所以location为classpath:test*.xml可能会找不到文件。

 

 

 



转载于:https://www.cnblogs.com/monkey0307/p/8436134.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/466542.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

JSON.parse()

JSON.parse() 方法用于将一个 JSON 字符串转换为对象。 语法 JSON.parse(text[, reviver]) 参数说明&#xff1a; text:必需&#xff0c; 一个有效的 JSON 字符串。 reviver: 可选&#xff0c;一个转换结果的函数&#xff0c; 将为对象的每个成员调用此函数。 返回值&#xf…

apache+php+mysql+phpadmin 服务环境搭建

前些日子&#xff0c;有朋友问如何用apache/php/mysql/phpadmin搭建PHP网站服务环境.现在就这个问题&#xff0c;我把我的一些经验与大家分享。各位同行有何建议&#xff0c;敬请在评论区指出&#xff0c;谢谢&#xff01; 1.下载apache软件&#xff0c;以apache2.2为例。安装…

单片机(MCU)如何才能不死机之对齐访问(Aligned Access)

从一个结构体说起。如下&#xff0c;在 STM32F0 的程序中&#xff0c;我们定义了一个结构体My_Struct &#xff0c;那么这个结构体占用多少内存呢&#xff1f;struct Struct_Def { uint8_t Var_B; uint16_t Var_W0; uint16_t Var_W1; uint32_t Var_DW; }; struct Struct_Def My…

c_str()的用法

c_str()的用法ing&#xff0c;所以函数c_str()就是将C的string转化为C的字符串数组&#xff0c;c_str()生成一个const char *指针&#xff0c;指向字符串的首地址。 char *ps[10]; string a“welcome”; strcpy(p,a.c_str()); cout<<p; 结果为"welcome".

常见的蓝牙4.0模块

常见的蓝牙4.0模块 1、TI公司的cc2540/cc2640&#xff1b;2、nordic的蓝牙4.0&#xff08;BLE&#xff09;芯片nRF51822&#xff1b;3、汇承公司的HC-05蓝牙转串口模块&#xff1a;主芯片CSR BlueCore5 MM。

Sub-process /usr/bin/dpkg returned an error code (1)解决办法

在ubuntu 16.04更新了一些东西之后用apt装ros bringup包会出现如下错误&#xff1a; Sub-process /usr/bin/dpkg returned an error code (1) 解决办法&#xff1a; cd /var/lib/dpkg sudo mv info info.bak #即备份一个info sudo mkdir info #新建一个新的info目录 sudo apt i…

小米的隔空充电,看起来好酷

昨天是1月29号&#xff0c;昨天小米发布了一个隔空充电技术&#xff0c;很火爆&#xff0c;大胆想&#xff0c;如果有一条无线充电的高速公路&#xff0c;那电动汽车还担心没有电吗&#xff1f;—— 雷总的微博原文隔空充电技术&#xff1a;如科幻电影一般&#xff0c;拿着手机…

网络基础知识 快速计算子网掩码的2种方法

网络基础知识快速计算子网掩码的2种方法<?xml:namespace prefix o ns "urn:schemas-microsoft-com:office:office" />IP地址(IP Address)的概念及其子网掩码(Subnet Mask)的计算对于首次学习网络知识的初学者来说是一件比较困难的事情。下文所述的是我个人的…

C++ 解析Json

C 解析Json——jsoncpp JSON(JavaScript Object Notation) 是一种轻量级的数据交换格式&#xff0c;和xml类似&#xff0c;本文主要对VS2008中使用Jsoncpp解析json的方法做一下记录。 Jsoncpp是个跨平台的开源库&#xff0c;下载地址&#xff1a;http://sourceforge.net/projec…

Linux下获取文件编译的时间

#include <stdio.h>#define MONTH_PER_YEAR 12 // 一年12月 #define YEAR_MONTH_DAY 20 // 年月日缓存大小 #define HOUR_MINUTES_SEC 20 // 时分秒缓存大小void GetCompileTime(void) {const char year_month[MONTH_PER_YEAR][4] { "Jan", "Fe…

同事用void把我给秀翻了!

1、聊一聊今天跟大家推荐的这首歌最近应该挺火的&#xff0c;不过没办法插入AGA的原版歌曲&#xff0c;大家觉得不错可以去找找原版歌曲收录一下。昨天建立了"最后一个bug"技术交流群,由于群成员超过200就无法直接通过群二维码加入&#xff0c;如果大家想加入扫描下面…

Spring Boot之自定义属性

选择Spring Boot主要是考虑到它既能兼顾Spring的强大功能&#xff0c;还能实现快速开发的便捷。我想大多数人也是出于这个原因选择了Spring Boot,如果不是特殊应用场景&#xff0c;就只需要在application.properties中完成一些属性配置就能开启各模块的应用。而不像传统的XML配…

编程大师论道:PHP的魅力和不足何

编程大师论道:PHP的魅力和不足何在2008-07-01 来自&#xff1a;lizhe1985 [收藏到我的网摘]SOA 案例分析系列学习和实验 SOA 的各方面知识、方法、技术及工具Snort&#xff1a;入侵检测工具Snort使用手册&#xff1a;保护和分析 Web 站点Snort 使用手册之第2部分通过设置 Snor…

WCF基础

第1章 WCF基础本章主要介绍WCF的基本概念、构建模块以及WCF体系架构&#xff0c;以指导读者构建一个简单的WCF服务。从本章的内容中&#xff0c;我们可以了解到WCF的基本术语&#xff0c;包括地址&#xff08;Address&#xff09;、绑定&#xff08;Binding&#xff09;、契约&…

检测ARM板的RTC

#include <stdio.h> #include <time.h>// 可以测试ARM板的RTC是不是好的 void GetLocalTime(void) {time_t now;struct tm *timenow;time(&now);timenow localtime(&now);printf("Local time is %s", asctime(timenow)); }int main(void) {Get…

一个老工程师的工作经历和思考

在这里不敢以”资深”工程师自居&#xff0c;因为学历和技术水平确实一般。为什么说“老”呢&#xff1f;因为工作时间确实够长&#xff0c;已经接近20年。下面把自身工作和学习经历和大家分享一下&#xff0c;使初学者能够得到一些有用的东西。2000年毕业&#xff0c;机械电子…

LA 3942 Remember the Word

(Remember the Word ,LA 3942) 题目来源&#xff1a;https://vjudge.net/problem/UVALive-3942 题意&#xff1a;给定一个字符串S以及n个单词&#xff0c;字符用这n个单词进行拆分&#xff0c;输出拆分的方案数。 思路&#xff1a;dp字典树 可以先将这n个单词存储于字典树中&a…

网线水晶头的接法

网线水晶头的接法网线水晶头的接法 制作线的时候&#xff0c;有一个一般性的原则就是&#xff1a; 同类&#xff08;连接线的两端是同一种设备&#xff09;交叉&#xff08;做交叉线&#xff09; 异类&#xff08;连接线的两端是不同的设备&#xff09;平行&#xff08;作平行线…

C++的Json解析库:jsoncpp和boost

JSON(JavaScript Object Notation)跟xml一样也是一种数据交换格式&#xff0c;了解json请参考其官网http://json.org&#xff0c;本文不再对json做介绍&#xff0c;将重点介绍c的json解析库的使用方法。json官网上列出了各种语言对应的json解析库&#xff0c;作者仅介绍自己使用…

电子工程师都在看什么?送你一份“修炼宝典”

现如今&#xff0c;形形色色的公众号如繁星一般让人眼花缭乱。近几年科技的飞速发展&#xff0c;让更多人开始关注科技&#xff0c;甚至成为极客。然而学习是永无止境的&#xff0c;如何才能追赶如此高速的发展&#xff1f;曾经&#xff0c;我也是不知道去看哪些&#xff0c;便…