前面我们介绍了线程(std::thread)和互斥量(std::mutex),互斥量是多线程间同时访问某一共享变量时,保证变量可被安全访问的手段。在多线程编程中,还有另一种十分常见的行为:线程同步。线程同步是指线程间需要按照预定的先后次序顺序进行的行为。C++11对这种行为也提供了有力的支持,这就是条件变量。条件变量位于头文件condition_variable下。本章我们将简要介绍一下该类,在文章的最后我们会综合运用std::mutex和std::condition_variable,实现一个chan类,该类可在多线程间安全的通信,具有广泛的应用场景。
1. std::condition_variable
条件变量提供了两类操作:wait和notify。这两类操作构成了多线程同步的基础。
1.1 wait
wait是线程的等待动作,直到其它线程将其唤醒后,才会继续往下执行。下面通过伪代码来说明其用法:
std::mutex mutex;
std::condition_variable cv;
// 条件变量与临界区有关,用来获取和释放一个锁,因此通常会和mutex联用。
std::unique_lock lock(mutex);
// 此处会释放lock,然后在cv上等待,直到其它线程通过cv.notify_xxx来唤醒当前线程,cv被唤醒后会再次对lock进行上锁,然后wait函数才会返回。
// wait返回后可以安全的使用mutex保护的临界区内的数据。此时mutex仍为上锁状态
cv.wait(lock)
需要注意的一点是, wait有时会在没有任何线程调用notify的情况下返回,这种情况就是有名的spurious wakeup 。因此当wait返回时,你需要再次检查wait的前置条件是否满足,如果不满足则需要再次wait。wait提供了重载的版本,用于提供前置检查。
template
void wait(unique_lock &lock, Predicate pred) {
while(!pred()) {
wait(lock);
}
}
除wait外, 条件变量还提供了wait_for和wait_until,这两个名称是不是看着有点儿眼熟,std::mutex也提供了_for和_until操作。在C++11多线程编程中,需要等待一段时间的操作,一般情况下都会有xxx_for和xxx_until版本。前者用于等待指定时长,后者用于等待到指定的时间。
1.2 notify
了解了wait,notify就简单多了:唤醒wait在该条件变量上的线程。notify有两个版本:notify_one和notify_all。
notify_one 唤醒等待的一个线程,注意只唤醒一个。
notify_all 唤醒所有等待的线程。使用该函数时应避免出现惊群效应。
其使用方式见下例:
std::mutex mutex;
std::condition_variable cv;
std::unique_lock lock(mutex);
// 所有等待在cv变量上的线程都会被唤醒。但直到lock释放了mutex,被唤醒的线程才会从wait返回。
cv.notify_all(lock)
2. 线程间通信 - chan的实现
有了上面的基础我们就可以设计我们的线程间通讯工具"chan"了。我们的设计目标:
在线程间安全的传递数据。golang社区有一句经典的话:不要通过共享内存来通信,要通过通信来共享内存。
消除线程线程同步带来的复杂性。
我们先来看一下chan的实际使用效果, 生产者-消费者(一个生产者,多个消费者)
#include
#include
#include "chan.h" // chan的头文件
using namespace std::chrono;
// 消费数据
void consume(chan ch, int thread_id) {
int n;
while(ch >> n) {
printf("[%d] %d\n", thread_id, n);
std::this_thread::sleep_for(milliseconds(100));
}
}
int main() {
chan chInt(3);
// 消费者
std::thread consumers[5];
for (int i = 0; i < 5; i++) {
consumers[i] = std::thread(consume, chInt, i+1);
}
// 生产数据
for (int i = 0; i < 16; i++) {
chInt << i;
}
chInt.close(); // 数据生产完毕
for (std::thread &thr: consumers) {
thr.join();
}
return 0;
}
附: 源码(可在github上下载到)
下面附上chan.simple.h的实现,是chan的较为简单的实现,完整实现请去github下载。该代码在g++和vc 2015下均编译通过,其它平台未验证。
// chan.simple.h
#pragma once
#include // std::condition_variable
#include // std::list
#include // std::mutex
template
class chan {
class queue_t {
mutable std::mutex mutex_;
std::condition_variable cv_;
std::list data_;
const size_t capacity_; // data_容量
const bool enable_overflow_;
bool closed_ = false; // 队列是否已关闭
size_t pop_count_ = 0; // 计数,累计pop的数量
public:
queue_t(size_t capacity) :
capacity_(capacity == 0 ? 1 : capacity),
enable_overflow_(capacity == 0) {
}
bool is_empty() const {
return data_.empty();
}
size_t free_count() const {
// capacity_为0时,允许放入一个,但_queue会处于overflow状态
return capacity_ - data_.size();
}
bool is_overflow() const {
return enable_overflow_ && data_.size() >= capacity_;
}
bool is_closed() const {
std::unique_lock lock(this->mutex_);
return this->closed_;
}
// close以后的入chan操作会返回false, 而出chan则在队列为空后才返回false
void close() {
std::unique_lock lock(this->mutex_);
this->closed_ = true;
if (this->is_overflow()) {
// 消除溢出
this->data_.pop_back();
}
this->cv_.notify_all();
}
template
bool pop(TR &data) {
std::unique_lock lock(this->mutex_);
this->cv_.wait(lock, [&]() { return !is_empty() || closed_; });
if (this->is_empty()) {
return false; // 已关闭
}
data = this->data_.front();
this->data_.pop_front();
this->pop_count_++;
if (this->free_count() == 1) {
// 说明以前是full或溢出状态
this->cv_.notify_all();
}
return true;
}
template
bool push(TR &&data) {
std::unique_lock lock(mutex_);
cv_.wait(lock, [this]() { return free_count() > 0 || closed_; });
if (closed_) {
return false;
}
data_.push_back(std::forward(data));
if (data_.size() == 1) {
cv_.notify_all();
}
// 当queue溢出,需等待queue回复正常
if (is_overflow()) {
const size_t old = this->pop_count_;
cv_.wait(lock, [&]() { return old != pop_count_ || closed_; });
}
return !this->closed_;
}
};
std::shared_ptr queue_;
public:
explicit chan(size_t capacity = 0) {
queue_ = std::make_shared(capacity);
}
// 支持拷贝
chan(const chan &) = default;
chan &operator=(const chan &) = default;
// 支持move
chan(chan &&) = default;
chan &operator=(chan &&) = default;
// 入chan,支持move语义
template
bool operator<<(TR &&data) {
return queue_->push(std::forward(data));
}
// 出chan(支持兼容类型的出chan)
template
bool operator>>(TR &data) {
return queue_->pop(data);
}
// close以后的入chan操作返回false, 而出chan则在队列为空后才返回false
void close() {
queue_->close();
}
bool is_closed() const {
return queue_->is_closed();
}
};
上一篇 C++11多线程-mutex(2)
目录
下一篇 C++11多线程-promise
你可能感兴趣的:(C++11多线程-条件变量(std::condition_variable))
JVM 内存模型深度解析:原子性、可见性与有序性的实现
练习时长两年半的程序员小胡
JVM 深度剖析:从面试考点到生产实践 jvm java 内存模型
在了解了JVM的基础架构和类加载机制后,我们需要进一步探索Java程序在多线程环境下的内存交互规则。JVM内存模型(JavaMemoryModel,JMM)定义了线程和主内存之间的抽象关系,它通过规范共享变量的访问方式,解决了多线程并发时的数据一致性问题。本文将从内存模型的核心目标出发,详解原子性、可见性、有序性的实现机制,以及volatile、synchronized等关键字在其中的作用。一、J
Java | 多线程经典问题 - 售票
Ada54
一、售票需求1)同一个票池2)多个窗口卖票,不能出售同一张票二、售票问题代码实现(线程与进程小总结,请戳:Java|线程和进程,创建线程)step1:定义SaleWindow类实现Runnable接口,覆盖run方法step2:实例化SaleWindow对象,创建Thread对象,将SaleWindow作为参数传给Thread类的构造函数,然后通过Thread.start()方法启动线程step3
深入理解汇编语言子程序设计与系统调用
网安spinage
汇编语言 开发语言 汇编 算法
本文将全面解析汇编语言中子程序设计的核心技术以及系统调用的实现方法,涵盖参数传递的多种方式、堆栈管理、API调用等关键知识点,并提供实际案例演示。一、子程序设计:参数传递的艺术1.寄存器传参:高效简洁.386.modelflat,stdcalloptioncasemap:none.dataxdd5;定义变量ydd6sumdd?.code;函数定义:addxy1addxy1procpushebpmo
CodeFoeces-450B
ss5smi
题目原题链接:B.JzzhuandSequences题意根据公式公式计算对应fn的值。参考了其他作者的代码和思路。找循环点。负数取余需要加取余数到>0为止才可取余。代码#includeusingnamespacestd;constintmod=1e9+7;intmain(){longlongf[10],x,y,n;cin>>x>>y>>n;x=(x+mod)%mod;y=(y+mod)%mod;f
Java并发核心:线程池使用技巧与最佳实践! | 多线程篇(五)
bug菌¹
Java实战(进阶版) java Java零基础入门 Java并发 线程池 多线程篇
本文收录于「Java进阶实战」专栏,专业攻坚指数级提升,希望能够助你一臂之力,帮你早日登顶实现财富自由;同时,欢迎大家关注&&收藏&&订阅!持续更新中,up!up!up!!环境说明:Windows10+IntelliJIDEA2021.3.2+Jdk1.8本文目录前言摘要正文何为线程池?为什么需要线程池?线程池的好处线程池使用场景如何创建线程池?线程池的常见配置源码解析案例分享案例代码演示案例运行
python笔记14介绍几个魔法方法
抢公主的大魔王
python python
python笔记14介绍几个魔法方法先声明一下各位大佬,这是我的笔记。如有错误,恳请指正。另外,感谢您的观看,谢谢啦!(1).__doc__输出对应的函数,类的说明文档print(print.__doc__)print(value,...,sep='',end='\n',file=sys.stdout,flush=False)Printsthevaluestoastream,ortosys.std
C++中std::variant的使用详解和实战代码示例
点云SLAM
C++ c++ 开发语言 variant C++泛型编程 联合体 C++ 类型擦除机制 C++17
std::variant是C++17引入的一个类型安全的联合体(type-safeunion),它可以在多个类型之间存储一个值,并在编译时进行类型检查。它是现代C++类型擦除与泛型编程的核心工具之一,适用于构建可变类型结构、消息传递系统、状态机等。一、基本概念#includestd::variantv;类似于联合体union,但类型安全。std::variant只能存储其中一个类型的值。默认构造时
CMS垃圾回收器+G1垃圾回收器+ZGC垃圾回收器详解及对比
weixin_43751710
jvm java 算法
一、CMS收集器CMS(ConcurrentMarkSweep)收集器是一种以获取最短回收停顿时间为目标的收集器,是一款针对老年代的垃圾回收器,一般和Parallel回收器(一款新生代回收器,是使用复制算法的收集器,又是并行的多线程收集器,收集时会Stoptheworld)配合使用。1.工作过程从名字(包含“MarkSweep”)上就可以看出CMS收集器是基于标记-清除算法实现的,它的运作整个过程
10.3 条件变量
百亿苍狗
Linux多进程 多线程 IO模型 linux
10.3条件变量不⾜:主线程(消费者线程)需要不断查询是否有产品可以消费,如果没有产品可以消费,也在运⾏程序,包括获得互斥锁、判断条件、释放互斥锁,⾮常消耗cpu资源条件变量允许⼀个线程就某个共享变量的状态变化通知其他线程,并让其他线程等待这⼀通知条件变量的本质为pthread_cond_t类型的变量,其他线程可以阻塞在这个条件变量上,或者唤醒阻塞在这个条件变量上的线程条件变量的初始化分为静态初始
C++11 列表初始化(initializer_list),pair
行十万里人生
C++ c++ list 开发语言 1024程序员节 数据结构 个人开发 蓝桥杯
1.{}初始化C++98中,允许使用{}对数组进行初始化。intarr[3]={0,1,2};C++11扩大了{}初始化的使用范围,使其可用于所有内置类型和自定义类型。structDate{int_year;int_month;int_day;Date(intyear,intmonth,intday):_year(year),_month(month),_day(day){}};in
C++98和C++11的构造和初始化、initializer_list以及decltype关键字(一般)
无聊看看天T^T
C++从入门到入土 c++ 开发语言
目录前言C++98的构造与初始化C++11的构造与初始化初始化列表的initializer_listdecltype关键字前言2003年C++标准委员会曾经提交了一份技术勘误表(简称TC1),使得C++03这个名字取代了C++98成为了C++11前最新的C++标准名称。不过由于C++03主要是对C++98标准中的漏洞进行修复,语言的核心部分则没有改动,因此人们习惯性的把两个标准合并成为C++98/
C++---初始化列表(initializer_list)
MzKyle
C/C++ c++ list java
在C++编程中,我们经常会用到形如vectorv={1,2,3,4};的语法——用花括号包裹一组元素直接初始化容器。这种直观且简洁的写法背后,依赖于C++11引入的一个特殊类型:std::initializer_list。它不仅是列表初始化的“桥梁”,更是C++标准库设计中连接语法糖与底层实现的关键机制。一、initializer_list的本质std::initializer_list是C++1
java callable 详解_详解Java Callable接口实现多线程的方式
想法臃肿
java callable 详解
在Java1.5以前,创建线程的2种方式,一种是直接继承Thread,另外一种就是实现Runnable接口。无论我们以怎样的形式实现多线程,都需要调用Thread类中的start方法去向操作系统请求io,cup等资源。因为线程run方法没有返回值,如果需要获取执行结果,就必须通过共享变量或者使用线程通信的方式来达到效果,这样使用起来就比较麻烦。而自从java1.5开始,就提供了Callable和F
Java CAS 分析
向梦而来
1概述CAS,CompareAndSwap,即比较并交换。DougLea大神在实现同步组件时,大量使用CAS技术,鬼斧神工地实现了Java多线程的并发操作。整个AQS同步组件、Atomic原子类操作等等都是基CAS实现的,甚至ConcurrentHashMap在JDK1.8的版本中,也调整为CAS+synchronized。可以说,CAS是整个J.U.C的基石。2017030900012CAS分析
为什么inet_ntoa会返回错误的IP地址?
dvlinker
C/C++实战专栏 整型IP 点式字符串IP inet_addr inet_ntoa inet_ntop Windows系统版本 NetWkstaGetInfo
目录1、调用inet_addr和inet_ntoa实现整型IP与点式字符串之间的转换1.1、调用inet_addr将点式字符串IP转换成整型IP1.2、调用inet_ntoa将整型IP转换成点式字符串IP2、调用inet_ntoa返回错误点式字符串IP的原因分析3、解决多线程调用inet_ntoa返回错误点式字符串IP的办法3.1、直接编写将32位整型转成点式字符串的代码3.2、使用线程安全的in
Python操作excel,工作效率提高篇_python对xlsx文档进行操作怎么提速
2401_84266286
程序员 python excel 网络
上面的代码是对工作簿最基本的操作,新建工作簿和保存工作簿,还有关闭当前工作簿。importosfile_path='table'file_list=os.listdir(file_path)foriinfile_list:print(i)列出文件夹下所有文件和子文件夹的名称,这是方便总结和查看文件的。importxlwingsasxwapp=xw.App(visible=False,add_boo
【Linux】线程——线程池、线程池的实现、线程安全的线程池、单例模式的概念、饿汉和懒汉模式、互斥锁、条件变量、信号量、自旋锁、读写锁
鳄鱼麻薯球
Linux linux 安全 单例模式
文章目录Linux线程7.线程池7.1线程池介绍7.2线程池的实现7.3线程安全的线程池7.3.1单例模式的概念7.3.2饿汉和懒汉模式8.常见锁使用汇总8.1互斥锁(Mutex)8.2条件变量(ConditionVariable)8.3信号量(Semaphore)8.4自旋锁(SpinLock)8.5读写锁(Read-WriteLock)Linux线程7.线程池 线程池是一种多线程编程中的技术
【Linux】多线程:线程池的创建、日志类、RAII互斥锁、单例模式:饿汉方式与懒汉方式
小白也有开发梦
Linux linux 单例模式 日志 多线程 线程池 c++ c语言
目录一、线程池概念二、线程的封装及线程池类成员变量的介绍三、单例模式饿汉方式(EagerInitialization)懒汉方式(LazyInitialization)四、RAII类型的互斥锁五、日志类的实现六、简单的任务类创建七、线程池的创建一、线程池概念线程池(ThreadPool)是一种基于池化技术的线程使用模式,它创建了一个线程的集合,这些线程可以被多个任务重复使用。线程池的主要目的是减少在
解决:RuntimeError: main thread is not in main loop
-米兰的小铁匠
python linux matplotlib bug
很久没更新了,分享一下新近遇到的bug。背景是在做一个demo,用到了多线程,其中一个子线程任务为绘图并保存图片。起初在Windows上运行正常,但将代码迁移至Linux服务器上,运行时发生报错RuntimeError:mainthreadisnotinmainloop。查阅了一下资料,如下:“该错误通常在使用GUI编程库(如Tkinter、PyQt、wxPython等)时出现。这个错误的原因通常
Python 的 GIL 时代即将终结,迈向真正的多线程时代
技术狂潮AI
Python开发实战 AI编程实战 AI应用实战 开发语言 GIL Python
Python功能强大、灵活且对程序员友好,广泛应用于从Web开发到机器学习的各个领域。根据引用次数最多的两项指标,Python甚至超越了Java和C等语言,成为最流行的编程语言。经过多年的流行,Python似乎势不可挡。但Python作为一种编程语言的未来发展至少面临一个重大障碍。它被称为GIL,即全局解释器锁,几十年来,Python开发人员一直试图将其从Python的默认实现中删除。虽然GIL在
Python 使用期物处理并发(显示下载进度并处理错误)
显示下载进度并处理错误前面说过,17.1节中的几个脚本没有处理错误,这样做是为了便于阅读和比较三种方案(依序、多线程和异步)的结构。为了处理各种错误,我创建了flags2系列示例。flags2_common.py这个模块中包含所有flags2示例通用的函数和设置,例如main函数,负责解析命令行参数、计时和报告结果。这个脚本中的代码其实是提供支持的,与本章的话题没有直接关系,因此我把源码放在附录A
【高频考点精讲】手写Web Worker通信:从主线程到子线程,掌握多线程编程技巧
全栈老李技术面试
前端高频考点精讲 前端 javascript html css 面试题 react vue
手写WebWorker通信:从主线程到子线程,掌握多线程编程技巧作者:全栈老李更新时间:2025年5月适合人群:前端初学者、进阶开发者版权:本文由全栈老李原创,转载请注明出处。今天咱们聊聊WebWorker这个前端性能优化的"核武器"。我是全栈老李,一个喜欢把复杂技术讲简单的技术博主。为什么需要WebWorker?想象你在餐厅点单,如果只有一个服务员(主线程),他既要接待顾客,又要去后厨炒菜,
告别UI卡顿:深入 Web Worker 与 Comlink,解锁浏览器多线程编程的真正威力
码力无边-OEC
ui 前端 web javascript
告别UI卡顿:深入WebWorker与Comlink,解锁浏览器多线程编程的真正威力你一定遇到过这样的场景:点击一个按钮后,页面突然“冻结”了,UI失去响应,动画卡住,滚动条也无法拖动。几秒钟后,页面才恢复正常。这种糟糕的体验,就是前端开发者永远的噩梦——主线程阻塞。Part1:痛点重现——当主线程不堪重负让我们先来制造一个经典的UI卡顿场景:计算大量斐波那契数。HTML(index.html):
结合Golang语言说明对多线程编程以及 select/epoll等网络模型的使用
zhoupenghui168
golang 计算机网络 golang 网络 数据库 select网络模型 epoll网络模型 多线程编程
首先介绍select和epoll这两个I/O多路复用的网络模型,然后介绍多线程编程,最后结合Go语言项目举例说明如何应用一、select和epoll的介绍1.select模型select是一种I/O多路复用技术,它允许程序同时监视多个文件描述符(通常是套接字),等待一个或多个描述符就绪(可读、可写或异常)然后进行相应的操作,它的跨平台兼容性好(Windows/Linux/macOS)核心原理:使用
leetcode0954. 二倍数对数组-medium
智趣代码实验室
Leetcode 算法 c++ leetcode 数据结构
1题目:二倍数对数组官方标定难度:中给定一个长度为偶数的整数数组arr,只有对arr进行重组后可以满足“对于每个0&arr){std::sort(arr.begin(),arr.end(),[](inta,intb){returnabs(a)>abs(b);});unordered_mapl;for(inti:arr){if(l[i*2]){l[i*2]--;}else{l[i]++;}}for(
【C语言网络编程基础】TCP并发网络编程:一请求一线程模型
(Charon)
网络 tcp/ip 网络协议
在实际开发中,一个TCP服务器往往要同时为多个客户端提供服务。最简单直观的方式,就是采用“一请求一线程”模型——每当有客户端连接进来,服务器就创建一个新线程专门负责这个客户端的收发任务。本文将介绍如何使用C语言+TCP+pthread多线程实现一个并发TCP服务器。一、TCP服务器的典型通信流程创建socket绑定IP和端口(bind)开始监听连接请求(listen)接收连接(accept)接收与
大数据量查询计算引发数据库CPU告警问题复盘
懒虫虫~
业务解决方案 大表治理
大数据量查询计算引发数据库CPU告警问题复盘一、背景二、根因分析三、解决方案方案1:多线程+缓存方案2:利用中间表+缓存四、总结一、背景2025年7月份某天,CDP系统每天不定时推送我们的Portal服务,生产环境运营看板会展示统计数据,发现接口响应缓慢,随之而来数据库监控告警,发现数据库CPU达到了80%。由于表数据量大,计算统计复杂,多线程使用不当,导致数据库服务器爆表。其中A表数据量达到1亿
[Flask] 异步非阻塞IO实现
_小老虎_
Flask默认是不支持非阻塞IO的,表现为:当请求1未完成之前,请求2是需要等待处理状态,效率非常低。在flask中非阻塞实现可以由2种:启用flask多线程机制#FlaskfromflaskimportFlask,request,gimportosimportsysreload(sys)sys.setdefaultencoding('utf-8')app=Flask(__name__)app.c
Java Lock使用
lijiabin417
java 开发语言
在编写代码时,使用Lock接口可以帮助你更好地控制多线程环境下的同步问题。Java提供了多种Lock实现,先介绍ReentrantLock//锁容器privateMapbackMap=newConcurrentHashMap<>();intcount=0;/***获取锁*@paramkey*@return*/privateLockgetLock(Stringkey){//根据key获取对应的锁,如
零基础学习性能测试第五章:JVM性能分析与调优-多线程检测与瓶颈分析
试着
性能测试 学习 jvm 零基础 性能测试
目录**一、多线程性能问题典型症状****二、核心分析工具链****1.基础诊断命令****2.高级可视化工具****三、多线程瓶颈四步分析法****步骤1:定位高负载线程****步骤2:分析线程阻塞原因****步骤3:锁竞争分析****步骤4:并发数据结构分析****四、高频瓶颈场景与调优方案****场景1:锁竞争激烈****场景2:线程池配置不当****场景3:ThreadLocal内存泄漏*
ASM系列五 利用TreeApi 解析生成Class
lijingyao8206
ASM 字节码动态生成 ClassNode TreeAPI
前面CoreApi的介绍部分基本涵盖了ASMCore包下面的主要API及功能,其中还有一部分关于MetaData的解析和生成就不再赘述。这篇开始介绍ASM另一部分主要的Api。TreeApi。这一部分源码是关联的asm-tree-5.0.4的版本。
在介绍前,先要知道一点, Tree工程的接口基本可以完
链表树——复合数据结构应用实例
bardo
数据结构 树型结构 表结构设计 链表 菜单排序
我们清楚:数据库设计中,表结构设计的好坏,直接影响程序的复杂度。所以,本文就无限级分类(目录)树与链表的复合在表设计中的应用进行探讨。当然,什么是树,什么是链表,这里不作介绍。有兴趣可以去看相关的教材。
需求简介:
经常遇到这样的需求,我们希望能将保存在数据库中的树结构能够按确定的顺序读出来。比如,多级菜单、组织结构、商品分类。更具体的,我们希望某个二级菜单在这一级别中就是第一个。虽然它是最后
为啥要用位运算代替取模呢
chenchao051
位运算 哈希 汇编
在hash中查找key的时候,经常会发现用&取代%,先看两段代码吧,
JDK6中的HashMap中的indexFor方法:
/**
* Returns index for hash code h.
*/
static int indexFor(int h, int length) {
最近的情况
麦田的设计者
生活 感悟 计划 软考 想
今天是2015年4月27号
整理一下最近的思绪以及要完成的任务
1、最近在驾校科目二练车,每周四天,练三周。其实做什么都要用心,追求合理的途径解决。为
PHP去掉字符串中最后一个字符的方法
IT独行者
PHP 字符串
今天在PHP项目开发中遇到一个需求,去掉字符串中的最后一个字符 原字符串1,2,3,4,5,6, 去掉最后一个字符",",最终结果为1,2,3,4,5,6 代码如下:
$str = "1,2,3,4,5,6,";
$newstr = substr($str,0,strlen($str)-1);
echo $newstr;
hadoop在linux上单机安装过程
_wy_
linux hadoop
1、安装JDK
jdk版本最好是1.6以上,可以使用执行命令java -version查看当前JAVA版本号,如果报命令不存在或版本比较低,则需要安装一个高版本的JDK,并在/etc/profile的文件末尾,根据本机JDK实际的安装位置加上以下几行:
export JAVA_HOME=/usr/java/jdk1.7.0_25  
JAVA进阶----分布式事务的一种简单处理方法
无量
多系统交互 分布式 事务
每个方法都是原子操作:
提供第三方服务的系统,要同时提供执行方法和对应的回滚方法
A系统调用B,C,D系统完成分布式事务
=========执行开始========
A.aa();
try {
B.bb();
} catch(Exception e) {
A.rollbackAa();
}
try {
C.cc();
} catch(Excep
安墨移动广 告:移动DSP厚积薄发 引领未来广 告业发展命脉
矮蛋蛋
hadoop 互联网
“谁掌握了强大的DSP技术,谁将引领未来的广 告行业发展命脉。”2014年,移动广 告行业的热点非移动DSP莫属。各个圈子都在纷纷谈论,认为移动DSP是行业突破点,一时间许多移动广 告联盟风起云涌,竞相推出专属移动DSP产品。
到底什么是移动DSP呢?
DSP(Demand-SidePlatform),就是需求方平台,为解决广 告主投放的各种需求,真正实现人群定位的精准广
myelipse设置
alafqq
IP
在一个项目的完整的生命周期中,其维护费用,往往是其开发费用的数倍。因此项目的可维护性、可复用性是衡量一个项目好坏的关键。而注释则是可维护性中必不可少的一环。
注释模板导入步骤
安装方法:
打开eclipse/myeclipse
选择 window-->Preferences-->JAVA-->Code-->Code
java数组
百合不是茶
java数组
java数组的 声明 创建 初始化; java支持C语言
数组中的每个数都有唯一的一个下标
一维数组的定义 声明: int[] a = new int[3];声明数组中有三个数int[3]
int[] a 中有三个数,下标从0开始,可以同过for来遍历数组中的数
javascript读取表单数据
bijian1013
JavaScript
利用javascript读取表单数据,可以利用以下三种方法获取:
1、通过表单ID属性:var a = document.getElementByIdx_x_x("id");
2、通过表单名称属性:var b = document.getElementsByName("name");
3、直接通过表单名字获取:var c = form.content.
探索JUnit4扩展:使用Theory
bijian1013
java JUnit Theory
理论机制(Theory)
一.为什么要引用理论机制(Theory)
当今软件开发中,测试驱动开发(TDD — Test-driven development)越发流行。为什么 TDD 会如此流行呢?因为它确实拥有很多优点,它允许开发人员通过简单的例子来指定和表明他们代码的行为意图。
TDD 的优点:
&nb
[Spring Data Mongo一]Spring Mongo Template操作MongoDB
bit1129
template
什么是Spring Data Mongo
Spring Data MongoDB项目对访问MongoDB的Java客户端API进行了封装,这种封装类似于Spring封装Hibernate和JDBC而提供的HibernateTemplate和JDBCTemplate,主要能力包括
1. 封装客户端跟MongoDB的链接管理
2. 文档-对象映射,通过注解:@Document(collectio
【Kafka八】Zookeeper上关于Kafka的配置信息
bit1129
zookeeper
问题:
1. Kafka的哪些信息记录在Zookeeper中 2. Consumer Group消费的每个Partition的Offset信息存放在什么位置
3. Topic的每个Partition存放在哪个Broker上的信息存放在哪里
4. Producer跟Zookeeper究竟有没有关系?没有关系!!!
//consumers、config、brokers、cont
java OOM内存异常的四种类型及异常与解决方案
ronin47
java OOM 内存异常
OOM异常的四种类型:
一: StackOverflowError :通常因为递归函数引起(死递归,递归太深)。-Xss 128k 一般够用。
二: out Of memory: PermGen Space:通常是动态类大多,比如web 服务器自动更新部署时引起。-Xmx
java-实现链表反转-递归和非递归实现
bylijinnan
java
20120422更新:
对链表中部分节点进行反转操作,这些节点相隔k个:
0->1->2->3->4->5->6->7->8->9
k=2
8->1->6->3->4->5->2->7->0->9
注意1 3 5 7 9 位置是不变的。
解法:
将链表拆成两部分:
a.0-&
Netty源码学习-DelimiterBasedFrameDecoder
bylijinnan
java netty
看DelimiterBasedFrameDecoder的API,有举例:
接收到的ChannelBuffer如下:
+--------------+
| ABC\nDEF\r\n |
+--------------+
经过DelimiterBasedFrameDecoder(Delimiters.lineDelimiter())之后,得到:
+-----+----
linux的一些命令 -查看cc攻击-网口ip统计等
hotsunshine
linux
Linux判断CC攻击命令详解
2011年12月23日 ⁄ 安全 ⁄ 暂无评论
查看所有80端口的连接数
netstat -nat|grep -i '80'|wc -l
对连接的IP按连接数量进行排序
netstat -ntu | awk '{print $5}' | cut -d: -f1 | sort | uniq -c | sort -n
查看TCP连接状态
n
Spring获取SessionFactory
ctrain
sessionFactory
String sql = "select sysdate from dual";
WebApplicationContext wac = ContextLoader.getCurrentWebApplicationContext();
String[] names = wac.getBeanDefinitionNames();
for(int i=0; i&
Hive几种导出数据方式
daizj
hive 数据导出
Hive几种导出数据方式
1.拷贝文件
如果数据文件恰好是用户需要的格式,那么只需要拷贝文件或文件夹就可以。
hadoop fs –cp source_path target_path
2.导出到本地文件系统
--不能使用insert into local directory来导出数据,会报错
--只能使用
编程之美
dcj3sjt126com
编程 PHP 重构
我个人的 PHP 编程经验中,递归调用常常与静态变量使用。静态变量的含义可以参考 PHP 手册。希望下面的代码,会更有利于对递归以及静态变量的理解
header("Content-type: text/plain");
function static_function () {
static $i = 0;
if ($i++ < 1
Android保存用户名和密码
dcj3sjt126com
android
转自:http://www.2cto.com/kf/201401/272336.html
我们不管在开发一个项目或者使用别人的项目,都有用户登录功能,为了让用户的体验效果更好,我们通常会做一个功能,叫做保存用户,这样做的目地就是为了让用户下一次再使用该程序不会重新输入用户名和密码,这里我使用3种方式来存储用户名和密码
1、通过普通 的txt文本存储
2、通过properties属性文件进行存
Oracle 复习笔记之同义词
eksliang
Oracle 同义词 Oracle synonym
转载请出自出处:http://eksliang.iteye.com/blog/2098861
1.什么是同义词
同义词是现有模式对象的一个别名。
概念性的东西,什么是模式呢?创建一个用户,就相应的创建了 一个模式。模式是指数据库对象,是对用户所创建的数据对象的总称。模式对象包括表、视图、索引、同义词、序列、过
Ajax案例
gongmeitao
Ajax jsp
数据库采用Sql Server2005
项目名称为:Ajax_Demo
1.com.demo.conn包
package com.demo.conn;
import java.sql.Connection;import java.sql.DriverManager;import java.sql.SQLException;
//获取数据库连接的类public class DBConnec
ASP.NET中Request.RawUrl、Request.Url的区别
hvt
.net Web C# asp.net hovertree
如果访问的地址是:http://h.keleyi.com/guestbook/addmessage.aspx?key=hovertree%3C&n=myslider#zonemenu那么Request.Url.ToString() 的值是:http://h.keleyi.com/guestbook/addmessage.aspx?key=hovertree<&
SVG 教程 (七)SVG 实例,SVG 参考手册
天梯梦
svg
SVG 实例 在线实例
下面的例子是把SVG代码直接嵌入到HTML代码中。
谷歌Chrome,火狐,Internet Explorer9,和Safari都支持。
注意:下面的例子将不会在Opera运行,即使Opera支持SVG - 它也不支持SVG在HTML代码中直接使用。 SVG 实例
SVG基本形状
一个圆
矩形
不透明矩形
一个矩形不透明2
一个带圆角矩
事务管理
luyulong
java spring 编程 事务
事物管理
spring事物的好处
为不同的事物API提供了一致的编程模型
支持声明式事务管理
提供比大多数事务API更简单更易于使用的编程式事务管理API
整合spring的各种数据访问抽象
TransactionDefinition
定义了事务策略
int getIsolationLevel()得到当前事务的隔离级别
READ_COMMITTED
基础数据结构和算法十一:Red-black binary search tree
sunwinner
Algorithm Red-black
The insertion algorithm for 2-3 trees just described is not difficult to understand; now, we will see that it is also not difficult to implement. We will consider a simple representation known
centos同步时间
stunizhengjia
linux 集群同步时间
做了集群,时间的同步就显得非常必要了。 以下是查到的如何做时间同步。 在CentOS 5不再区分客户端和服务器,只要配置了NTP,它就会提供NTP服务。 1)确认已经ntp程序包: # yum install ntp 2)配置时间源(默认就行,不需要修改) # vi /etc/ntp.conf server pool.ntp.o
ITeye 9月技术图书有奖试读获奖名单公布
ITeye管理员
ITeye
ITeye携手博文视点举办的9月技术图书有奖试读活动已圆满结束,非常感谢广大用户对本次活动的关注与参与。 9月试读活动回顾:http://webmaster.iteye.com/blog/2118112本次技术图书试读活动的优秀奖获奖名单及相应作品如下(优秀文章有很多,但名额有限,没获奖并不代表不优秀):
《NFC:Arduino、Andro