google-apis
1 | http_archive( |
google api common protos
1 | com_google_googleapis 包含了 common protos; 暂时保留 |
1 | http_archive( |
1 | com_google_googleapis 包含了 common protos; 暂时保留 |
1 | 参数 |
1 | 参数 |
1 | 示例 |
1 | 格式 |
1 | 示例 |
1 | tr: translate characters, 转换和删除字符 |
1 | 排序 |
1 | 删除重复行,一般与sort结合使用 |
1 | format |
1 | 美化 json 字符串 |
1 | 统计程序运行时间 |
1 | java -cp "./*.jar:lib/*.jar" path.to.my.class.Main # windows: "./*.jar;lib/*.jar" |
由程序负责任务切换,可以减少线程/进程上下文切换的消耗。用户态实现任务切换,无需进入内核态。
虽然 Python 有多线程的概念,但是由于 GIL 的存在,并不能利用多核资源。如果易不能充分利用单进程资源,可能会带来严重的性能问题。
python 默认只为主线程创建 loop。如下 tornado 代码实现了自动为创建 loop 的功能,使用 asyncio.set_event_loop_policy(AnyThreadEventLoopPolicy())
来生效。
1 | if sys.platform == "win32" and hasattr(asyncio, "WindowsSelectorEventLoopPolicy"): |
下面是使用协程实现的定时器。
1 | # coding: utf-8 |
下看下抛出异常的代码。
1 | def ensure_future(coro_or_future, *, loop=None): |
ServiceDiscovery
,创建 ServiceProvider
对象,首先需要有 ServiceDiscovery
;所有请求直接访问 zkServiceProvider
, 特定服务发现的封装,并集成了负载均衡策略;集成了 ServiceCache
,有节点监听和缓存ServiceCache
,会在本地内存缓存,并使用 watcher 来保持数据最新ServiceDiscovery
、ServiceProvider
需要调用 start
方法后可用。
ServiceDiscovery
的 registerService
注册服务后,只要 ServiceDiscovery
不 stop
,会一直保持节点注册stop
,没有及时调用 unregisterService
接口来取消注册,zk 节点会保存一段时间(几秒),然后由 zk 摘除ServiceProvider
的接口,会实时调用 zk 查询数据,ServiceCacheListener
有两个方法。
cacheChanged
当服务节点变化时,会调用该方法stateChanged
当 zk 连接状态变化时,会调用该方法gRPC 代码实现中,Channel 是一个虚拟类,是物理连接的逻辑概念。ManagedChannelImpl 和 ManagedChannelImpl2 继承了该类,并实现了 newCall 和 authority 接口。
SubChannel 是 LoadBalancer 的内部类,在了解 SubChannel 之前,需要先了解 SocketAddress 和 EquivalentAddressGroup。
SocketAddress 是一个虚拟类,代表一个不包含协议信息的 Socket 地址。
EquivalentAddressGroup 是一组 SocketAddress,在 Channel 创建连接时,其包含的 SocketAddress 被视为无差异的。
再回到 SubChannel,他表示和一个服务器或者 EquivalentAddressGroup 表示的一组等价服务器的 一个物理连接,这里需要注意的是,他至多有一个物理连接。在发起新的 RPCs 时,如果没有激活的 transport,在被安排 RPC 请求时,会创建 transport。调用 requestConnection ,会请求 Subchannel 在没有 transport 的情况下创建 transport。
SubChannel 有一个 List<EquivalentAddressGroup>
属性,可以通过 setAddresses(List<EquivalentAddressGroup> addrs)
和 setAddresses(EquivalentAddressGroup addrs)
设置。
InternalSubchannel 表示一个 SocketAddress 的 Transports 。他实现了 TransportProvider 接口,定义了 obtainActiveTransport
方法,该方法如果没有激活的 transports,则会调用 startNewTransport
进行创建。
在创建 transports 时,需要先获取 SocketAddress。在创建 InternalSubChannel 时,会传入 List<EquivalentAddressGroup>
。需要注意的是,InternalSubChannel 默认使用 第一个 EquivalentAddressGroup 的 第一个 SocketAddress ,只有在 transport 被关闭时,才会尝试下一个服务地址。
尝试完所有的地址,全部失败后,此时 InternalSubChannel 处于 TRANSIENT_FAILURE 状态,等待一个 delay 时间后,重新尝试。
NameResolver 是一个可插拔的组件(pluggable component),代码层面是一个接口,用来解析一个 target(URI),并返回给调用方一个地址列表,gRPC 内置了 DnsNameResolver。
地址的返回是基于 Listener 机制,NameResolver 的实现类,需要定义 start 方法,方法会传入一个 Listener,当服务列表发生变化时,调用 Listener 的 onResult 方法,通知 NameResolver 的持有方。
LoadBalancer 是一个可插拔的组件,接受 NameResolver 拿到的服务方列表,提供可用的 SubChannel。
从 RoundRobinLoadBalancer 的 handleResolvedAddresses 实现可以发现。
每次刷新时
Channel 去执行一次远程调用,是通过 newCall 方法,传入 **MethodDescriptor ** 和 CallOptions。对于 ManagedChannelImpl,其实现 Channel 接口,newCall 方法转而调用 InterceptorChannel 的 newCall,先简单说下 InterceptorChannel。
managedChannelImpl 字段是调用
ClientInterceptors.intercept(channel, interceptors)
构造,先说 InterceptorChannel 再说 ClientInterceptors。InterceptorChannel 将 Interceptor 和 Channel 的结合,由 channel + interceptor 构造,调用 channel 的 newCall 时,会执行 interceptor 的 interceptCall,该调用会传入 channel。对于一个原始 channel 和 多个 interceptor,先将 interceptor 倒序,然后依次创建 InterceptorChannel,进行包装。
1
2
3 for (ClientInterceptor interceptor : interceptors) {
channel = new InterceptorChannel(channel, interceptor);
}相比之下,ClientInterceptors 只是一个工具类。
接着,怎么用上 NameSolver 的。在构造 interceptorChannel 时,传入一个 channel。这个channel 是一个 RealChannel 对象。
RealChannel 实现了 Channel 接口。
这里需要再提一下,Channel 是逻辑连接,SubChannel 是物理连接。ManagedChannelImpl 实现了 Channel 接口,同时,有一个内部类 SubchannelImpl 实现 SubChannel。
首先调用 SubchannelImpl 的 requestConnection
,在方法内会调用 InternalSubchannel
的 obtainActiveTransport
创建和 SocketAddress 的连接。
在 ManagedChannelImpl 内,定义了 NameResolverListener,实现了NameResolver.Listener 接口,在 NameResolverListener 内做了 LoadBalancer 和 NameResolver 的结合。
在 NameResolverListener 的 onResult 方法内,当服务器地址变更时会执行改方法。首先,会从参数中获取服务端列 表List<EquivalentAddressGroup>
,接下来调用 AutoConfiguredLoadBalancer 的 tryHandleResolvedAddresses
方法,再调用 LoadBalancer 的 handleResolvedAddresses
。整个调用实现比较绕,直接了解内置的 LB 即可。
File > Preferences > Editor > Dynamically Adjust Node Size to Label Size
新建节点时,node 大小根据 label 大小动态调整。
不启用效果
启用效果
Tools > Fit Node To Label
假设,有如下 node。
调整后效果如下。
1 | brew install rsync |
1 | P 等价于 --partial --progress |
1 | rsync -Pav <local> <user@ip:remote-dist> |
使用工具 fswatch。
1 | 安装依赖 |