【面试 反思】Retrofit源码与设计 7 连问

news2025/3/3 4:09:21

前言

在实际项目中往往是使用Retrofit来做网络请求工作。Retrofit采用RESTful风格,本质上只是对OkHttp进行封装,今天我们根据几个问题来进一步学习一下Retrofit的源码与设计思想。

1. 使用方法

直接看一下官方介绍的使用方法。

public final class SimpleService {
  public static final String API_URL = "https://api.github.com";

  public static class Contributor {
    public final String login;
    public final int contributions;

    public Contributor(String login, int contributions) {
      this.login = login;
      this.contributions = contributions;
    }
  }

  public interface GitHub {
    @GET("/repos/{owner}/{repo}/contributors")
    Call<List<Contributor>> contributors(@Path("owner") String owner, @Path("repo") String repo);
  }

  public static void main(String... args) throws IOException {
    // Create a very simple REST adapter which points the GitHub API.
    Retrofit retrofit =
        new Retrofit.Builder()
            .baseUrl(API_URL)
            .client(new OkHttpClient().newBuilder().connectTimeout(10, TimeUnit.SECONDS).build())
            .addConverterFactory(GsonConverterFactory.create())
            .build();

    // Create an instance of our GitHub API interface.
    GitHub github = retrofit.create(GitHub.class);

    // Create a call instance for looking up Retrofit contributors.
    Call<List<Contributor>> call = github.contributors("square", "retrofit");

    // Fetch and print a list of the contributors to the library.
    List<Contributor> contributors = call.execute().body();
    for (Contributor contributor : contributors) {
      System.out.println(contributor.login + " (" + contributor.contributions + ")");
    }
  }
}

可以简单的概括成三步:

  1. 构建 retrofit 实例。
  2. 构建 API 接口实例。
  3. 执行请求,解析响应。

2. 流程解析

我们按照它的使用方法来分析一下它的流程。

2.1 构建 Retrofit 实例

从使用方法可以看出是使用建造者模式来构建实例。

Retrofit retrofit =
    new Retrofit.Builder()
        .baseUrl(API_URL)
        .client(new OkHttpClient().newBuilder().connectTimeout(10, TimeUnit.SECONDS).build())
        .addConverterFactory(GsonConverterFactory.create())
        .addCallAdapterFactory(RxJavaCallAdapterFactory.create())
        .build();

这一步就不具体展开了,看几个参数。

public static final class Builder {
    //实际的请求调用,如 okhttp3.OkHttpClient
    private @Nullable okhttp3.Call.Factory callFactory;
    //基础URL,如:域名
    private @Nullable HttpUrl baseUrl;
    //数据转换器列表
    private final List<Converter.Factory> converterFactories = new ArrayList<>();
    //请求适配器列表
    private final List<CallAdapter.Factory> callAdapterFactories = new ArrayList<>();

2.2 构建 API 接口实例

按照官方的使用方法介绍,我们会将我们的API方法放在一个接口中,然后通过注解来设置请求参数。在使用时,通过retrofit.create(Class<T>)方法将这个接口实例化,然后调用其方法。 如:

public interface GitHub {
    @GET("/repos/{owner}/{repo}/contributors")
    Call<List<Contributor>> contributors(@Path("owner") String owner, @Path("repo") String repo);
}

//实例化API接口
GitHub github = retrofit.create(GitHub.class);
//调用接口中某条API
Call<List<Contributor>> call = github.contributors("square", "retrofit");

看一下源码

public <T> T create(final Class<T> service) {
    //验证 api service
    validateServiceInterface(service);
    return (T)
        //这里采用了动态代理模式, service 就是被代理类
        //todo 为什么要采用动态代理,有什么好处吗?用别的行不行?
        Proxy.newProxyInstance(
            service.getClassLoader(),
            new Class<?>[] {service},
            new InvocationHandler() {
              private final Object[] emptyArgs = new Object[0];

              @Override
              public @Nullable Object invoke(Object proxy, Method method, @Nullable Object[] args)
                  throws Throwable {
                // If the method is a method from Object then defer to normal invocation.
                if (method.getDeclaringClass() == Object.class) {
                  return method.invoke(this, args);
                }
                args = args != null ? args : emptyArgs;
                Platform platform = Platform.get();
                //如果不是系统默认方法,通过loadServiceMethod()方法返回一个ServiceMethod,并调用invoke方法
                return platform.isDefaultMethod(method)
                    ? platform.invokeDefaultMethod(method, service, proxy, args)
                    : loadServiceMethod(method).invoke(args);
              }
            });
  }

做了两件事:

  1. 验证我们的API接口类。
  2. 利用动态代理在运行期间实例化API接口。
  private void validateServiceInterface(Class<?> service) {
    //service 必须是 interface,否则抛出异常
    if (!service.isInterface()) {
      throw new IllegalArgumentException("API declarations must be interfaces.");
    }

    ...省略代码...

    //是否立即验证API接口中的所有方法,由用户设置,默认为false
    if (validateEagerly) {
      Platform platform = Platform.get();
      //遍历 service 中定义的所有方法
      for (Method method : service.getDeclaredMethods()) {
        //如果该方法不是系统默认方法且方法修饰符不是静态方法就执行loadServiceMethod方法
        if (!platform.isDefaultMethod(method) && !Modifier.isStatic(method.getModifiers())) {
          //加载请求方法。
          loadServiceMethod(method);
        }
      }
    }
  }

从这我们也可以看出,我们的API方法必须方法接口中。如果开始验证接口,会遍历其声明的所有方法,过滤掉系统默认方法与静态方法,然后执行loadServiceMethod(method)

扩充一下:

getMethods(): 返回由类或接口声明的以及从超类和超接口继承的所有公共方法。 getDeclaredMethods(): 返回类声明的方法,包括 public, protected, default (package),但不包括继承的方法。所以,相对比于 getMethods 方法,getDeclaredMethods速度更快,尤其是在复杂的类中,如在Activity类中。

最终都是通过loadServiceMethod(method) 方法来加载一个ServiceMethod

看一下HttpServiceMethod.parseAnnotations()方法,我将其简化了一下,如下:

HttpServiceMethod.java

  static <ResponseT, ReturnT> HttpServiceMethod<ResponseT, ReturnT> parseAnnotations(
      Retrofit retrofit, Method method, RequestFactory requestFactory) {

    //获取方法的注解信息
    Annotation[] annotations = method.getAnnotations();
    //适配器类型,就是Retrofit.addCallAdapterFactory()添加的类型。
    Type adapterType;
    //方法的返回类型
    adapterType = method.getGenericReturnType();

    //实例化一个 CallAdapter 对象
    CallAdapter<ResponseT, ReturnT> callAdapter =
        createCallAdapter(retrofit, method, adapterType, annotations);
    //检查 responseType,如果不合格则抛出异常
    Type responseType = callAdapter.responseType();

    //实例化一个Converter对象,将 okhttp3.ResponseBody 转换成 ResponseT 类型
    Converter<ResponseBody, ResponseT> responseConverter =
        createResponseConverter(retrofit, method, responseType);

    okhttp3.Call.Factory callFactory = retrofit.callFactory;
    //不是kotlin挂起方法,返回 CallAdapted,其实也就是调用 callAdapter.adapter 方法
    return new CallAdapted<>(requestFactory, callFactory, responseConverter, callAdapter);
  }

实例化了 ServiceMethod 后,调用invoke方法。

HttpServiceMethod.java

  @Override
  final @Nullable ReturnT invoke(Object[] args) {
    //新建一个 OkHttpCall 请求
    Call<ResponseT> call = new OkHttpCall<>(requestFactory, args, callFactory, responseConverter);
    //然后调用 adapt 方法,CallAdapted 有重写 adapt 方法,然后调用 callAdapter.adapt(call) 方法
    return adapt(call, args);
  }

  protected abstract @Nullable ReturnT adapt(Call<ResponseT> call, Object[] args);

从上述代码中可以看出,invoke方法就是实例化一个Call请求,然后调用adapter方法,在这里adapter是一个抽象方法,所以具体实现方法就需要看它的具体实现类CallAdapter。 这里的 CallAdapter 就是通过.addCallAdapterFactory()方法所添加的CallAdapter,以及根据平台默认提供的DefaultCallAdapterFactory中的CallAdapter,执行其adapter方法,最终返回Call<Object>

2.3 执行请求,解析响应

在上一步中,我们对API接口进行了实例化,通过CallAdapter对请求进行适配,最终得到一个Call<Object>对象。

接着下一步,就是执行这个Call<Object>请求,最终得到我们想要的Object对象。

例如一开始使用方法中所介绍的:

//已经得到了Call<List<Contributor>>对象,执行call,得到List<Contributor>
List<Contributor> contributors = call.execute().body();

调用 execute 执行同步请求获取到Response,然后获取其请求体。

OkHttpCall.java

  @Override
  public Response<T> execute() throws IOException {
    okhttp3.Call call;

    synchronized (this) {
      //判断请求是否已经被执行,如果已被执行则抛出异常
      if (executed) throw new IllegalStateException("Already executed.");
      executed = true;
      //获取最原始的请求,通过createRawCall()创建okhttp3.Call
      call = getRawCall();
    }

    if (canceled) {
      call.cancel();
    }
    //执行请求,并且解析响应,将okhttp3.response 转换成 retrofit2.response
    return parseResponse(call.execute());
  }

  private okhttp3.Call createRawCall() throws IOException {
    //构造原始请求
    okhttp3.Call call = callFactory.newCall(requestFactory.create(args));
    if (call == null) {
      throw new NullPointerException("Call.Factory returned null.");
    }
    return call;
  }

  /**
   * 解析响应,就是就okhttp3.response 转换成 retrofit2.response
   */
  Response<T> parseResponse(okhttp3.Response rawResponse) throws IOException {

    ...省略代码...

    try {
      //利用converter转换成我们期望的类型
      T body = responseConverter.convert(catchingBody);
      return Response.success(body, rawResponse);
    } catch (RuntimeException e) {

  ...省略代码...

  }

从源码中也可以看出,请求的实际工作还是通过okhttp来完成的,这边Retrofit就是负责请求与响应转换工作,将retrofit2.Call转换成okhttp3.Call,将okhttp3.response转换成retrofit2.response

3. 为什么要引入CallAdapter与Converter?

如果你熟悉okHttp的话,你应该知道,当我们请求的时候,要先通过OkHttpClient.newCall(request)方法将request转换成Call对象,然后再执行这个Call对象拿到response

但是Retrofit不光光只支持Call,他还可以将请求适配成Observable类型,方便与RxJava2结合起来一起使用。这就是通过CallAdapter来进行适配工作的,例如通过默认的DefaultCallAdapterFactory将请求转换成Call<Object>,通过RxJava2CallAdapter将请求转换成Observable<Object>

回到okHttp,大部分业务情况下,我们在拿到响应体后都会将其进行反序列化成对象,方便调用。显然,Retrofit就考虑到了这一点,所以他默认提供了GsonConverterFactory,来帮助我们做这一步反序列化工作。这就是通过Converter来完成的,同时它也支持用户进行自定义。

4. CallAdapter 是如何工作的?

作为请求适配器,我们将CallAdapter工作流程分为三步:添加、匹配、工作。

添加

可以通过addCallAdapterFactory(CallAdapter.Factory)方法来添加请求适配器工厂类,添加成功后会被保存在callAdapterFactories列表中。另外,Retrofit会根据Platform来添加默认的请求适配器,例如:DefaultCallAdapterFactory等等,同样也加入到callAdapterFactories列表中。

匹配

思考一下:所有添加的请求适配器都会被保存在callAdapterFactories列表中,那在实际请求中是如何匹配出相对应的适配器的呢?

HttpServiceMethod.parseAnnotations()方法中,我们有实例化一个CallAdapter对象。(具体流程就不再次展开了,请回头看 2.2 构建 API 接口实例 中所介绍内容。)

HttpServiceMethod.java

  static <ResponseT, ReturnT> HttpServiceMethod<ResponseT, ReturnT> parseAnnotations(
      Retrofit retrofit, Method method, RequestFactory requestFactory) {

    //实例化一个 CallAdapter 对象
    CallAdapter<ResponseT, ReturnT> callAdapter =
        createCallAdapter(retrofit, method, adapterType, annotations);

  ···省略代码···

    //不是kotlin挂起方法,返回 CallAdapted,其实也就是调用 callAdapter.adapter 方法
    return new CallAdapted<>(requestFactory, callFactory, responseConverter, callAdapter);
  }

匹配工作其实就在createCallAdapter()方法中,一步步走下来,最终到Retrofit.nextCallAdapter()方法中:

Retrofit.java

  public CallAdapter<?, ?> nextCallAdapter(
      @Nullable CallAdapter.Factory skipPast, Type returnType, Annotation[] annotations) {

    int start = callAdapterFactories.indexOf(skipPast) + 1;
    for (int i = start, count = callAdapterFactories.size(); i < count; i++) {
      //通过方法的返回值类型与注解信息来找到匹配的CallAdapter
      CallAdapter<?, ?> adapter = callAdapterFactories.get(i).get(returnType, annotations, this);
      if (adapter != null) {
        return adapter;
      }
    }

  ···省略代码···

    //如果找不到匹配的CallAdapter,则抛出异常
    throw new IllegalArgumentException(builder.toString());
  }

简单概括一下,就是通过方法的返回值类型与注解信息,遍历callAdapterFactories列表,找到匹配的CallAdapter,如果找不到则抛出IllegalArgumentException异常。

工作

找到匹配的CallAdapter后,剩下就是看看他是如何工作的。

如上一步匹配过程所介绍,在找到匹配的callAdapter后,会通过它来实例化一个CallAdapted对象。

  static final class CallAdapted<ResponseT, ReturnT> extends HttpServiceMethod<ResponseT, ReturnT> {
    private final CallAdapter<ResponseT, ReturnT> callAdapter;

    CallAdapted(
        RequestFactory requestFactory,
        okhttp3.Call.Factory callFactory,
        Converter<ResponseBody, ResponseT> responseConverter,
        CallAdapter<ResponseT, ReturnT> callAdapter) {
      //将responseConverter传给父类。
      super(requestFactory, callFactory, responseConverter);
      this.callAdapter = callAdapter;
    }

    @Override
    protected ReturnT adapt(Call<ResponseT> call, Object[] args) {
      return callAdapter.adapt(call);
    }
  }

CallAdapted很简单,就是继承了HttpServiceMethod,然后复写了adapt方法。也就是说,最终执行的,其实就是我们上一步匹配到的CallAdapter对象的adapt方法。

比如匹配到的是DefaultCallAdapterFactory中的CallAdapter,最终执行的就是其adapt方法,具体代码细节这边就不展示了,有兴趣同学请自行查阅。

另外,我这边展示的是不支持kotlin挂起函数的情况,当然即使是kotlin挂起函数,过程也是一样的,也是执行其子类的adapt方法。

5. Converter 是如何工作的?

作为数据转换器,我们同样将Converter工作流程分为三步:添加、匹配、工作。

添加

可以通过addConverterFactory(Converter.Factory)方法来添加数据装换器工厂类,添加成功后会被保存在converterFactories列表中。另外,Retrofit会根据Platform来添加默认的数据转换器,例如OptionalConverterFactory,同样也加入到converterFactories列表中。

匹配

跟上述所介绍的 4. CallAdapter 是如何工作的 一样,同样在HttpServiceMethod.parseAnnotations()方法中,会实例化一个Converter对象。

匹配工作其实就在createResponseConverter()方法中,一步步走下来,最终到Retrofit.nextResponseBodyConverter()方法中:

  public <T> Converter<ResponseBody, T> nextResponseBodyConverter(
      @Nullable Converter.Factory skipPast, Type type, Annotation[] annotations) {

    int start = converterFactories.indexOf(skipPast) + 1;
    for (int i = start, count = converterFactories.size(); i < count; i++) {
      //通过转换类型与注解信息来找到匹配的Converter
      Converter<ResponseBody, ?> converter =
          converterFactories.get(i).responseBodyConverter(type, annotations, this);
      if (converter != null) {
        //noinspection unchecked
        return (Converter<ResponseBody, T>) converter;
      }
    }

  ···省略代码···

    //如果找不到匹配的Converter,则抛出异常
    throw new IllegalArgumentException(builder.toString());
  }

简单概括一下,就是通过转换类型与注解信息,遍历converterFactories列表,找到匹配的Converter,如果找不到则抛出IllegalArgumentException异常。

工作

跟上述所介绍的 4. CallAdapter 是如何工作的 一样,我们找到匹配的Converter后,通过它来实例化一个CallAdapted。但不同的是,我们会将responseConverter传给父类,也就是HttpServiceMethod,然后当其调用invoke方法时,我们通过responseConverter来实例化一个OkHttpCall对象,最终将这个OkHttpCall对象传给adapter方法执行。

最终当执行请求时,OkHttpCall执行parseResponse来解析响应,调用responseConverter.convert()方法,将ResponseBody数据转换我们想要的类型。

6. 说说使用到了哪些设计模式

动态代理模式

Retrofit内部通过动态代理+反射来拿到用户定义在接口中的请求参数,从而来构建实际请求。具体细节这边就不再次展开了,可以回头查看 2.2 构建 API 接口实例 这一部分内容。

为什么要使用动态代理来获取API方法?

不知道你们有没有这个疑问,为什么我们的API方法需要定义在interface中呢?又为什么要通过动态代理+反射的形式来拿到请求参数呢?

Retrofit按照RESTful风格设计并通过注解来定义API方法的请求参数,并将这些API方法放到interface中。因为interface是不能被实例化的,所以这里采用动态代理在运行期间实例化API接口,获取到方法的请求参数。

再进一步:

解耦,将实际业务与Retrofit隔离开来。用户只需通过注解方法来定义请求参数,而实际请求的构建则通过Retrofit内部来实现。

此时再反过来看为何放在interface中?相信你心中已有答案了吧。

策略模式

当针对同一类型问题有多种处理方式,仅仅是具体行为有差别时,就可以使用策略模式。

例如:Retrofit中的请求适配器

  • 抽象策略:CallAdapter
  • 策略具体实现:DefaultCallAdapterFactory.get()RxJava2CallAdapter

即提供默认的请求适配器,也支持用户自定义,符合开闭原则,达到很好的可扩展性。

适配器模式

Retrofit会帮我们构建实际请求,内部通过默认的DefaultCallAdapterFactory来将请求转换成Call<Object>,同时Retrofit也支持其它平台,比如为了适配RxJava特性,将请求转换成Observable<Object>

  • Target(目标角色): Call<Object>, Observable<Object>
  • adaptee(需要适配的对象): OkHttpCall
  • adapter(适配器):DefaultCallAdapterFactory.get()RxJava2CallAdapter

工厂方法模式

我们以Converter来举例。

  • 抽象工厂:Converter.Factory
  • 具体工厂:GsonConverterFactoryBuiltInConverters等等。
  • 抽象产品:Converter
  • 具体产品:GsonResponseBodyConverterGsonRequestBodyConverterToStringConverter等等。

这边就不具体展开分析各个类了,有兴趣的同学可自行查阅。

建造者模式

在构建Retrofit实例的时候,就用到了建造者模式。建造者模式在开源库中的出现的次数真的很频繁,为了适配不同的用户的各种需求,需提供各种各样的参数与方法来供用户自行选择,所以使用建造者模式,之所以很常见,是因为这样很合理。

7. 使用过程中踩过什么坑?

关于BaseUrl的使用曾经踩过坑,某天我将baseUrl改了一下,然后发现请求接口时服务器一直返回404,但是当我尝试用Postman去调试接口的时候,发现接口是好的,也就推测出来是我的代码出问题了。

最终发现,问题出在:拼接成完整的URLapi被删除了。

Base URL: http://example.com/api/
Endpoint: /foo/bar/ 
Result: http://example.com/foo/bar/

正确的使用方式为:Endpoint不以斜杠开头。

Base URL: http://example.com/api/ 
Endpoint: foo/bar/ 
Result: http://example.com/api/foo/bar/

总结

本文,我们以几个问题的形式展开来对Retrofit源码及设计思想进行解析,相信你对源码有了进一步的了解。Retrofit本质只是对okHttp进行封装,出发点肯定是让网络请求变得更加容易,考虑适配各种用户需求,Jake Wharton大神用了很多设计模式,真的太让人膜拜了。

到此,关于Retrofit的源码解析就结束啦。

推荐更多Android学习笔记参考

Android 性能优化篇:https://qr18.cn/FVlo89
Android 车载篇:https://qr18.cn/F05ZCM
Android 逆向安全学习笔记:https://qr18.cn/CQ5TcL
Android Framework底层原理篇:https://qr18.cn/AQpN4J
Android 音视频篇:https://qr18.cn/Ei3VPD
Jetpack全家桶篇(内含Compose):https://qr18.cn/A0gajp
Kotlin 篇:https://qr18.cn/CdjtAF
Gradle 篇:https://qr18.cn/DzrmMB
OkHttp 源码解析笔记:https://qr18.cn/Cw0pBD
Flutter 篇:https://qr18.cn/DIvKma
Android 八大知识体:https://qr18.cn/CyxarU
Android 核心笔记:https://qr21.cn/CaZQLo
Android 往年面试题锦:https://qr18.cn/CKV8OZ
2023年最新Android 面试题集:https://qr18.cn/CgxrRy
Android 车载开发岗位面试习题:https://qr18.cn/FTlyCJ
音视频面试题锦:https://qr18.cn/AcV6Ap

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

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

相关文章

STM32之17.PWM脉冲宽度调制

一LED0脉冲宽度调制在TIM14_CHI&#xff0c;先将LED&#xff08;PF9&#xff09;代码配置为AF推挽输出模式&#xff0c;将PF9引脚连接到TIM14&#xff0c; #include <stm32f4xx.h>static GPIO_InitTypeDef GPIO_InitStruct;void Led_init(void) {//打开端口F的硬件时钟&a…

021-从零搭建微服务-短信服务(一)

写在最前 如果这个项目让你有所收获&#xff0c;记得 Star 关注哦&#xff0c;这对我是非常不错的鼓励与支持。 源码地址&#xff08;后端&#xff09;&#xff1a;https://gitee.com/csps/mingyue 源码地址&#xff08;前端&#xff09;&#xff1a;https://gitee.com/csps…

Git gui教程---第八篇 Git gui的使用 创建一个分支

一般情况下一个主分支下代码稳定的情况下会新建出一个分支&#xff0c;然后在分支上修改&#xff0c;修改完成稳定后再合并到主分支上。 或者几个人合作写一份代码&#xff0c;每个人各一个分支&#xff0c;测试稳定再合并到主分支上。 在git gui选择菜单栏“分支”&#xff0…

centos7安装hadoop 单机版

1.解压 &#xff08;1&#xff09;将hadoop压缩包复制到/opt/software路径下 &#xff08;2&#xff09;解压hadoop到/opt/module目录下 [rootkb135 software]# tar -zxvf hadoop-3.1.3.tar.gz -C /opt/module/ &#xff08;3&#xff09;修改hadoop属主和属组 [rootkb135 m…

VirtualBox虚拟机安装Centos7详细教程图解

一、前期准备工作&#xff1a; 1、虚拟机下载 VirtualBox版本&#xff1a;7.0.6 下载 VirtualBox 的下载页面&#xff1a;https://www.virtualbox.org/wiki/Downloads VMWare 虚拟机软件&#xff08;收费的&#xff0c;要使用请购买正版软件&#xff09;的官网&#xff1a;…

Springboot中sharding-jdbc的API模式并使用自定义算法

Springboot中sharding-jdbc的API模式并使用自定义算法 可配合AbstractRoutingData使用切换数据源 程序用到了AbstractRoutingData来切换数据源&#xff08;数据源是自定义的格式编写并没有用springboot的自动装配的格式写&#xff09;&#xff0c;但是又用到sharding-jdbc进行…

pycharm添加虚拟环境以及虚拟环境安装pytorch

file、settings、interpreter、add interpreter、add local interpreter 记住不要勾选inherit&#xff0c;不然会把主环境的东西继承到虚拟环境。 创建前可以先点existing看看有没有已经建好的虚拟环境 有的时候pycharm有问题&#xff0c;创建了虚拟环境没有显示。找一个.py文…

交通网络分析性能再升级,SuperMap iServer新增开启SSC分析模型

导语 SSC分析模型&#xff0c;全名SuperMap Short Cut&#xff0c;底层采用Contraction Hierarchies&#xff08;简称CH&#xff09;算法&#xff0c;该算法旨在通过对图形进行预处理和优化来降低最佳路径分析的时间复杂度。SuperMap iServer 11i&#xff08;2023&#xff09;&…

scikit-learn中OneHotEncoder用法

One-Hot编码&#xff0c;又称为一位有效编码&#xff0c;是分类变量作为二进制向量的表示。这首先要求将分类值映射到整数值&#xff0c;然后&#xff0c;每个整数值被表示为二进制向量&#xff0c;将整数索引标记为1&#xff0c;其余都标为0。 OneHotEncoder()常用参数解释 …

【Winform学习笔记(十)】TextBox文本框控件 光标定位到文本末尾的方法

TextBox文本框控件 光标定位到文本末尾的方法 前言正文1、具体代码2、使用示例 前言 本文中主要介绍实现 TextBox 文本框控件的光标定位到文本末尾的方法. 正文 1、具体代码 /// <summary> /// TextBox文本框控件 光标定位到文本末尾的方法 /// </summary> ///…

next.js报错点

next.js报错点 1.类型“{ children: ReactNode; }”与类型“IntrinsicAttributes”不具有相同的属性。2. 不能将类型“void[]”分配给类型“ReactNode”&#xff1f;3.useRouter only works in Client Components. Add the "use client" directive at the top of the…

IDEA的maven想显示层级关系,而非平级

新版和旧版的IDEA的位置不一样&#xff0c;2023.2.1的版本在右上角的“” 这个位置 如图所示&#xff1a; 然后点击按模块分组&#xff1a;

[Stable Diffusion教程] 第一课 原理解析+配置需求+应用安装+基本步骤

第一课 原理解析配置需求应用安装基本步骤 本次内容记录来源于B站的一个视频 以下是自己安装过程中整理的问题及解决方法&#xff1a; 问题&#xff1a;stable-diffusion-webui启动No Python at ‘C:\xxx\xxx\python.exe‘ 解答&#xff1a;打开webui.bat 把 if not de…

从零开始的自动化测试框架——Web篇01

Selenium 谈到web自动化&#xff0c;逃不开的一定会是Selenium。这是最为主流&#xff0c;也是最广为人知的一项web自动化产物。但目前业内web自动化其实主要分为以下方向&#xff1a; Selenium&#xff0c;核心主流自动化技术&#xff0c;功能齐全&#xff0c;一般是搭配web…

【算法日志】动态规划刷题:01背包问题,多重背包问题(day37,day38)

代码随想录刷题60Day 目录 前言 目标和&#xff08;01背包&#xff09; 一和零&#xff08;01背包&#xff09; 零钱兑换&#xff08;多重背包&#xff09; 排列总和&#xff08;多重背包&#xff09; 前言 这两天都是背包问题&#xff0c;其中的01背包的一些应用问题需要…

迈步从头越——同为科技(TOWE)22周年庆典活动圆满举行

8月23日&#xff0c;同为科技&#xff08;TOWE&#xff09;迎来公司成立22周年纪念日&#xff0c;这是属于TOWE品牌的又一里程碑时刻。从2001到2023&#xff0c;从产品到品牌&#xff0c;从技术到服务&#xff0c;同为科技&#xff08;TOWE&#xff09;始终秉承着专注于终端设备…

电子合同9问9答,君子签为您答疑解惑

随着互联网的发展&#xff0c;人们交易行为的改变、电子合同的推广&#xff0c;买卖合同、租房合同、采购合同、劳动合同、招投标文件等都开始在线上签署。签署过程中&#xff0c;君子签针对大家的一些疑虑&#xff0c;整理了相关解答。 1、 签电子合同时&#xff0c;如何确…

MMEdu库的下载和安装(Python版)

为保证程序的顺利运行&#xff0c;这里提供MMEdu库的下载安装包。 OpenXLabEdu开源资源&#xff1a;OpenXLabEdu帮助文档 — OpenXLabEdu 文档 现MMEdu一键安装包已升级为XEdu一键安装包&#xff0c;下文提到的MMEdu文件&#xff0c;应理解为XEdu一键安装包。 第一步&#x…

融合算法综述

融合算法 前言一、概念二、原理三、融合的先决条件四、融合分类4.1、前融合和后融合4.2 、数据级融合、特征级融合和决策级融合 五、典型融合算法 多传感器信息融合&#xff08;Multi-sensor Information Fusion,MSIF&#xff09;&#xff1a;利用计算机技术将来自多传感器或多…

小红书旅游种草笔记:如何打造爆款内容,吸引万千粉丝?

随着社交媒体的飞速发展&#xff0c;小红书作为国内知名的分享平台&#xff0c;已经成为越来越多人展示生活、交流心得的阵地。在众多分类中&#xff0c;旅游种草笔记尤为热门&#xff0c;吸引了大量用户关注。那么&#xff0c;如何在这片竞争激烈的红海中&#xff0c;打造出爆…