生产者与消费者以及ActiveMQ

一、 多线程实现生产者与消费者

1.1 生产者与消费者头文件

#pragma once

#include <iostream>

#include <mutex>

#include <thread>

#include <condition_variable>

class ProAndConClass

{

public:

ProAndConClass();

~ProAndConClass();

void printThread();

void addThread(int num);

bool g_flag = false;

private:

std::condition_variable g_cond_add_enable; //计算条件变量

std::condition_variable g_cond_print_enable;//打印条件

std::mutex g_mutex;

int g_value = 0;

bool g_print_able = false; //是否可以打印

};

1.2 实现cpp文件

#include "ProduceAndConsume.h"

//若不在堆上创建类的实例。需要将如下变量定义在全局数据区,不能放在类的内部,类成员变量可能分配在堆或者栈上,而线程是独享栈区的

//std::condition_variable g_cond_add_enable; //计算条件变量

//std::condition_variable g_cond_print_enable;//打印条件

//std::mutex g_mutex;

//int g_value = 0;

//bool g_print_able = false; //是否可以打印

ProAndConClass::ProAndConClass()

{

}

ProAndConClass::~ProAndConClass()

{

}

void ProAndConClass::addThread(int numThread)

{

std::cout << "add thread begin" << std::endl;

while (g_value < numThread)

{

std::unique_lock<std::mutex>my_lock(g_mutex);

g_cond_add_enable.wait(my_lock,

[=] {

return !g_print_able;

});

g_value++;

g_print_able = true;

std::cout << "++add thread" << g_value << std::endl;

g_cond_print_enable.notify_one(); //增加打印

}

//g_flag = false;

std::cout << "add thread leave"<<std::endl;

}

void ProAndConClass::printThread()

{

std::cout << "print thread begin" << std::endl;

while (g_flag)

{

std::unique_lock<std::mutex> my_lock(g_mutex);

g_cond_print_enable.wait(my_lock,

[=] {

return g_print_able;

});

g_print_able = false;

std::cout << "-- print thread" << g_value << std::endl;

g_cond_add_enable.notify_one();//通知增加线程

}

std::cout << "print thread leave" << std::endl;

}

1.3 主函数main

void testProAndCon()

{

ProAndConClass *proandcon = new(std::nothrow)ProAndConClass();//分配在堆上

proandcon->g_flag = true;

/*ProAndConClass proandcon;//如此,需要把变量定义到全局数据段.因为这种形式对象构

//造在栈区,而线程独享栈区

proandcon.g_flag = true;*/

//线程的初始化三种方式:普通函数、类成员函数、函数对象

std::thread thread_add(&ProAndConClass::addThread, proandcon, 10); // 生产者

std::thread thread_print(&ProAndConClass::printThread, proandcon); //消费者

//getchar();

Sleep(1000);

if (thread_add.joinable())

{

std::cout << "join add thread" << std::endl;

thread_add.join();

}

if (thread_print.joinable())

{

std::cout << "join print thread" << std::endl;

thread_print.join();

}

return;

}

1.4 生产者消费者小结

    使用两个线程,一个生产一个消费。两个线程是独享栈区的,所以测试的变量要放在他们可以共同可以操作的到的地方。

二、基于生产者消费者的MQ

2.1 代码调用过程

//Mq也分为生产者和消费者两个类

class
HelloWorldProducer : public Runnable {

private:

Connection* connection;

Session* session;

Destination* destination;

MessageProducer* producer;

int numMessages;

bool useTopic;

bool sessionTransacted;

std::string brokerURI;

private:

HelloWorldProducer(const
HelloWorldProducer&);

HelloWorldProducer& operator=(const
HelloWorldProducer&);

public:

HelloWorldProducer(const std::string&
brokerURI, int numMessages, bool useTopic = false, bool sessionTransacted =
false) :

connection(NULL),

session(NULL),

destination(NULL),

producer(NULL),

numMessages(numMessages),

useTopic(useTopic),

sessionTransacted(sessionTransacted),

brokerURI(brokerURI) {

}

virtual ~HelloWorldProducer(){

cleanup();

}

void close() {

this->cleanup();

}

virtual void run() {

try {

//1 Create a ConnectionFactory

auto_ptr<ConnectionFactory>
connectionFactory(

ConnectionFactory::createCMSConnectionFactory(brokerURI));

// 2Create a Connection

connection =
connectionFactory->createConnection();

connection->start();

// 3Create a Session

if (this->sessionTransacted) {

session =
connection->createSession(Session::SESSION_TRANSACTED);

} else {

session = connection->createSession(Session::AUTO_ACKNOWLEDGE);

}

// 4Create the destination (Topic or Queue)

if (useTopic) {

destination =
session->createTopic("TEST.FOO");

} else {

destination =
session->createQueue("TEST.FOO");

}

// 5Create a MessageProducer from the Session to the Topic
or Queue

producer =
session->createProducer(destination);

producer->setDeliveryMode(DeliveryMode::NON_PERSISTENT);

//6 Create the Thread Id String

string threadIdStr =
Long::toString(Thread::currentThread()->getId());

//7 Create a messages

string text = (string) "Hello
world! from thread " + threadIdStr;

for (int ix = 0; ix <
numMessages; ++ix) {

std::auto_ptr<TextMessage>
message(session->createTextMessage(text));

message->setIntProperty("Integer", ix);

printf("Sent message #%d
from thread %s\n", ix + 1, threadIdStr.c_str());

producer->send(message.get());

}

} catch (CMSException& e) {

e.printStackTrace();

}

}

private:

void cleanup() {

if (connection != NULL) {

try {

connection->close();

} catch (cms::CMSException& ex)
{

ex.printStackTrace();

}

}

// Destroy resources.

try {

delete destination;

destination = NULL;

delete producer;

producer = NULL;

delete session;

session = NULL;

delete connection;

connection = NULL;

} catch (CMSException& e) {

e.printStackTrace();

}

}

};

//消费者类

class
HelloWorldConsumer : public ExceptionListener,

public
MessageListener,

public Runnable {

private:

CountDownLatch latch;

CountDownLatch doneLatch;

Connection* connection;

Session* session;

Destination* destination;

MessageConsumer* consumer;

long waitMillis;

bool useTopic;

bool sessionTransacted;

std::string brokerURI;

private:

HelloWorldConsumer(const
HelloWorldConsumer&);

HelloWorldConsumer& operator=(const
HelloWorldConsumer&);

public:

HelloWorldConsumer(const std::string&
brokerURI, int numMessages, bool useTopic = false, bool sessionTransacted =
false, int waitMillis = 30000) :

latch(1),

doneLatch(numMessages),

connection(NULL),

session(NULL),

destination(NULL),

consumer(NULL),

waitMillis(waitMillis),

useTopic(useTopic),

sessionTransacted(sessionTransacted),

brokerURI(brokerURI) {

}

virtual ~HelloWorldConsumer() {

cleanup();

}

void close() {

this->cleanup();

}

void waitUntilReady() {

latch.await();

}

virtual void run() {

try {

//1 Create a ConnectionFactory

auto_ptr<ConnectionFactory>
connectionFactory(

ConnectionFactory::createCMSConnectionFactory(brokerURI));

// 2Create a Connection

connection =
connectionFactory->createConnection();

connection->start();

connection->setExceptionListener(this);

//3 Create a Session

if (this->sessionTransacted ==
true) {

session =
connection->createSession(Session::SESSION_TRANSACTED);

} else {

session =
connection->createSession(Session::AUTO_ACKNOWLEDGE);

}

//4 Create the destination (Topic or Queue)

if (useTopic) {

destination =
session->createTopic("TEST.FOO");

} else {

destination =
session->createQueue("TEST.FOO");

}

//5 Create a MessageConsumer from the Session to the Topic
or Queue

consumer =
session->createConsumer(destination);

consumer->setMessageListener(this);

std::cout.flush();

std::cerr.flush();

//6 Indicate we are ready for messages.

latch.countDown();

// 7Wait while asynchronous messages come in.

doneLatch.await(waitMillis);

} catch (CMSException& e) {

// Indicate we are ready for
messages.

latch.countDown();

e.printStackTrace();

}

}

//8 Called from the consumer since this class is a registered
MessageListener.

virtual void onMessage(const Message*
message) {

static int count = 0;

try {

count++;

const TextMessage* textMessage =
dynamic_cast<const TextMessage*> (message);

string text = "";

if (textMessage != NULL) {

text =
textMessage->getText();

} else {

text = "NOT A
TEXTMESSAGE!";

}

printf("Message #%d Received:
%s\n", count, text.c_str());

} catch (CMSException& e) {

e.printStackTrace();

}

// Commit all messages.

if (this->sessionTransacted) {

session->commit();

}

// No matter what, tag the count down
latch until done.

doneLatch.countDown();

}

// If something bad happens you see it here
as this class is also been

// registered as an ExceptionListener with
the connection.

virtual void onException(const
CMSException& ex AMQCPP_UNUSED) {

printf("CMS Exception
occurred.  Shutting down
client.\n");

ex.printStackTrace();

exit(1);

}

private:

void cleanup() {

if (connection != NULL) {

try {

connection->close();

} catch (cms::CMSException& ex)
{

ex.printStackTrace();

}

}

// Destroy resources.

try {

delete destination;

destination = NULL;

delete consumer;

consumer = NULL;

delete session;

session = NULL;

delete connection;

connection = NULL;

} catch (CMSException& e) {

e.printStackTrace();

}

}

};

//main

int main(int
argc AMQCPP_UNUSED, char* argv[] AMQCPP_UNUSED) {

activemq::library::ActiveMQCPP::initializeLibrary();

{

std::cout <<
"=====================================================\n";

std::cout << "Starting the
example:" << std::endl;

std::cout << "-----------------------------------------------------\n";

std::string brokerURI =

"failover:(tcp://localhost:61616"

//        "?wireFormat=openwire"

//       
"&transport.useInactivityMonitor=false"

//       
"&connection.alwaysSyncSend=true"

//        "&connection.useAsyncSend=true"

//       
"?transport.commandTracingEnabled=true"

//       
"&transport.tcpTracingEnabled=true"

//       
"&wireFormat.tightEncodingEnabled=true"

")";

bool useTopics = true;

bool sessionTransacted = false;

int numMessages = 2000;

long long startTime =
System::currentTimeMillis();

HelloWorldProducer  producer(brokerURI, numMessages, useTopics);

HelloWorldConsumer consumer(brokerURI,
numMessages, useTopics, sessionTransacted);

// Start the consumer thread.

Thread consumerThread(&consumer);

consumerThread.start();

// Wait for the consumer to indicate that
its ready to go.

consumer.waitUntilReady();

// Start the producer thread.

Thread producerThread(&producer);

producerThread.start();

// Wait for the threads to complete.

producerThread.join();

consumerThread.join();

long long endTime =
System::currentTimeMillis();

double totalTime = (double)(endTime -
startTime) / 1000.0;

consumer.close();

producer.close();

std::cout << "Time to completion
= " << totalTime << " seconds." << std::endl;

std::cout <<
"-----------------------------------------------------\n";

std::cout << "Finished with the
example." << std::endl;

std::cout << "=====================================================\n";

}

activemq::library::ActiveMQCPP::shutdownLibrary();

}

// END SNIPPET:
demo

生产者与消费者以及ActiveMQ的更多相关文章

  1. 【Active入门-2】ActiveMQ学习-生产者与消费者

    1个生产者,1个消费者,使用Queue: 方式1: 生产者将消息发送到Queue中,退出: 然后运行消费者: . 可以看到,可以接收到消息. 方式2: 先运行消费者程序: 然后运行生产者: 消费者见下 ...

  2. 【原创】JMS生产者和消费者【PTP同步接收消息】

    一般步骤: 请求一个JMS连接工i厂. 是用连接工厂创建连接. 启动JMS连接. 通过连接创建session. 获取一个目标. 创建一个生产者,或a.创建一个生产者,b.创建一条JMS消息并发送到目标 ...

  3. 0032ActiveMQ之java编码实现生产者和消费者操作队列queue

    今天学习了入门级的用java编写生产者producer和消费者consumer操作activemq的queue队列,为了之后复习回顾,现做整理如下: maven工程的搭建此处不再讲解,在maven工程 ...

  4. 第3月第2天 find symbolicatecrash 生产者-消费者 ice 引用计数

    1.linux find export find /Applications/Xcode.app/ -name symbolicatecrash -type f export DEVELOPER_DI ...

  5. LMAX Disruptor—多生产者多消费者中,消息复制分发的高性能实现

    解决的问题 当我们有多个消息的生产者线程,一个消费者线程时,他们之间如何进行高并发.线程安全的协调? 很简单,用一个队列. 当我们有多个消息的生产者线程,多个消费者线程,并且每一条消息需要被所有的消费 ...

  6. JAVA并发框架之Semaphore实现生产者与消费者模型

    分类: Java技术      锁和信号量(Semaphore)是实现多线程同步的两种常用的手段.信号量需要初始化一个许可值,许可值可以大于0,也可以小于0,也可以等于0.      如果大于0,表示 ...

  7. java 22 - 19 多线程之生产者和消费者的代码优化

    在之前,是把生产者录入数据和消费者获取数据的所有代码都分别写在各自的类中. 这样不大好 这次把生产者和消费者部分关键代码都写入资源类中: package zl_Thread; public class ...

  8. java 22 - 16 多线程之生产者和消费者的问题

    生产者和消费者问题的描述图 通过上图,我们可以发现: 生产者和消费者使用的都是同一个资源(肉包子) 所以,当使用线程的时候,这两类的锁也是同一把锁(为了避免出现线程安全问题) 例子:学生信息的录入和获 ...

  9. Java实现生产者和消费者

    生产者和消费者问题是操作系统的经典问题,在实际工作中也常会用到,主要的难点在于协调生产者和消费者,因为生产者的个数和消费者的个数不确定,而生产者的生成速度与消费者的消费速度也不一样,同时还要实现生产者 ...

随机推荐

  1. HttpRunner(1)自我介绍

    前言 首先,我们无论学习哪个框架,都要带着问题,带着思考去学习 思考1:HttpRunner是什么? 思考2:HttpRunner的设计模式是什么? 思考3:为什么我们要学习HttpRunner?他的 ...

  2. 2019 Multi-University Training Contest 2 Harmonious Army(最小割)

    题意:给你n个点 每个点都有两种选择 成为战士或者法师 现在给你m个关系 对应这两个人的对应关系的权值A,B,C 思路:按照下面的思路建图跑最小割(要注意权值要乘2 可能存在不整除的情况) #incl ...

  3. UVA442 矩阵链乘 Matrix Chain Multiplication

    题意: 这道题也是在不改变原序列每个元素位置的前提下,看每个元素与他身边的两个元素那个先结合能得到最大的能量 题解: 很明显这是一道区间dp的题目,这道题要断环成链,这道题需要考虑在这个区间上某个元素 ...

  4. PHP的常用函数 持续更新

    PHP的常用函数 前言: 由于害怕遗忘,故在此记录下常用的php函数,以便复习 1 define函数 作用:定义常量 用法 <?php define('a',100); ?> 2 intv ...

  5. Dcoker命令使用详解

    Docker语法说明 docker [OPTIONS] COMMAND [arg...] OPTIONS --config=~/.docker :指定本地客户端配置文件. -D, --debug :开 ...

  6. K8S(01)二进制部署实践-1.15.5

    系列文章说明 本系列文章,可以基本算是 老男孩2019年王硕的K8S周末班课程 笔记,根据视频来看本笔记最好,否则有些地方会看不明白 需要视频可以联系我 目录 系列文章说明 1 部署架构 1.1 架构 ...

  7. python = 赋值顺序 && C++ side effect

    title: python = 赋值顺序 && C++ side effect date: 2020-03-17 15:00:00 categories: [python][c++] ...

  8. 1005E1 Median on Segments (Permutations Edition) 【思维+无序数组求中位数】

    题目:戳这里 百度之星初赛原题:戳这里 题意:n个不同的数,求中位数为m的区间有多少个. 解题思路: 此题的中位数就是个数为奇数的数组中,小于m的数和大于m的数一样多,个数为偶数的数组中,小于m的数比 ...

  9. 网站资源被盗链的:预防方法 VS 网站资源防盗链的:破解技巧

    1 www.cnblogs.com 发布文章使用:只允许注册用户才可以访问! 1 资源被盗链:(简明定义) 下载者不是从你的网站直接下载资源,而是通过其他盗链网站提供的你的下载资源链接进行下载你的服务 ...

  10. TDD & Unit testing

    TDD & Unit testing TDD jest https://github.com/facebook/jest https://facebook.github.io/jest/zh- ...