Composer 镜像原理 (2) —— composer.json
相关文章
有使用PHP组件的朋友, 应该会注意到组件里头会有一个文件 composer.json, 它描述了组件的信息: 名称, 描述, 关键词, 作者, GitHub仓库地址...还有它所依赖的子组件, 是 Composer 工作的核心.
拿一个大家都知道的日志组件 monolog 的 composer.json 为例, 我说下一些比较重要的字段:
{
"name": "monolog/monolog",
"description": "Sends your logs to files, sockets, inboxes, databases and various web services",
"keywords": ["log", "logging", "psr-3"],
"homepage": "http://github.com/Seldaek/monolog",
"type": "library",
"license": "MIT",
"authors": [
{
"name": "Jordi Boggiano",
"email": "j.boggiano@seld.be",
"homepage": "http://seld.be"
}
],
"require": {
"php": "^7.0",
"psr/log": "^1.0.1"
},
"require-dev": {
"phpunit/phpunit": "^5.7",
"graylog2/gelf-php": "^1.4.2",
"sentry/sentry": "^0.13",
"ruflin/elastica": ">=0.90 <3.0",
"doctrine/couchdb": "~1.0@dev",
"aws/aws-sdk-php": "^2.4.9 || ^3.0",
"php-amqplib/php-amqplib": "~2.4",
"swiftmailer/swiftmailer": "^5.3|^6.0",
"php-console/php-console": "^3.1.3",
"jakub-onderka/php-parallel-lint": "^0.9",
"predis/predis": "^1.1",
"phpspec/prophecy": "^1.6.1"
},
"suggest": {
"graylog2/gelf-php": "Allow sending log messages to a GrayLog2 server",
"sentry/sentry": "Allow sending log messages to a Sentry server",
"doctrine/couchdb": "Allow sending log messages to a CouchDB server",
"ruflin/elastica": "Allow sending log messages to an Elastic Search server",
"php-amqplib/php-amqplib": "Allow sending log messages to an AMQP server using php-amqplib",
"ext-amqp": "Allow sending log messages to an AMQP server (1.0+ required)",
"ext-mongodb": "Allow sending log messages to a MongoDB server (via driver)",
"mongodb/mongodb": "Allow sending log messages to a MongoDB server (via library)",
"aws/aws-sdk-php": "Allow sending log messages to AWS services like DynamoDB",
"rollbar/rollbar": "Allow sending log messages to Rollbar",
"php-console/php-console": "Allow sending log messages to Google Chrome"
},
"autoload": {
"psr-4": {"Monolog\\": "src/Monolog"}
},
"autoload-dev": {
"psr-4": {"Monolog\\": "tests/Monolog"}
},
"provide": {
"psr/log-implementation": "1.0.0"
},
"extra": {
"branch-alias": {
"dev-master": "2.0.x-dev"
}
},
"scripts": {
"test": [
"parallel-lint . --exclude vendor",
"phpunit"
]
}
}
安装依赖的时候, 最重要的字段是 name, require 以及 require-dev.
name
该字段标识了组件的名称, 在 所有 的组件中, 它是唯一的;
require
该字段列举出该组件 所需的运行环境 以及 依赖的子组件的版本, 安装该组件时, 会检测运行环境, 并安装该组件的子组件, 以及这些子组件的所有子组件...直到子组件不再依赖任何组件为止;
require-dev
该字段不是必须的, 一般来说不安装里面的依赖, 也是可以用的, 通常都是用来跑单元测试. 依赖的安装同 require 字段.
其他字段对于理解镜像的原理没什么帮助, 有兴趣可以看下 这篇文章.
安装依赖的过程, 其实就是请求服务器, 要求拿到该组件的 composer.json 文件, 然后 JSON 解析, 得到 require 和 require-dev 字段的组件, 一直遍历下去, 根据文件描述的仓库地址 git clone 到本地.
看过我 上一篇文章 的朋友就知道, 文章末尾我们配置了 国内的composer镜像, 用来加速我们安装组件的过程, 它缓存了所有包的 composer.json, 并把仓库的每一个分支源码, 打包为 zip 压缩包, 并结合 cdn 加速.
镜像服务器提供了让我们得到 composer.json 的接口, 我们只需提交一个包名, 还有请求结果的哈希值(是不是很懵逼, 我怎么知道结果的哈希值), 镜像服务器会返回一个 JSON, 它包含了很多 composer.json (至少一个), 这些 composer.json 里面就有我们要找的组件的 composer.json (根据name字段), 也包括了其他包的, 为什么会带有其他包的呢, 我捣鼓了挺多次, 发现是当 require 字段存在时, 它就顺带返回了, 不过也不是绝对的, 可能考虑体积关系, 也不会返回太多.
说了这么多, 看下 psr/log 组件的请求结果吧, 比较长, 它包含了5个包的信息:
hackification/log
mobio/target
notadd/wechat
psr/log
wedeto/log
{
"packages":{
"hackification/log":{
"1.0.2":{
"name":"hackification/log",
"description":"Common interface for logging libraries",
"keywords":[
"log",
"psr",
"psr-3",
"hack",
"hacklang"
],
"homepage":"https://github.com/hackification/log",
"version":"1.0.2",
"version_normalized":"1.0.2.0",
"license":[
"MIT"
],
"authors":[
{
"name":"PHP-FIG",
"homepage":"http://www.php-fig.org/"
}
],
"source":{
"type":"git",
"url":"https://github.com/hackification/log.git",
"reference":"75b02e14bd8e5b8ded75de91d7eb9df6046f7fa7"
},
"dist":{
"type":"zip",
"url":"https://files.phpcomposer.com/files/hackification/log/75b02e14bd8e5b8ded75de91d7eb9df6046f7fa7.zip",
"reference":"75b02e14bd8e5b8ded75de91d7eb9df6046f7fa7",
"shasum":""
},
"type":"library",
"time":"2016-11-08T15:32:34+00:00",
"autoload":{
"psr-4":{
"Psr\Log\":"Psr/Log/"
}
},
"extra":{
"branch-alias":{
"dev-master":"1.0.x-dev"
}
},
"require":{
"hhvm":">=3.0.0"
},
"replace":{
"psr/log":"*"
},
"uid":1072563
},
"dev-master":{
"name":"hackification/log",
"description":"Common interface for logging libraries",
"keywords":[
"log",
"psr",
"psr-3",
"hack",
"hacklang"
],
"homepage":"https://github.com/hackification/log",
"version":"dev-master",
"version_normalized":"9999999-dev",
"license":[
"MIT"
],
"authors":[
{
"name":"PHP-FIG",
"homepage":"http://www.php-fig.org/"
}
],
"source":{
"type":"git",
"url":"https://github.com/hackification/log.git",
"reference":"75b02e14bd8e5b8ded75de91d7eb9df6046f7fa7"
},
"dist":{
"type":"zip",
"url":"https://files.phpcomposer.com/files/hackification/log/75b02e14bd8e5b8ded75de91d7eb9df6046f7fa7.zip",
"reference":"75b02e14bd8e5b8ded75de91d7eb9df6046f7fa7",
"shasum":""
},
"type":"library",
"time":"2016-11-08T15:32:34+00:00",
"autoload":{
"psr-4":{
"Psr\Log\":"Psr/Log/"
}
},
"extra":{
"branch-alias":{
"dev-master":"1.0.x-dev"
}
},
"require":{
"hhvm":">=3.0.0"
},
"replace":{
"psr/log":"*"
},
"uid":1072564
}
},
"mobio/target":{
"0.0.5":{
"name":"mobio/target",
"description":"PHP library for myTarget API",
"keywords":[
"php",
"myTarget"
],
"homepage":"",
"version":"0.0.5",
"version_normalized":"0.0.5.0",
"license":[
"MIT"
],
"authors":[
],
"source":{
"type":"git",
"url":"https://github.com/MobioInc/target.git",
"reference":"5baeaae1aa7d85c5b5fd4e33a06608a1de93d73b"
},
"dist":{
"type":"zip",
"url":"https://files.phpcomposer.com/files/MobioInc/target/5baeaae1aa7d85c5b5fd4e33a06608a1de93d73b.zip",
"reference":"5baeaae1aa7d85c5b5fd4e33a06608a1de93d73b",
"shasum":""
},
"type":"library",
"time":"2016-10-31T08:52:52+00:00",
"autoload":{
"psr-4":{
"Mobio\Target\":"src/"
}
},
"require":{
"php":">=5.5.0",
"psr/log":"~1.0",
"guzzlehttp/guzzle":"^6.1"
},
"require-dev":{
"phpunit/phpunit":"^5.5"
},
"provide":{
"psr/log":"1.0.0"
},
"uid":1060012
},
"dev-master":{
"name":"mobio/target",
"description":"PHP library for myTarget API",
"keywords":[
"php",
"myTarget"
],
"homepage":"",
"version":"dev-master",
"version_normalized":"9999999-dev",
"license":[
"MIT"
],
"authors":[
],
"source":{
"type":"git",
"url":"https://github.com/MobioInc/target.git",
"reference":"70aa382ca6d3ba3b5a834bbe85d3fc2cbfec965f"
},
"dist":{
"type":"zip",
"url":"https://files.phpcomposer.com/files/MobioInc/target/70aa382ca6d3ba3b5a834bbe85d3fc2cbfec965f.zip",
"reference":"70aa382ca6d3ba3b5a834bbe85d3fc2cbfec965f",
"shasum":""
},
"type":"library",
"time":"2017-02-13T18:53:10+00:00",
"autoload":{
"psr-4":{
"Mobio\Target\":"src/"
}
},
"require":{
"php":">=5.5.0",
"guzzlehttp/guzzle":"^6.1",
"psr/log":"~1.0"
},
"require-dev":{
"phpunit/phpunit":"^5.5"
},
"provide":{
"psr/log":"1.0.0"
},
"uid":700331
}
},
"notadd/wechat":{
"dev-master":{
"name":"notadd/wechat",
"description":"Notadd's Wechat Module.",
"keywords":[
"framework",
"cms",
"member",
"notadd"
],
"homepage":"https://notadd.com",
"version":"dev-master",
"version_normalized":"9999999-dev",
"license":[
"Apache-2.0"
],
"authors":[
{
"name":"Notadd",
"email":"notadd@ibenchu.com"
}
],
"source":{
"type":"git",
"url":"https://github.com/notadd/wechat.git",
"reference":"e3f684cd225f3fadf21953c0289cb8426baad0e5"
},
"dist":{
"type":"zip",
"url":"https://files.phpcomposer.com/files/notadd/wechat/e3f684cd225f3fadf21953c0289cb8426baad0e5.zip",
"reference":"e3f684cd225f3fadf21953c0289cb8426baad0e5",
"shasum":""
},
"type":"notadd-module",
"time":"2017-11-13T04:23:05+00:00",
"autoload":{
"psr-4":{
"Notadd\Wechat\":"src/"
}
},
"require":{
"php":">=7.0",
"overtrue/wechat":"~3.1"
},
"require-dev":{
"notadd/installers":"0.14.*",
"notadd/testing":"0.4.*",
"phpunit/phpunit":"~6.0"
},
"replace":{
"guzzlehttp/guzzle":"*",
"guzzlehttp/promises":"*",
"guzzlehttp/psr7":"*",
"monolog/monolog":"*",
"psr/container":"*",
"psr/http-message":"*",
"psr/log":"*",
"symfony/http-foundation":"*",
"symfony/polyfill-mbstring":"*",
"symfony/psr-http-message-bridge":"*"
},
"uid":1108963
}
},
"psr/log":{
"1.0.0":{
"name":"psr/log",
"description":"Common interface for logging libraries",
"keywords":[
"log",
"psr",
"psr-3"
],
"homepage":"",
"version":"1.0.0",
"version_normalized":"1.0.0.0",
"license":[
"MIT"
],
"authors":[
{
"name":"PHP-FIG",
"homepage":"http://www.php-fig.org/"
}
],
"source":{
"type":"git",
"url":"https://github.com/php-fig/log.git",
"reference":"fe0936ee26643249e916849d48e3a51d5f5e278b"
},
"dist":{
"type":"zip",
"url":"https://files.phpcomposer.com/files/php-fig/log/fe0936ee26643249e916849d48e3a51d5f5e278b.zip",
"reference":"fe0936ee26643249e916849d48e3a51d5f5e278b",
"shasum":""
},
"type":"library",
"time":"2012-12-21T11:40:51+00:00",
"autoload":{
"psr-0":{
"Psr\Log\":""
}
},
"uid":29358
},
"1.0.1":{
"name":"psr/log",
"description":"Common interface for logging libraries",
"keywords":[
"log",
"psr",
"psr-3"
],
"homepage":"https://github.com/php-fig/log",
"version":"1.0.1",
"version_normalized":"1.0.1.0",
"license":[
"MIT"
],
"authors":[
{
"name":"PHP-FIG",
"homepage":"http://www.php-fig.org/"
}
],
"source":{
"type":"git",
"url":"https://github.com/php-fig/log.git",
"reference":"5277094ed527a1c4477177d102fe4c53551953e0"
},
"dist":{
"type":"zip",
"url":"https://files.phpcomposer.com/files/php-fig/log/5277094ed527a1c4477177d102fe4c53551953e0.zip",
"reference":"5277094ed527a1c4477177d102fe4c53551953e0",
"shasum":""
},
"type":"library",
"time":"2016-09-19T16:02:08+00:00",
"autoload":{
"psr-4":{
"Psr\Log\":"Psr/Log/"
}
},
"extra":{
"branch-alias":{
"dev-master":"1.0.x-dev"
}
},
"require":{
"php":">=5.3.0"
},
"uid":1000789
},
"1.0.2":{
"name":"psr/log",
"description":"Common interface for logging libraries",
"keywords":[
"log",
"psr",
"psr-3"
],
"homepage":"https://github.com/php-fig/log",
"version":"1.0.2",
"version_normalized":"1.0.2.0",
"license":[
"MIT"
],
"authors":[
{
"name":"PHP-FIG",
"homepage":"http://www.php-fig.org/"
}
],
"source":{
"type":"git",
"url":"https://github.com/php-fig/log.git",
"reference":"4ebe3a8bf773a19edfe0a84b6585ba3d401b724d"
},
"dist":{
"type":"zip",
"url":"https://files.phpcomposer.com/files/php-fig/log/4ebe3a8bf773a19edfe0a84b6585ba3d401b724d.zip",
"reference":"4ebe3a8bf773a19edfe0a84b6585ba3d401b724d",
"shasum":""
},
"type":"library",
"time":"2016-10-10T12:19:37+00:00",
"autoload":{
"psr-4":{
"Psr\Log\":"Psr/Log/"
}
},
"extra":{
"branch-alias":{
"dev-master":"1.0.x-dev"
}
},
"require":{
"php":">=5.3.0"
},
"uid":1029935
},
"dev-master":{
"name":"psr/log",
"description":"Common interface for logging libraries",
"keywords":[
"log",
"psr",
"psr-3"
],
"homepage":"https://github.com/php-fig/log",
"version":"dev-master",
"version_normalized":"9999999-dev",
"license":[
"MIT"
],
"authors":[
{
"name":"PHP-FIG",
"homepage":"http://www.php-fig.org/"
}
],
"source":{
"type":"git",
"url":"https://github.com/php-fig/log.git",
"reference":"4ebe3a8bf773a19edfe0a84b6585ba3d401b724d"
},
"dist":{
"type":"zip",
"url":"https://files.phpcomposer.com/files/php-fig/log/4ebe3a8bf773a19edfe0a84b6585ba3d401b724d.zip",
"reference":"4ebe3a8bf773a19edfe0a84b6585ba3d401b724d",
"shasum":""
},
"type":"library",
"time":"2016-10-10T12:19:37+00:00",
"autoload":{
"psr-4":{
"Psr\Log\":"Psr/Log/"
}
},
"extra":{
"branch-alias":{
"dev-master":"1.0.x-dev"
}
},
"require":{
"php":">=5.3.0"
},
"uid":29285
}
},
"wedeto/log":{
"v0.9.1":{
"name":"wedeto/log",
"description":"Wedeto Platform - Logger",
"keywords":[
],
"homepage":"https://wedeto.net/",
"version":"v0.9.1",
"version_normalized":"0.9.1.0",
"license":[
"MIT"
],
"authors":[
{
"name":"Egbert van der Wal",
"email":"ewal@pointpro.nl"
}
],
"source":{
"type":"git",
"url":"https://github.com/Wedeto/Log.git",
"reference":"acc7f4aa66965dd2627a3886c9ac8b0d77b0a268"
},
"dist":{
"type":"zip",
"url":"https://files.phpcomposer.com/files/Wedeto/Log/acc7f4aa66965dd2627a3886c9ac8b0d77b0a268.zip",
"reference":"acc7f4aa66965dd2627a3886c9ac8b0d77b0a268",
"shasum":""
},
"type":"library",
"time":"2017-04-10T12:31:55+00:00",
"autoload":{
"psr-4":{
"Wedeto\Log\":"src/"
}
},
"require":{
"psr/log":"1.0.*",
"wedeto/util":"0.9.*",
"php":">=7.0.0"
},
"require-dev":{
"mikey179/vfsstream":"~1"
},
"provide":{
"psr/log":"1.0.*"
},
"uid":1336127
},
"v0.9.2":{
"name":"wedeto/log",
"description":"Wedeto Platform - Logger",
"keywords":[
],
"homepage":"https://wedeto.net/",
"version":"v0.9.2",
"version_normalized":"0.9.2.0",
"license":[
"MIT"
],
"authors":[
{
"name":"Egbert van der Wal",
"email":"ewal@pointpro.nl"
}
],
"source":{
"type":"git",
"url":"https://github.com/Wedeto/Log.git",
"reference":"55e6b03f0b446b7054078fd8a680ad1499d26264"
},
"dist":{
"type":"zip",
"url":"https://files.phpcomposer.com/files/Wedeto/Log/55e6b03f0b446b7054078fd8a680ad1499d26264.zip",
"reference":"55e6b03f0b446b7054078fd8a680ad1499d26264",
"shasum":""
},
"type":"library",
"time":"2017-04-10T18:15:36+00:00",
"autoload":{
"psr-4":{
"Wedeto\Log\":"src/"
}
},
"require":{
"psr/log":"1.0.*",
"wedeto/util":"0.9.*",
"php":">=7.0.0"
},
"require-dev":{
"mikey179/vfsstream":"~1"
},
"provide":{
"psr/log":"1.0.*"
},
"uid":1336697
}
}
}
}
可以看到, packages 字段里面有5个包, 里面的 psr/log 字段就是我们要找的, 而里面有各个分支的 composer.json, 以分支 1.0.0 为例, 里面有两个很关键的字段, source 和 dist:
{
"source":{
"type":"git",
"url":"https://github.com/php-fig/log.git",
"reference":"fe0936ee26643249e916849d48e3a51d5f5e278b"
},
"dist":{
"type":"zip",
"url":"https://files.phpcomposer.com/files/php-fig/log/fe0936ee26643249e916849d48e3a51d5f5e278b.zip",
"reference":"fe0936ee26643249e916849d48e3a51d5f5e278b",
"shasum":""
}
}
dist
该字段其实就是加速的 zip 压缩包, 无需 git clone, 只需把 zip 下载到本地, 解压完, 分支 1.0.0 就装好了.
source
这个字段的作用, 就是万一 dist 字段的 zip 下载不了, 不会马上中断整个安装流程, 而是接着 git clone. 也就是说, dist 字段失败, 或者压根就没有 dist 字段, 就走 source 字段.
看到这里, 对 Composer 的了解应该多了很多吧? 还记得 请求结果的哈希值 吗? 这个哈希哪里来的, 为什么我可以提前知道这个请求的 JSON 的哈希值? 还有, 接口在哪里? 镜像服务器的官方网站, 并没有提供啊...
下一篇文章再告诉你.
相关文章
文章来源于本人博客,发布于 2017-12-05,原文链接:https://imlht.com/archives/81/
Composer 镜像原理 (2) —— composer.json的更多相关文章
- Packagist 镜像使用方法--composer
镜像用法 有两种方式启用本镜像服务: 系统全局配置: 即将配置信息添加到 Composer 的全局配置文件 config.json 中.见“方法一” 单个项目配置: 将配置信息添加到某个项目的 com ...
- composer镜像安装laravel
博主最近在学习Laravel的框架的相关知识,对于Laravel的许多新特性,大家最好还是去查看官网文档最好,Laravel的文档非常完善,中文英文的都有,可以很好的解决你的困惑. 但是我们会发现学习 ...
- 配置和查看composer镜像
composer 默认地址改为中国镜像地址,以及中国镜像地址还原成默认地址 一.查看当前镜像地址 在命令行输入如下命令,即可查看全局镜像地址: $ composer config -g repo.pa ...
- 启用composer镜像服务
使用composer下载东西,需要FQ时,可使用其镜像服务 安装composer后,命令行执行全局配置 composer config -g repo.packagist composer https ...
- 如何把Composer镜像迁移到Laravel China 维护的镜像?
今天在更新Laravel-admin:1.6.0提示没有对应的包,后面才发现需要使用官方或者 Laravel-China 的 composer 镜像,phpcomposer 镜像已经停止维护了.怎么从 ...
- laravel中composer镜像服务的方式
系统全局配置:即将配置信息添加到Composer的全局配置文件config.json中. 单个项目配置:将配置信息添加到某个项目的composer.json文件中. 例1:修改Composer的全局配 ...
- 设置composer镜像地址为阿里云的方法
所有项目都会使用该镜像地址: composer config -g repo.packagist composer https://mirrors.aliyun.com/composer/ 取消配置: ...
- 阿里云composer 镜像
2019年12月2日13:54:32 https://developer.aliyun.com/composer 阿里云的镜像更新时间比较及时 本镜像与 Packagist 官方实时同步,推荐使用最新 ...
- 【composer】 PHP composer 镜像地址更换
如果你使用的是 laravel-china.org 得 composer 镜像.那么近期执行更新时候就会报错: 莫慌,这是因为 laravel-china.org 已经停止了对composer得更新. ...
- 更换composer镜像源为阿里云
说一说我为什么会更换镜像源,今天我准备给公司的项目添加一个 Excel 导出的功能,需要 PhpSpreadsheet 插件来实现我的功能.输入命令发现提示我 Authentication req ...
随机推荐
- ArcGIS Pro发布地图服务(影像、矢量)
做GIS一般都是用ArcMap发布影像或者矢量服务,由于ArcGIS后续不在更新ArcMap,改用ArcGIS Pro,本文对ArcGIS Pro发布服务进行说明. 本文示例使用(因为portal的授 ...
- 利用Velero对K8S备份还原与集群迁移实战
一.简介 Velero 是一款云原生时代的灾难恢复和迁移工具,采用 Go 语言编写,并在 github 上进行了开源,利用 velero 用户可以安全的备份.恢复和迁移 Kubernetes 集群资源 ...
- Node工程的依赖包管理方式
作者:京东零售 陈震 在前端工程化中,JavaScript 依赖包管理是非常重要的一环.依赖包通常是项目所依赖的第三方库.工具和框架等资源,它们能够帮助我们减少重复开发.提高效率并且确保项目可以正确的 ...
- es6 新增 set 和 map数据结构以及应用场景
https://www.cnblogs.com/sexintercourse/p/12269580.html Set数据容器,这是一个能够存储无重复值的有序列表. 1. 创建集合 set let se ...
- 从不均匀性角度浅析AB实验
作者:京东零售 路卫强 本篇的目的是从三个不均匀性的角度,对AB实验进行一个认知的普及,最终着重讲述AB实验的一个普遍的问题,即实验准确度问题. 一.AB实验场景 在首页中,我们是用红色基调还是绿色基 ...
- 聊一聊 dotnet-trace 调查 lock锁竞争
一:背景 1. 讲故事 最近在分析一个 linux 上的 dump,最后的诱因是大量的lock锁诱发的高频上下文切换,虽然问题告一段落,但我还想知道一点信息,所谓的高频到底有多高频?锁竞争到底是一个怎 ...
- #Powerquery 数据结构基础 一维数据与二维数据
本文参考了采悟老师的文章,推荐大家看原文,本文为笔记随笔 https://mp.weixin.qq.com/s?__biz=MzA4MzQwMjY4MA==&mid=2484068871&am ...
- 2022-07-14:以下go语言代码输出什么?A:1;B:3;C:4;D:编译错误。 package main import ( “fmt“ ) func main() { a
2022-07-14:以下go语言代码输出什么?A:1:B:3:C:4:D:编译错误. package main import ( "fmt" ) func main() { a ...
- docker安装es和kibana,单机模式
操作系统:mac系统 1.安装es docker pull elasticsearch:7.14.0 docker run --name es -p 9200:9200 -p 9300:9300 -e ...
- CogSci 2017-Learning to reinforcement learn
Key 元学习系统(监督+从属)扩展于RL设置 LSTM用强化学习算法进行训练,可以使agent获得一定的学习适应能力 解决的主要问题 DRL受限于特定的领域 DRL训练需要大量的数据 作者参考了Ho ...