目录
项目结构
retrofit 使用
Retrofit #create
ServiceMethod #parseAnnotations
HttpServiceMethod#parseAnnotations
第二种 非Kotlin协程情况
DefaultCallAdapterFactory#get
第一种 Kotlin协程情况
总结
项目结构
把源码 clone 下来 , 可以看到 retrofit 整体结构如下
图 http包目录下就是一些http协议常用接口 , 比如 请求方法 url , 请求体, 请求行 之类的
retrofit 使用
把retrofit使用作为分析的切入口吧 , retrofit单元测试使用如下
public final class BasicCallTest {
@Rule public final MockWebServer server = new MockWebServer();
interface Service {
@GET("/") Call getBody();
}
@Test public void responseBody() throws IOException {
Retrofit retrofit = new Retrofit.Builder()
.baseUrl(server.url("/"))
.build();
Service example = retrofit.create(Service.class);
server.enqueue(new MockResponse().setBody("1234"));
Response response = example.getBody().execute();
assertEquals("1234", response.body().string());
}
}
Retrofit 构建 , 以构建者模式构建出Retrofit
可以看到builer可以配置baseUrl , 回调线程池 , 还有一些适配器的工厂 , 这些适配器的作用后面说
Retrofit #create
从create 方法开始分析 , 跟进看下create 方法
public T create(final Class service) {
validateServiceInterface(service);
return (T)
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();
return platform.isDefaultMethod(method)
? platform.invokeDefaultMethod(method, service, proxy, args)
: loadServiceMethod(method).invoke(args);
}
});
}
service 是请求的接口的class , 动态代理只能是接口 , 所以validateServiceInterface 先验证是不是接口 , 不是接口则抛异常。
method.getDeclaringClass() 获取声明类的Class。
比如 A类 有个method , method.getDeclaringClass() 返回为A.class , 如果method声明类是Object.class 则直接method.invoke , 往下执行毫无意义。
Platform#get()会根据当前平台获取Platform 。
有点类似状态模式思想 , 根据当前的平台选择合适的子类
public boolean isDefaultMethod(Method method) {
return method.isDefault();
}
isDefault , 在接口类型中以default关键字声明 则返回true, 比如
interface InterfaceWithDefault {
void firstMethod();
default void newMethod() {
System.out.println("newMethod");
}
}
所以此处会返回 false 接着调用 loadServiceMethod。
ServiceMethod #parseAnnotations
跟进ServiceMethod #parseAnnotations
static ServiceMethod parseAnnotations(Retrofit retrofit, Method method) {
RequestFactory requestFactory = RequestFactory.parseAnnotations(retrofit, method);
return HttpServiceMethod.parseAnnotations(retrofit, method, requestFactory);
}
根据当前的方法信息构建出RequestFactory , 然后把具体实现细节交给HttpServiceMethod 处理 , HttpServiceMethod 继承自ServiceMethod , 有三个子类 。
我们在Api.class定义的方法 , 解析并不是由HttpServiceMethod 完成 , 而是由RequestFactory去处理的 , 比如解析方法的注解。
更多的解析方法如下
方法解析的细节就不说了 , 继续看RequestFactory 这个类 , 这个类的作用难道就是负责方法信息的解析 , 感觉和名字不太符合 , RequestFactory 顾名思义应该是用来构建Request的工厂 , 果不其然内部还有个 create 方法 , 用来构建okhttp3.Request的
//RequestFactory #create
okhttp3.Request create(Object[] args) throws IOException {
return requestBuilder.get().tag(Invocation.class, new Invocation(method, argumentList)).build();
}
就只有这一个create方法 , 难道retrofit 就只能使用okhttp来负责网络请求 ? 答案是肯定的 , 从最开始的 loadServiceMethod(method).invoke(args)也可以看出 , 方法里面只构建出OkHttpCall 没提供api可以让我们切换到其他的网络请求库。
但是 , Call 又抽象成接口的形式 ,如下, 这么做的目的可能是以后便于框架的维护
沿途风景再美丽 , 也要回到主线路 , 继续分析 HttpServiceMethod#parseAnnotations
HttpServiceMethod#parseAnnotations
这个方法太长 , 贴关键代码吧
static HttpServiceMethod parseAnnotations(
Retrofit retrofit, Method method, RequestFactory requestFactory) {
boolean isKotlinSuspendFunction = requestFactory.isKotlinSuspendFunction;
boolean continuationWantsResponse = false;
boolean continuationBodyNullable = false;
boolean continuationIsUnit = false;
Annotation[] annotations = method.getAnnotations();
Type adapterType;
if (isKotlinSuspendFunction) {
Type[] parameterTypes = method.getGenericParameterTypes();
Type responseType =
Utils.getParameterLowerBound(
0, (ParameterizedType) parameterTypes[parameterTypes.length - 1]);
if (getRawType(responseType) == Response.class && responseType instanceof ParameterizedType) {
continuationWantsResponse = true;
}
} else {
//非kt 协程情况
adapterType = method.getGenericReturnType();
}
CallAdapter callAdapter =
createCallAdapter(retrofit, method, adapterType, annotations);
Type responseType = callAdapter.responseType();
Converter responseConverter =
createResponseConverter(retrofit, method, responseType);
okhttp3.Call.Factory callFactory = retrofit.callFactory;
if (!isKotlinSuspendFunction) {
//非kt 协程情况
return new CallAdapted<>(requestFactory, callFactory, responseConverter, callAdapter);
} else if (continuationWantsResponse) {
return (HttpServiceMethod)
new SuspendForResponse<>(
requestFactory,
callFactory,
responseConverter,
(CallAdapter>) callAdapter);
} else {
return (HttpServiceMethod)
new SuspendForBody<>(
requestFactory,
callFactory,
responseConverter,
(CallAdapter>) callAdapter,
continuationBodyNullable,
continuationIsUnit);
}
}
构建出HttpServiceMethod分两种情况 :
第二种 非Kotlin协程情况
第一种情况稍许复杂 , 先分析第二种
adapterType = method.getGenericReturnType();
@GET("/") Call getBody();
如果是上面代码 , method.getGenericReturnType() = Call , 然后根据方法的返回值类型 / 方法注解信息 , 构建出CallAdapter 。
createCallAdapter() 方法会使用 CallAdapter.Factory 构建CallAdapter , 因为初始化retrofit的时候没有配置CallAdapter.Factory , 所以会使用默认的DefaultCallAdapterFactory。
最终会进入DefaultCallAdapterFactory#get 。
DefaultCallAdapterFactory#get
这个方法作用就是返回CallAdapter , 修改下源码加入两个打印。
public @Nullable CallAdapter get(
Type returnType, Annotation[] annotations, Retrofit retrofit) {
final Type responseType = Utils.getParameterUpperBound(0, (ParameterizedType) returnType);
System.out.println("TAG" + " ->" + "returnType = " +getRawType(returnType) .getSimpleName());
System.out.println("TAG" + " ->" + "responseType = " +getRawType(responseType) .getSimpleName());
return new CallAdapter>() {
@Override
public Type responseType() {
return responseType;
}
@Override
public Call adapt(Call call) {
return executor == null ? call : new ExecutorCallbackCall<>(executor, call);
}
};
}
运行可以看到以下打印信息
returnType = Call ,responseType =ResponseBody 。
总结一下 , returnType就是方法返回值 , responseType 就是方法返回值上的泛型 DefaultCallAdapterFactory会根据平台环境去构建。
以Android24分析 , DefaultCallAdapterFactory(Executor callbackExecutor) , 构造方法中 , 线程池为主线程池 , 在retrofit初始化的时候添加到 到callAdapterFactories 集合中。
至此 , CallAdapterFactory 和 CallAdapter 分析完了 , 总结下就是给Call (retrofit内存只有OkHttpCall 作为唯一实现类)做适配 , 让其可以在 Rxjava / 协程 等各个环境中使用 Call。
非kt 协程情况下 , parseAnnotations 方法最终返回的是将requestFactory , callFactory , responseConverter, callAdapter 封装好的CallAdapted 对象。
再次回到梦开始的地方Retrofit#create 方法 , loadServiceMethod获取的ServiceMethod最终实现类为CallAdapted , 获取之后会调用invoke方法 , invoke是一个final方法 , 里面构建了OkHttpCall , 然后调用了adapt方法 , adapt中调用了callAdapter.adapt(call)。
@Override
protected ReturnT adapt(Call<ResponseT> call, Object[] args) {
return callAdapter.adapt(call);
}
这里的ReturnT 就是ExecutorCallbackCall<>(executor, call) 对象 , 所以 example.getBody().execute() 就是调用ExecutorCallbackCall#execute方法
//ExecutorCallbackCall#execute
public Response execute() throws IOException {
return delegate.execute();
}
delegate为OkHttpCall , 所以就调用到OkHttpCallCall#execute方法 , 这里就转给Okhttp去请求网络加载数据了 , 代码就不贴了 , 我们看下网络请求之后 , 数据Response 的处理 , 关键代码OkHttpCall#parseResponse。
Response parseResponse(okhttp3.Response rawResponse) throws IOException {
ResponseBody rawBody = rawResponse.body();
ExceptionCatchingResponseBody catchingBody = new ExceptionCatchingResponseBody(rawBody);
try {
T body = responseConverter.convert(catchingBody);
return Response.success(body, rawResponse);
}
}
responseConverter 在 HttpServiceMethod#parseAnnotations 方法中获取 , 回应数据转换器 , 把数据转换成我们可以直接使用的对象 , 比如我们常用的 GsonConverterFactory。
最后把转换好之后的数据 , 封装成Response对象返回。
response.body()就是responseConverter 转换后的数据 来张大致流程图感受下吧
第一种 Kotlin协程情况
其实大致流程第二种情况分析的差不多了 , 接下来分析下Retrofit对于kotlin的特殊处理吧。
if (Utils.getRawType(parameterType) == Continuation.class) {
isKotlinSuspendFunction = true;
return null;
}
协程挂起方法 , 第一个参数为Continuation , 所以判断是不是挂起方法也很简单 , 根据ResponseType 去构建协程专用的HttpServiceMethod , 主要有两类。
SuspendForResponse , 对应type为Continuation
SuspendForBody , 对应type为Continuation
这里看下 SuspendForBody 实现 , 套娃情况就不分析了。
如果是这样使用 , 最终会调到SuspendForBody #adapt。
@Override
protected Object adapt(Call call, Object[] args) {
call = callAdapter.adapt(call);
Continuation continuation = (Continuation) args[args.length - 1];
try {
//去掉干扰代码 , 仅保留这个
return KotlinExtensions.awaitNullable(call, continuation);
}
}
这个地方就很关键了 , java 直接调kotlin 协程 suspend 方法。
KotlinExtensions.awaitNullable 会调到KotlinExtensions#await方法。
retrofit与协程适配的细节都在 KotlinExtensions这个类里。
进入await , 可以看到使用suspendCancellableCoroutine把回调装换成协程。
@JvmName("awaitNullable")
suspend fun Call.await(): T? {
return suspendCancellableCoroutine { continuation ->
continuation.invokeOnCancellation {
cancel()
}
enqueue(object : Callback {
override fun onResponse(call: Call, response: Response) {
if (response.isSuccessful) {
continuation.resume(response.body())
} else {
continuation.resumeWithException(HttpException(response))
}
}
override fun onFailure(call: Call, t: Throwable) {
continuation.resumeWithException(t)
}
})
}
}
其实内部也是调用 OkHttp Call.enqueue() , 只不过是用suspendCancellableCoroutine给协程做了一层包装处理
通过 suspendCancellableCoroutine包装之后使用就很简单了。
GlobalScope.launch {
try {
val result = xxxApi.getXxx()
} catch (exception: Exception) {
}
}
总结
Call 这个接口用于与网络请求库做适配 , 比如Okhttp。
CallAdapter 用于retrofit 与各种环境搭配使用做适配 , 比如rxjava / 协程 / java。
Converter 用于将请求结果转换实体类Bean 或者其他。
用到的设计模式有: 动态代理/静态代理 / 构建者 / 工厂 / 适配器 / 状态 等。
以上就是Android开发Retrofit源码分析的详细内容,更多关于Android Retrofit源码分析的资料请关注脚本之家其它相关文章!
你可能感兴趣的:(Android开发Retrofit源码分析)
Ubuntu 下 nginx-1.24.0 源码分析 - cycle->modules[i]->ctx
若云止水
ubuntu nginx linux
ngx_module_s结构体中的ctx字段是Nginx模块系统的核心设计之一。这个字段的主要作用是提供特定类型模块的上下文信息。ctx字段实际上是一个指向模块特定上下文结构的指针。根据模块类型的不同,ctx可以指向不同的上下文结构,比如:对于HTTP模块,指向ngx_http_module_t对于Core模块,指向ngx_core_module_t对于Event模块,指向ngx_event_mo
DeepSeek告诉你,2025年Android开发和鸿蒙开发,哪个更有前途?
工程师老罗
android harmonyos 华为
本书定位于原生Android开发的入门书籍,力求做到看得懂、能练习、能学会,提供**视频、源代码、课件、**习题、试卷、教案等,免费在线答疑,教学两用以下问题答案来自腾讯元宝里的DeepSeekR1,仅供参考先说Deepseek得出的结论:最终建议:若聚焦国内高增长赛道(如车联网、政务数字化),选择鸿蒙更易把握职业红利;若追求技术通用性或出海机会,Android仍是更稳妥的选择。两者并非完全互斥,
180.HTML、CSS、JavaScript
我也念过晚霞
html css javascript
———————————————————————————————【1.引言】在Android开发过程中,前端技术(HTML、CSS、JavaScript)逐渐成为不可或缺的技能之一。无论是通过WebView显示网页内容,还是构建混合应用,掌握前端基础将有助于提升整体产品体验。本文在保证内容全面性的前提下,剔除了多余的推荐信息,直接进入技术核心讲解。通过详细介绍HTML、CSS与JavaScript的
如何在Android中实现网络请求
python资深爱好者
android 网络
在Android中实现网络请求有多种方法,常用的几种方法包括使用HttpURLConnection、OkHttp库和Retrofit库。下面分别介绍这几种方法:1.使用HttpURLConnectionHttpURLConnection是Java标准库提供的类,用于处理HTTP请求。下面是一个简单的例子:java复制代码importjava.io.BufferedReader;importjava
android多个usb摄像头,Android中多USB摄像头解决方案——UVCCamera源码分析(四)
桔子毛
android多个usb摄像头
经过前几章的学习,我们大概了解了整个UVCCamera初始化、开始预览的过程。那么接着我们将来看看UVCCamera是如何实现拍照功能的。本章内容相对比较简单,均是Java层的实现。我们直接来看代码:@OverridepublicvoidcaptureStill(finalStringpath,OnCaptureListenerlistener){super.captureStill(path,l
android多个usb摄像头,Android中多USB摄像头解决方案——UVCCamera源码分析(三)
spRachel雷切爾
android多个usb摄像头
前面几章我们分析了UVCCamera的初始化、预览相关的准备工作,本章我们则来看看startPreview的整个流程。按照惯例我们先大概看下调用的时序图:startPreview接着之前开启预览过程最终走到AbstractUVCCameraHandler.CameraThread的handleStartPreview方法,继而调用UVCCamera的startPreview,如上图所示,UVCCa
Android 动态布局的权重问题
t0_54manong
android 个人开发
在Android开发中,动态创建布局是常见的需求,特别是当我们需要根据运行时条件来调整界面时。今天我们来探讨一个常见的问题:在动态创建的LinearLayout中,如何正确地设置子View的权重(weight),以实现预期的布局效果。问题描述假设我们需要创建一个父LinearLayout,其中包含两个子LinearLayout。父布局的方向是垂直的,我们希望第一个子布局占用30%的空间,第二个子布
用户认证与登录机制:原理与实践
thankgoodneww
网络 java spring spring boot
内测之家一款功能强大且全面的应用内测与管理平台、分发平台,专为iOS和Android开发者打造,旨在为用户提供便捷高效、安全可靠的一站式服务。无论是从资源安全到传输安全,还是从数据保护到应用管理、统计分析,内测之家都展现出卓越的能力与优势。引言在数字化时代,用户认证是系统安全的基石。无论是社交应用、企业系统还是电商平台,都需要通过高效且安全的认证机制来验证用户身份并保护数据隐私。本文将从基础原理出
【android开发-16】android中文件和sharedpreferences数据存储详解
孤舟簔笠翁
android应用开发 android
1,文件读写方式的数据存储下面是一个简单的示例,演示如何在Android中使用内部存储来保存和读取文件:保存文件:try{Stringdata="这是要保存的数据";FileOutputStreamfos=openFileOutput("myFile",Context.MODE_PRIVATE);
面试基础---分布式架构基础:CAP 理论与 BASE
WeiLai1112
后端 面试 redis junit java 架构 分布式 后端
分布式架构基础:CAP理论与BASE理论深度解析引言在互联网大厂的高并发、高可用场景下,分布式系统的设计是至关重要的。CAP理论和BASE理论是分布式系统设计的基石,理解这些理论对于设计高可用、高性能的分布式系统至关重要。本文将深入探讨CAP理论和BASE理论,结合实际项目案例和源码分析,帮助读者深入理解其实现原理。1.CAP理论CAP理论是分布式系统设计的基础理论之一,由EricBrewer在2
Android wpa_supplicant源码分析--conf配置文件
水木无痕
http://blog.csdn.net/cuijiyue/article/details/514288351配置文件conf文件作为wpa_supplicant的配置文件,一般叫做wpa_supplicant.conf。其中存储着wpa_supplicant的运行参数和以保存的网络列表。conf文件的路径,通过启动wpa_supplicant时的–c参数传入,初始化过程中赋值到wpa_s->co
面试基础---MySQL 分布式 ID 方案深度解析
WeiLai1112
mysql vue.js
MySQL分布式ID方案深度解析:UUID、自增ID与雪花算法引言在分布式系统中,生成全局唯一的ID是一个常见的需求。MySQL作为最流行的关系型数据库之一,如何在高并发、分布式环境下生成唯一ID是一个重要的技术挑战。本文将深入探讨MySQL分布式ID的生成方案,包括UUID、自增ID和雪花算法,结合实际项目案例和源码分析,帮助读者深入理解其实现原理。1.分布式ID的需求与挑战在分布式系统中,生成
Android 开发必备:BaseActivity、BaseFragment 和 BaseApplication 的封装与优化
tangweiguo03051987
android android
在Android开发中,基类(BaseClasses)是提高开发效率、减少重复代码的重要手段。以下是BaseActivity、BaseFragment和BaseApplication的实现,涵盖了常用功能,如生命周期管理、Toast提示、权限处理、Fragment管理等。BaseApplicationBaseApplication是应用的全局基类,用于初始化全局配置、工具类等。importandr
Fragment 懒加载的优化方案
tangweiguo03051987
android android java
懒加载方案:使用Lifecycle监听Fragment的生命周期,简化逻辑。结合ViewPager2和FragmentStateAdapter,兼容现代Android开发。封装通用基类,减少重复代码,提高可维护性。这种方式不仅代码简洁,而且性能更好,适合现代Android应用开发以下是完整的Fragment懒加载实现代码,包括基类封装、具体Fragment实现以及ViewPager2的集成。代码经
ADB 和 Monkey 进行 Android 应用的测试和调试
小赖同学啊
自动化测试(app pc API) adb android
ADB(AndroidDebugBridge)和Monkey是Android开发和测试中常用的工具。ADB用于与Android设备通信,而Monkey是一个压力测试工具,可以模拟用户随机操作。以下是它们的高级用法,帮助您更高效地进行Android应用测试和调试。一、ADB的高级用法1.设备管理查看连接的设备:adbdevices连接远程设备:adbconnect断开远程设备:adbdisconne
Android中的AtomicLong:原理、使用与实战指南
jiet_h
Android 夯实基础 android
本文结合生产环境实战案例,带你彻底搞懂AtomicLong在Android多线程开发中的应用。全文包含大量Kotlin代码示例,建议收藏备用。一、为什么需要AtomicLong?在Android开发中,当多个线程同时操作同一个Long型变量时,你可能会遇到这样的诡异场景:varcounter=0Lfunincrement(){//这个操作在并发场景下会出错!counter++}这个简单的自增操作,
容器运行时 源码分析
梅梅与彤彤
kubernetes源码分析 kubernetes 容器
源码地址https://github.com/opencontainers/runctagv1.2.5整体流程一个容器启动主要分为三大部分,如文章题目所示create:主要是为了解析、组装容器启动的配置和与子进程的消息通道等;init:主要根据容器配置启动容器整个运行环境,包括熟知ns,cgroups,seccomp,apparmor,caps等;start:主要是为了通知init进程启动容器;r
登录认证流程分析与总结
thankgoodneww
java 安全 gateway 后端 架构 代理模式 模板方法模式
内测之家一款功能强大且全面的应用内测与管理平台、分发平台,专为iOS和Android开发者打造,旨在为用户提供便捷高效、安全可靠的一站式服务。无论是从资源安全到传输安全,还是从数据保护到应用管理、统计分析,内测之家都展现出卓越的能力与优势。一、登录认证类型分类根据用户输入的认证方式,登录类型可分为以下类别:类型特点适用场景账密登录用户输入账号(邮箱、手机号、自定义账号)和密码,后端校验密码哈希值。
深入源码分析spring AOP
萌新coder
Java基础知识 spring java 后端
深入源码分析springAOP一、SpringAOP核心概念AOP(面向切面编程)是Spring框架的核心功能之一,它通过动态代理技术,在不修改源代码的情况下,为业务逻辑横向添加通用功能(如日志、事务、权限等)。其核心思想是将业务代码与非业务代码解耦,例如://业务代码publicvoidtransferMoney(){//转账逻辑...}//非业务代码(日志记录)publicvoidlog(){
Android Studio安装与配置详解
Ya-Jun
android studio android ide
AndroidStudio安装与配置详解前言作为一名Android开发者,AndroidStudio是我们日常开发中最重要的工具。本文将详细介绍AndroidStudio的安装配置过程,帮助你搭建一个高效的开发环境。一、AndroidStudio下载与安装1.1下载AndroidStudio访问Android开发者官网(https://developer.android.google.cn/stu
面试基础---Redis 延迟队列深度解析
WeiLai1112
Redis 面试 redis junit java 架构 分布式 后端
Redis延迟队列深度解析:基于ZSet和Lua脚本的实现引言在互联网大厂的高并发场景下,延迟队列是一种常见的需求,用于处理需要延迟执行的任务,如订单超时取消、消息重试等。Redis作为高性能的内存数据库,通过ZSet(有序集合)和Lua脚本可以实现高效的延迟队列。本文将深入探讨Redis延迟队列的实现原理,结合实际项目案例和源码分析,帮助读者深入理解其实现细节。1.延迟队列的需求与挑战1.1延迟
百万架构师第二十二课:源码分析:Spring 源码分析:Spring经典面试答疑|JavaGuide
大雄野比
spring 面试 java
Spring面试解答上半节:面试中需要注意的细节动脑子,面试是一种交流面试的时候,要用心去感受当时面试场景了解自己,自己的长处、自己的短处(巧妙地扬长避短)了解1.公司的业务场景2.你是去面试什么岗位的?Java高级工程师实际工作经验是1年(如实填写)1、请描述SpringIOC的工作原理答:定位加载注册BeanFactoryBeanDefintion...1-3年1+ApplicationCon
React Native 源码分析(一)——启动流程
薛瑄
ReactNative react native 源码分析
本系列文章,是分析Android的ReactNative的源码,主要包括以下文章,和以往的源码系列一样,分析主流程的代码,不会细致到每一行(但相比上一篇的Gradle源码分析,要细致很多),会涉及到java、C++、js等源码。前三篇RN版本是0.64.0,后面是0.72.01、ReactNative源码分析(一)——启动流程2、ReactNative源码分析(二)——NativeModules桥
软件架构设计与模式之:模块化设计与组件化架构
AI天才研究院
AI大模型企业级应用开发实战 大数据 人工智能 语言模型 Java Python 架构设计
作者:禅与计算机程序设计艺术1.背景介绍模块化、组件化、微服务架构已经成为云计算的主流架构模式,软件架构设计如何满足需求并持续优化性能是一个非常重要的课题。本文将阐述模块化、组件化、微服务架构的概念及优缺点,并从面向对象的角度阐述模块化设计与组件化架构的过程。同时介绍一些基于Java语言的开源框架(如Spring)所实现的模块化、组件化、微服务架构,并深入到源码分析其实现机制和原理。本文的主要读者
分布式锁—4.Redisson的联锁和红锁一
东阳马生架构
分布式锁原理与源码 分布式锁 Redission
大纲1.Redisson联锁MultiLock概述2.Redisson联锁MultiLock的加锁与释放锁3.Redisson红锁RedLock的算法原理4.Redisson红锁RedLock的源码分析1.Redisson联锁MultiLock概述(1)MultiLock的简介(2)MultiLock的使用(3)MultiLock的初始化(1)MultiLock的简介一.一次性要锁定多个资源的场景
Android Room数据库中的主键重置
t0_54manong
android 数据库 个人开发
在Android开发中,尤其是使用Room数据库时,管理数据库的主键(PrimaryKey)是非常重要的一个环节。本文将详细探讨如何在重新安装应用后重置Room数据库中的主键,并结合实际操作实例来阐释整个过程。Room数据库的主键在Room中,主键通常由@PrimaryKey注解定义。如果你希望主键自动生成,可以使用autoGenerate=true属性。以下是两个简单的实体类示例:@Entity
Android Kotlin中Room数据库的参数传递
t0_54coder
android 数据库 kotlin 个人开发
在Android开发中,Room数据库是一个非常强大且灵活的持久化存储解决方案。通过Room库,我们可以轻松地与SQLite数据库进行交互。然而,当我们尝试在查询中传递参数时,可能会遇到一些问题。本文将通过一个实际案例来讨论如何在Kotlin中使用Room数据库时正确传递参数。问题背景假设我们正在开发一个待办事项(To-DoList)应用,使用Kotlin编写。应用中,我们需要根据ID查找特定的待
Android中的依赖注入
LWJAVA
Android android
前言最近在重构项目,用到依赖注入,查了很多资料,这里主要是记录一下自己的理解。正文依赖注入的优势:1.代码的可复用性2.便于重构3.易于测试什么是依赖注入?依赖注入(DI)是一种广泛的应用编程的技术,非常适合Android开发,通过遵循DI的原则,您可以为良好的应用程序架构打下基础。类通常需要对其他类的引用,举个例子,Car(汽车)类可能需要对Engine(引擎)类的引用。这些必需的类称为依赖项,
kubevirt源码分析之谁分配了gpu_device(3)
生命不息折腾不止
kubernetes 容器 云原生
目标当一个launcherpod被创建时,它会请求资源,如下Requests:cpu:16devices.kubevirt.io/kvm:1devices.kubevirt.io/tun:1devices.kubevirt.io/vhost-net:1ephemeral-storage:50Mhugepages-2Mi:8Gimemory:1574961152nvidia.com/GA102_GE
Redis 源码分析-内部数据结构 intset
笨手笨脚の
# Redis redis 数据结构 set intset 并差集
Redis源码分析-内部数据结构intsetintset是用于实现集合(set)这种对外的数据结构。它包含的元素无序,且不能重复。当插入的元素都是整形,底层使用intset存储,否则使用dict。intset结构和部分函数分析结构体定义如下://intset结构体typedefstructintset{uint32_tencoding;//数据编码,表示intset中的每个数据元素用几个字节(2、
ztree设置禁用节点
3213213333332132
JavaScript ztree json setDisabledNode Ajax
ztree设置禁用节点的时候注意,当使用ajax后台请求数据,必须要设置为同步获取数据,否者会获取不到节点对象,导致设置禁用没有效果。
$(function(){
showTree();
setDisabledNode();
});
JVM patch by Taobao
bookjovi
java HotSpot
在网上无意中看到淘宝提交的hotspot patch,共四个,有意思,记录一下。
7050685:jsdbproc64.sh has a typo in the package name
7058036:FieldsAllocationStyle=2 does not work in 32-bit VM
7060619:C1 should respect inline and
将session存储到数据库中
dcj3sjt126com
sql PHP session
CREATE TABLE sessions (
id CHAR(32) NOT NULL,
data TEXT,
last_accessed TIMESTAMP NOT NULL,
PRIMARY KEY (id)
);
<?php
/**
* Created by PhpStorm.
* User: michaeldu
* Date
Vector
171815164
vector
public Vector<CartProduct> delCart(Vector<CartProduct> cart, String id) {
for (int i = 0; i < cart.size(); i++) {
if (cart.get(i).getId().equals(id)) {
cart.remove(i);
各连接池配置参数比较
g21121
连接池
排版真心费劲,大家凑合看下吧,见谅~
Druid
DBCP
C3P0
Proxool
数据库用户名称 Username Username User
数据库密码 Password Password Password
驱动名
[简单]mybatis insert语句添加动态字段
53873039oycg
mybatis
mysql数据库,id自增,配置如下:
<insert id="saveTestTb" useGeneratedKeys="true" keyProperty="id"
parameterType=&
struts2拦截器配置
云端月影
struts2拦截器
struts2拦截器interceptor的三种配置方法
方法1. 普通配置法
<struts>
<package name="struts2" extends="struts-default">
&
IE中页面不居中,火狐谷歌等正常
aijuans
IE中页面不居中
问题是首页在火狐、谷歌、所有IE中正常显示,列表页的页面在火狐谷歌中正常,在IE6、7、8中都不中,觉得可能那个地方设置的让IE系列都不认识,仔细查看后发现,列表页中没写HTML模板部分没有添加DTD定义,就是<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3
String,int,Integer,char 几个类型常见转换
antonyup_2006
html sql .net
如何将字串 String 转换成整数 int?
int i = Integer.valueOf(my_str).intValue();
int i=Integer.parseInt(str);
如何将字串 String 转换成Integer ?
Integer integer=Integer.valueOf(str);
如何将整数 int 转换成字串 String ?
1.
PL/SQL的游标类型
百合不是茶
显示游标(静态游标) 隐式游标 游标的更新和删除 %rowtype ref游标(动态游标)
游标是oracle中的一个结果集,用于存放查询的结果;
PL/SQL中游标的声明;
1,声明游标
2,打开游标(默认是关闭的);
3,提取数据
4,关闭游标
注意的要点:游标必须声明在declare中,使用open打开游标,fetch取游标中的数据,close关闭游标
隐式游标:主要是对DML数据的操作隐
JUnit4中@AfterClass @BeforeClass @after @before的区别对比
bijian1013
JUnit4 单元测试
一.基础知识
JUnit4使用Java5中的注解(annotation),以下是JUnit4常用的几个annotation: @Before:初始化方法 对于每一个测试方法都要执行一次(注意与BeforeClass区别,后者是对于所有方法执行一次)@After:释放资源 对于每一个测试方法都要执行一次(注意与AfterClass区别,后者是对于所有方法执行一次
精通Oracle10编程SQL(12)开发包
bijian1013
oracle 数据库 plsql
/*
*开发包
*包用于逻辑组合相关的PL/SQL类型(例如TABLE类型和RECORD类型)、PL/SQL项(例如游标和游标变量)和PL/SQL子程序(例如过程和函数)
*/
--包用于逻辑组合相关的PL/SQL类型、项和子程序,它由包规范和包体两部分组成
--建立包规范:包规范实际是包与应用程序之间的接口,它用于定义包的公用组件,包括常量、变量、游标、过程和函数等
--在包规
【EhCache二】ehcache.xml配置详解
bit1129
ehcache.xml
在ehcache官网上找了多次,终于找到ehcache.xml配置元素和属性的含义说明文档了,这个文档包含在ehcache.xml的注释中!
ehcache.xml : http://ehcache.org/ehcache.xml
ehcache.xsd : http://ehcache.org/ehcache.xsd
ehcache配置文件的根元素是ehcahe
ehcac
java.lang.ClassNotFoundException: org.springframework.web.context.ContextLoaderL
白糖_
java eclipse spring tomcat Web
今天学习spring+cxf的时候遇到一个问题:在web.xml中配置了spring的上下文监听器:
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
随后启动
angular.element
boyitech
AngularJS AngularJS API angular.element
angular.element
描述: 包裹着一部分DOM element或者是HTML字符串,把它作为一个jQuery元素来处理。(类似于jQuery的选择器啦) 如果jQuery被引入了,则angular.element就可以看作是jQuery选择器,选择的对象可以使用jQuery的函数;如果jQuery不可用,angular.e
java-给定两个已排序序列,找出共同的元素。
bylijinnan
java
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
public class CommonItemInTwoSortedArray {
/**
* 题目:给定两个已排序序列,找出共同的元素。
* 1.定义两个指针分别指向序列的开始。
* 如果指向的两个元素
sftp 异常,有遇到的吗?求解
Chen.H
java jcraft auth jsch jschexception
com.jcraft.jsch.JSchException: Auth cancel
at com.jcraft.jsch.Session.connect(Session.java:460)
at com.jcraft.jsch.Session.connect(Session.java:154)
at cn.vivame.util.ftp.SftpServerAccess.connec
[生物智能与人工智能]神经元中的电化学结构代表什么?
comsci
人工智能
我这里做一个大胆的猜想,生物神经网络中的神经元中包含着一些化学和类似电路的结构,这些结构通常用来扮演类似我们在拓扑分析系统中的节点嵌入方程一样,使得我们的神经网络产生智能判断的能力,而这些嵌入到节点中的方程同时也扮演着"经验"的角色....
我们可以尝试一下...在某些神经
通过LAC和CID获取经纬度信息
dai_lm
lac cid
方法1:
用浏览器打开http://www.minigps.net/cellsearch.html,然后输入lac和cid信息(mcc和mnc可以填0),如果数据正确就可以获得相应的经纬度
方法2:
发送HTTP请求到http://www.open-electronics.org/celltrack/cell.php?hex=0&lac=<lac>&cid=&
JAVA的困难分析
datamachine
java
前段时间转了一篇SQL的文章(http://datamachine.iteye.com/blog/1971896),文章不复杂,但思想深刻,就顺便思考了一下java的不足,当砖头丢出来,希望引点和田玉。
-----------------------------------------------------------------------------------------
小学5年级英语单词背诵第二课
dcj3sjt126com
english word
money 钱
paper 纸
speak 讲,说
tell 告诉
remember 记得,想起
knock 敲,击,打
question 问题
number 数字,号码
learn 学会,学习
street 街道
carry 搬运,携带
send 发送,邮寄,发射
must 必须
light 灯,光线,轻的
front
linux下面没有tree命令
dcj3sjt126com
linux
centos p安装
yum -y install tree
mac os安装
brew install tree
首先来看tree的用法
tree 中文解释:tree
功能说明:以树状图列出目录的内容。
语 法:tree [-aACdDfFgilnNpqstux][-I <范本样式>][-P <范本样式
Map迭代方式,Map迭代,Map循环
蕃薯耀
Map循环 Map迭代 Map迭代方式
Map迭代方式,Map迭代,Map循环
>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>
蕃薯耀 2015年
Spring Cache注解+Redis
hanqunfeng
spring
Spring3.1 Cache注解
依赖jar包:
<!-- redis -->
<dependency>
<groupId>org.springframework.data</groupId>
<artifactId>spring-data-redis</artifactId>
Guava中针对集合的 filter和过滤功能
jackyrong
filter
在guava库中,自带了过滤器(filter)的功能,可以用来对collection 进行过滤,先看例子:
@Test
public void whenFilterWithIterables_thenFiltered() {
List<String> names = Lists.newArrayList("John"
学习编程那点事
lampcy
编程 android PHP html5
一年前的夏天,我还在纠结要不要改行,要不要去学php?能学到真本事吗?改行能成功吗?太多的问题,我终于不顾一切,下定决心,辞去了工作,来到传说中的帝都。老师给的乘车方式还算有效,很顺利的就到了学校,赶巧了,正好学校搬到了新校区。先安顿了下来,过了个轻松的周末,第一次到帝都,逛逛吧!
接下来的周一,是我噩梦的开始,学习内容对我这个零基础的人来说,除了勉强完成老师布置的作业外,我已经没有时间和精力去
架构师之流处理---------bytebuffer的mark,limit和flip
nannan408
ByteBuffer
1.前言。
如题,limit其实就是可以读取的字节长度的意思,flip是清空的意思,mark是标记的意思 。
2.例子.
例子代码:
String str = "helloWorld";
ByteBuffer buff = ByteBuffer.wrap(str.getBytes());
Sy
org.apache.el.parser.ParseException: Encountered " ":" ": "" at line 1, column 1
Everyday都不同
$ 转义 el表达式
最近在做Highcharts的过程中,在写js时,出现了以下异常:
严重: Servlet.service() for servlet jsp threw exception
org.apache.el.parser.ParseException: Encountered " ":" ": "" at line 1,
用Java实现发送邮件到163
tntxia
java实现
/*
在java版经常看到有人问如何用javamail发送邮件?如何接收邮件?如何访问多个文件夹等。问题零散,而历史的回复早已经淹没在问题的海洋之中。
本人之前所做过一个java项目,其中包含有WebMail功能,当初为用java实现而对javamail摸索了一段时间,总算有点收获。看到论坛中的经常有此方面的问题,因此把我的一些经验帖出来,希望对大家有些帮助。
此篇仅介绍用
探索实体类存在的真正意义
java小叶檀
POJO
一. 实体类简述
实体类其实就是俗称的POJO,这种类一般不实现特殊框架下的接口,在程序中仅作为数据容器用来持久化存储数据用的
POJO(Plain Old Java Objects)简单的Java对象
它的一般格式就是
public class A{
private String id;
public Str