当前位置:   article > 正文

oracle S3工具类 OCI S3工具类 解决S3生成预签名url前端跨域问题 S3 objectstorage api工具栏_s3上传跨域

s3上传跨域

objectstorage API 官方文档: https://docs.oracle.com/en-us/iaas/api/#/en/objectstorage/20160918/

介绍

使用对象存储和存档存储 API 来管理存储桶、对象和相关资源。有关详细信息,请参阅对象存储概述和存档存储概述。
Oracle 云基础设施对象存储服务是一个互联网规模的高性能存储平台,可提供可靠且经济高效的数据持久性。 对象存储服务可以存储无限量的任何内容类型的非结构化数据,包括分析数据和丰富的内容,如图像和视频。

注意事项

生成URL前端跨域问题
1. Amazon(亚马逊) 解决方案:

官方解决跨域文档地址: https://docs.aws.amazon.com/zh_cn/AmazonS3/latest/userguide/enabling-cors-examples.html
问题解决及工具类: https://blog.csdn.net/ayunnuo/article/details/127211608

2. OCI(Oracle) 解决方案: 使用OCI原生服务: API Amazon Simple Storage

官方文档地址: https://docs.amazonaws.cn/AmazonS3/latest/userguide/upload-objects.html

POM依赖


 <!-- 依赖声明 -->
 <dependencyManagement>
     <dependencies>
     
         <!-- oci -->
         <dependency>
             <groupId>com.oracle.oci.sdk</groupId>
             <artifactId>oci-java-sdk-bom</artifactId>
             <version>2.44.0</version>
             <type>pom</type>
             <scope>import</scope>
         </dependency>
         
     </dependencies>
 </dependencyManagement>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16

    <!-- https://mvnrepository.com/artifact/com.oracle.oci.sdk/oci-java-sdk-objectstorage -->
     <dependency>
         <groupId>com.oracle.oci.sdk</groupId>
         <artifactId>oci-java-sdk-objectstorage</artifactId>
     </dependency>
 
     <!-- lombok  -->
     <dependency>
         <groupId>org.projectlombok</groupId>
         <artifactId>lombok</artifactId>
         <optional>true</optional>
     </dependency>

     <!--    hutool工具类    -->
     <dependency>
         <groupId>cn.hutool</groupId>
         <artifactId>hutool-all</artifactId>
         <version>5.8.3</version>
     </dependency>
     
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21

DTO

1. S3BaseConfig

import cn.hutool.json.JSONUtil;
import io.swagger.annotations.ApiModel;
import io.swagger.annotations.ApiModelProperty;
import lombok.AllArgsConstructor;
import lombok.Getter;
import lombok.NoArgsConstructor;
import lombok.Setter;

/**
 * s3基础配置Bean
 *
 * @author yunnuo <a href="2552846359@qq.com">E-Mail: 2552846359@qq.com</a>
 * @since 1.0.0
 */
@Getter
@Setter
@NoArgsConstructor
@AllArgsConstructor
@ApiModel(value = "S3基础配置 Bean")
public class S3BaseConfig {

    /**
     * s3秘密访问密钥
     */
    @ApiModelProperty(value = "s3秘密访问密钥")
    private String s3SecretAccessKey;

    /**
     * s3访问密钥id
     */
    @ApiModelProperty(value = "s3访问密钥id")
    private String s3AccessKeyId;

    /**
     * s3 bucket
     */
    @ApiModelProperty(value = "s3 bucket")
    private String s3Bucket;

    /**
     * 地区
     */
    @ApiModelProperty(value = "地区")
    private String regions;

    /**
     * 类型 0=aws环境 1=oracle环境, default= 0
     */
    @ApiModelProperty(value = "类型 0=aws环境 1=oracle环境, default=0")
    private Integer type = 0;

    /**
     * aws环境可空, oci环境使用
     */
    @ApiModelProperty(value = "namespace")
    private String namespace;

    @Override
    public String toString() {
        return JSONUtil.toJsonStr(this);
    }
}


  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
2. S3CommonConfig

import cn.hutool.json.JSONUtil;
import io.swagger.annotations.ApiModelProperty;
import lombok.*;
import lombok.experimental.Accessors;

/**
 * s3 配置中心通用配置
 *
 * @author yunnuo <a href="2552846359@qq.com">Email: 2552846359@qq.com</a>
 * @see S3CommonConfig : {@code 参考配置中心key: bt-user.s3.images.config}
 * @since 1.0.0
 */
@Getter
@Setter
@Builder
@NoArgsConstructor
@AllArgsConstructor
@Accessors(chain = true)
public class S3CommonConfig extends S3BaseConfig {

    /**
     * 标题
     */
    @ApiModelProperty(value = "标题")
    private String title;

    /**
     * 描述
     */
    @ApiModelProperty(value = "描述")
    private String desc;

    /**
     * cdn前缀
     */
    @ApiModelProperty(value = "cdn前缀")
    private String cdnPrefix;


    @Override
    public String toString() {
        return JSONUtil.toJsonStr(this);
    }

}

  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47

OCI Object Storage Util 工具类

注意: 此工具类生产的预前面上传url 支持跨域操作


import com.kabak.rongsheng.domain.dto.s3.S3BaseConfig;
import com.kabak.rongsheng.domain.dto.s3.S3CommonConfig;
import com.oracle.bmc.ConfigFileReader;
import com.oracle.bmc.auth.ConfigFileAuthenticationDetailsProvider;
import com.oracle.bmc.objectstorage.ObjectStorageClient;
import com.oracle.bmc.objectstorage.model.CreatePreauthenticatedRequestDetails;
import com.oracle.bmc.objectstorage.model.PreauthenticatedRequest;
import com.oracle.bmc.objectstorage.model.RenameObjectDetails;
import com.oracle.bmc.objectstorage.requests.*;
import com.oracle.bmc.objectstorage.responses.*;

import java.io.IOException;
import java.io.InputStream;
import java.util.Date;

/**
 * oci Object Storage 版本工具类
 *
 * @author yunnuo <a href="2552846359@qq.com">Email: 2552846359@qq.com</a>
 * @since 1.0.0
 */
public class OciS3Utils {

    public static final String PRE_URL = "https://objectstorage.<region_ID>.oraclecloud.com<access-uri><objectName>";

    /**
     * 复制对象
     *
     * @param client            客户端
     * @param copyObjectRequest 复制对象请求
     * @return {@link CopyObjectResponse}
     */
    public static CopyObjectResponse copyObject(ObjectStorageClient client, CopyObjectRequest copyObjectRequest) {
        return client.copyObject(copyObjectRequest);
    }


    /**
     * 重命名对象
     *
     * @param client     客户端
     * @param namespace  名称空间
     * @param bucketName bucket名称
     * @param sourceName 原名称
     * @param newName    新名字
     * @return {@link RenameObjectResponse}
     */
    public static RenameObjectResponse renameObject(ObjectStorageClient client, String namespace, String bucketName, String sourceName, String newName) {
        RenameObjectDetails renameObjectDetails = RenameObjectDetails.builder()
                .sourceName(sourceName)
                .newName(newName).build();
        RenameObjectRequest renameObjectRequest = RenameObjectRequest.builder()
                .namespaceName(namespace)
                .bucketName(bucketName)
                .renameObjectDetails(renameObjectDetails).build();
        return client.renameObject(renameObjectRequest);
    }

    /**
     * 重命名对象
     *
     * @param client              客户端
     * @param renameObjectRequest 重命名对象请求
     * @return {@link RenameObjectResponse}
     */
    public static RenameObjectResponse renameObject(ObjectStorageClient client, RenameObjectRequest renameObjectRequest) {
        return client.renameObject(renameObjectRequest);
    }

    /**
     * 获取对象
     *
     * @param client     客户端
     * @param namespace  名称空间
     * @param bucketName bucket名称
     * @param key        key
     * @return {@link GetObjectResponse}
     */
    public static GetObjectResponse getObject(ObjectStorageClient client, String namespace, String bucketName, String key) {
        GetObjectRequest getObjectRequest = GetObjectRequest.builder()
                .namespaceName(namespace)
                .bucketName(bucketName)
                .objectName(key).build();
        return client.getObject(getObjectRequest);
    }

    /**
     * 获取对象
     *
     * @param client        客户端
     * @param objectRequest 对象请求
     * @return {@link GetObjectResponse}
     */
    public static GetObjectResponse getObject(ObjectStorageClient client, GetObjectRequest objectRequest) {
        return client.getObject(objectRequest);
    }

    /**
     * 删除对象
     *
     * @param client     客户端
     * @param namespace  名称空间
     * @param bucketName bucket名称
     * @param key        key
     * @return {@link DeleteObjectResponse}
     */
    public static DeleteObjectResponse deleteObject(ObjectStorageClient client, String namespace, String bucketName, String key) {
        DeleteObjectRequest deleteObjectRequest = DeleteObjectRequest.builder()
                .namespaceName(namespace)
                .bucketName(bucketName)
                .objectName(key).build();
        return client.deleteObject(deleteObjectRequest);
    }

    /**
     * 删除对象
     *
     * @param client              客户端
     * @param deleteObjectRequest 删除对象请求
     * @return {@link DeleteObjectResponse}
     */
    public static DeleteObjectResponse deleteObject(ObjectStorageClient client, DeleteObjectRequest deleteObjectRequest) {
        return client.deleteObject(deleteObjectRequest);
    }

    /**
     * 获取上传单个文件 默认:bundle/date/key 预授权url
     * <p color='red'> 注意: 如果需要上传多个文件到同一个目录下, 请使用{@code getPreAuthAnyObjectPathReadWriteURL} </p>
     *
     * @param client      客户端
     * @param config      配置
     * @param name        名字(主要是在Console里面起显示作用, 不允许重复)
     * @param timeExpires 过期时间
     * @param key         key
     * @return {@link String}
     */
    public static String getPreAuthObjDefaultBundleDateURL(ObjectStorageClient client, S3CommonConfig config, String name, Date timeExpires, String... key) {
        String filePath = S3CommonUtils.getBundleCurrentDateKey(config.getS3Bucket(), S3CommonUtils.formatFilePath(key));
        CreatePreauthenticatedRequestResponse response = getPreAuthAnyObjectReadWriteResponse(client, config.getS3Bucket(), config.getNamespace(), name, filePath, timeExpires);
        return getPreAuthUrl(config.getRegions(), response.getPreauthenticatedRequest());
    }

    /**
     * 获取上传单个文件 预授权url
     * <p color='red'> 注意: 如果需要上传多个文件到同一个目录下, 请使用{@code getPreAuthAnyObjectPathReadWriteURL} </p>
     *
     * @param client      客户端
     * @param config      配置
     * @param name        名字(主要是在Console里面起显示作用, 不允许重复)
     * @param timeExpires 过期时间
     * @param key         key
     * @return {@link String}
     */
    public static String getPreAuthAnyObjectReadWriteURL(ObjectStorageClient client, S3BaseConfig config, String name, Date timeExpires, String... key) {
        String filePath = S3CommonUtils.formatFilePath(key);
        CreatePreauthenticatedRequestResponse response = getPreAuthAnyObjectReadWriteResponse(client, config.getS3Bucket(), config.getNamespace(), name, filePath, timeExpires);
        return getPreAuthUrl(config.getRegions(), response.getPreauthenticatedRequest());
    }

    /**
     * 获取预身份验证对象路径 [权限: 读写]  默认:bundle/date 预授权url
     *
     * @param client      客户端
     * @param config      配置
     * @param name        名字(主要是在Console里面起显示作用, 不允许重复)
     * @param timeExpires 过期时间
     * @return {@link String}
     */
    public static String getPreAuthObjPathDefaultBundleDateURL(ObjectStorageClient client, S3BaseConfig config, String name, String bundle, Date timeExpires) {
        String filePath = S3CommonUtils.getBundleCurrentDateKey(bundle, "");
        CreatePreauthenticatedRequestResponse response = getPreAuthAnyObjectReadWriteResponse(client, config.getS3Bucket(), config.getNamespace(), name, filePath, timeExpires);
        return getPreAuthUrl(config.getRegions(), response.getPreauthenticatedRequest());
    }

    /**
     * 获取预身份验证对象路径[权限: 读写] URL
     * <p color='red'>解决性能问题: prefixKey可以传一个前缀路径, 然后通过生成的url地址拼接对象名称进行上传,防止上传多个文件到同一个文件夹下申请多次url</p>
     *
     * @param client      客户端
     * @param config      配置
     * @param name        名字(主要是在Console里面起显示作用, 不允许重复)
     * @param prefixKey   前缀路径
     * @param timeExpires 过期时间
     * @return {@link String}
     */
    public static String getPreAuthAnyObjectPathReadWriteURL(ObjectStorageClient client, S3BaseConfig config, String name, String prefixKey, Date timeExpires) {
        CreatePreauthenticatedRequestResponse response = getPreAuthAnyObjectReadWriteResponse(client, config.getS3Bucket(), config.getNamespace(), name, prefixKey, timeExpires);
        return getPreAuthUrl(config.getRegions(), response.getPreauthenticatedRequest());
    }


    /**
     * 获取预身份验证对象路径[权限: 读写] URL
     * <p color='red'>解决性能问题: prefixKey可以传一个前缀路径, 然后通过生成的url地址拼接对象名称进行上传,防止上传多个文件到同一个文件夹下申请多次url</p>
     *
     * @param client      客户端
     * @param config      配置
     * @param name        名字(主要是在Console里面起显示作用, 不允许重复)
     * @param prefixKey   前缀key
     * @param timeExpires 过期时间
     * @return {@link String}
     */
    public static String getPreAuthAnyObjectPathReadWriteURL(ObjectStorageClient client, S3CommonConfig config, String name, String prefixKey, Date timeExpires) {
        CreatePreauthenticatedRequestResponse response = getPreAuthAnyObjectReadWriteResponse(client, config.getS3Bucket(), config.getNamespace(), name, prefixKey, timeExpires);
        return getPreAuthUrl(config.getRegions(), response.getPreauthenticatedRequest());
    }


    /**
     * 获取预身份验证对象[权限: 读写]响应
     *
     * @param client        客户端
     * @param bucketName    bucket名称
     * @param namespaceName 名称空间名字
     * @param name          名字(主要是在Console里面起显示作用, 不允许重复)
     * @param key           key
     * @param timeExpires   过期时间
     * @return {@link CreatePreauthenticatedRequestResponse}
     */
    public static CreatePreauthenticatedRequestResponse getPreAuthAnyObjectReadWriteResponse(ObjectStorageClient client, String bucketName, String namespaceName, String name, String key, Date timeExpires) {
        CreatePreauthenticatedRequestDetails createPreauthenticatedRequestDetails = CreatePreauthenticatedRequestDetails.builder()
                .name(name)
                .bucketListingAction(PreauthenticatedRequest.BucketListingAction.ListObjects)
                .objectName(key)
                .accessType(CreatePreauthenticatedRequestDetails.AccessType.AnyObjectReadWrite)
                .timeExpires(timeExpires).build();

        CreatePreauthenticatedRequestRequest createPreauthenticatedRequestRequest = CreatePreauthenticatedRequestRequest.builder()
                .namespaceName(namespaceName)
                .bucketName(bucketName)
                .createPreauthenticatedRequestDetails(createPreauthenticatedRequestDetails).build();

        return client.createPreauthenticatedRequest(createPreauthenticatedRequestRequest);
    }

    /**
     * 上传对象到默认包/日期/key下
     *
     * @param config        配置
     * @param client        客户端
     * @param bundle        包
     * @param contentLength 内容长度
     * @param contentType   内容类型
     * @param key           key
     * @param objectContent 对象内容
     * @return {@link PutObjectResponse}
     */
    public static PutObjectResponse putObjectDefaultBundleDate(S3BaseConfig config, ObjectStorageClient client, String bundle, InputStream objectContent, Long contentLength, String contentType, String... key) {
        final String bundleCurrentDateKey = S3CommonUtils.getBundleCurrentDateKey(bundle, S3CommonUtils.formatFilePath(key));
        PutObjectRequest putObjectRequest = PutObjectRequest.builder()
                .namespaceName(config.getNamespace())
                .bucketName(config.getS3Bucket())
                .objectName(bundleCurrentDateKey)
                .putObjectBody(objectContent)
                .contentLength(contentLength)
                .contentType(contentType).build();
        return getPutObjectRequest(client, putObjectRequest);
    }

    /**
     * 上传对象到默认包/日期/key下
     *
     * @param config        配置
     * @param client        客户端
     * @param bundle        包
     * @param contentLength 内容长度
     * @param contentType   内容类型
     * @param key           key
     * @param objectContent 对象内容
     * @return {@link PutObjectResponse}
     */
    public static PutObjectResponse putObjectDefaultBundleDate(S3CommonConfig config, ObjectStorageClient client, String bundle, InputStream objectContent, Long contentLength, String contentType, String... key) {
        final String bundleCurrentDateKey = S3CommonUtils.getBundleCurrentDateKey(bundle, S3CommonUtils.formatFilePath(key));
        PutObjectRequest putObjectRequest = PutObjectRequest.builder()
                .namespaceName(config.getNamespace())
                .bucketName(config.getS3Bucket())
                .objectName(bundleCurrentDateKey)
                .putObjectBody(objectContent)
                .contentLength(contentLength)
                .contentType(contentType).build();
        return getPutObjectRequest(client, putObjectRequest);
    }

    /**
     * 上传对象
     *
     * @param config        配置
     * @param client        客户端
     * @param contentLength 内容长度
     * @param contentType   内容类型
     * @param key           key
     * @param objectContent 对象内容
     * @return {@link PutObjectResponse}
     */
    public static PutObjectResponse putObject(S3BaseConfig config, ObjectStorageClient client, InputStream objectContent, Long contentLength, String contentType, String... key) {
        String filePath = S3CommonUtils.formatFilePath(key);
        PutObjectRequest putObjectRequest = PutObjectRequest.builder()
                .namespaceName(config.getNamespace())
                .bucketName(config.getS3Bucket())
                .objectName(filePath)
                .putObjectBody(objectContent)
                .contentLength(contentLength)
                .contentType(contentType).build();
        return getPutObjectRequest(client, putObjectRequest);
    }


    /**
     * 上传对象
     *
     * @param config        配置
     * @param client        客户端
     * @param contentLength 内容长度
     * @param contentType   内容类型
     * @param key           key
     * @param objectContent 对象内容
     * @return {@link PutObjectResponse}
     */
    public static PutObjectResponse putObject(S3CommonConfig config, ObjectStorageClient client, InputStream objectContent, Long contentLength, String contentType, String... key) {
        String filePath = S3CommonUtils.formatFilePath(key);
        PutObjectRequest putObjectRequest = PutObjectRequest.builder()
                .namespaceName(config.getNamespace())
                .bucketName(config.getS3Bucket())
                .objectName(filePath)
                .putObjectBody(objectContent)
                .contentLength(contentLength)
                .contentType(contentType).build();
        return getPutObjectRequest(client, putObjectRequest);
    }

    /**
     * 获取 上传对象请求
     *
     * @param client           客户端
     * @param putObjectRequest 把对象请求
     * @return {@link PutObjectResponse}
     */
    public static PutObjectResponse getPutObjectRequest(ObjectStorageClient client, PutObjectRequest putObjectRequest) {
        return client.putObject(putObjectRequest);
    }


    /**
     * 获取默认client
     *
     * @return {@link ObjectStorageClient}
     * @throws IOException ioexception
     */
    public static ObjectStorageClient getDefaultClient() throws IOException {
        final ConfigFileAuthenticationDetailsProvider provider = getProvider(ConfigFileReader.parseDefault());
        return getClient(provider);
    }

    /**
     * 获取client
     *
     * @param configFilePath 配置文件路径
     * @return {@link ObjectStorageClient}
     * @throws IOException ioexception
     */
    public static ObjectStorageClient getClient(String configFilePath) throws IOException {
        final ConfigFileAuthenticationDetailsProvider provider = getProvider(configFilePath);
        return getClient(provider);
    }

    /**
     * 获取client
     *
     * @param configFile 配置文件
     * @return {@link ObjectStorageClient}
     */
    public static ObjectStorageClient getClient(ConfigFileReader.ConfigFile configFile) {
        final ConfigFileAuthenticationDetailsProvider provider = getProvider(configFile);
        return getClient(provider);
    }

    /**
     * 获取client
     *
     * @param provider 提供者
     * @return {@link ObjectStorageClient}
     */
    public static ObjectStorageClient getClient(ConfigFileAuthenticationDetailsProvider provider) {
        return new ObjectStorageClient(provider);
    }


    /**
     * 获取提供者
     *
     * @param configFilePath 配置文件路径
     * @return {@link ConfigFileAuthenticationDetailsProvider}
     * @throws IOException ioexception
     */
    public static ConfigFileAuthenticationDetailsProvider getProvider(String configFilePath) throws IOException {
        final ConfigFileReader.ConfigFile configFile = ConfigFileReader.parse(configFilePath);
        return getProvider(configFile);

    }

    /**
     * 获取提供者
     *
     * @param configFile 配置文件
     * @return {@link ConfigFileAuthenticationDetailsProvider}
     */
    public static ConfigFileAuthenticationDetailsProvider getProvider(ConfigFileReader.ConfigFile configFile) {
        return new ConfigFileAuthenticationDetailsProvider(configFile);

    }


    /**
     * 获取预身份验证url
     *
     * @param region                  地区
     * @param preauthenticatedRequest preauthenticated请求
     * @return {@link String}
     */
    public static String getPreAuthUrl(String region, PreauthenticatedRequest preauthenticatedRequest) {
        return getPreAuthUrl(region, preauthenticatedRequest.getAccessUri(), preauthenticatedRequest.getObjectName());
    }

    /**
     * 获取预身份验证url
     *
     * @param accessUri  uri访问
     * @param objectName 对象名称
     * @param region     地区
     * @return {@link String}
     */
    public static String getPreAuthUrl(String region, String accessUri, String objectName) {
        return PRE_URL.replace("<region_ID>", region)
                .replace("<access-uri>", accessUri)
                .replace("<objectName>", objectName);
    }


}

  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71
  • 72
  • 73
  • 74
  • 75
  • 76
  • 77
  • 78
  • 79
  • 80
  • 81
  • 82
  • 83
  • 84
  • 85
  • 86
  • 87
  • 88
  • 89
  • 90
  • 91
  • 92
  • 93
  • 94
  • 95
  • 96
  • 97
  • 98
  • 99
  • 100
  • 101
  • 102
  • 103
  • 104
  • 105
  • 106
  • 107
  • 108
  • 109
  • 110
  • 111
  • 112
  • 113
  • 114
  • 115
  • 116
  • 117
  • 118
  • 119
  • 120
  • 121
  • 122
  • 123
  • 124
  • 125
  • 126
  • 127
  • 128
  • 129
  • 130
  • 131
  • 132
  • 133
  • 134
  • 135
  • 136
  • 137
  • 138
  • 139
  • 140
  • 141
  • 142
  • 143
  • 144
  • 145
  • 146
  • 147
  • 148
  • 149
  • 150
  • 151
  • 152
  • 153
  • 154
  • 155
  • 156
  • 157
  • 158
  • 159
  • 160
  • 161
  • 162
  • 163
  • 164
  • 165
  • 166
  • 167
  • 168
  • 169
  • 170
  • 171
  • 172
  • 173
  • 174
  • 175
  • 176
  • 177
  • 178
  • 179
  • 180
  • 181
  • 182
  • 183
  • 184
  • 185
  • 186
  • 187
  • 188
  • 189
  • 190
  • 191
  • 192
  • 193
  • 194
  • 195
  • 196
  • 197
  • 198
  • 199
  • 200
  • 201
  • 202
  • 203
  • 204
  • 205
  • 206
  • 207
  • 208
  • 209
  • 210
  • 211
  • 212
  • 213
  • 214
  • 215
  • 216
  • 217
  • 218
  • 219
  • 220
  • 221
  • 222
  • 223
  • 224
  • 225
  • 226
  • 227
  • 228
  • 229
  • 230
  • 231
  • 232
  • 233
  • 234
  • 235
  • 236
  • 237
  • 238
  • 239
  • 240
  • 241
  • 242
  • 243
  • 244
  • 245
  • 246
  • 247
  • 248
  • 249
  • 250
  • 251
  • 252
  • 253
  • 254
  • 255
  • 256
  • 257
  • 258
  • 259
  • 260
  • 261
  • 262
  • 263
  • 264
  • 265
  • 266
  • 267
  • 268
  • 269
  • 270
  • 271
  • 272
  • 273
  • 274
  • 275
  • 276
  • 277
  • 278
  • 279
  • 280
  • 281
  • 282
  • 283
  • 284
  • 285
  • 286
  • 287
  • 288
  • 289
  • 290
  • 291
  • 292
  • 293
  • 294
  • 295
  • 296
  • 297
  • 298
  • 299
  • 300
  • 301
  • 302
  • 303
  • 304
  • 305
  • 306
  • 307
  • 308
  • 309
  • 310
  • 311
  • 312
  • 313
  • 314
  • 315
  • 316
  • 317
  • 318
  • 319
  • 320
  • 321
  • 322
  • 323
  • 324
  • 325
  • 326
  • 327
  • 328
  • 329
  • 330
  • 331
  • 332
  • 333
  • 334
  • 335
  • 336
  • 337
  • 338
  • 339
  • 340
  • 341
  • 342
  • 343
  • 344
  • 345
  • 346
  • 347
  • 348
  • 349
  • 350
  • 351
  • 352
  • 353
  • 354
  • 355
  • 356
  • 357
  • 358
  • 359
  • 360
  • 361
  • 362
  • 363
  • 364
  • 365
  • 366
  • 367
  • 368
  • 369
  • 370
  • 371
  • 372
  • 373
  • 374
  • 375
  • 376
  • 377
  • 378
  • 379
  • 380
  • 381
  • 382
  • 383
  • 384
  • 385
  • 386
  • 387
  • 388
  • 389
  • 390
  • 391
  • 392
  • 393
  • 394
  • 395
  • 396
  • 397
  • 398
  • 399
  • 400
  • 401
  • 402
  • 403
  • 404
  • 405
  • 406
  • 407
  • 408
  • 409
  • 410
  • 411
  • 412
  • 413
  • 414
  • 415
  • 416
  • 417
  • 418
  • 419
  • 420
  • 421
  • 422
  • 423
  • 424
  • 425
  • 426
  • 427
  • 428
  • 429
  • 430
  • 431
  • 432
  • 433
  • 434
  • 435
  • 436
  • 437
  • 438
  • 439
  • 440
  • 441
声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/小小林熬夜学编程/article/detail/707154
推荐阅读
相关标签
  

闽ICP备14008679号