带有Kafka和ZeroMQ的分布式类星体演员

因此,您已经有了使用actor的精美设计,选择了JVM和Quasar在该主题上的强大而忠实的观点。 所有明智的决定,但是在集群上进行分配时您有什么选择呢?

星系

Galaxy是一个非常酷的选择:快速的内存中数据网格,针对数据局部性进行了优化,具有复制,可选的持久性,分布式参与者注册表,甚至参与者之间的参与者迁移! 只有一个警告:要发布正式版的生产质量的银河版,还需要几个月的时间。 不建议将当前版本的Galaxy用于生产。

如果我们需要在那之前上线怎么办?

幸运的是,Quasar Actors的阻塞编程模型非常简单,以至于将其与大多数消息传递解决方案集成起来都是轻而易举的,并且为了证明让我们用两种快速,流行且截然不同的模型来做到这一点: Apache Kafka和ØMQ 。

代码和计划

可以在GitHub上找到以下所有示例,只需快速阅读简短的README ,即可立即运行它们。

Kafka和ØMQ分别有两个示例:

  • 快速而肮脏的人直接进行发布/投票或发送/接收演员的呼叫。
  • 详细介绍了代理角色,这些代理角色将您的代码与消息传递API隔离开。 为了证明我没有在说谎,该程序对两种技术使用了相同的生产者和消费者参与者类 ,并且几乎使用了相同的引导程序。

卡夫卡

Apache Kafka的采用率急剧上升,这是由于其基于持久性提交日志和用于并行消息使用的使用者组的独特设计:这种结合形成了快速,可靠,灵活和可扩展的代理。

该API包括两种类型的生产者:sync和async;一种消费者(仅sync); Comsat包括社区贡献的,对光纤友好的Kafka生产商集成 。

Kafka生产者句柄是线程安全的,在共享时表现最佳,并且可以像这样在actor主体(或其他任何地方)中轻松获得和使用:

final Properties producerConfig = new Properties();
producerConfig.put("bootstrap.servers", "localhost:9092");
producerConfig.put("client.id", "DemoProducer");
producerConfig.put("key.serializer", "org.apache.kafka.common.serialization.IntegerSerializer");
producerConfig.put("value.serializer", "org.apache.kafka.common.serialization.ByteArraySerializer");try (final FiberKafkaProducer<Integer, byte[]> producer = new FiberKafkaProducer<>(new KafkaProducer<>(producerConfig))) {final byte[] myBytes = getMyBytes(); // ...final Future<RecordMetaData> res = producer.send(new ProducerRecord<>("MyTopic", i, myBytes));res.get(); // Optional, blocks the fiber until the record is persisted; thre's also `producer.flush()`
}

我们用Comsat的FiberKafkaProducer包装了KafkaProducer对象,以便找回光纤阻塞的未来。

但是,使用者句柄不是线程安全的1,并且仅是线程阻塞的:

final Properties producerConfig = new Properties();
consumerConfig = new Properties();
consumerConfig.put(ConsumerConfig.BOOTSTRAP_SERVERS_CONFIG, BOOTSTRAP);
consumerConfig.put(ConsumerConfig.GROUP_ID_CONFIG, "DemoConsumer");
consumerConfig.put(ConsumerConfig.ENABLE_AUTO_COMMIT_CONFIG, "true");
consumerConfig.put(ConsumerConfig.AUTO_COMMIT_INTERVAL_MS_CONFIG, "1000");
consumerConfig.put(ConsumerConfig.SESSION_TIMEOUT_MS_CONFIG, "30000");
consumerConfig.put(ConsumerConfig.KEY_DESERIALIZER_CLASS_CONFIG, "org.apache.kafka.common.serialization.IntegerDeserializer");
consumerConfig.put(ConsumerConfig.VALUE_DESERIALIZER_CLASS_CONFIG, "org.apache.kafka.common.serialization.ByteArrayDeserializer");try (final Consumer<Integer, byte[]> consumer = new KafkaConsumer<>(consumerConfig)) {consumer.subscribe(Collections.singletonList(TOPIC));final ConsumerRecords<Integer, byte[]> records = consumer.poll(1000L);for (final ConsumerRecord<Integer, byte[]> record : records) {final byte[] v = record.value();useMyBytes(v); // ...}
}

由于我们不想阻塞光纤的基础线程池(除了卡夫卡在doRun的线程池,我们无法对其做太多的事情),因此在我们的actor的doRun我们将使用FiberAsync.runBlocking代替FiberAsync.runBlocking来喂入固定的FiberAsync.runBlocking具有异步任务的size执行程序,该任务将阻塞光纤直到poll (将在给定的池中执行)返回之前:

final ExecutorService e = Executors.newFixedThreadPool(2);try (final Consumer<Integer, byte[]> consumer = new KafkaConsumer<>(consumerConfig)) {consumer.subscribe(Collections.singletonList(TOPIC));final ConsumerRecords<Integer, byte[]> records = call(e, () -> consumer.poll(1000L));for (final ConsumerRecord<Integer, byte[]> record : records) {final byte[] v = record.value();useMyBytes(v); // ...}
}

其中call是一个定义如下的实用程序方法(如果不是此Java编译器bug,则没有必要):

@Suspendable
public static <V> V call(ExecutorService es, Callable<V> c) throws InterruptedException, SuspendExecution {try {return runBlocking(es, (CheckedCallable<V, Exception>) c::call);} catch (final InterruptedException | SuspendExecution e) {throw e;} catch (final Exception e) {throw new RuntimeException(e);}
}

在第一个完整的示例中,我们将从生产者角色向消费者发送一千个序列化消息。

ØMQ

ØMQ(或ZeroMQ)不是集中的代理解决方案,而更多地是各种通信模式(请求/答复,发布/订阅等)的套接字的一般化。 在我们的示例中,我们将使用最简单的请求-答复模式。 这是我们的新生产者代码:

try (final ZMQ.Context zmq = ZMQ.context(1 /* IO threads */);final ZMQ.Socket trgt = zmq.socket(ZMQ.REQ)) {trgt.connect("tcp://localhost:8000");final byte[] myBytes = getMyBytes(); // ...trgt.send(baos.toByteArray(), 0 /* flags */)trgt.recv(); // Reply, e.g. ACK
}

如您所见,上下文充当套接字工厂,并传递了要使用的I / O线程数:这是因为ØMQ套接字不是连接绑定的OS句柄,而是用于处理的机器的简单前端重试连接,多个连接,高效的并发I / O甚至为您排队。 这就是为什么send调用几乎永远不会阻塞,而recv调用不是连接上的I / O调用,而是线程与专门的I / O任务之间的同步的原因,该任务将从一个或多个连接中传入字节。

不过,我们将在角色中阻塞光纤,而不是线程,因此让我们在read调用上使用FiberAsync.runBlocking ,以防万一它阻塞甚至在send时阻塞:

final ExecutorService ep = Executors.newFixedThreadPool(2);try (final ZMQ.Context zmq = ZMQ.context(1 /* IO threads */);final ZMQ.Socket trgt = zmq.socket(ZMQ.REQ)) {exec(e, () -> trgt.connect("tcp://localhost:8000"));final byte[] myBytes = getMyBytes(); // ...call(e, trgt.send(myBytes, 0 /* flags */));call(e, trgt::recv); // Reply, e.g. ACK
}

这是消费者:

try (final ZMQ.Context zmq = ZMQ.context(1 /* IO threads */);final ZMQ.Socket src = zmq.socket(ZMQ.REP)) {exec(e, () -> src.bind("tcp://*:8000"));final byte[] v = call(e, src::recv);exec(e, () -> src.send("ACK"));useMyBytes(v); // ...
}

exec是另一个实用程序函数,类似于call

@Suspendable
public static void exec(ExecutorService es, Runnable r) throws InterruptedException, SuspendExecution {try {runBlocking(es, (CheckedCallable<Void, Exception>) () -> { r.run(); return null; });} catch (final InterruptedException | SuspendExecution e) {throw e;} catch (final Exception e) {throw new RuntimeException(e);}
}

这是完整的第一个示例 。

在不改变逻辑的情况下进行分配:与救援人员的松散耦合

很简单,不是吗? 但是,有些令人讨厌的事情:我们与网络另一端的参与者打交道的方式与本地参与者不同。 无论他们位于何处或如何连接,这些都是我们愿意写的演员:

public final class ProducerActor extends BasicActor<Void, Void> {private final ActorRef<Msg> target;public ProducerActor(ActorRef<Msg> target) {this.target = target;}@Overrideprotected final Void doRun() throws InterruptedException, SuspendExecution {for (int i = 0; i < MSGS; i++) {final Msg m = new Msg(i);System.err.println("USER PRODUCER: " + m);target.send(m);}System.err.println("USER PRODUCER: " + EXIT);target.send(EXIT);return null;}
}
public final class ConsumerActor extends BasicActor<Msg, Void> {@Overrideprotected final Void doRun() throws InterruptedException, SuspendExecution {for (;;) {final Msg m = receive();System.err.println("USER CONSUMER: " + m);if (EXIT.equals(m))return null;}}
}

幸运的是,每个演员,无论做什么,都具有相同的非常基本的接口:传入消息队列,称为信箱 。 这意味着我们可以在两个通信参与者之间插入任意数量的中间参与者或代理 ,尤其是我们希望有一个发送代理,它将通过中间件将消息获取到目标主机,并在其中接收接收代理,以捕获传入的消息。并将它们放入目标目的地的邮箱中。

因此,在我们的主程序中,我们将为我们的ProducerActor提供合适的发送代理,然后让ConsumerActor从合适的接收代理接收:

final ProducerActor pa = Actor.newActor(ProducerActor.class, getSendingProxy()); // ...
final ConsumerActor ca = Actor.newActor(ConsumerActor.class);
pa.spawn();
System.err.println("USER PRODUCER started");
subscribeToReceivingProxy(ca.spawn()); // ...
System.err.println("USER CONSUMER started");
pa.join();
System.err.println("USER PRODUCER finished");
ca.join();
System.err.println("USER CONSUMER finished");

让我们看看如何首先使用Kafka然后使用ØMQ来实现这些代理。

卡夫卡男演员代理

代理参与者的工厂将与特定的Kafka主题相关联:这是因为可以对主题进行分区 ,以使多个使用者可以同时读取该主题。 我们希望能够最佳地利用每个主题的最大级别或并发性:

/* ... */ KafkaProxies implements AutoCloseable {/* ... */ KafkaProxies(String bootstrap, String topic) { /* ... */ }// ...
}

当然,我们希望对多个参与者使用一个主题,因此发送代理将指定接收者参与者ID,接收代理将仅将消息转发给绑定到该ID的用户参与者:

/* ... */ <M> ActorRef<M> create(String actorID) { /* ... */ }
/* ... */ void drop(ActorRef ref) throws ExecutionException, InterruptedException { /* ... */ }
/* ... */ <M> void subscribe(ActorRef<? super M> consumer, String actorID) { /* ... */ }
/* ... */ void unsubscribe(ActorRef<?> consumer, String actorID) { /* ... */ }

关闭AutoClosable工厂将告诉所有代理终止,并清理簿记参考:

/* ... */ void close() throws Exception { /* ... */ }

生产者实现是非常简单和无趣的,同时给消费者带来了更多的乐趣,因为它将使用Quasar Actors的选择性接收将传入消息保留在其邮箱中,直到至少有一个订阅的用户actor可以使用它们为止:

@Override
protected Void doRun() throws InterruptedException, SuspendExecution {//noinspection InfiniteLoopStatementfor (;;) {// Try extracting from queuefinal Object msg = tryReceive((Object m) -> {if (EXIT.equals(m))return EXIT;if (m != null) {//noinspection uncheckedfinal ProxiedMsg rmsg = (ProxiedMsg) m;final List<ActorRef> l = subscribers.get(rmsg.actorID);if (l != null) {boolean sent = false;for (final ActorRef r : l) {//noinspection uncheckedr.send(rmsg.payload);sent = true;}if (sent) // Someone was listening, remove from queuereturn m;}}return null; // No subscribers (leave in queue) or no messages});// Something from queueif (msg != null) {if (EXIT.equals(msg)) {return null;}continue; // Go to next cycle -> precedence to queue}// Try receiving//noinspection Convert2Lambdafinal ConsumerRecords<Void, byte[]> records = call(e, () -> consumer.get().poll(100L));for (final ConsumerRecord<Void, byte[]> record : records) {final byte[] v = record.value();try (final ByteArrayInputStream bis = new ByteArrayInputStream(v);final ObjectInputStream ois = new ObjectInputStream(bis)) {//noinspection uncheckedfinal ProxiedMsg rmsg = (ProxiedMsg) ois.readObject();final List<ActorRef> l = subscribers.get(rmsg.actorID);if (l != null && l.size() > 0) {for (final ActorRef r : l) {//noinspection uncheckedr.send(rmsg.payload);}} else {ref().send(rmsg); // Enqueue}} catch (final IOException | ClassNotFoundException e) {e.printStackTrace();throw new RuntimeException(e);}}
}

由于我们还需要处理邮箱,因此我们以足够小的超时来轮询Kafka。 还要注意,许多参与者可以订阅相同的ID,传入的消息将广播给所有参与者。 每个主题创建的接收actor代理(即光纤)的数量,以及池线程和Kafka使用者句柄( consumer是本地线程,因为Kafka使用者不是线程安全的)的数量将等于每个主题的分区数:这使接收吞吐量达到最大。

目前,此实现使用Java序列化在字节之间来回转换消息,但是当然可以使用其他框架,例如Kryo 。

ØMQ演员代理

ØMQ模型是完全去中心化的:既没有经纪人,也没有话题,因此我们可以简单地将ØMQ地址/端点与一组参与者等同,而无需使用额外的参与者ID:

/* ... */ ZeroMQProxies implements AutoCloseable {/* ... */ ZeroMQProxies(int ioThreads) { /* ... */ }/* ... */ <M> ActorRef<M> to(String trgtZMQAddress) { /* ... */ }/* ... */ void drop(String trgtZMQAddress)/* ... */ void subscribe(ActorRef<? super M> consumer, String srcZMQEndpoint) { /* ... */ }/* ... */ void unsubscribe(ActorRef<?> consumer, String srcZMQEndpoint) { /* ... */ }/* ... */ void close() throws Exception { /* ... */ }
}

同样,在这种情况下,并且由于与以前相同的原因,使用者有点有趣,但幸运的是,线程安全性方面的任何问题都因为ØMQ套接字在多个线程中可以正常工作:

@Override
protected Void doRun() throws InterruptedException, SuspendExecution {try(final ZMQ.Socket src = zmq.socket(ZMQ.REP)) {System.err.printf("PROXY CONSUMER: binding %s\n", srcZMQEndpoint);Util.exec(e, () -> src.bind(srcZMQEndpoint));src.setReceiveTimeOut(100);//noinspection InfiniteLoopStatementfor (;;) {// Try extracting from queuefinal Object m = tryReceive((Object o) -> {if (EXIT.equals(o))return EXIT;if (o != null) {//noinspection uncheckedfinal List<ActorRef> l = subscribers.get(srcZMQEndpoint);if (l != null) {boolean sent = false;for (final ActorRef r : l) {//noinspection uncheckedr.send(o);sent = true;}if (sent) // Someone was listening, remove from queuereturn o;}}return null; // No subscribers (leave in queue) or no messages});// Something processable is thereif (m != null) {if (EXIT.equals(m)) {return null;}continue; // Go to next cycle -> precedence to queue}System.err.println("PROXY CONSUMER: receiving");final byte[] msg = Util.call(e, src::recv);if (msg != null) {System.err.println("PROXY CONSUMER: ACKing");Util.exec(e, () -> src.send(ACK));final Object o;try (final ByteArrayInputStream bis = new ByteArrayInputStream(msg);final ObjectInputStream ois = new ObjectInputStream(bis)) {o = ois.readObject();} catch (final IOException | ClassNotFoundException e) {e.printStackTrace();throw new RuntimeException(e);}System.err.printf("PROXY CONSUMER: distributing '%s' to %d subscribers\n", o, subscribers.size());//noinspection uncheckedfor (final ActorRef s : subscribers.getOrDefault(srcZMQEndpoint, (List<ActorRef>) Collections.EMPTY_LIST))//noinspection uncheckeds.send(o);} else {System.err.println("PROXY CONSUMER: receive timeout");}}}
}

更多功能

这篇简短的文章有望使人们一眼就可以看出,由于Quasar的Actor具有顺畅的顺序流程的特性,因此可以无缝地将Quasar的Actor与消息传递解决方案进行接口连接。 当然,可以更进一步,例如:

  • 演员查找和发现 :我们如何提供全球演员命名/发现服务? 例如,Kafka使用ZooKeeper,因此可能值得利用,但ØMQ大量下注于去中心化,故意不提供预先打包的基础。
  • Actor故障管理 :我们如何支持在不同节点中运行的actor之间的故障管理链接和监视?
  • 消息路由 :如何在不更改参与者内部逻辑的情况下动态调整节点与参与者之间的消息流?
  • 角色移动性 :我们如何将角色移动到其他节点,例如,使其更靠近消息源,以提高性能或移动到具有不同安全性的位置?
  • 可伸缩性和容错性 :如何管理参与者节点的添加,删除,死亡和分区? 像Galaxy这样的分布式IMDG和像Kafka这样的基于代理的解决方案通常已经做到了,但是像ØMQ这样的结构级解决方案通常不这样做。
  • 安全性 :我们如何支持相关的信息安全性属性?
  • 测试,记录,监视 :我们如何方便地整体测试,跟踪和监视分布式参与者集合?

这些主题尤其是分布式系统设计的“硬核”,尤其是分布式参与者,因此,有效地解决它们可能需要大量的精力。 Galaxy解决了所有这些问题,但Quasar参与者提供了一个SPI ,涵盖了上述一些主题,并允许与发行技术更紧密地集成。 您可能还对Akka与Quasar + Galaxy之间的比较感兴趣,该比较涵盖了许多此类方面。

就是这样,请与您分布的Quasar演员一起玩乐,并在Quasar-Pulsar用户组中留下有关您的旅程的注释!

  1. 实际上,它也禁止除第一个线程外的任何线程使用。

翻译自: https://www.javacodegeeks.com/2016/05/distributed-quasar-actors-kafka-zeromq.html

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

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

相关文章

JAVA中断线程的方法

JAVA中断线程的方法 Thread.stop, Thread.suspend, Thread.resume 和Runtime.runFinalizersOnExit 这些终止线程运行的方法已经被废弃&#xff0c;使用它们是极端不安全的&#xff01; 现在&#xff0c;如果你要安全有效地终止一个线程&#xff0c;应该采用以下这些方法&a…

最常见的Java异常及其对Java开发人员的评价

我知道我说在完成视频之前我不会张贴帖子&#xff0c;但这一直困扰着我。 至于视频的更新&#xff0c;我有些懒散&#xff0c;有些忙碌&#xff0c;但是我已经准备好记录第一集&#xff0c;这是我第一次有机会&#xff0c;而且我不需要太多的编辑。 无论如何&#xff0c;还是今…

日志系统设计

日志系统设计 2009-12-11 00:46:58| 分类&#xff1a; 技术 | 标签&#xff1a; |字号大中小 订阅 一、重要性日志系统在整个系统架构中的重要性可以称得上基础的基础&#xff0c;但是这一点&#xff0c;都容易被大多数人所忽视。因为日志在很多人看来只是printf。在系…

bzoj 1124 [POI2008]枪战Maf 贪心

[POI2008]枪战Maf Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 741 Solved: 295[Submit][Status][Discuss]Description 有n个人&#xff0c;每个人手里有一把手枪。一开始所有人都选定一个人瞄准&#xff08;有可能瞄准自己&#xff09;。然后他们按某个顺序开枪&#x…

核心API最佳实践——JDK日志分级

核心API最佳实践——JDK日志分级 时间:2005-10-29 08:00 来源:网管之家bitsCN.com 字体:[大 中 小]日志&#xff08;Log&#xff09;是什么&#xff1f;字典对其的解释是"对某种机器工作情况或某项任务进展情况的记载"。对于应用系统来说&#xff0c;日志就应该记录应…

20165234 《Java程序设计》第五周学习总结

第五周学习总结 教材学习内容总结 第七章 内部类与异常类 内部类内部类&#xff1a;在一个类中定义另一个类。 外嵌类&#xff1a;包含内部类的类&#xff0c;称为内部类的外嵌类。 内部类的类体中不能声明类变量和类方法。外嵌类的类体中可以用内部类声明对象&#xff0c;作为…

Java日志操作总结

Java日志操作总结 (2008-04-21 17:39:06)标签&#xff1a; 杂谈 . 使用Jakarta Commons Logging(JCL) 1.1. 概述 Apache的开源日志组件Jakarta CommonsLogging(JCL)提供的是一个日志(Log)接口(Interface)&#xff0c;同时兼顾轻量级和不依赖于具体的日志实现工具。它提供给中…

4-2日装饰器,带参数的装饰器

1&#xff0c;函数的有用信息 from functools import wraps#引用模块 def wrapper(f): # f func1wraps(f)def inner(*args,**kwargs): #聚合#args &#xff08;1,2,3&#xff09;执行函数之前的相关操作ret f(*args,**kwargs) # 打散 1,2,3执行函数之后的相关操作return re…

Apache Storm:如何使用Flux配置KafkaBolt

微型框架中的助焊剂可以帮助我们定义和部署Storm拓扑。 Flux有各种包装器&#xff0c;可帮助您定义所需的流并初始化Bolts和Spouts&#xff08;使用带有或不带有参数的构造函数&#xff0c;并通过反射自动调用自定义配置方法&#xff09;。 您只需要使用Flux就是将其作为依赖…

java 获取当前函数名

import java.text.SimpleDateFormat; import java.util.Date; /** * Java实现类似C/C中的__FILE__、__FUNC__、__LINE__等,主要用于日志等功能中。 * * version 1.0 2011-07-13 * */ public abstract class CommonFunction { /** * 打印日志时获取当前的程序文件名、行号、方法…

美国华尔街拥抱区块链是最大的威胁

Overstock的首席执行官帕特里克伯恩宣布证券交易委员会已批准其计划在区块链上发行股票&#xff0c;该区块链是推动比特币数字货币的巨大在线分类账。 这是一个重要的时刻。至少在理论上&#xff0c;区块链可以更有效&#xff0c;准确和公开地跟踪股票&#xff0c;债券和其他金…

java枚举和枚举类_Java枚举:您拥有优雅,优雅和力量,这就是我所爱!

java枚举和枚举类当Java 8即将面世时&#xff0c;您确定您对Java 5中引入的枚举很熟悉吗&#xff1f; Java枚举仍然被低估了&#xff0c;很可惜&#xff0c;因为它们比您想象的要有用&#xff0c;它们不仅仅用于通常的枚举常量&#xff01; Java枚举是多态的 Java枚举是可以包…

Java关键字final、static使用总结

Java关键字final、static使用总结 一、final 根据程序上下文环境&#xff0c;Java关键字final有“这是无法改变的”或者“终态的”含义&#xff0c;它可以修饰非抽象类、非抽象类成员方法和变量。你可能出于两种理解而需要阻止改变&#xff1a;设计或效率。 final类不能被继承…

【SpringCloud】第五篇: 路由网关(zuul)

前言: 必需学会SpringBoot基础知识 简介: spring cloud 为开发人员提供了快速构建分布式系统的一些工具&#xff0c;包括配置管理、服务发现、断路器、路由、微代理、事件总线、全局锁、决策竞选、分布式会话等等。它运行环境简单&#xff0c;可以在开发人员的电脑上跑。 工具:…

ubuntu shell简介

ubuntu shell简介 (2012-03-13 19:48:09) 标签&#xff1a; 杂谈 分类&#xff1a; 虚拟机下的ubuntu8.04 1.什么是shell&#xff1f; shell是接受用户或应用层的命令&#xff0c;并将这些命令解释给底层的系统内核层&#xff0c;由这些内核完成相应的工作&#xff0c;并将结果…

如何更新Jenkins作业发布config.xml

最近&#xff0c;我想更新Cloudbees中的一些作业&#xff08;未使用DSL定义&#xff09;&#xff0c;为每个作业添加一些属性。 好吧&#xff0c;我在使其工作时遇到了一些麻烦&#xff0c;这是我的注意事项&#xff08;我使用的是Jenkins 1.651.2.1&#xff0c;但有可能它应与…

月下“毛景树”

Description 毛毛虫经过及时的变形&#xff0c;最终逃过的一劫&#xff0c;离开了菜妈的菜园。 毛毛虫经过千山万水&#xff0c;历尽千辛万苦&#xff0c;最后来到了小小的绍兴一中的校园里。爬啊爬~爬啊爬~~毛毛虫爬到了一颗小小的“毛景树”下面&#xff0c;发现树上长着他最…

自由口通信模式下计算机读写PLC存储区的程序

自由口通信模式下计算机读写PLC存储区的程序 2008-8-20 8:51:00 来源&#xff1a;摘 要&#xff1a;本文介绍了在自由口通信模式下&#xff0c;用计算机读写S7-200 PLC存储区内相邻的多个字节数据的通信程序设计方法&#xff0c;程序设计中采用了多种可靠性措施。 关键词&…

Zookeeper入门

ZooKeeper 是一个开源的分布式协调架&#xff0c;主要用来解决分布式集群中应用系统的一致性问题 本质 分布式的文件存储系统(Zookeeper文件系统监听机制)&#xff0c;是一个基于观察者模式设计的分布式服务管理框架 zookeeper的数据结构 Zookeeper的层次模型称作Data Tree,…

【laravel5.4】重定向带参数

1、 2、重定向回上一页面 3、返回上一页面带参数 转载于:https://www.cnblogs.com/xuzhengzong/p/8715463.html