百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术资源 > 正文

OpenResty 介绍与实战讲解(nginx&lua)

off999 2025-02-16 22:27 79 浏览 0 评论

一、概述

OpenResty是一个基于 NginxLua 的高性能 Web 平台,其内部集成了大量精良的 Lua 库、第三方模块以及大多数的依赖项。用于方便地搭建能够处理超高并发、扩展性极高的动态 Web 应用、Web 服务和动态网关。

简单地说 OpenResty 的目标是让你的Web服务直接跑在 Nginx 服务内部,充分利用 Nginx 的非阻塞 I/O 模型,不仅仅对 HTTP 客户端请求,甚至于对远程后端诸如 MySQL、PostgreSQL、Memcached 以及 Redis 等都进行一致的高性能响应。

从上面官网的描述信息中,可以看出OpenResty主要包含两方面的技术:

  • Nginx:一款轻量级、高性能、高并发的Web服务器。
  • Lua:一种轻量、小巧、可移植、快速的脚本语言;LuaJIT即时编译器会将频繁执行的Lua代码编译成本地机器码交给CPU直接执行,执行效率更高,OpenResty 会默认启用 LuaJIT

官方网站:http://openresty.org/
Github组织:https://github.com/openresty

nginx 与 lua 介绍与安装可以参考我以下几篇文章:

二、OpenResty 安装

yum install pcre-devel openssl-devel gcc curl
wget https://openresty.org/package/centos/openresty.repo
sudo mv openresty.repo /etc/yum.repos.d/openresty.repo

# update the yum index:
sudo yum check-update
yum -y install openresty

验证安装

/usr/local/openresty/nginx/sbin/nginx -v

启动 OpenResty

/usr/local/openresty/nginx/sbin/nginx

测试

# 创建一个 NGINX 配置文件(例如 /usr/local/openresty/nginx/conf/nginx.conf)并添加一个包含 Lua 代码的 location 块:
server {
    listen 80;
    server_name localhost;

    location /test {
        content_by_lua_block {
            ngx.say("Hello, LuaJIT!")
        }
    }
}

保存配置文件,然后重新加载 NGINX:

# 先检查语法
sudo /usr/local/openresty/nginx/sbin/nginx -t
# 重新加载配置
sudo /usr/local/openresty/nginx/sbin/nginx -s reload

访问 http://localhost/test 应该返回 "Hello, LuaJIT!"

三、OpenResty 的工作原理

OpenResty 是基于 Nginx 的高性能Web平台,所以其高效运行与 Nginx 密不可分。
Nginx 处理HTTP请求有11个执行阶段,我们可以从ngx_http_core_module.h 的源码中看到:

typedef enum {
    NGX_HTTP_POST_READ_PHASE = 0,

    NGX_HTTP_SERVER_REWRITE_PHASE,

    NGX_HTTP_FIND_CONFIG_PHASE,
    NGX_HTTP_REWRITE_PHASE,
    NGX_HTTP_POST_REWRITE_PHASE,

    NGX_HTTP_PREACCESS_PHASE,

    NGX_HTTP_ACCESS_PHASE,
    NGX_HTTP_POST_ACCESS_PHASE,

    NGX_HTTP_PRECONTENT_PHASE,

    NGX_HTTP_CONTENT_PHASE,

    NGX_HTTP_LOG_PHASE
} ngx_http_phases;

巧合的是,OpenResty 也有 11*_by_lua 指令,它们和 NGINX11个执行阶段有很大的关联性。指令是使用Lua编写Nginx脚本的基本构建块,用于指定用户编写的Lua代码何时运行以及运行结果如何使用等。下图显示了不同指令的执行顺序,这张图可以帮助理清我们编写的脚本是按照怎样的逻辑运行的。



四、OpenResty 核心模块

OpenResty 是一个基于 NGINX 的全功能 Web 平台,它集成了许多模块和库,为 NGINX 增加了额外的功能和能力。以下是 OpenResty 的一些核心模块:

1)ngx_lua 模块

ngx_lua 模块是 OpenResty 的核心模块之一,提供了对 Lua 脚本的支持。它允许开发者在 NGINX 配置中嵌入Lua代码,实现高级的请求处理逻辑、动态内容生成、访问控制等功能。

ngx_lua 模块示例:

server {
    listen 80;
    server_name example.com;

    location /lua_example {
        default_type 'text/plain';
        content_by_lua_block {
            ngx.say("Hello, ngx_lua!")
        }
    }
}

在这个例子中,当访问 http://example.com/lua_example 时,将返回 "Hello, ngx_lua!"。这里使用了 ngx_lua 模块的 content_by_lua_block 指令,将 Lua 代码嵌入 NGINX 配置文件。

2)ngx_stream_lua 模块

ngx_stream_lua 模块 与 ngx_lua 类似,但专门用于处理 TCPUDP 流量。它允许开发者在 NGINX 配置中嵌入 Lua 代码以处理流量。

以下是一个简单的 ngx_stream_lua 模块的示例:

stream {
    server {
        listen 12345;

        content_by_lua_block {
            local data, err = ngx.req.socket()
            if not data then
                ngx.log(ngx.ERR, "Failed to read request: ", err)
                return
            end

            ngx.say("Received data: ", data)
        }
    }
}

在这个示例中:

  • 使用 content_by_lua_block 指令定义了一个 Lua 代码块,用于处理 TCP 流量。
  • 通过 ngx.req.socket() 获取连接的套接字,然后读取请求数据。
  • 输出接收到的数据。

此配置监听在端口 12345 上,当有 TCP 连接到达时,Lua 代码将读取并输出接收到的数据。

3)ngx_http_lua_module 模块

ngx_http_lua_module 模块是 ngx_lua 模块的一部分,为 NGINX 提供了强大的 HTTP 服务和 Lua 扩展。

以下是一个简单的 ngx_http_lua_module 模块的示例:

server {
    listen 80;
    server_name example.com;

    location /lua_example {
        default_type 'text/plain';
        content_by_lua_block {
            ngx.say("Hello, ngx_http_lua!")
        }
    }

    location /lua_variable {
        default_type 'text/plain';
        set $my_variable "NGINX with Lua";
        content_by_lua_block {
            local my_variable = ngx.var.my_variable
            ngx.say("Value of my_variable: ", my_variable)
        }
    }
}

这个示例中:

  • /lua_example 路径下的请求将返回 "Hello, ngx_http_lua!"。这里使用了 content_by_lua_block 指令,将 Lua 代码嵌入 NGINX 配置文件,实现了简单的响应内容输出。
  • /lua_variable 路径下的请求将输出一个自定义变量的值。使用了 set 指令设置了一个名为 my_variable 的变量,然后在 Lua 代码块中通过 ngx.var 获取并输出了这个变量的值。

4)ngx_http_headers_more 模块

ngx_http_headers_more 模块提供了更多的操作 HTTP 头部的指令,使得在 NGINX 中更容易操作和修改 HTTP 头信息。

以下是一个简单的使用 ngx_http_headers_more 模块的示例:

server {
    listen 80;
    server_name example.com;

    location /add_custom_header {
        more_set_headers "Custom-Header: OpenResty";
        return 200 "Custom header added!";
    }

    location /remove_server_header {
        more_clear_headers Server;
        return 200 "Server header removed!";
    }
}

在这个示例中:

  • /add_custom_header 路径下的请求将返回 "Custom header added!",同时响应头中包含了一个自定义的头部 "Custom-Header: OpenResty"。这是通过 more_set_headers 指令添加的。
  • /remove_server_header 路径下的请求将返回 "Server header removed!",同时响应头中不再包含 "Server" 头部。这是通过 more_clear_headers 指令移除的默认的 "Server" 头部。

这些指令允许你更灵活地配置 NGINX 的响应头信息,添加或删除特定的头部字段。请注意,使用这些指令时应谨慎,确保符合安全性和隐私性的最佳实践。

5)ngx_http_echo 模块

ngx_http_echo 模块 提供了更丰富的内容输出和变量替换的功能。可以通过指定响应内容、设置 HTTP 状态码等,以及支持类似 PHP 的变量替换。

server {
    listen 80;
    server_name example.com;

    location /echo_example {
        echo "Hello, ngx_http_echo!";
    }
}

在这个示例中:

  • /echo_example 路径下的请求将返回 "Hello, ngx_http_echo!"。这是通过 ngx_http_echo 模块的 echo 指令实现的。
  • ngx_http_echo 模块的 echo 指令允许你更方便地输出内容,支持类似 PHP 的变量替换等功能。你可以在 echo 指令中直接使用变量,也可以使用其他模块提供的一些特殊的变量。

6)ngx_http_lua_upstream 模块

ngx_http_lua_upstream 模块用于在 Lua 脚本中进行向上游服务器(后端服务器)的请求,并处理来自上游的响应。

以下是一个简单的示例,演示了如何使用 ngx_http_lua_upstream 模块向上游服务器发起请求,并处理来自上游服务器的响应:

http {
    upstream backend {
        server backend1;
        server backend2;
    }

    server {
        listen 80;
        server_name example.com;

        location /proxy_example {
            content_by_lua_block {
                -- 初始化 upstream 对象
                local upstream = require "ngx.upstream"
                local backend = upstream.backend

                -- 创建一个新的请求对象
                local request = upstream.request()

                -- 设置请求的方法、URI和头部
                request.method = ngx.HTTP_GET
                request.uri = "/path/to/resource"
                request.headers["Host"] = "backend.example.com"

                -- 发送请求到上游服务器
                local status, headers, body = request.send(backend)

                -- 处理上游服务器的响应
                if status == 200 then
                    ngx.say("Response from backend: ", body)
                else
                    ngx.say("Error from backend. Status: ", status)
                end
            }
        }
    }
}

在这个示例中:

  • 定义了一个名为 backend 的上游服务器块,包含两个后端服务器 backend1backend2
  • /proxy_example 路径下的请求中,通过 Lua 脚本使用 ngx_http_lua_upstream 模块创建了一个新的上游请求对象。
  • 设置了请求的方法、URI和头部信息。
  • 调用 request.send(backend) 发送请求到上游服务器。
  • 根据上游服务器的响应状态码,输出响应内容或显示错误信息。

请注意,实际使用时,你可能需要根据具体的业务需求和上游服务器的特性进行更复杂的 Lua 脚本编写。此示例仅为演示基本用法。

7)ngx_http_redis 模块

ngx_http_redis 模块提供了与 Redis 数据库的交互功能,允许 NGINX 通过 Lua 脚本与 Redis 通信。

以下是一个简单的示例,演示了如何使用 ngx_http_redis 模块与 Redis 交互:

http {
    server {
        listen 80;
        server_name example.com;

        location /redis_example {
            # 定义 Redis 服务器的地址和端口
            set $redis_host "127.0.0.1";
            set $redis_port 6379;

            # 使用 ngx_http_redis 模块向 Redis 发送 GET 请求
            redis_pass $redis_host:$redis_port;
            redis_query GET my_key;

            # 处理 Redis 的响应
            content_by_lua_block {
                local redis = require "ngx.redis"
                local red = redis:new()

                red:set_timeout(1000)  -- 设置超时时间

                local ok, err = red:connect(ngx.var.redis_host, ngx.var.redis_port)
                if not ok then
                    ngx.say("Failed to connect to Redis: ", err)
                    return
                end

                local res, err = red:get("my_key")
                if not res then
                    ngx.say("Failed to get value from Redis: ", err)
                    return
                end

                ngx.say("Value from Redis: ", res)

                -- 释放连接
                local ok, err = red:set_keepalive(10000, 100)
                if not ok then
                    ngx.say("Failed to set keepalive: ", err)
                    return
                end
            }
        }
    }
}

在这个示例中:

  • /redis_example 路径下的请求首先通过 redis_passredis_query 指令设置了 Redis 服务器的地址和端口,并发送了一个 GET 请求获取键为 "my_key" 的值。
  • Lua 脚本中,使用 ngx.redis 模块创建了一个 Redis 对象,连接到 Redis 服务器,并通过 get 方法获取了 "my_key" 的值。
  • 最后,输出从 Redis 获取的值。

请注意,实际使用时,你需要根据你的 Redis 服务器配置和业务需求进行适当的修改。此示例仅为演示基本用法。

8)ngx_http_proxy_connect_module 模块

ngx_http_proxy_connect_module 模块允许 NGINX 充当 CONNECT 代理,用于处理 TLS/SSL 连接的代理请求。

以下是一个简单的使用示例:

http {
    server {
        listen 80;
        server_name example.com;

        location /proxy_connect_example {
            proxy_pass http://backend;
        }
    }

    server {
        listen 443 ssl;
        server_name example.com;

        ssl_certificate /path/to/certificate.crt;
        ssl_certificate_key /path/to/private_key.key;

        location /proxy_connect_example {
            proxy_pass http://backend;
        }
    }
}

在这个示例中:

  • 配置了两个 server 块,分别监听了 80 端口和 443 端口。第一个 server 块用于处理 HTTP 请求,第二个 server 块用于处理 HTTPS 请求。
  • 对于 /proxy_connect_example 路径,使用了 proxy_pass 指令,将请求代理到名为 backend 的上游服务器。
  • 对于 HTTPS 的情况,需要提供 SSL 证书和私钥的路径。实际上,这个配置使 NGINX 具有 CONNECT 代理的能力,可以处理 TLS/SSL 握手并将请求代理到上游服务器。

9)ngx_http_js_module 模块

ngx_http_js_module 模块提供了对 JavaScript 的支持,使得可以在 NGINX 中使用 JavaScript 来编写请求处理逻辑。

以下是一个简单的使用 ngx_http_js_module 模块的示例:

http {
    server {
        listen 80;
        server_name example.com;

        location /js_example {
            js_content main;
        }
    }

    js_include main;

    function main(r) {
        r.return(200, "Hello, ngx_http_js_module!");
    }
}

在这个示例中:

  • 定义了一个监听 80 端口的 server 块。
  • /js_example 路径下的请求中,使用了 js_content 指令,将请求处理的逻辑指定为 JavaScript 脚本。
  • 使用 js_include 指令引入了一个名为 mainJavaScript 函数。
  • main 函数中使用了 r.return 方法返回了一个 HTTP 200 响应,并带有相应的消息。

请注意,使用 ngx_http_js_module 模块时,需要确保已经正确安装并启用了该模块。此示例只是一个基本的演示,实际使用时可能需要根据具体的业务需求编写更复杂的 JavaScript 脚本。

10)ngx_http_geoip2_module 模块

ngx_http_geoip2_module 模块用于通过 MaxMind GeoIP2 数据库来获取客户端的地理位置信息。

以下是一个简单的示例,演示了如何使用 ngx_http_geoip2_module 模块获取客户端的地理位置信息:

http {
    geoip2 /path/to/GeoIP2-City.mmdb {
        $geoip2_city_country_iso_code country iso_code;
        $geoip2_city_country country names en;
        $geoip2_city_city city names en;
        $geoip2_city_latitude latitude;
        $geoip2_city_longitude longitude;
    }

    server {
        listen 80;
        server_name example.com;

        location /geoip_example {
            default_type 'text/plain';
            content_by_lua_block {
                local country = ngx.var.geoip2_city_country
                local city = ngx.var.geoip2_city_city
                local latitude = ngx.var.geoip2_city_latitude
                local longitude = ngx.var.geoip2_city_longitude

                ngx.say("Country: ", country)
                ngx.say("City: ", city)
                ngx.say("Latitude: ", latitude)
                ngx.say("Longitude: ", longitude)
            }
        }
    }
}

ngx_http_geoip2_module 模块用于在 NGINX 中获取客户端的地理位置信息,基于 MaxMind 的 GeoIP2 数据库。以下是一个简单的示例,演示了如何使用 ngx_http_geoip2_module 模块获取客户端的地理位置信息:

http {
    geoip2 /path/to/GeoIP2-City.mmdb {
        $geoip2_city_country_iso_code country iso_code;
        $geoip2_city_country country names en;
        $geoip2_city_city city names en;
        $geoip2_city_latitude latitude;
        $geoip2_city_longitude longitude;
    }

    server {
        listen 80;
        server_name example.com;

        location /geoip_example {
            default_type 'text/plain';
            content_by_lua_block {
                local country = ngx.var.geoip2_city_country
                local city = ngx.var.geoip2_city_city
                local latitude = ngx.var.geoip2_city_latitude
                local longitude = ngx.var.geoip2_city_longitude

                ngx.say("Country: ", country)
                ngx.say("City: ", city)
                ngx.say("Latitude: ", latitude)
                ngx.say("Longitude: ", longitude)
            }
        }
    }
}

在这个示例中:

  • 使用 geoip2 指令配置了 GeoIP2 数据库的路径,并定义了一些变量用于存储地理位置信息,如国家 ISO 代码、国家名称、城市名称、纬度和经度。
  • /geoip_example 路径下的请求中,通过 Lua 脚本获取了客户端的地理位置信息,并输出了国家、城市、纬度和经度等信息。

请注意,在实际使用中,你需要确保已经获取并配置了正确的 GeoIP2 数据库文件路径。此示例只是一个基本的演示,实际场景中可能需要根据业务需求进一步处理和使用这些地理位置信息。

11)ngx_brotli 模块

ngx_brotli 模块用于支持 Brotli 压缩算法,提供更高效的内容压缩。

以下是一个简单的使用 ngx_brotli 模块的示例:

http {
    brotli on;
    brotli_comp_level 6;
    brotli_static on;

    server {
        listen 80;
        server_name example.com;

        location /brotli_example {
            default_type 'text/plain';
            content_by_lua_block {
                ngx.say("Hello, ngx_brotli!");
            }
        }
    }
}

在这个示例中:

  • 使用 brotli 指令开启了 Brotli 压缩功能。
  • 使用 brotli_comp_level 指令设置了 Brotli 压缩级别。
  • 使用 brotli_static 指令开启了静态文件的 Brotli 压缩。
  • /brotli_example 路径下的请求中,返回了一个简单的文本内容 "Hello, ngx_brotli!"

请注意,在实际使用中,你需要确保已经安装了支持 Brotli 压缩的库,并且 NGINX 配置中启用了对应的模块。此示例只是一个基本的演示,实际配置中可能需要根据业务需求调整压缩级别等参数。

这些模块共同构成了 OpenResty 的核心,使得 NGINX 变得更加强大和灵活。使用这些模块,你可以实现更高级的请求处理、负载均衡、反向代理、动态内容生成等功能。详细信息和用法可以参考 OpenResty 官方文档:https://openresty.org/。

五、OpenResty 示例讲解

OpenResty 是一个基于 NGINX 的全功能 Web 平台,集成了大量的第三方模块和库,其中最重要的是 ngx_lua 模块,它允许在 NGINX 配置中嵌入 Lua 脚本,实现高级的请求处理逻辑、动态内容生成、访问控制等功能。下面是一个简单的 OpenResty 示例:

http {
    server {
        listen 80;
        server_name example.com;

        location /hello {
            default_type 'text/plain';
            content_by_lua_block {
                ngx.say("Hello, OpenResty!")
            }
        }
    }
}

在这个示例中:

  • 配置了一个监听 80 端口的 server 块,处理 example.com 的请求。
  • 当访问路径 /hello 时,通过 content_by_lua_block 指令执行 Lua 代码,输出 "Hello, OpenResty!"

这只是一个最简单的演示,OpenResty 的强大之处在于它允许在 NGINX 配置中使用 Lua 脚本,从而实现更复杂的逻辑。以下是一个稍复杂的示例,演示了如何通过 OpenResty 实现简单的 API 访问控制:

http {
    lua_shared_dict my_limit 10m;

    server {
        listen 80;
        server_name api.example.com;

        location /api {
            access_by_lua_block {
                local limit = ngx.shared.my_limit
                local key = ngx.var.remote_addr
                local reqs, err = limit:get(key)
                if reqs then
                    if reqs > 10 then
                        ngx.exit(ngx.HTTP_TOO_MANY_REQUESTS)
                    else
                        limit:incr(key, 1)
                    end
                else
                    limit:set(key, 1, 60)
                end
            }

            default_type 'application/json';
            content_by_lua_block {
                ngx.say('{"message": "API response"}')
            }
        }
    }
}

在这个示例中:

  • 配置了一个共享内存字典 my_limit 用于存储请求计数。
  • 当访问路径 /api 时,通过 access_by_lua_block 指令执行 Lua 代码,实现了一个简单的请求频率限制,每个 IP 地址在 60 秒内最多允许 10 次请求。
  • 如果超过请求限制,将返回 HTTP 429 (TOO MANY REQUESTS) 状态码;否则,继续执行后续 Lua 代码返回 JSON 响应。

这只是 OpenResty 的一小部分功能展示,实际使用中可以结合更多的模块和功能,如 ngx_http_lua_upstream、ngx_http_headers_more、ngx_stream_lua 等,以实现更复杂的 Web 应用和服务。


OpenResty 介绍与实战讲解就先到这里了,有任何疑问也可关注我公众号:大数据与云原生技术分享,进行技术交流,如本篇文章对您有所帮助,麻烦帮忙一键三连(点赞、转发、收藏)~

相关推荐

安全教育登录入口平台(安全教育登录入口平台官网)

122交通安全教育怎么登录:122交通网的注册方法是首先登录网址http://www.122.cn/,接着打开网页后,点击右上角的“个人登录”;其次进入邮箱注册,然后进入到注册页面,输入相关信息即可完...

大鱼吃小鱼经典版(大鱼吃小鱼经典版(经典版)官方版)

大鱼吃小鱼小鱼吃虾是于谦跟郭麒麟的《我的棒儿呢?》郭德纲说于思洋郭麒麟作诗的相声,最后郭麒麟做了一首,师傅躺在师母身上大鱼吃小鱼小鱼吃虾虾吃水水落石出师傅压师娘师娘压床床压地地动山摇。...

谷歌地球下载高清卫星地图(谷歌地球地图下载器)
  • 谷歌地球下载高清卫星地图(谷歌地球地图下载器)
  • 谷歌地球下载高清卫星地图(谷歌地球地图下载器)
  • 谷歌地球下载高清卫星地图(谷歌地球地图下载器)
  • 谷歌地球下载高清卫星地图(谷歌地球地图下载器)
哪个软件可以免费pdf转ppt(免费的pdf转ppt软件哪个好)
哪个软件可以免费pdf转ppt(免费的pdf转ppt软件哪个好)

要想将ppt免费转换为pdf的话,我们建议大家可以下一个那个wps,如果你是会员的话,可以注册为会员,这样的话,在wps里面的话,就可以免费将ppt呢转换为pdfpdf之后呢,我们就可以直接使用,不需要去直接不需要去另外保存,为什么格式转...

2026-02-04 09:03 off999

电信宽带测速官网入口(电信宽带测速官网入口app)

这个网站看看http://www.swok.cn/pcindex.jsp1.登录中国电信网上营业厅,宽带光纤,贴心服务,宽带测速2.下载第三方软件,如360等。进行在线测速进行宽带测速时,尽...

植物大战僵尸95版手机下载(植物大战僵尸95 版下载)

1可以在应用商店或者游戏平台上下载植物大战僵尸95版手机游戏。2下载教程:打开应用商店或者游戏平台,搜索“植物大战僵尸95版”,找到游戏后点击下载按钮,等待下载完成即可安装并开始游戏。3注意:确...

免费下载ppt成品的网站(ppt成品免费下载的网站有哪些)

1、Chuangkit(chuangkit.com)直达地址:chuangkit.com2、Woodo幻灯片(woodo.cn)直达链接:woodo.cn3、OfficePlus(officeplu...

2025世界杯赛程表(2025世界杯在哪个国家)

2022年卡塔尔世界杯赛程公布,全部比赛在卡塔尔境内8座球场举行,2022年,决赛阶段球队全部确定。揭幕战于当地时间11月20日19时进行,由东道主卡塔尔对阵厄瓜多尔,决赛于当地时间12月18日...

下载搜狐视频电视剧(搜狐电视剧下载安装)

搜狐视频APP下载好的视频想要导出到手机相册里方法如下1、打开手机搜狐视频软件,进入搜狐视频后我们点击右上角的“查找”,找到自已喜欢的视频。2、在“浏览器页面搜索”窗口中,输入要下载的视频的名称,然后...

pubg免费下载入口(pubg下载入口官方正版)
  • pubg免费下载入口(pubg下载入口官方正版)
  • pubg免费下载入口(pubg下载入口官方正版)
  • pubg免费下载入口(pubg下载入口官方正版)
  • pubg免费下载入口(pubg下载入口官方正版)
永久免费听歌网站(丫丫音乐网)

可以到《我爱音乐网》《好听音乐网》《一听音乐网》《YYMP3音乐网》还可以到《九天音乐网》永久免费听歌软件有酷狗音乐和天猫精灵,以前要跳舞经常要下载舞曲,我从QQ上找不到舞曲下载就从酷狗音乐上找,大多...

音乐格式转换mp3软件(音乐格式转换器免费版)

有两种方法:方法一在手机上操作:1、进入手机中的文件管理。2、在其中选择“音乐”,将显示出手机中的全部音乐。3、点击“全选”,选中所有音乐文件。4、点击屏幕右下方的省略号图标,在弹出菜单中选择“...

电子书txt下载(免费的最全的小说阅读器)

1.Z-library里面收录了近千万本电子书籍,需求量大。2.苦瓜书盘没有广告,不需要账号注册,使用起来非常简单,直接搜索预览下载即可。3.鸠摩搜书整体风格简洁清晰,书籍资源丰富。4.亚马逊图书书籍...

最好免费观看高清电影(播放免费的最好看的电影)

在目前的网上选择中,IMDb(互联网电影数据库)被认为是最全的电影网站之一。这个网站提供了各种类型的电影和电视节目的海量信息,包括剧情介绍、演员表、评价、评论等。其还提供了有关电影制作背后的详细信息,...

孤单枪手2简体中文版(孤单枪手2简体中文版官方下载)

要将《孤胆枪手2》游戏的征兵秘籍切换为中文,您可以按照以下步骤进行操作:首先,打开游戏设置选项,通常可以在游戏主菜单或游戏内部找到。然后,寻找语言选项或界面选项,点击进入。在语言选项中,选择中文作为游...

取消回复欢迎 发表评论: