为Android组件化项目搭建Maven私服

news2024/11/20 6:16:56

概览

文章目录

      • 概览
      • 前言
      • 搭建 maven 私服
        • 服务器环境
        • jdk安装配置
        • nexus安装配置
        • 管理创建存储点、仓库
      • 项目中使用 maven 私服
        • 上传 module 到仓库
          • 自动发布 module
          • 手动上传单个aar包
        • 引用仓库中的 module
          • build.gradle引入远程module
      • FAQ
        • 开发阶段有些module用远程依赖,有些module用本地依赖,造成BuildConfig中DEBUG环境变量不一致?
        • 发布的组件中远程依赖引用不到了?
        • 引用的第三方aar无需根据编译环境分别打包正式/测试版,如何处理?
      • 参考链接

前言

本文根据网络上的文章以及个人实践,综合整理而成。
做这件事的目的,是为了缩短APP的编译时间。
记得接手公司项目的时候,项目已经使用了当时流行的组件化方案,当时项目刚起步,组件不多,比较轻量化,编译一次正式版用时大概五分钟左右。这几年随着项目和技术的发展,组件化方案中间重构更换了一次,项目中的组件也发展到了二十多个,现在编译一次正式版,不算加固时间,用时十七分钟左右。
开发调试阶段目前也饱受困扰。由于所有组件都依赖base_lib,只要base中改了一个字符,debug调试编译时所有组件就会重新编译一遍,及其浪费时间。

搭建 maven 私服

要正常安装使用 maven 私服,需要几个前置条件,如果你的服务器都满足的话,可以跳过这几个配置:

  • 根据nexus需求,安装合适的 java 环境
  • 安装合适的 nexus 版本
  • 打开服务器8081端口,用以支持在浏览器访问nexus管理页面
服务器环境
jdk安装配置

Download jdk 11这里我下载的是Linux x64 Compressed Archive-jdk-11.0.22这个版本
在这里插入图片描述

  1. 使用sftp上传到服务器/opt/jdk目录
    在这里插入图片描述

  2. 解压到当前目录

    cd /opt/jdk
    tar -zxvf jdk-11.0.22_linux-x64_bin.tar.gz
    

    在这里插入图片描述

  3. 使用sftp修改环境配置文件:/etc/profile

    #jdk config(在文档结尾插入)
    export JAVA_HOME=/opt/jdk/jdk-11.0.22
    export PATH=$PATH:$JAVA_HOME/bin
    export CLASSPATH=.:$JAVA_HIOME/jre/lib/rt.jar:$JAVA_HOME/lib/dt.jar:$JAVA_HOME/lib/tools.jar
    

    在这里插入图片描述

  4. 重启 /etc/profile 使之重新生效:

    source /etc/profile
    
  5. 验证 JDK 是否安装成功

    java --version
    

    在这里插入图片描述

nexus安装配置

Download Sonatype Nexus Repository根据安装的java版本选择对应的安装包,这里我下载的是/nexus-3.68.1-02-java11-unix这个版本,对应java11
在这里插入图片描述

  1. 使用sftp上传到服务器/opt/nexus目录

  2. 解压到当前目录

    cd /opt/nexus
    tar -zxvf nexus-3.68.1-02-java11-unix.tar.gz
    

    在这里插入图片描述

  3. 使用sftp编辑配置文件/opt/nexus/nexus-3.68.1-02/bin/nexus.rc

    run_as_user="root"
    

    在这里插入图片描述

  4. 查看并开放8081端口

    # 查看防火墙某个端口是否开放
    firewall-cmd --query-port=8081/tcp
    # 开放防火墙端口 8081
    firewall-cmd --add-port=8081/tcp --permanent
    # 重新加载防火墙规则
    firewall-cmd --reload
    # 再次查看防火墙某个端口是否开放
    firewall-cmd --query-port=8081/tcp
    

    在这里插入图片描述

  5. 进入bin目录启动nexus

    cd /opt/nexus/nexus-3.68.1-02/bin
    ./nexus start
    

    在这里插入图片描述

  6. 浏览器进入管理页面,http://[服务器ip]:8081
    在这里插入图片描述

  7. 登录账号,admin 的密码在/opt/nexus/sonatype-work/nexus3/admin.password文件中
    在这里插入图片描述

管理创建存储点、仓库
  1. 创建存储点
    默认存储点是default,你也可以直接用默认的,不创建新的
    登录账号,依次进入Setting->Repository->Blob Stores
    在这里插入图片描述

  2. 创建仓库
    在创建仓库前,先介绍一下仓库的三种存储类型Release、Snapshot和Mixed,创建仓库时会使用到
    Release(正式版): 正式版,适合项目稳定后正式发布使用,特性看下面的原理简介。
    Snapshot(快照版): 快照版,适合项目在开发调试中使用,特性看下面的原理简介。
    Mixed(混合版): 可包含release和snapshot版本

    ReleaseSnapshot的原理简介:
    它们的主要区别在于本地获取这些依赖的机制不同。

    • 对于Release版本的项目: 当项目中添加了一个正式版的version为x的依赖时,构建工具在构建项目时会从远程仓库中下载这个version为x的依赖到本地仓库缓存起来,下次再构建项目时,构建工具会从本地仓库中查找是否存在这个version为x的依赖,存在就不会在去远程仓库中拉取。这种特性会导致在团队开发中,如果你发布一个正式版的项目时,仍然使用的是同一个version,就可能出现其他使用这个version项目的成员根本接收不到项目的最新变更,这是糟糕的,为了生命安全,大家尽量不要这么做。所以,在每次发布正式版项目时,必须更新version。
    • 对于Snapshot版本的项目: 当项目中添加了一个快照版的version为x的依赖时,不管本地仓库中是否存在这个version为x的依赖,构建工具都会尝试从远程仓库中查看这个version为x的依赖是否最新。这样就能保证团队开发时,所有使用这个version为x的依赖的项目都能获取到依赖最新的变更,而且不用不停的迭代依赖的version。最后一还有一点,Snapshot版项目发布时的version一定要以-SNAPSHOT结尾,英文字母必须大写。

    依次进入Setting->Repository->Repositories
    在这里插入图片描述
    在这里插入图片描述

项目中使用 maven 私服

上传 module 到仓库
自动发布 module
  1. 在module的 build.gradle 同级创建publish.gradle

    apply plugin: "maven-publish"
    
    afterEvaluate {
        def ext = rootProject.ext
        publishing {
            publications {
                release(MavenPublication) {
    
                    groupId 'com.wln100.future'
                    artifactId 'fragmentation'
                    version '1.0.0-SNAPSHOT'
    
                    if (isAndroidEnv(project)) {
                        from project.components.release
                    } else {
                        from project.components.java
                    }
    
                    pom {
                        name = 'fragmentation'
                        description = 'fragmentation'
                        url = ext.nexusMavenUrl
                    }
                }
            }
    
            repositories {
                maven {
                    allowInsecureProtocol true
                    url = ext.nexusMavenUrl
                    credentials {
                        username ext.nexusMavenUsername
                        password ext.nexusMavenPassword
                    }
                }
            }
        }
    }
    
    static def isAndroidEnv(Project project) {
        return project.getPlugins().hasPlugin('com.android.application') || project.getPlugins().hasPlugin('com.android.library')
    }
    
  2. 在 module 的 build.gradle 引入 publish.gradle

    apply plugin: 'com.android.library'
    
    ...
    android {
        compileSdk compileVersion
    
        defaultConfig {
            ...
        }
    
        buildTypes {
            ...
        }
        ...
    
    }
    
    dependencies {
        ...
    }
    //引入自动发布gradle
    apply from: project.file('publish.gradle')
    
  3. 同步代码,运行GradleTask
    在这里插入图片描述

  4. 之后在仓库管理页面可以看到上传成功的module
    在这里插入图片描述

手动上传单个aar包
  1. 管理页面进入Upload->android-examonline-hosted
    在这里插入图片描述

  2. 上传aar,填写必要参数
    在这里插入图片描述
    在这里插入图片描述

引用仓库中的 module

首先需要在项目级的build.gradle中添加仓库地址:

buildscript {
    ...
    repositories {
        //本地私服仓
        maven {
            allowInsecureProtocol true
            url 'http://[服务器ip]:8081/repository/[仓库名称]/'
        }
        ...
    }
    ...
}
...
allprojects {
    repositories {
        //本地私服仓
        maven {
            allowInsecureProtocol true
            url 'http://[服务器ip]:8081/repository/[仓库名称]/'
        }
        ...
    }
}
...
build.gradle引入远程module
dependencies {
    //引入组件module  
    api 'com.wln100.future:fragmentation:1.0.0-SNAPSHOT'
    //引入单个aar
    api 'me.yokeyword.fragmentation:fragmentation_core:1.0'
}

FAQ

开发阶段有些module用远程依赖,有些module用本地依赖,造成BuildConfig中DEBUG环境变量不一致?

我的解决方式为,发布module时,分别发布debug和release两包到两个仓库中;引入依赖时,根据编译命令动态切换对应的私服仓库:

  1. 发布时分别发布debug和release
    //publish.gradle
    afterEvaluate {
        publishing {
            publications {
                release(MavenPublication) {
                    ...
                    if (isAndroidEnv(project)) {
                        //打包release module
                        from project.components.release
                    } else {
                        from project.components.java
                    }
                    ...
                }
    
                debug(MavenPublication) {
                    ...
                    if (isAndroidEnv(project)) {
                    //打包debug module
                        from project.components.debug
                    } else {
                        from project.components.java
                    }
                    ...
                }
            }
    
            repositories {
                maven {
                    allowInsecureProtocol true
                    //根据编译任务动态设置发布仓库
                    if (isReleaseBuild()) {
                        url = NEXUS_URL
                    } else {
                        url = NEXUS_DEBUG_URL
                    }
                    ...
                }
            }
        }
        
    // 定义一个函数来判断是否为release版本
    def isReleaseBuild() {
        def taskNames = gradle.startParameter.taskNames
        println "isReleaseBuild>>>>taskNames:$taskNames"
        return taskNames.any { it.toLowerCase().contains("release") }
    }
    
    
    
  2. 项目根目录build.gradle动态设置仓库
    //project build.gradle
    ...
    allprojects {
        apply from: "$rootDir/gradle/maven-utils.gradle"
        repositories {
            ...
            //本地私服仓-自有组件
            maven {
                allowInsecureProtocol true
                if (isReleaseBuild()) {
                    url NEXUS_URL
                } else {
                    url NEXUS_DEBUG_URL
                }
            }
            google()
            ...
            maven { url 'https://jitpack.io' }
        }
    }
    
发布的组件中远程依赖引用不到了?

发布时配置发布参数withxml

afterEvaluate {

    publishing {
        publications {
            release(MavenPublication) {
                ...
                pom {
                    ...
                    withXml {
                        def dependenciesNode = asNode().getAt('dependencies')[0] ?:
                                asNode().appendNode('dependencies')
                        configurations.api.getDependencies().each {
                            dep -> addDependency(project, dependenciesNode, dep, "compile")
                        }
                        configurations.implementation.getDependencies().each {
                            dep -> addDependency(project, dependenciesNode, dep, "runtime")
                        }
                    }
                }
            }

            debug(MavenPublication) {
                ...
                pom {
                    ...
                    withXml {
                        def dependenciesNode = asNode().getAt('dependencies')[0] ?:
                                asNode().appendNode('dependencies')

                        configurations.api.getDependencies().each {
                            dep -> addDependency(project, dependenciesNode, dep, "compile")
                        }
                        configurations.implementation.getDependencies().each {
                            dep -> addDependency(project, dependenciesNode, dep, "runtime")
                        }
                    }
                }
            }
        }
        ...
    }
}

private void addDependency(Project project, def dependenciesNode, Dependency dep, String scope) {
    if (dep.group == null || dep.version == null || dep.name == null || dep.name == "unspecified") {
        return
    }
    if (dep.version == 'unspecified') {
        return
    }

    final dependencyNode = dependenciesNode.appendNode('dependency')
    dependencyNode.appendNode('scope', scope)
    dependencyNode.appendNode('groupId', dep.group)
    dependencyNode.appendNode('artifactId', dep.name)
    dependencyNode.appendNode('version', dep.version)
    println("publish -> library <${dep.group}:${dep.name}:${dep.version}>")

    if (!dep.transitive) {
        // In case of non transitive dependency,
        // all its dependencies should be force excluded from them POM file
        final exclusionNode = dependencyNode.appendNode('exclusions').appendNode('exclusion')
        exclusionNode.appendNode('groupId', '*')
        exclusionNode.appendNode('artifactId', '*')
    } else if (!dep.properties.excludeRules.empty) {
        // For transitive with exclusions, all exclude rules should be added to the POM file
        final exclusions = dependencyNode.appendNode('exclusions')
        dep.properties.excludeRules.each { ExcludeRule rule ->
            final exclusionNode = exclusions.appendNode('exclusion')
            exclusionNode.appendNode('groupId', rule.group ?: '*')
            exclusionNode.appendNode('artifactId', rule.module ?: '*')
        }
    }
}
引用的第三方aar无需根据编译环境分别打包正式/测试版,如何处理?

新建一个专门存放三方aar库的私服仓库,单独上传依赖三方aar。

//project build.gradle
...
allprojects {
    apply from: "$rootDir/gradle/maven-utils.gradle"
    repositories {

        //本地私服仓-三方lib
        maven{
            allowInsecureProtocol true
            url 'http://x.x.x.x:8081/repository/android-third-lib-hosted/'
        }
        ...
        google()
        ...
    }
}

参考链接

Centos开放端口以及查看端口和防火墙配置命令
centos7下搭建nexus
CentOS 7 安装 JDK11(注意版本号要与自己的版本一致)
centos7安装Nexus(Maven私服)与配置使用教程
【github】Blankj/AndroidUtilCode-publish.gradle

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

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

相关文章

苹果警告部分 iPhone 用户称他们遭到雇佣间谍软件攻击

苹果警告部分 iPhone 用户称他们遭到“雇佣间谍软件攻击 苹果正在提醒用户注意针对 iPhone 的新一轮已识别雇佣间谍软件攻击。可能的受害者已经收到来自苹果的电子邮件&#xff0c;其中描述了该攻击如何“远程破坏 iPhone”。据路透社报道&#xff0c;印度和其他 91 个国家的受…

内容产品运营方案业务架构解析与实践探索

### 背景 在信息爆炸的时代背景下&#xff0c;内容产品运营成为各行各业竞争的重要环节。构建合理的内容产品运营方案业务架构&#xff0c;能够帮助企业更好地管理内容生产、推广和变现&#xff0c;提升品牌影响力和商业价值。 ### 业务架构设计 #### 1. 内容生产与管理 建立…

一套java开发的(未来工厂核心MES系统成品源码)技术架构:java+springboot 支撑多端管理,可商用

MES定义为“位于上层的计划管理系统与底层的工业控制之间的面向车间层的管理信息系统” 20世纪90年代初期&#xff0c;中国就开始对MES以及ERP的跟踪研究、告知或试点&#xff0c;而且曾经发言 “管控一体化”&#xff0c;“人、财、物、产、供、销”等颇具中国独具一格的CIMS、…

input组件 type为nickname pc端获取不到这个绑定的值?

一、input组件 type为nickname pc端获取不到这个绑定的值&#xff1f; 在pc端 使用input blur 事件获取选择昵称结果失败 代码如下&#xff1a; <input type"nickname" bindblur"binname" name"nickName" placeholder"请输入昵称"…

2024年政治经济学与社会科学国际会议(ICPESS 2024)

2024年政治经济学与社会科学国际会议 2024 International Conference on Political Economy and Social Sciences 会议简介 2024年政治经济学与社会科学国际会议是一个致力于探讨政治经济学与社会科学交叉领域前沿问题的国际盛会。本次会议汇聚了全球顶尖的专家学者、研究人员和…

传统产品经理AI产品经理

前言 随着科技的发展&#xff0c;技术的革新&#xff0c;AI技术当今已经渗入到各个行业里边&#xff0c;身处其中的产品经理也面临的新的挑战和机遇&#xff0c;下面是笔者整理分享的关于传统的产品经理如何顺应时代发展&#xff0c;成功转换成AI产品经理的相关内容&#xff0…

vcruntime140.dll干嘛的?丢失了vcruntime140.dll要咋办?

vcruntime140.dll干嘛的&#xff1f;vcruntime140.dll就是一个dll文件&#xff0c;它对于很多程序都是有用的&#xff0c;如果没有了它&#xff0c;那么你的有些程序是打不开的&#xff01;所以当你丢失的时候&#xff0c;你就要想办法去修复vcruntime140.dll文件&#xff0c;假…

那个会用AI绘画的设计师,刚刚加薪了!不会AI的打工人却只能被优化?

大家好&#xff0c;我是向阳。 随着ChatGPT引领人工智能的热潮&#xff0c;AI绘画也悄然崭露头角&#xff01; 插画师、建筑师、平面设计师等艺术创作领域的从业者&#xff0c;感受到了前所未有的压力 当一个设计师能有多难&#xff1f; 设计师需要在&#xff1a;主管/经理…

SQL面试问题集

目录 Q.左连接和右连接的区别 Q.union 和 union all的区别 1、取结果的交集 2、获取结果后的操作 Q.熟悉开窗函数吗&#xff1f;讲一下row_number和dense_rank的区别。 Q.hive行转列怎么操作的 Q.要求手写的题主要考了聚合函数和窗口函数&#xff0c;row_number()&#…

(2024,ViT,小波变换,图像标记器,稀疏张量)基于小波的 ViT 图像标记器

Wavelet-Based Image Tokenizer for Vision Transformers 公和众和号&#xff1a;EDPJ&#xff08;进 Q 交流群&#xff1a;922230617 或加 VX&#xff1a;CV_EDPJ 进 V 交流群&#xff09; 目录 0 摘要 1 引言 3 基于小波的图像压缩简介 4 图像标记器 4.1 像素空间标记嵌…

【Python机器学习】主成分分析(PCA)

主成分分析&#xff08;PCA&#xff09;是一种旋转数据集的方法&#xff0c;旋转后的数特征在统计上不相关。在做完这种旋转之后&#xff0c;通常是根据新特征对解释数据的重要性来选择它的一个子集。 举例&#xff1a; import mglearn.plots import matplotlib.pyplot as pl…

AWS S3存储桶中如何下载文件

AWS S3存储桶中如何下载文件 1.单个下载 AWS S3 控制台提供了下载单个文件的功能&#xff0c;但是不支持直接在控制台中进行批量下载文件。您可以通过以下步骤在 AWS S3 控制台上下载单个文件&#xff1a;   1.1登录 AWS 管理控制台。   1.2转到 S3 服务页面。   1.3单击…

seerfar丨OZON运营工具,OZON选品插件

随着全球电商市场的蓬勃发展&#xff0c;OZON作为俄罗斯及东欧地区的重要电商平台&#xff0c;吸引了众多中国商家的目光。然而&#xff0c;如何在OZON平台上脱颖而出&#xff0c;实现高效的商品运营&#xff0c;成为了众多商家亟待解决的问题。在这样的背景下&#xff0c;seer…

流批一体计算引擎-10-[Flink]中的常用算子和DataStream转换

pyflink 处理 kafka数据 1 DataStream API 示例代码 从非空集合中读取数据&#xff0c;并将结果写入本地文件系统。 from pyflink.common.serialization import Encoder from pyflink.common.typeinfo import Types from pyflink.datastream import StreamExecutionEnviron…

什么是阴道菌群CST分型,不同的分型代表哪些女性健康问题

谷禾健康 人体内的各个部位&#xff0c;如皮肤、口腔、肠道和阴道等&#xff0c;都是微生物的重要栖息地&#xff0c;这些微生物与人体健康紧密相关&#xff0c;并能反映人体的疾病状态。这些部位因受基因、环境和生活方式等影响&#xff0c;具有独特的菌群特征。 女性生殖系统…

Linux磁盘分区(fdisk)和卷管理详解(VG-LV-PV)

先看整体图&#xff0c;再讲解概念 一、磁盘分区 一个磁盘disk可以分多个区part&#xff0c;用fdisk命令&#xff0c;举例把/dev/vdb划分为/dev/vdb1和/dev/vdb2 二、创建虚拟卷 LVM是逻辑盘卷管理&#xff08;Logical Volume Manager&#xff09;的简称&#xff0c;他是磁盘…

Cortex-M7——NVIC

Cortex-M7——NVIC 小狼http://blog.csdn.net/xiaolangyangyang 一、NVIC架构 二、中断及异常编号 三、中断屏蔽寄存器&#xff08;__disable_irq和__enable_irq操作的是PRIMASK寄存器&#xff09; 四、中断分组寄存器&#xff08;SCB->AIRCR[10:8]&#xff09; 五、NVIC寄…

「动态规划」如何求最小路径和?

64. 最小路径和https://leetcode.cn/problems/minimum-path-sum/description/ 给定一个包含非负整数的m x n网格grid&#xff0c;请找出一条从左上角到右下角的路径&#xff0c;使得路径上的数字总和为最小。说明&#xff1a;每次只能向下或者向右移动一步。 输入&#xff1a;…

DNF手游攻略:主C职业推荐,云手机强力辅助!

在《地下城与勇士》手游中&#xff0c;你是否厌倦了重复刷图和无休止的手动操作&#xff1f;利用VMOS云手机&#xff0c;你可以一键解决这些烦恼&#xff0c;实现自动打怪、一机多开&#xff0c;让游戏变得更加轻松愉快。下面我们将介绍如何使用VMOS云手机&#xff0c;以及推荐…

servlet实现图片上传和下载

图片上传 前端 后端 protected void dopost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {Part filePart request.getPart("file"); // 获取上传的文件String fileName filePart.getSubmittedFileName(); …