欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java Retrofit源碼層深入分析

 更新時間:2023年01月13日 09:28:51   作者:AllenC6  
這篇文章主要介紹了Java Retrofit源碼層分析,Retrofit是一個RESTful的HTTP網(wǎng)絡(luò)請求框架的封裝,網(wǎng)絡(luò)請求的工作本質(zhì)上是OkHttp完成,而Retrofit僅負責(zé)網(wǎng)絡(luò)請求接口的封裝

提醒:看的過程一定要自己點開源碼,跟著一步步走,只看容易懵逼

一、自己對Retrofit的理解

Retrofit的中文翻譯是改造,改造什么呢?

我認為是對OkHttp的使用、RxJava的使用進行改造。

具體體現(xiàn)在哪里?

在使用OkHttp請求前:

1.用注解統(tǒng)一配置網(wǎng)絡(luò)請求頭和請求參數(shù)

2.通過動態(tài)代理統(tǒng)一獲取注解的請求頭和請求參數(shù)然后一致組裝適配成請求的request,交給okHttp進行請求

使用OkHttp結(jié)果返回后:

1.線程切換

線程切換分為兩種一種是用Retrofit默認的,另一種是使用RxJava

2.把返回的數(shù)據(jù)json適配成javabean

簡單總結(jié),Retrofit就是為了讓OkHttp、RxJava使用的更加簡潔的一個封裝

二、Retrofit的簡單使用

沒有添加RxJava

class RetrofitActivity : AppCompatActivity() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        //初始化一個Retrofit對象
        val retrofit = Retrofit.Builder()
            .baseUrl("https://api.github.com/")
            .addConverterFactory(GsonConverterFactory.create())
            .client(OkHttpClientProvider.client()) // 這個OkHttpClientProvider.client()是我自己封裝的,就是簡單提供一個OkHttpClient
            .build()
        //創(chuàng)建出GitHubApiService這個GitHubApiService的動態(tài)代理對象service 
        val service = retrofit.create(GitHubApiService::class.java)
        //返回一個Okhttp的請求 Call 對象repos 
        val repos = service.listRepos("octocat")
        //調(diào)用 enqueue 方法在回調(diào)方法里處理結(jié)果
        repos.enqueue(object : Callback<List<Repo>?> {
            override fun onFailure(call: Call<List<Repo>?>, t: Throwable) {
                                t.printStackTrace()
            }
            override fun onResponse(call: Call<List<Repo>?>, response: Response<List<Repo>?>) {
                "response.code() = ${response.code()}".logE()
            }
        })
    }
}
//自己定義的 API 請求接口
interface GitHubApiService {
    @GET("users/{user}/repos")
    fun listRepos(@Path("user") user: String?): Call<List<Repo>>
}

三、請求前Retrofit所做的工作

以統(tǒng)一的方式為網(wǎng)絡(luò)請求準備各種參數(shù)

(1).Retrofit的創(chuàng)建,把BaseUrl、OkHttpClient和Gson的實例保存在Retrofit對象中

Retrofit.Builder()的build()方法:

    public Retrofit build() {
      if (baseUrl == null) {
        throw new IllegalStateException("Base URL required.");
      }
      okhttp3.Call.Factory callFactory = this.callFactory;
      if (callFactory == null) {
        callFactory = new OkHttpClient();
      }
      Executor callbackExecutor = this.callbackExecutor;
      if (callbackExecutor == null) {
        callbackExecutor = platform.defaultCallbackExecutor();
      }
      // Make a defensive copy of the adapters and add the default Call adapter.
      List<CallAdapter.Factory> adapterFactories = new ArrayList<>(this.adapterFactories);
      adapterFactories.add(platform.defaultCallAdapterFactory(callbackExecutor));
      // Make a defensive copy of the converters.
      List<Converter.Factory> converterFactories = new ArrayList<>(this.converterFactories);
      return new Retrofit(callFactory, baseUrl, converterFactories, adapterFactories,
          callbackExecutor, validateEagerly);
    }

首先看這個callFactory,因為我們已經(jīng).client(OkHttpClientProvider.client())方法設(shè)置了我們的okhttpclient,所以這里callFactory不為null:

    public Builder client(OkHttpClient client) {
      return callFactory(checkNotNull(client, "client == null"));
    }
    public Builder callFactory(okhttp3.Call.Factory factory) {
      this.callFactory = checkNotNull(factory, "factory == null");
      return this;
    }

我們沒有添加CallbackExecutor(這個看名字就知道是用來處理返回值的),所以這里用的是默認的:

Executor callbackExecutor = this.callbackExecutor;
if (callbackExecutor == null) {
callbackExecutor = platform.defaultCallbackExecutor();
}

記住這個CallbackExecutor,后面會用到

接下來重點看:

List<CallAdapter.Factory> adapterFactories = new ArrayList<>(this.adapterFactories);
adapterFactories.add(platform.defaultCallAdapterFactory(callbackExecutor));

這個很重要,我們沒有調(diào)用addCallAdapterFactory方法另外添加AdapterFactory,所以這里第一句的this.adapterFactories是一個空集合,之后調(diào)用add方法添加了一個默認的platform.defaultCallAdapterFactory(callbackExecutor),這個很重要需要記住。

接下來把我們的Gson解析器添加到了converterFactories中

最后創(chuàng)建了Retrofit對象,把上面的對象都維護在了Retrofit中,這些都是網(wǎng)絡(luò)請求需要的工具。

(2).用注解來表示請求頭和請求的參數(shù)等

如果不知道注解的本質(zhì)的小伙伴點擊傳送門:

Java 注解

看一下 注解的具體使用:

    @GET("users/{user}/repos")
    fun listRepos(@Path("user") user: String?): Call<List<Repo>>

這個就是注解提供的請求方式、請求地址、請求參數(shù)等,這個是接口類中的一個方法,后面這個接口類會被動態(tài)代理來代理這個接口。

(3).用動態(tài)代理來把Retrofit和代理接口的方法上的請求信息,組裝成一個OkHttp的請求

如果不知道動態(tài)代理本質(zhì)的小伙伴點擊傳送門:

Java動態(tài)代理與靜態(tài)代理

使用動態(tài)代理,組裝請求的代碼:

        //創(chuàng)建出GitHubApiService這個GitHubApiService的動態(tài)代理對象service 
        val service = retrofit.create(GitHubApiService::class.java)
        //返回一個Okhttp的請求 Call 對象repos 
        val repos = service.listRepos("octocat")

看Retrofit的create方法:

  public <T> T create(final Class<T> service) {
    Utils.validateServiceInterface(service);
    if (validateEagerly) {
      eagerlyValidateMethods(service);
    }
    return (T) Proxy.newProxyInstance(service.getClassLoader(), new Class<?>[] { service },
        new InvocationHandler() {
          private final Platform platform = Platform.get();
          @Override public Object invoke(Object proxy, Method method, @Nullable Object[] args)
              throws Throwable {
......
            ServiceMethod<Object, Object> serviceMethod =
                (ServiceMethod<Object, Object>) loadServiceMethod(method);
            OkHttpCall<Object> okHttpCall = new OkHttpCall<>(serviceMethod, args);
            return serviceMethod.callAdapter.adapt(okHttpCall);
          }
        });
  }

可以看到就是創(chuàng)建了一個動態(tài)代理,動態(tài)代理的作用就是代理類的任何一個方法的調(diào)用,都會走這個動態(tài)代理的invoke方法,即service.listRepos("octocat")的調(diào)用會走這個invoke方法。

下面我們具體分析,動態(tài)代理的invoke方法如何把各種請求的信息,組裝成一個OkHttpCall的:

ServiceMethod<Object, Object> serviceMethod =

(ServiceMethod<Object, Object>) loadServiceMethod(method);

這個ServiceMethod是一個重點,它是真正執(zhí)行把所有的請求信息,拼成一個Okhttp的請求Call的類:

  ServiceMethod<?, ?> loadServiceMethod(Method method) {
    ServiceMethod<?, ?> result = serviceMethodCache.get(method);
    if (result != null) return result;
    synchronized (serviceMethodCache) {
      result = serviceMethodCache.get(method);
      if (result == null) {
        result = new ServiceMethod.Builder<>(this, method).build();
        serviceMethodCache.put(method, result);
      }
    }
    return result;
  }

這里有一個緩存,serviceMethodCache是一個ConcurrentHashMap來緩存serviceMethod。

然后是ServiceMethod的創(chuàng)建,主要看build方法:

    public ServiceMethod build() {
      callAdapter = createCallAdapter();
      responseType = callAdapter.responseType();
......
      responseConverter = createResponseConverter();
      for (Annotation annotation : methodAnnotations) {
        parseMethodAnnotation(annotation);
      }
......
      int parameterCount = parameterAnnotationsArray.length;
      parameterHandlers = new ParameterHandler<?>[parameterCount];
      for (int p = 0; p < parameterCount; p++) {
        Type parameterType = parameterTypes[p];
......
        Annotation[] parameterAnnotations = parameterAnnotationsArray[p];
......
        parameterHandlers[p] = parseParameter(p, parameterType, parameterAnnotations);
      }
......
      return new ServiceMethod<>(this);
    }

可以看出來,這里都在把之前的注解、請求參數(shù)、請求地址等解析出來,存到serviceMethod中:

  ServiceMethod(Builder<R, T> builder) {
    this.callFactory = builder.retrofit.callFactory();
    this.callAdapter = builder.callAdapter;
    this.baseUrl = builder.retrofit.baseUrl();
    this.responseConverter = builder.responseConverter;
    this.httpMethod = builder.httpMethod;
    this.relativeUrl = builder.relativeUrl;
    this.headers = builder.headers;
    this.contentType = builder.contentType;
    this.hasBody = builder.hasBody;
    this.isFormEncoded = builder.isFormEncoded;
    this.isMultipart = builder.isMultipart;
    this.parameterHandlers = builder.parameterHandlers;
  }

這個ServiceMethod大家現(xiàn)在不用細看它,知道它是解析了我們之前統(tǒng)一用注解格式表示的請求信息,后保存在了自己serviceMethoed的對象中。后面用到的時候,大家自然能體會到。

然后回到動態(tài)代理的invoke方法:

OkHttpCall<Object> okHttpCall = new OkHttpCall<>(serviceMethod, args);

這個OkHttpCall就是具體發(fā)出請求的類,它持有serviceMethod,這里我們先不分析它,繼續(xù)往下看:

return serviceMethod.callAdapter.adapt(okHttpCall);

這里就和我們調(diào)用的地方聯(lián)系起來了,這個return的是一個okHttp的call:

        //創(chuàng)建出GitHubApiService這個GitHubApiService的動態(tài)代理對象service 
        val service = retrofit.create(GitHubApiService::class.java)
        //返回一個Okhttp的請求 Call 對象repos ???注意這里變化了
        val repos = serviceMethod.callAdapter.adapt(okHttpCall)

注意上面service.listRepos("octocat")等于動態(tài)代理調(diào)用invoke方法的返回值即:

serviceMethod.callAdapter.adapt(okHttpCall)

那我們重點分析:

serviceMethod.callAdapter.adapt(okHttpCall)

首先這個serviceMethod.callAdapter是什么:

callAdapter是在serviceMethod中的build方法中賦值的:

    public ServiceMethod build() {
      callAdapter = createCallAdapter();
......
    }
    private CallAdapter<T, R> createCallAdapter() {
......
      try {
        //noinspection unchecked
        return (CallAdapter<T, R>) retrofit.callAdapter(returnType, annotations);
      } catch (RuntimeException e) { // Wide exception range because factories are user code.
        throw methodError(e, "Unable to create call adapter for %s", returnType);
      }
    }
  public CallAdapter<?, ?> callAdapter(Type returnType, Annotation[] annotations) {
    return nextCallAdapter(null, returnType, annotations);
  }
  public CallAdapter<?, ?> nextCallAdapter(@Nullable CallAdapter.Factory skipPast, Type returnType,
      Annotation[] annotations) {
......
    int start = adapterFactories.indexOf(skipPast) + 1;
    for (int i = start, count = adapterFactories.size(); i < count; i++) {
      CallAdapter<?, ?> adapter = adapterFactories.get(i).get(returnType, annotations, this);
      if (adapter != null) {
        return adapter;
      }
    }
......
  }

上面是整個callAdapter的調(diào)用鏈,最后是:

adapterFactories.get(i).get(returnType, annotations, this);

這句確定的CallAdapter:

adapterFactoris之前我們提到過要記住的,在Retrofit的build方法中:

List<CallAdapter.Factory> adapterFactories = new ArrayList<>(this.adapterFactories);
adapterFactories.add(platform.defaultCallAdapterFactory(callbackExecutor));

上面分析了,adapterFactories中只有platform.defaultCallAdapterFactory(callbackExecutor),這一個CallAdapter,adapterFactories.get(i)就是獲取platform.defaultCallAdapterFactory(callbackExecutor)的。

接下來看platform.defaultCallAdapterFactory(callbackExecutor)是什么:

    @Override CallAdapter.Factory defaultCallAdapterFactory(@Nullable Executor callbackExecutor) {
      if (callbackExecutor == null) throw new AssertionError();
      return new ExecutorCallAdapterFactory(callbackExecutor);
    }

所以我們得到結(jié)論,adapterFactories.get(i)是new ExecutorCallAdapterFactory(callbackExecutor)

注意我們要的是adapterFactories.get(i).get(returnType, annotations, this),所以我們要看ExecutorCallAdapterFactory的get方法:

  public CallAdapter<?, ?> get(Type returnType, Annotation[] annotations, Retrofit retrofit) {
    if (getRawType(returnType) != Call.class) {
      return null;
    }
    final Type responseType = Utils.getCallResponseType(returnType);
    return new CallAdapter<Object, Call<?>>() {
      @Override public Type responseType() {
        return responseType;
      }
      @Override public Call<Object> adapt(Call<Object> call) {
        return new ExecutorCallbackCall<>(callbackExecutor, call);
      }
    };
  }

返回的是:

new CallAdapter<Object, Call<?>>() {
      @Override public Type responseType() {
        return responseType;
      }
      @Override public Call<Object> adapt(Call<Object> call) {
        return new ExecutorCallbackCall<>(callbackExecutor, call);
      }
    };

所以:

serviceMethod.callAdapter是adapterFactories.get(i).get(returnType, annotations, this)是上面這段代碼new CallAdapter

那我們要分析的serviceMethod.callAdapter.adapt(okHttpCall),所以我們要的是new CallAdapter的adaptger方法:它返回的是new ExecutorCallbackCall<>(callbackExecutor, call)

這里要注意,這兩個參數(shù):

callbackExecutor是我們上面提到過的處理返回值的:

Executor callbackExecutor = this.callbackExecutor;
if (callbackExecutor == null) {
callbackExecutor = platform.defaultCallbackExecutor();
}

call是:

OkHttpCall<Object> okHttpCall = new OkHttpCall<>(serviceMethod, args);

好現(xiàn)在我們知道了:

        //創(chuàng)建出GitHubApiService這個GitHubApiService的動態(tài)代理對象service 
        val service = retrofit.create(GitHubApiService::class.java)
        //返回一個Okhttp的請求 Call 對象repos ???注意這里變化了
        val repos = serviceMethod.callAdapter.adapt(okHttpCall)

其實是:

        //創(chuàng)建出GitHubApiService這個GitHubApiService的動態(tài)代理對象service 
        val service = retrofit.create(GitHubApiService::class.java)
        //返回一個Okhttp的請求 Call 對象repos ???注意這里變化了
        Call repos = new ExecutorCallbackCall<>(callbackExecutor, call)

到這里我們分析了第三小節(jié)的題目:如何用動態(tài)代理來把Retrofit和代理接口的方法上的請求信息組裝成一個OkHttp的請求

(4).發(fā)起請求的調(diào)用鏈細節(jié)

我們知道發(fā)起請求的代碼:

        val service = retrofit.create(GitHubApiService::class.java)
        //返回一個Okhttp的請求 Call 對象repos 
        val repos = service.listRepos("octocat")
        //調(diào)用 enqueue 方法在回調(diào)方法里處理結(jié)果
        repos.enqueue(object : Callback<List<Repo>?> {
            override fun onFailure(call: Call<List<Repo>?>, t: Throwable) {
                                t.printStackTrace()
            }
            override fun onResponse(call: Call<List<Repo>?>, response: Response<List<Repo>?>) {
                "response.code() = ${response.code()}".logE()
            }
        })

這里的repos已經(jīng)分析了是ExecutorCallbackCall,調(diào)用的enqueue方法,接下來我們來看調(diào)用鏈ExecutorCallbackCall的enqueue方法:

    final Executor callbackExecutor;
    final Call<T> delegate;
    ExecutorCallbackCall(Executor callbackExecutor, Call<T> delegate) {
      this.callbackExecutor = callbackExecutor;
      this.delegate = delegate;
    }
 @Override public void enqueue(final Callback<T> callback) {
      checkNotNull(callback, "callback == null");
      delegate.enqueue(new Callback<T>() {
        @Override public void onResponse(Call<T> call, final Response<T> response) {
          callbackExecutor.execute(new Runnable() {
            @Override public void run() {
              if (delegate.isCanceled()) {
                // Emulate OkHttp's behavior of throwing/delivering an IOException on cancellation.
                callback.onFailure(ExecutorCallbackCall.this, new IOException("Canceled"));
              } else {
                callback.onResponse(ExecutorCallbackCall.this, response);
              }
            }
          });
        }
        @Override public void onFailure(Call<T> call, final Throwable t) {
          callbackExecutor.execute(new Runnable() {
            @Override public void run() {
              callback.onFailure(ExecutorCallbackCall.this, t);
            }
          });
        }
      });
    }

首先我們已經(jīng)知道構(gòu)造方法的兩個參數(shù)是什么:

delegate 是OkHttpCall

callbackExecutor是platform.defaultCallbackExecutor();

然后我們繼續(xù)看delegate即OkHttpCall的enqueue方法(上面提到的掠過的OkhttpCall這里用到了):

  @Override public void enqueue(final Callback<T> callback) {
    checkNotNull(callback, "callback == null");
    okhttp3.Call call;
    Throwable failure;
    synchronized (this) {
      if (executed) throw new IllegalStateException("Already executed.");
      executed = true;
      call = rawCall;
      failure = creationFailure;
      if (call == null && failure == null) {
        try {
          //?? 重點
          call = rawCall = createRawCall();
        } catch (Throwable t) {
          failure = creationFailure = t;
        }
      }
    }
    if (failure != null) {
      callback.onFailure(this, failure);
      return;
    }
    if (canceled) {
      call.cancel();
    }
    //?? 重點
    call.enqueue(new okhttp3.Callback() {
      @Override public void onResponse(okhttp3.Call call, okhttp3.Response rawResponse)
          throws IOException {
        Response<T> response;
        try {
          response = parseResponse(rawResponse);
        } catch (Throwable e) {
          callFailure(e);
          return;
        }
        //?? 重點
        callSuccess(response);
      }
      @Override public void onFailure(okhttp3.Call call, IOException e) {
        try {
          callback.onFailure(OkHttpCall.this, e);
        } catch (Throwable t) {
          t.printStackTrace();
        }
      }
      private void callFailure(Throwable e) {
        try {
          callback.onFailure(OkHttpCall.this, e);
        } catch (Throwable t) {
          t.printStackTrace();
        }
      }
      private void callSuccess(Response<T> response) {
        try {
          callback.onResponse(OkHttpCall.this, response);
        } catch (Throwable t) {
          t.printStackTrace();
        }
      }
    });
  }

上面我們標(biāo)記了三個重點:

1.第一個重點call = rawCall = createRawCall();

  private okhttp3.Call createRawCall() throws IOException {
    Request request = serviceMethod.toRequest(args);
    okhttp3.Call call = serviceMethod.callFactory.newCall(request);
    if (call == null) {
      throw new NullPointerException("Call.Factory returned null.");
    }
    return call;
  }

這里用到了上面提到了serviceMethod,之前我們已經(jīng)總結(jié),它是保存了所有請求需要的信息和請求需要的工具,這里就要使用了:

Request request = serviceMethod.toRequest(args);
okhttp3.Call call = serviceMethod.callFactory.newCall(request);

就這兩句,先把保存的請求頭和參數(shù)等信息,拼湊成OkHttpClient需要的Request,然后調(diào)用serviceMethod.callFactory.newCall(request),這個serviceMethod.callFactory就是我們之前設(shè)置的OkHttpClient,調(diào)用的它的newCall方法,得到一個OkHttp的Call請求對象,其實是RealCall對象

2.第二個重點call.enqueue(new okhttp3.Callback() {}

這是調(diào)用的OkHttpClient的enqueue方法,所以我們之前總結(jié)Retrofit就是對OkHttp請求之前和之后的工作的封裝,使其有一致、簡單、可復(fù)用等等。

3.第三個重點callSuccess(response);

     private void callSuccess(Response<T> response) {
        try {
          callback.onResponse(OkHttpCall.this, response);
        } catch (Throwable t) {
          t.printStackTrace();
        }
      }

callback.onResponse(OkHttpCall.this, response);,這個callback是之前delegate調(diào)用enqueue方法時的參數(shù):

    final Executor callbackExecutor;
    final Call<T> delegate;
    ExecutorCallbackCall(Executor callbackExecutor, Call<T> delegate) {
      this.callbackExecutor = callbackExecutor;
      this.delegate = delegate;
    }
 @Override public void enqueue(final Callback<T> callback) {
      checkNotNull(callback, "callback == null");
      delegate.enqueue(new Callback<T>() {
        @Override public void onResponse(Call<T> call, final Response<T> response) {
          callbackExecutor.execute(new Runnable() {
            @Override public void run() {
              if (delegate.isCanceled()) {
                // Emulate OkHttp's behavior of throwing/delivering an IOException on cancellation.
                callback.onFailure(ExecutorCallbackCall.this, new IOException("Canceled"));
              } else {
                callback.onResponse(ExecutorCallbackCall.this, response);
              }
            }
          });
        }
        @Override public void onFailure(Call<T> call, final Throwable t) {
          callbackExecutor.execute(new Runnable() {
            @Override public void run() {
              callback.onFailure(ExecutorCallbackCall.this, t);
            }
          });
        }
      });
    }

注意這個callback是delegate.enqueue方法的參數(shù),不是外層enqueue方法的callback,所以這個回調(diào)執(zhí)行的這個callback的onResponse方法代碼:

callbackExecutor.execute(new Runnable() {
  @Override public void run() {
    if (delegate.isCanceled()) {
      // Emulate OkHttp's behavior of throwing/delivering an IOException on cancellation.
      callback.onFailure(ExecutorCallbackCall.this, new IOException("Canceled"));
    } else {
      callback.onResponse(ExecutorCallbackCall.this, response);
    }
  }
});

callbackExecutor之前提到過是platform.defaultCallbackExecutor():

    @Override public Executor defaultCallbackExecutor() {
      return new MainThreadExecutor();
    }
    static class MainThreadExecutor implements Executor {
      private final Handler handler = new Handler(Looper.getMainLooper());
      @Override public void execute(Runnable r) {
        handler.post(r);
      }
    }

可以看到就是把Runnable來交給handler執(zhí)行,實現(xiàn)的線程切換

到這里發(fā)起請求的調(diào)用邏輯分析清楚了。

到此這篇關(guān)于Java Retrofit源碼層深入分析的文章就介紹到這了,更多相關(guān)Java Retrofit內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 詳解Java中的pinpoint1.8.5安裝及使用指南

    詳解Java中的pinpoint1.8.5安裝及使用指南

    pinpoint是開源在github上的一款A(yù)PM監(jiān)控工具,它是用Java編寫的,用于大規(guī)模分布式系統(tǒng)監(jiān)控。這篇文章主要介紹了pinpoint1.8.5安裝及使用指南,非常不錯,具有一定的參考借鑒價值,需要的朋友可以參考下
    2019-10-10
  • 三種Java自定義DNS解析器方法與實踐

    三種Java自定義DNS解析器方法與實踐

    這篇文章主要分享三種Java自定義DNS解析器方法與實踐,對于高性能的測試機(54C96G * 3)而言,可任意通過自定義Java DNS解析器來實現(xiàn)接口請求,下文內(nèi)容的實現(xiàn),需要的小伙伴可以參考一下
    2022-02-02
  • SpringBoot項目打包war包時無法運行問題的解決方式

    SpringBoot項目打包war包時無法運行問題的解決方式

    在開發(fā)工程中,使用啟動類啟動能夠正常啟動并測試,下面這篇文章主要給大家介紹了關(guān)于SpringBoot項目打包war包時無法運行問題的解決方式,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下
    2022-06-06
  • Java文件基本操作總結(jié)

    Java文件基本操作總結(jié)

    今天給大家?guī)淼氖顷P(guān)于Java基礎(chǔ)的相關(guān)知識,文章圍繞著Java文件操作展開,文中有非常詳細的介紹及代碼示例,需要的朋友可以參考下
    2021-06-06
  • java如何從地址串中解析提取省市區(qū)(完美匹配中國所有地址)

    java如何從地址串中解析提取省市區(qū)(完美匹配中國所有地址)

    這篇文章主要給大家介紹了關(guān)于java如何從地址串中解析提取省市區(qū)的相關(guān)資料,通過這個方法可以完美匹配中國所有地址,文中通過示例代碼介紹的非常詳細,需要的朋友可以參考下
    2022-07-07
  • Java?Apache?common-pool對象池介紹

    Java?Apache?common-pool對象池介紹

    這篇文章主要介紹了Java Apache?common-pool對象池介紹,文章通過圍繞主題展開詳細的內(nèi)容介紹,具有一定的參考價值,感興趣的小伙伴可以參考一下
    2022-09-09
  • Spring MVC請求參數(shù)的傳遞方式

    Spring MVC請求參數(shù)的傳遞方式

    Spring MVC是一種基于Model-View-Controller(MVC)設(shè)計模式的輕量級Web框架,用于Java應(yīng)用程序的開發(fā),在處理HTTP請求時,Spring MVC會涉及到請求參數(shù)的傳遞,本文給大家介紹了Spring MVC請求參數(shù)的傳遞方式,需要的朋友可以參考下
    2024-10-10
  • Java自定義異常與異常使用的最佳方式

    Java自定義異常與異常使用的最佳方式

    這篇文章主要介紹了Java自定義異常與異常使用的最佳方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-08-08
  • java swing 創(chuàng)建一個簡單的QQ界面教程

    java swing 創(chuàng)建一個簡單的QQ界面教程

    這篇文章主要介紹了java swing 創(chuàng)建一個簡單的QQ界面教程,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-09-09
  • Spring Cache + Caffeine的整合與使用示例詳解

    Spring Cache + Caffeine的整合與使用示例詳解

    對于一些項目里需要對數(shù)據(jù)庫里的某些數(shù)據(jù)一直重復(fù)請求的,且這些數(shù)據(jù)基本是固定的,在這種情況下,可以借助簡單使用本地緩存來緩存這些數(shù)據(jù),本文介紹一下Spring Cache和Caffeine的使用,感興趣的朋友一起看看吧
    2023-12-12

最新評論