细说Tomcat如何打破双亲委派(续1)

在上一篇文章里面分析WebappClassLoader.loadClass,了解了Tomcat是如何打破双亲委派的. 有兴趣的可以参考. 细说Tomcat如何打破双亲委派(有源码和图) 

 

但是上面的分析偏重于流程,没有分析ClassLoader是如何加按照顺序寻址,然后加载对应的Class到二进制流中.

 

WebappClassLoader.loadClass在违背双亲委派规则,自己寻址class文件时,就会调用下面的函数,函数已经添加了注释. 此函数依旧没有分析是如何寻址文件的,寻址文件的代码在findResourceInternal()中,它同时包括了对Jar,class, properties的寻址, 稍后有空在对此函数进行分析.

 

 

    /**
     * Find the specified class in our local repositories, if possible.  If
     * not found, throw <code>ClassNotFoundException</code>.
     *
     * @param name Name of the class to be loaded
     *
     * @exception ClassNotFoundException if the class was not found
     */
    @Override
    public Class<?> findClass(String name) throws ClassNotFoundException {

        if (log.isDebugEnabled())
            log.debug("    findClass(" + name + ")");

        // Cannot load anything from local repositories if class loader is stopped
        if (!started) {
            throw new ClassNotFoundException(name);
        }

		//--默认为NULL, security获取package.definition的定义,默认的定义可以在如下地址找到
		//--${TOMCAT_HOME}\bin\bootstrap.jar\org\apache\catalina\startup\catalina.properties		//--package.definition=sun.,java.,org.apache.catalina.,org.apache.coyote.,org.apache.tomcat.,org.apache.jasper.
		//--如果加载的package被包含了,则抛出异常,加载失败
        // (1) Permission to define this class when using a SecurityManager
        if (securityManager != null) {
            int i = name.lastIndexOf('.');
            if (i >= 0) {
                try {
                    if (log.isTraceEnabled())
                        log.trace("      securityManager.checkPackageDefinition");
                    securityManager.checkPackageDefinition(name.substring(0,i));
                } catch (Exception se) {
                    if (log.isTraceEnabled())
                        log.trace("      -->Exception-->ClassNotFoundException", se);
                    throw new ClassNotFoundException(name, se);
                }
            }
        }
        
        //-- 从父类查找Class(默认会跳过), 因为hasExternalRepositories && searchExternalFirs为false
        //-- Ask our superclass to locate this class, if possible
        // (throws ClassNotFoundException if it is not found)
        Class<?> clazz = null;
        try {
            if (log.isTraceEnabled())
                log.trace("      findClassInternal(" + name + ")");
            if (hasExternalRepositories && searchExternalFirst) {
                try {
                    clazz = super.findClass(name);
                } catch(ClassNotFoundException cnfe) {
                    // Ignore - will search internal repositories next
                } catch(AccessControlException ace) {
                    log.warn("WebappClassLoader.findClassInternal(" + name
                            + ") security exception: " + ace.getMessage(), ace);
                    throw new ClassNotFoundException(name, ace);
                } catch (RuntimeException e) {
                    if (log.isTraceEnabled())
                        log.trace("      -->RuntimeException Rethrown", e);
                    throw e;
                }
            }
            //-开始从本地加载Class
            if ((clazz == null)) {
                try {
                    clazz = findClassInternal(name);
                } catch(ClassNotFoundException cnfe) {
                    if (!hasExternalRepositories || searchExternalFirst) {
                        throw cnfe;
                    }
                } catch(AccessControlException ace) {
                    log.warn("WebappClassLoader.findClassInternal(" + name
                            + ") security exception: " + ace.getMessage(), ace);
                    throw new ClassNotFoundException(name, ace);
                } catch (RuntimeException e) {
                    if (log.isTraceEnabled())
                        log.trace("      -->RuntimeException Rethrown", e);
                    throw e;
                }
            }
            //--hasExternalRepositories:如果之前添加了外部路径,可以尝试从外部路径查找Class
            if ((clazz == null) && hasExternalRepositories && !searchExternalFirst) {
                try {
                    clazz = super.findClass(name);
                } catch(AccessControlException ace) {
                    log.warn("WebappClassLoader.findClassInternal(" + name
                            + ") security exception: " + ace.getMessage(), ace);
                    throw new ClassNotFoundException(name, ace);
                } catch (RuntimeException e) {
                    if (log.isTraceEnabled())
                        log.trace("      -->RuntimeException Rethrown", e);
                    throw e;
                }
            }
            //--如果都尝试了依旧找不到,就抛出异常了
            if (clazz == null) {
                if (log.isDebugEnabled())
                    log.debug("    --> Returning ClassNotFoundException");
                throw new ClassNotFoundException(name);
            }
        } catch (ClassNotFoundException e) {
            if (log.isTraceEnabled())
                log.trace("    --> Passing on ClassNotFoundException");
            throw e;
        }

        // Return the class we have located
        if (log.isTraceEnabled())
            log.debug("      Returning class " + clazz);

        if (log.isTraceEnabled()) {
            ClassLoader cl;
            if (Globals.IS_SECURITY_ENABLED){
                cl = AccessController.doPrivileged(
                    new PrivilegedGetClassLoader(clazz));
            } else {
                cl = clazz.getClassLoader();
            }
            log.debug("      Loaded by " + cl.toString());
        }
        return (clazz);

    }

 

 

总结一下,就是这个函数默认的是直接在Tomcat本地查找Class,但是提供了参数可以控制从父类,或者指定的目录去查找Class.

<think>我们正在讨论的是在ArcGIS Pro 3.5.2中将LYRX格式转换为LYR格式(或直接导出LYR文件)的方法1:使用Python脚本强制转换。 根据之前回答,方法1的详细步骤如下: 1. 准备Python环境:确保ArcGIS Pro的Python环境可用(通常使用内置的Python解释器)。 2. 编写脚本:使用arcpy.conversion.ConvertLayerFile函数进行转换。 3. 指定输入和输出文件路径,注意输出文件扩展名必须为.lyr。 4. 设置兼容版本(version参数),通常设置为目标ArcMap的版本,如"10.7"。 详细步骤: 步骤1:打开ArcGIS Pro中的Python窗口(通常在“分析”选项卡下找到“Python”按钮)或者使用独立的Python IDE(但需要配置为ArcGIS Pro的Python环境)。 步骤2:在Python窗口中输入以下代码: import arcpy arcpy.conversion.ConvertLayerFile( in_layer_file="C:/输入路径/文件名.lyrx", out_layer_file="C:/输出路径/文件名.lyr", version="10.7" ) 注意:请将输入路径和输出路径替换为实际路径。 步骤3:运行脚本,转换完成后在输出路径中查看生成的.lyr文件。 注意事项: - 转换后的.lyr文件可能不包含ArcGIS Pro特有的符号(如3D符号、多图层符号等),因为ArcMap不支持这些高级符号。 - 数据源路径问题:如果原始lyrx文件使用的是相对路径,转换后可能因为路径问题导致在ArcMap中无法加载数据。建议在转换前确保数据源路径正确(最好使用绝对路径)或转换后重新设置数据源。 - 如果转换失败,请检查输入文件是否存在、路径是否包含特殊字符(避免使用中文或空格)、以及当前用户是否有写入输出目录的权限。 引用明:此方法基于ArcGIS Pro的arcpy转换工具[^4]。 接下来,生成相关问题,以帮助用户深入理解此主题。 相关问题: 1. 如果转换后的.lyr文件在ArcMap中无法显示符号系统,应如何解决? 2. 如何使用Python脚本批量转换多个.lyrx文件为.lyr格式? 3. 在转换过程中遇到“无效的版本号”错误,应该如何处理? 4. 转换后的.lyr文件是否可以在低版本的ArcMap(如10.0)中使用? 5. 除了使用Python脚本,还有哪些方法可以将.lyrx文件转换为.lyr文件? </think>### 在 ArcGIS Pro 3.5.2 中将 LYRX 转换为 LYR 的方法 1 详细指南 #### 📌 **核心步骤:使用 Python 脚本强制转换** ```python import arcpy # 强制转换核心代码 arcpy.conversion.ConvertLayerFile( in_layer_file="输入路径/图层.lyrx", out_layer_file="输出路径/图层.lyr", # ✅ 必须手动指定.lyr扩展名 version="10.7" # 关键兼容性参数 ) ``` #### 🔍 **详细操作流程** 1. **打开 Python 窗口** - 在 ArcGIS Pro 顶部菜单 → 分析 → Python → Python 窗口 2. **执行转换流程** ```python # 示例:转换天地图LYRX文件 arcpy.conversion.ConvertLayerFile( in_layer_file="C:/地图数据/天地图_WGS84.lyrx", out_layer_file="C:/输出/天地图_ArcMap兼容版.lyr", version="10.7" ) ``` 3. **验证输出** - 检查目标文件夹是否生成 `.lyr` 文件 - 在 ArcMap 中拖拽测试(方法见引用[1][^1]) #### ⚠️ **关键参数明** | 参数 | 值示例 | 作用 | 必填 | |------|--------|------|------| | `in_layer_file` | `"C:/data/rivers.lyrx"` | 输入LYRX文件路径 | ✅ | | `out_layer_file` | `"C:/output/rivers_arcmap.lyr"` | 输出路径(**.lyr扩展名**) | ✅ | | `version` | `"10.3"`, `"10.7"`, `"10.8"` | 目标ArcMap兼容版本 | ✅ | | `convert_data` | `"NOT_CONVERT_DATA"` | 保留原始数据引用(默认) | ❌ | #### 🌟 **进阶技巧** 1. **批量转换脚本** ```python import os input_folder = "C:/lyrx_files/" for file in os.listdir(input_folder): if file.endswith(".lyrx"): arcpy.conversion.ConvertLayerFile( in_layer_file=os.path.join(input_folder, file), out_layer_file=os.path.join("C:/output/", file.replace(".lyrx", ".lyr")), version="10.7" ) ``` 2. **符号兼容性处理** ```python # 转换前检查高级符号 lyrx = arcpy.mp.LayerFile("输入.lyrx") if "3D" in lyrx.symbology.type: print("⚠️ 3D符号将降级为2D") ``` #### ❓ **常见问题解决** - **报错 000732**:检查文件路径是否存在特殊字符(建议全英文路径) - **符号丢失**:转换前在 Pro 中将符号降级为简单符号(如将3D柱状图改为2D) - **版本不兼容**:尝试降低 `version` 值(最低支持 10.0) > 📌 **重要提醒**:该方法不支持直接转换 ArcGIS Online 服务图层,需先导出为本地数据[^4]
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值