Dubbo 3.x源码(29)—Dubbo Consumer服务调用源码(1)服务调用入口
基于Dubbo 3.1,详细介绍了Dubbo Consumer服务调用源码。
此前我们学习了Dubbo服务的导出和引入的源码,现在我们来学习Dubbo服务调用的源码。
此前的文章中我们讲过了最上层代理的调用逻辑(服务引用bean的获取以及懒加载原理):业务引入的接口代理对象(ReferenceBean内部的lazyProxy对象)-> 代理目标对象(ReferenceConfig内部的接口代理对象ref),代理目标对象的请求都会被统一转发到内部的InvokerInvocationHandler#invoke方法中去。
所以,我们学习Dubbo服务调用源码入口就是InvokerInvocationHandler#invoke方法。
Dubbo 3.x服务调用源码:
- Dubbo 3.x源码(29)—Dubbo Consumer服务调用源码(1)服务调用入口
- Dubbo 3.x源码(30)—Dubbo Consumer服务调用源码(2)发起远程调用
Dubbo 3.x服务引用源码:
- Dubbo 3.x源码(11)—Dubbo服务的发布与引用的入口
- Dubbo 3.x源码(18)—Dubbo服务引用源码(1)
- Dubbo 3.x源码(19)—Dubbo服务引用源码(2)
- Dubbo 3.x源码(20)—Dubbo服务引用源码(3)
- Dubbo 3.x源码(21)—Dubbo服务引用源码(4)
- Dubbo 3.x源码(22)—Dubbo服务引用源码(5)服务引用bean的获取以及懒加载原理
- Dubbo 3.x源码(23)—Dubbo服务引用源码(6)MigrationRuleListener迁移规则监听器
- Dubbo 3.x源码(24)—Dubbo服务引用源码(7)接口级服务发现订阅refreshInterfaceInvoker
- Dubbo 3.x源码(25)—Dubbo服务引用源码(8)notify订阅服务通知更新
- Dubbo 3.x源码(26)—Dubbo服务引用源码(9)应用级服务发现订阅refreshServiceDiscoveryInvoker
- Dubbo 3.x源码(27)—Dubbo服务引用源码(10)subscribeURLs订阅应用级服务url
Dubbo 3.x服务发布源码:
- Dubbo 3.x源码(11)—Dubbo服务的发布与引用的入口
- Dubbo 3.x源码(12)—Dubbo服务发布导出源码(1)
- Dubbo 3.x源码(13)—Dubbo服务发布导出源码(2)
- Dubbo 3.x源码(14)—Dubbo服务发布导出源码(3)
- Dubbo 3.x源码(15)—Dubbo服务发布导出源码(4)
- Dubbo 3.x源码(16)—Dubbo服务发布导出源码(5)
- Dubbo 3.x源码(17)—Dubbo服务发布导出源码(6)
- Dubbo 3.x源码(28)—Dubbo服务发布导出源码(7)应用级服务接口元数据发布
文章目录
- InvokerInvocationHandler#invoke调用入口
- InvocationUtil#invoke发起调用
- getUrl获取consumerUrl
- MigrationInvoker#invoke决策调用Invoker
- MockClusterInvoker#invoker本地mock调用
- AbstractCluster#invoke继续调用
- CallbackRegistrationInvoker#invoke回调注册
- CopyOfFilterChainNode#invoke过滤器链式调用
- ConsumerContextFilter封装信息
- FutureFilter执行回调方法
- MonitorFilter收集监控数据
- RouterSnapshotFilter路由快照zhichi
- AbstractClusterInvoker#invoke集群容错rpc调用
- FailoverClusterInvoker#doInvoke失败重试调用
- AbstractClusterInvoker#select选择invoker
- AbstractClusterInvoker#doSelect继续选择invoker
- FailoverClusterInvoker#invokeWithContext调用服务
- 总结
InvokerInvocationHandler#invoke调用入口
InvokerInvocationHandler#invoke可以看作是消费者发起调用的入口方法。
该方法中,将会构建一个RpcInvocation作为接口方法调用抽象,包含方法名字,接口名,方法参数等信息,然后通过InvocationUtil#invoke方法来发起真正的远程or本地调用。
/*** InvokerInvocationHandler的方法* <p>* 执行代理对象的方法的转发** @param proxy 在其上调用方法的代理实例* @param method 在代理实例上调用的接口方法* @param args 一个对象数组,包含在代理实例的方法调用中传递的参数值,如果接口方法不接受参数,则为null* @return 调用代理对象方法执行结果*/
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {//对于object类的方法,直接反射调用if (method.getDeclaringClass() == Object.class) {return method.invoke(invoker, args);}//获取方法名和方法参数类型数组String methodName = method.getName();Class<?>[] parameterTypes = method.getParameterTypes();//特殊方法的调用,直接调用invoker对象的同名方法if (parameterTypes.length == 0) {if ("toString".equals(methodName)) {return invoker.toString();} else if ("$destroy".equals(methodName)) {invoker.destroy();return null;} else if ("hashCode".equals(methodName)) {return invoker.hashCode();}} else if (parameterTypes.length == 1 && "equals".equals(methodName)) {return invoker.equals(args[0]);}//构建一个RpcInvocation作为接口方法调用抽象,包含方法名字,接口名,方法参数等信息RpcInvocation rpcInvocation = new RpcInvocation(serviceModel, method.getName(), invoker.getInterface().getName(), protocolServiceKey, method.getParameterTypes(), args);if (serviceModel instanceof ConsumerModel) {rpcInvocation.put(Constants.CONSUMER_MODEL, serviceModel);rpcInvocation.put(Constants.METHOD_MODEL, ((ConsumerModel) serviceModel).getMethodModel(method));}//通过InvocationUtil.invoke方法来发起真正的远程or本地调用return InvocationUtil.invoke(invoker, rpcInvocation);
}
InvocationUtil#invoke发起调用
这里的Invoker是经过层层封装之后的Invoker(注意MockClusterInvoker上面还有一个MigrationInvoker没画出来),我们将按照这个顺序从上向下讲解。
public class InvocationUtil {private static final Logger logger = LoggerFactory.getLogger(InvokerInvocationHandler.class);/*** 消费者调用** @param invoker 可执行器* @param rpcInvocation 接口方法调用抽* @return 执行结果*/public static Object invoke(Invoker<?> invoker, RpcInvocation rpcInvocation) throws Throwable {//消费者url,consumer://10.253.45.126/org.apache.dubbo.demo.GreetingService?application=demo-consumer&background=false&check=false&dubbo=2.0.2&group=greeting&init=false&interface=org.apache.dubbo.demo.GreetingService&methods=hello&pid=72363®ister.ip=10.253.45.126&release=&revision=1.0.0&side=consumer&sticky=false×tamp=1668394930680&unloadClusterRelated=false&version=1.0.0URL url = invoker.getUrl();//服务key,{group}/{serviceInterface}:{version}String serviceKey = url.getServiceKey();//设置目标服务的唯一名称rpcInvocation.setTargetServiceUniqueName(serviceKey);//设置消费者url到rpc调用上下文中吗格式一个内部的线程本地变量// invoker.getUrl() returns consumer url.RpcServiceContext.getServiceContext().setConsumerUrl(url);//调用性能解析器if (ProfilerSwitch.isEnableSimpleProfiler()) {//线程本地变量ProfilerEntry parentProfiler = Profiler.getBizProfiler();ProfilerEntry bizProfiler;if (parentProfiler != null) {//接收请求。客户端调用开始。bizProfiler = Profiler.enter(parentProfiler,"Receive request. Client invoke begin. ServiceKey: " + serviceKey + " MethodName:" + rpcInvocation.getMethodName());} else {//接收请求。客户端调用开始。bizProfiler = Profiler.start("Receive request. Client invoke begin. ServiceKey: " + serviceKey + " " + "MethodName:" + rpcInvocation.getMethodName());}rpcInvocation.put(Profiler.PROFILER_KEY, bizProfiler);try {/** 执行invoker#invoker方法实现调用*/return invoker.invoke(rpcInvocation).recreate();} finally {//释放Profiler.release(bizProfiler);int timeout;//timeout附加信息,远程服务调用超时时间(毫秒),默认1000Object timeoutKey = rpcInvocation.getObjectAttachmentWithoutConvert(TIMEOUT_KEY);if (timeoutKey instanceof Integer) {timeout = (Integer) timeoutKey;} else {timeout = url.getMethodPositiveParameter(rpcInvocation.getMethodName(),TIMEOUT_KEY,DEFAULT_TIMEOUT);}//调用执行时间long usage = bizProfiler.getEndTime() - bizProfiler.getStartTime();//是否超时if ((usage / (1000_000L * ProfilerSwitch.getWarnPercent())) > timeout) {StringBuilder attachment = new StringBuilder();rpcInvocation.foreachAttachment((entry) -> {attachment.append(entry.getKey()).append("=").append(entry.getValue()).append(";\n");});//超时打印日志logger.warn(String.format("[Dubbo-Consumer] execute service %s#%s cost %d.%06d ms, this invocation almost (maybe already) timeout. Timeout: %dms\n" + "invocation context:\n%s" + "thread info: \n%s",rpcInvocation.getProtocolServiceKey(),rpcInvocation.getMethodName(),usage / 1000_000,usage % 1000_000,timeout,attachment,Profiler.buildDetail(bizProfiler)));}}}/** 抛出异常后重新执行invoker#invoker方法实现调用*/return invoker.invoke(rpcInvocation).recreate();}
}
getUrl获取consumerUrl
该方法获取消费者url。例如:consumer://10.253.45.126/org.apache.dubbo.demo.GreetingService?application=demo-consumer&background=false&check=false&dubbo=2.0.2&group=greeting&init=false&interface=org.apache.dubbo.demo.GreetingService&methods=hello&pid=72363®ister.ip=10.253.45.126&release=&revision=1.0.0&side=consumer&sticky=false×tamp=1668394930680&unloadClusterRelated=false&version=1.0.0
/*** MigrationInvoker的方法*/
@Override
public URL getUrl() {if (currentAvailableInvoker != null) {//当前激活的Invokerreturn currentAvailableInvoker.getUrl();} else if (invoker != null) {//接口级别invokerreturn invoker.getUrl();} else if (serviceDiscoveryInvoker != null) {//应用级别invokerreturn serviceDiscoveryInvoker.getUrl();}//消费者urlreturn consumerUrl;
}
MigrationInvoker#invoke决策调用Invoker
MigrationInvoker是可迁移Invoker,也是最上层Invoker,它可用于选择不同的Invoker执行调用,用于应用级服务发现和接口级服务发现之间的迁移,以及实现灰度调用。
该方法决策使用应用级Invoker还是接口级Invoker执行调用,如果设置了激活的invoker并且类型为应用级优先,那么还会进入灰度策略,如果随机决策值(0-100)大于灰度值,那么走接口级订阅模式,否则走应用级订阅模式。灰度比例功能仅在应用级优先状态下生效。
/*** MigrationInvoker的方法* * 决策使用应用级Invoker还是接口级Invoker执行调用** @param invocation RpcInvocation* @return 调用结果*/
@Override
public Result invoke(Invocation invocation) throws RpcException {//如果设置了当前激活的invokerif (currentAvailableInvoker != null) {//如果类型为应用级优先if (step == APPLICATION_FIRST) {//如果随机决策值(0-100)大于灰度值,那么走接口模式//灰度比例功能仅在应用级优先状态下生效if (promotion < 100 && ThreadLocalRandom.current().nextDouble(100) > promotion) {//退回到接口模式调用return invoker.invoke(invocation);}//每次检查调用程序是否可用,应用级invoker优先,然后继续使用invoker调用return decideInvoker().invoke(invocation);}//其他类型return currentAvailableInvoker.invoke(invocation);}//如果没有当前激活的invokerswitch (step) {case APPLICATION_FIRST://每次检查调用程序是否可用,应用级invoker优先currentAvailableInvoker = decideInvoker();break;case FORCE_APPLICATION://应用级invokercurrentAvailableInvoker = serviceDiscoveryInvoker;break;case FORCE_INTERFACE:default://接口级invokercurrentAvailableInvoker = invoker;}//调用return currentAvailableInvoker.invoke(invocation);
}
MockClusterInvoker#invoker本地mock调用
dubbo除了限流措施之外,还支持mock本地伪装。本地伪装常被用于服务降级和熔断。比如某验权服务,当服务提供方全部挂掉后,假如此时服务消费方发起了一次远程调用,那么本次调用将会失败并抛出一个 RpcException 异常。
为了避免出现这种直接抛出异常的情况出现,那么客户端就可以利用本地伪装来提供 Mock 数据返回授权失败。mock的大概配置如下:
- false、0、null、N/A:如果没有设置mock属性,或者设置了这些值,表示不启用mock,直接发起远程调用,这是默认策略。
- force:强制服务降级,如果mock属性值以“force”开头则使用该策略。服务调用方在调用该接口服务时候会直接执行客户端本地的mock逻辑,不会远程调用服务提供者。比如使用mock=force:return+null表示消费方对该服务的方法调用都直接返回null值,不发起远程调用。
- fail策略:服务失败降级,如果mock属性值不符合上面所有的类型,则使用该策略。表示服务调用彻底失败并抛出RPCException之后执行本地mock逻辑,不一定会抛出异常。服务彻底失败具体和设置的集群容错方式有关。如果设置了retries,则是在全部重试次数使用完毕并且抛出RPCException异常之后才会执行mock的逻辑。
- true/default:如果mock属性值为true或者default,那么在最终失败并抛出RPCException之后,会在接口同路径下查找interfaceName+Mock名字的类,随后通过无参构造器实例化,随后调用该Mock类对象的对应方法,该方法中提供对应的逻辑。
详细介绍参见官方文档:https://dubbo.apache.org/zh/docs3-v2/java-sdk/advanced-features-and-usage/service/local-mock/#%E5%BC%80%E5%90%AF-mock-%E9%85%8D%E7%BD%AE。
/*** MockClusterInvoker的方法* <p>* mock调用** @param invocation RpcInvocation* @return 调用结果*/
@Override
public Result invoke(Invocation invocation) throws RpcException {Result result;//消费者url获取mock参数String value = getUrl().getMethodParameter(invocation.getMethodName(), MOCK_KEY, Boolean.FALSE.toString()).trim();//false、0、null、N/A,表示不需要mockif (ConfigUtils.isEmpty(value)) {//no mock/** 继续调用下层Invoker#invoke*/result = this.invoker.invoke(invocation);}//以force开头,表示强制服务降级else if (value.startsWith(FORCE_KEY)) {if (logger.isWarnEnabled()) {logger.warn("force-mock: " + invocation.getMethodName() + " force-mock enabled , url : " + getUrl());}//force:direct mock//服务调用方在调用该接口服务时候会直接执行客户端本地的mock逻辑,不会远程调用服务提供者。//比如使用mock=force:return+null表示消费方对该服务的方法调用都直接返回null值,不发起远程调用。result = doMockInvoke(invocation, null);}//其他值,先先发起远程调用,当远程服务调用失败时,才会根据配置降级执行mock功能else {//fail-mocktry {/** 继续调用下层Invoker#invoke*/result = this.invoker.invoke(invocation);//fix:#4585//最终失败并抛出RPCExceptionif (result.getException() != null && result.getException() instanceof RpcException) {RpcException rpcException = (RpcException) result.getException();if (rpcException.isBiz()) {throw rpcException;} else {/** 执行本地mock调用*/result = doMockInvoke(invocation, rpcException);}}} catch (RpcException e) {if (e.isBiz()) {throw e;}if (logger.isWarnEnabled()) {logger.warn("fail-mock: " + invocation.getMethodName() + " fail-mock enabled , url : " + getUrl(), e);}/** 执行本地mock调用*/result = doMockInvoke(invocation, e);}}return result;
}
AbstractCluster#invoke继续调用
继续向下到这里,首先执行inoker的过滤操作。
/*** AbstractCluster的方法*/
@Override
public Result invoke(Invocation invocation) throws RpcException {return filterInvoker.invoke(invocation);
}
CallbackRegistrationInvoker#invoke回调注册
filterInvoker实际类型为FilterChainBuilder的内部类CallbackRegistrationInvoker,它的invoke方法用于添加一个回调,它可以在RPC调用完成时触发,在这回调函数将会倒序执行filters中的过滤器。
/*** FilterChainBuilder的内部类CallbackRegistrationInvoker的方法*/
@Override
public Result invoke(Invocation invocation) throws RpcException {/** 继续调用下层invoker#invoke*/Result asyncResult = filterInvoker.invoke(invocation);//添加一个回调,它可以在RPC调用完成时触发。asyncResult.whenCompleteWithContext((r, t) -> {RuntimeException filterRuntimeException = null;//过滤器倒序执行for (int i = filters.size() - 1; i >= 0; i--) {FILTER filter = filters.get(i);try {InvocationProfilerUtils.releaseDetailProfiler(invocation);if (filter instanceof ListenableFilter) {//执行过滤器ListenableFilter listenableFilter = ((ListenableFilter) filter);Filter.Listener listener = listenableFilter.listener(invocation);try {if (listener != null) {if (t == null) {listener.onResponse(r, filterInvoker, invocation);} else {listener.onError(t, filterInvoker, invocation);}}} finally {listenableFilter.removeListener(invocation);}} else if (filter instanceof FILTER.Listener) {FILTER.Listener listener = (FILTER.Listener) filter;if (t == null) {listener.onResponse(r, filterInvoker, invocation);} else {listener.onError(t, filterInvoker, invocation);}}} catch (RuntimeException runtimeException) {LOGGER.error(String.format("Exception occurred while executing the %s filter named %s.", i, filter.getClass().getSimpleName()));if (LOGGER.isDebugEnabled()) {LOGGER.debug(String.format("Whole filter list is: %s", filters.stream().map(tmpFilter -> tmpFilter.getClass().getSimpleName()).collect(Collectors.toList())));}filterRuntimeException = runtimeException;t = runtimeException;}}if (filterRuntimeException != null) {throw filterRuntimeException;}});return asyncResult;
}
CopyOfFilterChainNode#invoke过滤器链式调用
FilterChainBuilder的内部类,一个CopyOfFilterChainNode实例表示一个过滤器节点,将会首先执行全部的过滤器,执行完毕之后,才会继续后面真正的rpc调用。
这个版本默认4个filter,分别是:ConsumerContextFilter、FutureFilter、MonitorFilter、RouterSnapshotFilter。
ConsumerContextFilter封装信息
ConsumerContextFilter为consumer invoker序设置当前RpcContext,包括invoker,invocation, local host, remote host and port。它这样做是为了让执行线程的RpcContext包含这些必要的信息,方便后续直接获取。
/*** ConsumerContextFilter的方法** 始终在实现中调用invoker.invoke()将请求转交给下一个筛选器节点。** @param invoker 下一个invoker节点*/
@Override
public Result invoke(Invoker<?> invoker, Invocation invocation) throws RpcException {RpcContext.RestoreServiceContext originServiceContext = RpcContext.storeServiceContext();try {//设置相关属性RpcContext.getServiceContext()//下一个invoker.setInvoker(invoker)//调用抽象.setInvocation(invocation)//本地地址.setLocalAddress(NetUtils.getLocalHost(), 0);RpcContext context = RpcContext.getClientAttachment();context.setAttachment(REMOTE_APPLICATION_KEY, invoker.getUrl().getApplication());if (invocation instanceof RpcInvocation) {//下一个invoker((RpcInvocation) invocation).setInvoker(invoker);}if (CollectionUtils.isNotEmpty(supportedSelectors)) {for (PenetrateAttachmentSelector supportedSelector : supportedSelectors) {Map<String, Object> selected = supportedSelector.select();if (CollectionUtils.isNotEmptyMap(selected)) {((RpcInvocation) invocation).addObjectAttachments(selected);}}} else {((RpcInvocation) invocation).addObjectAttachments(RpcContext.getServerAttachment().getObjectAttachments());}Map<String, Object> contextAttachments = RpcContext.getClientAttachment().getObjectAttachments();if (CollectionUtils.isNotEmptyMap(contextAttachments)) {/*** invocation.addAttachmentsIfAbsent(context){@link RpcInvocation#addAttachmentsIfAbsent(Map)}should not be used here,* because the {@link RpcContext#setAttachment(String, String)} is passed in the Filter when the call is triggered* by the built-in retry mechanism of the Dubbo. The attachment to update RpcContext will no longer work, which is* a mistake in most cases (for example, through Filter to RpcContext output traceId and spanId and other information).*/((RpcInvocation) invocation).addObjectAttachments(contextAttachments);}// pass default timeout set by end user (ReferenceConfig)Object countDown = context.getObjectAttachment(TIME_COUNTDOWN_KEY);if (countDown != null) {TimeoutCountDown timeoutCountDown = (TimeoutCountDown) countDown;if (timeoutCountDown.isExpired()) {return AsyncRpcResult.newDefaultAsyncResult(new RpcException(RpcException.TIMEOUT_TERMINATE,"No time left for making the following call: " + invocation.getServiceName() + "."+ invocation.getMethodName() + ", terminate directly."), invocation);}}RpcContext.removeServerContext();//调用下一个invoker节点return invoker.invoke(invocation);} finally {RpcContext.restoreServiceContext(originServiceContext);}
}
FutureFilter执行回调方法
这个过滤器主要是获取异步方法调用信息AsyncMethodInfo,AsyncMethodInfo内部包含在执行Invoker的各个阶段的回调方法和实例,这里将会执行invoker调用时(调用之前)的回调方法。
/*** FutureFilter的方法** 始终在实现中调用invoker.invoke()将请求转交给下一个筛选器节点。** @param invoker 下一个invoker节点*/
@Override
public Result invoke(final Invoker<?> invoker, final Invocation invocation) throws RpcException {//获取Invoker回调方法,并且执行fireInvokeCallback(invoker, invocation);// need to configure if there's return value before the invocation in order to help invoker to judge if it's// necessary to return future.//调用下一个invoker节点return invoker.invoke(invocation);
}private void fireInvokeCallback(final Invoker<?> invoker, final Invocation invocation) {//获取异步方法调用信息AsyncMethodInfo,并且执行invoker调用时的回调方法final AsyncMethodInfo asyncMethodInfo = getAsyncMethodInfo(invoker, invocation);//如果为null直接返回if (asyncMethodInfo == null) {return;}//调用异步调用时的回调方法final Method onInvokeMethod = asyncMethodInfo.getOninvokeMethod();//调用异步调用时的回调实例final Object onInvokeInst = asyncMethodInfo.getOninvokeInstance();if (onInvokeMethod == null && onInvokeInst == null) {return;}if (onInvokeMethod == null || onInvokeInst == null) {throw new IllegalStateException("service:" + invoker.getUrl().getServiceKey() + " has a oninvoke callback config , but no such " + (onInvokeMethod == null ? "method" : "instance") + " found. url:" + invoker.getUrl());}if (!onInvokeMethod.isAccessible()) {onInvokeMethod.setAccessible(true);}Object[] params = invocation.getArguments();try {//反射执行invoker调用时的回调方法onInvokeMethod.invoke(onInvokeInst, params);} catch (InvocationTargetException e) {fireThrowCallback(invoker, invocation, e.getTargetException());} catch (Throwable e) {fireThrowCallback(invoker, invocation, e);}
}
MonitorFilter收集监控数据
监控拦截器,它将收集关于此调用的调用数据并将其发送到监控中心。
/*** MonitorFilter的方法** 始终在实现中调用invoker.invoke()将请求转交给下一个筛选器节点。** @param invoker 下一个invoker节点*/
@Override
public Result invoke(Invoker<?> invoker, Invocation invocation) throws RpcException {//如果存在monitor,则设置监控数据if (invoker.getUrl().hasAttribute(MONITOR_KEY)) {invocation.put(MONITOR_FILTER_START_TIME, System.currentTimeMillis());invocation.put(MONITOR_REMOTE_HOST_STORE, RpcContext.getServiceContext().getRemoteHost());// count upgetConcurrent(invoker, invocation).incrementAndGet();}//调用下一个invoker节点return invoker.invoke(invocation);
}
RouterSnapshotFilter路由快照zhichi
匹配路由器快照切换器RouterSnapshotSwitcher,用于打印日志以及路由快照。
/*** RouterSnapshotFilter的方法** 始终在实现中调用invoker.invoke()将请求转交给下一个筛选器节点。** @param invoker 下一个invoker节点*/
@Override
public Result invoke(Invoker<?> invoker, Invocation invocation) throws RpcException {//默认不支持if (!switcher.isEnable()) {//调用下一个invoker节点//如果后面没有其他filter,那么该节点就是真正的invoker节点return invoker.invoke(invocation);}if (!logger.isInfoEnabled()) {return invoker.invoke(invocation);}if (!switcher.isEnable(invocation.getServiceModel().getServiceKey())) {return invoker.invoke(invocation);}RpcContext.getServiceContext().setNeedPrintRouterSnapshot(true);return invoker.invoke(invocation);
}
如果后面没有其他filter,那么下一个节点就是真正的invoker节点,即具有集群容错策略的ClusterInvoker,默认FailoverClusterInvoker。
AbstractClusterInvoker#invoke集群容错rpc调用
该方法是执行rpc调用的骨干方法。大概步骤为:
- 调用list方法,从服务目录Directory中根据路由规则或滤出满足规则的服务提供者invoker列表。最后会调用每个路由器的route方法进行服务路由,如果没有路由器则返回全部invoker列表。具体源码后面再说。
- 调用initLoadBalance方法,获取负载均衡策略实例,默认RandomLoadBalance。具体源码后面再说。
- 调用doinvoke方法,继续向下执行rpc调用的逻辑。
/*** AbstractClusterInvoker的方法* <p>* rpc调用的模版方法实现*/@Overridepublic Result invoke(final Invocation invocation) throws RpcException {//销毁检测checkWhetherDestroyed();// binding attachments into invocation.
// Map<String, Object> contextAttachments = RpcContext.getClientAttachment().getObjectAttachments();
// if (contextAttachments != null && contextAttachments.size() != 0) {
// ((RpcInvocation) invocation).addObjectAttachmentsIfAbsent(contextAttachments);
// }InvocationProfilerUtils.enterDetailProfiler(invocation, () -> "Router route.");/** 1 从服务目录中根据路由规则或滤出满足规则的服务提供者invoker列表*/List<Invoker<T>> invokers = list(invocation);InvocationProfilerUtils.releaseDetailProfiler(invocation);/** 2 获取负载均衡策略实现*/LoadBalance loadbalance = initLoadBalance(invokers, invocation);RpcUtils.attachInvocationIdIfAsync(getUrl(), invocation);InvocationProfilerUtils.enterDetailProfiler(invocation, () -> "Cluster " + this.getClass().getName() + " invoke.");try {/** 3 继续执行rpc调用*/return doInvoke(invocation, invokers, loadbalance);} finally {InvocationProfilerUtils.releaseDetailProfiler(invocation);}}
FailoverClusterInvoker#doInvoke失败重试调用
此前我们通过服务路由获取了符合规则的服务提供者invoker列表,以及获取了服务负载均衡实例,下面将会进行带有容错机制的rpc调用。
FailoverClusterInvoker采用失败自动切换机制,Dubbo默认的容错策略。服务消费方调用失败后自动切换到其他服务提供者的服务器进行重试。客户端等待服务端的处理时间超过了设定的超时时间时,也算做失败,将会重试。可通过 retries属性来设置重试次数(不含第一次),默认重试两次。
通常用于读操作或者具有幂等的写操作,需要注意的是重试会带来更长延迟。大概步骤为:
- 计算最大调用次数,默认3,包括两次失败重试,最小1次。在一个循环中执行调用。
- 每次重试前,重新调用list方法从服务目录中根据路由规则或滤出满足规则的服务提供者invoker列表。可想而知,如果此时服务有变更,那么invoked列表将失去准确性。
- 基于负载均衡策略选择一个服务提供者invoker。
- 通过服务提供者invoker执行rpc调用获取结果。
/*** FailoverClusterInvoker的方法* * 失败重试的容错策略** @param invocation 方法调用抽象* @param invokers 服务提供者invoker列表,已经经过了路由过滤* @param loadbalance 负载均衡* @return 执行结果*/
@Override
@SuppressWarnings({"unchecked", "rawtypes"})
public Result doInvoke(Invocation invocation, final List<Invoker<T>> invokers, LoadBalance loadbalance) throws RpcException {List<Invoker<T>> copyInvokers = invokers;//可用invoker列表不能为空,否则直接抛出异常checkInvokers(copyInvokers, invocation);//方法名String methodName = RpcUtils.getMethodName(invocation);/** 计算最大调用次数,默认3,包括两次失败重试,最小1次* retries属性*/int len = calculateInvokeTimes(methodName);// retry loop.RpcException le = null; // last exception.//已调用过的invoker列表List<Invoker<T>> invoked = new ArrayList<Invoker<T>>(copyInvokers.size()); // invoked invokers.//已调用过的provider列表Set<String> providers = new HashSet<String>(len);for (int i = 0; i < len; i++) {//重试前重新从服务目录中根据路由规则或滤出满足规则的服务提供者invoker列表//如果此时服务有变更,那么invoked列表将失去准确性if (i > 0) {checkWhetherDestroyed();//重新服务路由copyInvokers = list(invocation);// check againcheckInvokers(copyInvokers, invocation);}/** 基于负载均衡策略选择一个invoker*/Invoker<T> invoker = select(loadbalance, invocation, copyInvokers, invoked);//添加到已调用列表invoked.add(invoker);RpcContext.getServiceContext().setInvokers((List) invoked);boolean success = false;try {/** 使用服务提供者invoker执行rpc服务调用*/Result result = invokeWithContext(invoker, invocation);if (le != null && logger.isWarnEnabled()) {logger.warn("Although retry the method " + methodName+ " in the service " + getInterface().getName()+ " was successful by the provider " + invoker.getUrl().getAddress()+ ", but there have been failed providers " + providers+ " (" + providers.size() + "/" + copyInvokers.size()+ ") from the registry " + directory.getUrl().getAddress()+ " on the consumer " + NetUtils.getLocalHost()+ " using the dubbo version " + Version.getVersion() + ". Last error is: "+ le.getMessage(), le);}success = true;return result;} catch (RpcException e) {if (e.isBiz()) { // biz exception.throw e;}le = e;} catch (Throwable e) {le = new RpcException(e.getMessage(), e);} finally {//如果未成功if (!success) {//加入已调用过的provider列表providers.add(invoker.getUrl().getAddress());}}}throw new RpcException(le.getCode(), "Failed to invoke the method "+ methodName + " in the service " + getInterface().getName()+ ". Tried " + len + " times of the providers " + providers+ " (" + providers.size() + "/" + copyInvokers.size()+ ") from the registry " + directory.getUrl().getAddress()+ " on the consumer " + NetUtils.getLocalHost() + " using the dubbo version "+ Version.getVersion() + ". Last error is: "+ le.getMessage(), le.getCause() != null ? le.getCause() : le);
}
AbstractClusterInvoker#select选择invoker
基于负载均衡策略选择一个invoker。
/*** AbstractClusterInvoker的方法* * 基于负载均衡策略选择一个invoker** @param loadbalance 负载均衡策略* @param invocation 调用方法抽象* @param invokers 全部服务提供者invoker列表,已经经过了路由过滤* @param selected 已被调用过的服务提供者invoker列表* @return* @throws RpcException*/
protected Invoker<T> select(LoadBalance loadbalance, Invocation invocation,List<Invoker<T>> invokers, List<Invoker<T>> selected) throws RpcException {if (CollectionUtils.isEmpty(invokers)) {return null;}//调用方法名String methodName = invocation == null ? StringUtils.EMPTY_STRING : invocation.getMethodName();//默认false,设置true 该接口上的所有方法使用同一个provider.如果需要更复杂的规则,请使用路由boolean sticky = invokers.get(0).getUrl().getMethodParameter(methodName, CLUSTER_STICKY_KEY, DEFAULT_CLUSTER_STICKY);//ignore overloaded method 忽略重载方法if (stickyInvoker != null && !invokers.contains(stickyInvoker)) {stickyInvoker = null;}//ignore concurrency problem 忽略并发问题if (sticky && stickyInvoker != null && (selected == null || !selected.contains(stickyInvoker))) {if (availableCheck && stickyInvoker.isAvailable()) {return stickyInvoker;}}/** 继续选择invoker*/Invoker<T> invoker = doSelect(loadbalance, invocation, invokers, selected);if (sticky) {stickyInvoker = invoker;}return invoker;
}
AbstractClusterInvoker#doSelect继续选择invoker
- 如果只有一个服务提供者invoker,那么返回。
- 调用loadbalance#select方法基于负载均衡策略选择一个invoker。这是核心方法,Dubbo提供有很多的负载均衡实现,具体的源码我们后面单独分析。
- 如果invoker在已被调用过的服务提供者invoker列表中,那么调用reselect方法重新选择一个。
- 重新选择仍然失败,采用兜底策略。检查当前选中的invoker的索引,如果不是最后一个,则选择索引+1的invoker,否则返回第一个。
/*** AbstractClusterInvoker的方法* <p>* 基于负载均衡策略选择一个invoker** @param loadbalance 负载均衡策略* @param invocation 调用方法抽象* @param invokers 全部服务提供者invoker列表,已经经过了路由过滤* @param selected 已被调用过的服务提供者invoker列表*/
private Invoker<T> doSelect(LoadBalance loadbalance, Invocation invocation,List<Invoker<T>> invokers, List<Invoker<T>> selected) throws RpcException {if (CollectionUtils.isEmpty(invokers)) {return null;}//如果只有一个服务提供者invoker,那么返回if (invokers.size() == 1) {Invoker<T> tInvoker = invokers.get(0);checkShouldInvalidateInvoker(tInvoker);return tInvoker;}/** 基于负载均衡策略选择一个invoker* 核心方法*/Invoker<T> invoker = loadbalance.select(invokers, getUrl(), invocation);//If the `invoker` is in the `selected` or invoker is unavailable && availablecheck is true, reselect.//如果invoker在selected中,则重新选择。boolean isSelected = selected != null && selected.contains(invoker);//如果availableCheck=true,并且invoker不可用,则重新选择。boolean isUnavailable = availableCheck && !invoker.isAvailable() && getUrl() != null;if (isUnavailable) {invalidateInvoker(invoker);}//重新选择if (isSelected || isUnavailable) {try {/** 重新基于负载均衡策略选择一个invoker*/Invoker<T> rInvoker = reselect(loadbalance, invocation, invokers, selected, availableCheck);if (rInvoker != null) {invoker = rInvoker;} //重新选择仍然失败,兜底策略else {//Check the index of current selected invoker, if it's not the last one, choose the one at index+1.//检查当前选中的invoker的索引,如果不是最后一个,则选择索引+1的invokerint index = invokers.indexOf(invoker);try {//Avoid collisioninvoker = invokers.get((index + 1) % invokers.size());} catch (Exception e) {logger.warn("2-5", "select invokers exception", "", e.getMessage() + " may because invokers list dynamic change, ignore.", e);}}} catch (Throwable t) {logger.error("2-5", "failed to reselect invokers", "", "cluster reselect fail reason is :" + t.getMessage() + " if can not solve, you can set cluster.availablecheck=false in url", t);}}return invoker;
}
FailoverClusterInvoker#invokeWithContext调用服务
将当前invoker设置到服务调用上下文RpcContext中,RpcContext是一个线程本地变量。随后使用服务提供者invoker#invoke执行rpc服务调用。
/*** AbstractClusterInvoker* <p>* 使用服务提供者invoker执行rpc服务调用** @param invoker 服务提供者invoker* @param invocation 方法调用抽象* @return 调用结果*/
protected Result invokeWithContext(Invoker<T> invoker, Invocation invocation) {//将当前invoker设置到服务调用上下文RpcContext中,RpcContext是一个线程本地变量setContext(invoker);Result result;try {if (ProfilerSwitch.isEnableSimpleProfiler()) {InvocationProfilerUtils.enterProfiler(invocation, "Invoker invoke. Target Address: " + invoker.getUrl().getAddress());}/** 使用服务提供者invoker执行rpc服务调用*/result = invoker.invoke(invocation);} finally {//清除上下文中的invoker属性clearContext(invoker);InvocationProfilerUtils.releaseSimpleProfiler(invocation);}return result;
}
总结
本次我们学习了,Dubbo 发起服务调用的上半部分源码,实际上就是按照Invoker一层层的调用,每一层的Invoker又不同的功能,通过Invoker层层包装实现一些类似于AOP的能力。
调用过程中也会执行一个Filter链表,用于执行一些额外的逻辑,最终默认会执行到FailoverClusterInvoker,这是一个失败重试的Invoker,是Dubbo默认的容错策略,会给予负载均衡策略选择一个真实的服务提供者Invoker发起远程RPC调用。
调用过程中,Invoker全程参与,这里我们也能明白为什么Invoker被称为可执行体,因为其内部封装了Dubbo远程RPC调用的各种逻辑:服务路由、负载均衡、失败容错等等,也能明白Invoker作为Dubbo核心模型的重要性了。
整体看下来,Dubbo服务调用的源码是不是比此前服务注册和服务发现的源码简单得多了呢?下文我们学习Dubbo 发起服务调用的下半部分源码,也就是真正的RPC调用的源码。
相关文章:
Dubbo 3.x源码(29)—Dubbo Consumer服务调用源码(1)服务调用入口
基于Dubbo 3.1,详细介绍了Dubbo Consumer服务调用源码。 此前我们学习了Dubbo服务的导出和引入的源码,现在我们来学习Dubbo服务调用的源码。 此前的文章中我们讲过了最上层代理的调用逻辑(服务引用bean的获取以及懒加载原理):业务引入的接口…...
Linux内核同步机制:确保系统稳定与高效
在复杂而庞大的 Linux 系统世界中,内核就如同一位有条不紊的指挥官,协调着各种任务和资源的分配。而其中,内核同步机制则是确保整个系统稳定与高效运行的关键要素。想象一下,众多的进程和线程在 Linux 内核的舞台上同时登场&#…...
firebase简介
Firebase 是一个由 Google 提供的移动应用开发平台,旨在帮助开发者快速构建和管理应用程序。它提供了一系列强大的工具和服务,特别适合用于开发和管理 Web 和移动应用。以下是 Firebase 的一些核心功能: 实时数据库:Firebase 提供…...
利用Termux在安卓手机中安装 PostgreSQL
利用Termux在安卓手机中安装 PostgreSQL ⬇️Termux下载 点击下载 在 Termux 中安装 PostgreSQL 可以按照以下步骤进行: 1. 更新 Termux 包管理器 先更新软件包列表和已安装的软件包: pkg update && pkg upgrade -y2. 安装 PostgreSQL 使…...
windows安装WSL完整指南
本文首先介绍WSL,然后一步一步安装WSL及Ubuntu系统,最后讲解如何在两个系统之间访问和共享文件信息。通过学习该完整指南,能帮助你快速安装WSL,解决安装和使用过程中的常见问题。 理解WSL(Windows Subsystem for Linux…...
Windows Docker笔记-安装docker
安装环境 操作系统:Windows 11 家庭中文版 docker版本:Docker Desktop version: 4.36.0 (175267) 注意: Docker Desktop 支持以下Windows操作系统: 支持的版本:Windows 10(家庭版、专业版、企业版、教育…...
ReactNative进阶(五十九):存量 react-native 项目适配 HarmonyOS NEXT
文章目录 一、前言二、ohos_react_native2.1 Fabric2.2 TurboModule2.2.1 ArkTSTurboModule2.2.2 cxxTurboModule: 三、拓展阅读 一、前言 2024年10月22日19:00,华为在深圳举办“原生鸿蒙之夜暨华为全场景新品发布会”,主题为“星河璀璨&…...
[x86 ubuntu22.04]进入S4失败
目录 1 问题描述 2 解决过程 2.1 查看内核日志 2.2 新建一个交换分区 2.3 指定交换分区的位置 1 问题描述 CPU:G6900E OS:ubuntu22.04 Kernel:6.8.0-49-generic 使用“echo disk > /sys/power/state”命令进入 S4,但是无法…...
Java面试题-MySQL数据库
文章目录 1.事务1.事务的特性 ACID2.并发事务问题3.undo log 和redo log的区别?4.事务的隔离性是如何保证的呢?解释一下MVCC? 2.索引1.如何定位慢查询?2.explain3.了解过索引吗?索引的底层数据结构B树和B树对比4.什么是…...
为什么Vue的data属性是函数而不是对象
Vue中data属性设计为函数而非对象的原因是解决组件复用时的数据隔离问题。确保每个实例维护独立的数据副本,避免数据共享导致的状态污染。 而根实例因为只会被创建一次(不会被复用),所以可以直接用对象,不会有这个问题。 组件一般都会被多个实…...
网络工程师 (26)TCP/IP体系结构
一、层次 四层: 网络接口层:TCP/IP协议的最底层,负责网络层与硬件设备间的联系。该层协议非常多,包括逻辑链路和媒体访问控制,负责与物理传输的连接媒介打交道,主要功能是接收数据报,并把接收到…...
MySQL部署基于二进制日志文件位置的主从复制集群
MySQL主从复制介绍 MySQL 主从复制(Master-Slave Replication) 作为一种经典的数据库复制方案,被广泛应用于企业生产环境,尤其是在提升数据库性能、实现数据备份和分布式扩展方面具有重要作用。 官方文档:https://de…...
【系统设计】Spring、SpringMVC 与 Spring Boot 技术选型指南:人群、场景与实战建议
在 Java 开发领域,Spring 生态的技术选型直接影响项目的开发效率、维护成本和长期扩展性。然而,面对 Spring、SpringMVC 和 Spring Boot 这三个紧密关联的框架,开发者常常陷入纠结:该从何入手?如何根据团队能力和业务需…...
【CAPL实战】LIN调度表操作
文章目录 前言1、linChangeSchedTable切换调度表2、linStartScheduler开启调度表3、linStopScheduler停止调度表 前言 在LIN调度表Schedule Table文章中,详细介绍了LIN调度表的信息,那么如何在CAPL脚本测试中进行LIN调度表的操作呢? 1、linC…...
「vue3-element-admin」告别 vite-plugin-svg-icons!用 @unocss/preset-icons 加载本地 SVG 图标
🚀 作者主页: 有来技术 🔥 开源项目: youlai-mall ︱vue3-element-admin︱youlai-boot︱vue-uniapp-template 🌺 仓库主页: GitCode︱ Gitee ︱ Github 💖 欢迎点赞 👍 收藏 ⭐评论 …...
[图文]课程讲解片段-Fowler分析模式的剖析和实现01
解说: GJJ-004-1,分析模式高阶Fowler分析模式的剖析和实现,这个课是针对Martin Fowler的《分析模式》那本书里面的模式来讲解,对里面的模式来剖析,然后用代码来实现。 做到这一步的,我们这个是世界上独…...
element-plus el-tree-select 修改 value 字段
element-plus el-tree-select 修改 value 字段 ,不显示label 需要注意两个地方: <el-tree-select v-model"value" :data"data" multiple :render-after-expand"false" show-checkbox style"width: 240px" …...
软件测评实验室CNAS认证能力验证什么时机做?如何查询能力验证相关信息?
能力验证是软件测评实验室申请CNAS认证前必须要做的一类质量活动。CNAS软件测评实验室初次认可和扩大认可范围时,申请认可的每个子领域应至少参加过一次相关领域的能力验证且获得满意结果。通过认定认可后,只要存在可获得的能力验证,不同类目…...
Spring Boot 3.4 中 MockMvcTester 的新特性解析
引言 在 Spring Boot 3.4 版本中,引入了一个全新的 MockMvcTester 类,使 MockMvc 测试可以直接支持 AssertJ 断言。本文将深入探讨这一新特性,分析它如何优化 MockMvc 测试并提升测试的可读性。 Spring MVC 示例 为了演示 MockMvcTester 的…...
网安加·百家讲坛 | 刘志诚:以业务为中心的网络安全挑战与机遇
作者简介:刘志诚,乐信集团信息安全中心总监、OWASP广东区域负责人、网安加社区特聘专家。专注于企业数字化过程中网络空间安全风险治理,对大数据、人工智能、区块链等新技术在金融风险治理领域的应用,以及新技术带来的技术风险治理…...
配置 VS Code 调试 ROS Python 脚本:完整步骤
在 Ubuntu 系统上使用 ROS 和 VS Code 进行 Python 开发时,可能会遇到一些环境配置的问题,特别是当需要加载 ROS 环境变量以及确保正确使用 Python 3 环境时。以下是如何配置 launch.json 和 tasks.json 来确保 VS Code 调试环境能够正确加载 ROS 和 Pyt…...
HTTP4种方法(GET、POST、 PUT和DELETE)
一、GET 和 POST 1. GET方法 特点: 用途:用于从服务器获取数据。 参数传递方式:参数会附加在URL后面,以 keyvalue的形式,通过查询字符串传递,例如: http://example.com/page?nameJohn&…...
AnythingLLM开发者接口API测试
《Win10OllamaAnythingLLMDeepSeek构建本地多人访问知识库》见上一篇文章,本文在上篇基础上进行。 1.生成本地API 密钥 2.打开API测试页面(http://localhost:3001/api/docs/) 就可以在页面测试API了 2.测试获取用户接口(/v1/admin/users) 3…...
CSS定位简介
目录 一、静态定位(Static Positioning) 二、相对定位(Relative Positioning) 三、绝对定位(Absolute Positioning) 四、固定定位(Fixed Positioning) 五、黏性定位(…...
CentOS服务器部署Docker+Jenkins持续集成环境
一、准备工作 一台运行 CentOS 的服务器,确保有足够的磁盘空间、内存资源,并且网络连接稳定。建议使用 CentOS 7 或更高版本,本文以 CentOS 7 为例进行讲解。 拥有服务器的 root 权限,因为后续安装软件包、配置环境等操作需要较…...
React受控组件的核心原理与实战精要
在 React 中,受控组件(Controlled Component) 是一种重要的模式,用于通过组件的状态来管理表单元素的值。这种模式不仅确保了数据的一致性和可预测性,还便于与其他功能(如验证和格式化)集成。本…...
基于python多线程多进程爬虫的maa作业站技能使用分析
基于python多线程多进程爬虫的maa作业站技能使用分析 技能使用分析 多线程(8核) import json import multiprocessing import requests from multiprocessing.dummy import Pooldef maa(st):url "https://prts.maa.plus/copilot/get/"m …...
Android studio怎么创建assets目录
在Android Studio中创建assets文件夹是一个简单的步骤,通常用于存储不需要编译的资源文件,如文本文件、图片、音频等 main文件夹,邮件new->folder-assets folder...
解锁 DeepSeek 模型高效部署密码:蓝耘平台全解析
💖亲爱的朋友们,热烈欢迎来到 青云交的博客!能与诸位在此相逢,我倍感荣幸。在这飞速更迭的时代,我们都渴望一方心灵净土,而 我的博客 正是这样温暖的所在。这里为你呈上趣味与实用兼具的知识,也…...
【Spring相关知识】Spring应用如何优雅使用消息队列
文章目录 概述**核心概念****使用场景****快速入门**1. 添加依赖2. 配置 Binder3. 定义消息通道4. 发送和接收消息5. 运行应用 **高级特性****优点****适用场景** 概述 Spring Cloud Stream 是一个用于构建消息驱动微服务的框架,它基于 Spring Boot 和 Spring Inte…...
2025牛客寒假算法基础集训营4(补题)
C Tokitsukaze and Balance String (hard) 一道规律题。赛时以为是难的算法题,就没去碰了,实际上把几种情况列出来后可能就会发现,只有首尾相同的字符串才是平衡的。 首先我们容易发现,连续的1或者0是多余的,因为他们…...
.net一些知识点5
1.dot Net带out的参数如何使用 string name;//假设这个参数带out TestMethod(1,out name);//一定要有out 方法体中,一定要有out参数的赋值,并且能输出 2.参数的传递方式有哪些 a.值传递 b.引用传递 ref c.输出传递 out 3.设计模式知道哪些 3.us…...
基于Servlet简易学生信息管理系统
本次设计的学生信息管理系统,能提供以下功能: (1) 输入入学生信息并保存 (2) 显示所有学生信息 (3) 查询学生信息 (4) 修改学生信息并保存 (…...
IDEA编写SpringBoot项目时使用Lombok报错“找不到符号”的原因和解决
目录 概述|背景 报错解析 解决方法 IDEA配置解决 Pom配置插件解决 概述|背景 报错发生背景:在SpringBoot项目中引入Lombok依赖并使用后出现"找不到符号"的问题。 本文讨论在上述背景下发生的报错原因和解决办法,如果仅为了解决BUG不论原…...
JVM图文入门
往期推荐 【已解决】redisCache注解失效,没写cacheConfig_com.howbuy.cachemanagement.client.redisclient#incr-CSDN博客 【已解决】OSS配置问题_keyuewenhua.oss-cn-beijing.aliyuncs-CSDN博客 【排坑】云服务器docker部署前后端分离项目域名解析OSS-CSDN博客 微服…...
【算法】动态规划专题⑨ —— 二维费用背包问题 python
目录 前置知识进入正题实战演练 前置知识 【算法】动态规划专题⑤ —— 0-1背包问题 滚动数组优化 python 进入正题 二维费用背包问题 方法思路 二维费用背包问题在传统背包问题的基础上增加了第二个维度的限制(如重量)。 每个物品具有两种费用&#x…...
链表专题-02
链表专题 /*** 链表的节点* param <E>*/ public class ListNode<E> {public E element;public ListNode<E> next;public ListNode() {}public ListNode(E element) {this.element element;}public ListNode(E element, ListNode<E> next) {this.eleme…...
亚远景-精通ASPICE:专业咨询助力汽车软件开发高效合规
在竞争日益激烈的汽车行业,软件开发已成为决定成败的关键因素。ASPICE(汽车软件过程改进和能力确定) 作为行业公认的软件开发框架,为汽车制造商和供应商提供了实现高效、合规开发的路线图。 然而,ASPICE 的实施并非易…...
HALCON 数据结构
目录 1. HALCON基本数据分类 1.1 图像相关数据 1.1.1 Image(图片) 1.1.2 Region(区域) 1.1.3 XLD(轮廓) 1.2 控制类数据 1.2.1 基本控制数据类型 1.2.2 handle(句柄) 2. 数组与字典 2.1 数组类型及特点 2.1.1 Iconic数组(Objects) 2.1.2 Control数组(Tu…...
动手写ORM框架 - GeeORM第一天 database/sql 基础
文章目录 1 初识 SQLite2 database/sql 标准库3 实现一个简单的 log 库4 核心结构 Session本文是7天用Go从零实现ORM框架GeeORM的第一篇。介绍了 SQLite 的基础操作(连接数据库,创建表、增删记录等)。使用 Go 语言标准库 database/sql 连接并操作 SQLite 数据库,并简单封装…...
ubuntu conda运行kivy时报“No matching FB config found”
错误描述:本人使用ubuntu自带的python环境运行kivy是没有问题的,就是在使用conda时发生了错误,去网上寻找报错原因,却一直没有头绪(这个问题有诸多问题导致的,不敢说用我的这个方法100%能好) 1…...
SSM开发(十一) mybatis关联关系多表查询(嵌套查询,举例说明)
目录 一、背景介绍 二、一对一查询(嵌套查询) 三、一对多查询(嵌套查询) 四、嵌套查询效率评估 注:关联查询则是指在一个查询中涉及到多个表的联合查询 一、背景介绍 当对数据库的操作涉及到多张表,这在面向对象语言如Java中就涉及到了对象与对象之间的关联关系。针对多…...
【AIGC】冷启动数据与多阶段训练在 DeepSeek 中的作用
博客主页: [小ᶻ☡꙳ᵃⁱᵍᶜ꙳] 本文专栏: AIGC | ChatGPT 文章目录 💯前言💯冷启动数据的作用冷启动数据设计 💯多阶段训练的作用阶段 1:冷启动微调阶段 2:推理导向强化学习(RL࿰…...
Spring(26) spring-security-oauth2 官方表结构解析
目录 一、什么是 spring-security-oauth2?二、spring-security-oauth2 的表结构2.1 oauth_client_details 客户端详细信息表2.2 oauth_access_token 认证授权Token记录表2.3 oauth_refresh_token 刷新授权Token记录表2.4 oauth_code 授权Code记录表 一、什么是 spri…...
WPS如何接入DeepSeek(通过JS宏调用)
WPS如何接入DeepSeek 一、文本扩写二、校对三、翻译 本文介绍如何通过 WPS JS宏调用 DeepSeek 大模型,实现自动化文本扩写、校对和翻译等功能。 一、文本扩写 1、随便打开一个word文档,点击工具栏“工具”。 2、点击“开发工具”。 3、点击“查看代码”…...
项目的虚拟环境的搭建与pytorch依赖的下载
文章目录 配置环境 pytorch的使用需要安装对应的cuda 在PyTorch中使用CUDA, pytorch与cuda不同版本对应安装指南,查看CUDA版本,安装对应版本pytorch 【超详细教程】2024最新Pytorch安装教程(同时讲解安装CPU和GPU版本) 配置环境…...
[每周一更]-(第133期):Go中MapReduce架构思想的使用场景
文章目录 **MapReduce 工作流程**Go 中使用 MapReduce 的实现方式:**Go MapReduce 的特点****哪些场景适合使用 MapReduce?**使用场景1. 数据聚合2. 数据过滤3. 数据排序4. 数据转换5. 数据去重6. 数据分组7. 数据统计8.**统计文本中单词出现次数****代码…...
C 移位运算符
宏定义 #define GET_BIT(n) ((1 << (n))) 用于生成一个整数,该整数在第 n 位上是 1,其余位都是 0。这个宏通常用于位操作,比如设置、清除或检查某个特定位置的标志位。 1 << (n):这是位移操作符。它将数字 1 左移 n …...
redis高级数据结构布隆过滤器
文章目录 背景什么是布隆过滤器Redis 中的布隆过滤器布隆过滤器使用注意事项实现原理空间占用估计 背景 我们在使用新闻客户端看新闻时,它会给我们不停地推荐新的内容,它每次推荐时要去重,去掉那些已经看过的内容。问题来了,新闻…...
活动预告 |【Part1】Microsoft 安全在线技术公开课:安全性、合规性和身份基础知识
课程介绍 通过参加“Microsoft 安全在线技术公开课:安全性、合规性和身份基础知识”活动提升你的技能。在本次免费的介绍性活动中,你将获得所需的安全技能和培训,以创造影响力并利用机会推动职业发展。你将了解安全性、合规性和身份的基础知识…...