Memcached线程模型详解:初始化与事件注册

需积分: 0 1 下载量 179 浏览量 更新于2024-08-19 收藏 580KB PPT 举报
本文主要介绍了Memcached的网络模型,特别是初始化workers线程和连接处理机制。Memcached采用主-工作线程模型,其中主线程负责监听客户端连接,而工作者线程处理已建立连接的读写事件。 在Memcached启动过程中,主线程首先初始化,创建一个event_base实例`main_base`,接着初始化所有工作线程。每个工作线程有自己的事件库和一个由主线程分配的连接队列(CQ队列)。主线程还会创建监听socket,并将监听socket的libevent事件注册到`main_base`上,进入事件循环等待连接请求。 当有新连接到来时,主线程accept连接,然后将新建立的socket描述符(sfd)和相关数据放入工作线程的CQ队列中。每个工作线程会监控自己的CQ队列,当有可读事件时,调用`conn_new()`函数。`conn_new()`函数从队列中取出CQ_ITEM,使用libevent注册`event_handler`函数,绑定到该socket的读事件上。这样,当连接有可读数据时,`event_handler`会被调用,进一步驱动`drive_machine`执行,处理客户端的请求。 工作线程结构体包含以下几个关键部分: 1. `thread_id`:线程的唯一标识。 2. `CQ队列`:存放主线程传递过来的待处理连接。 3. `通知pipe`:用于主线程与工作线程之间的通信。 4. `libevent实例`:每个工作线程有自己的事件库,用于处理连接事件。 5. `suffix_cache`:用于CAS和GETS指令的数据一致性检查。 `conn_new()`函数的核心是将新连接注册到工作线程的libevent事件库,设置为EV_READ|EV_PERSIST,确保即使在网络断开后,libevent仍然能够处理连接关闭的事件。 整个Memcached网络模型确保了高并发和低延迟,主线程负责接收新连接,工作线程处理已有连接,充分利用多核处理器的能力,提高了服务性能。通过管道和连接队列,主线程和工作线程间实现了无锁通信,减少了同步开销。每个工作线程独立的libevent实例保证了线程间的隔离,避免了竞态条件,从而提高了系统的稳定性和效率。

根据以下api和数据结构写出一个将adc转换出来的数据通过GATT发给手机端的代码void ble_controller_init(uint8_t task_priority) int hci_driver_init(void) int bt_enable(bt_ready_cb_t cb)int bt_le_adv_start(const struct bt_le_adv_param *param,const struct bt_data *ad, size_t ad_len, const struct bt_data *sd, size_t sd_len)int bt_le_adv_update_data(const struct bt_data *ad, size_t ad_len,const struct bt_data *sd, size_t sd_len)int bt_le_adv_stop(void)int bt_le_scan_start(const struct bt_le_scan_param *param, bt_le_scan_cb_t cb)int bt_le_scan_stop(void)int bt_le_whitelist_add(const bt_addr_le_t *addr)int bt_le_whitelist_rem(const bt_addr_le_t *addr)int bt_le_whitelist_clear(void)int bt_le_set_chan_map(u8_t chan_map[5])int bt_unpair(u8_t id, const bt_addr_le_t *addr)int bt_conn_get_info(const struct bt_conn *conn, struct bt_conn_info *info)int bt_conn_get_remote_dev_info(struct bt_conn_info *info)int bt_conn_le_param_update(struct bt_conn *conn,const struct bt_le_conn_param *param)int bt_conn_disconnect(struct bt_conn *conn, u8_t reason)struct bt_conn *bt_conn_create_le(const bt_addr_le_t *peer,const struct bt_le_conn_param *param)int bt_conn_create_auto_le(const struct bt_le_conn_param *param)int bt_conn_create_auto_stop(void)int bt_le_set_auto_conn(const bt_addr_le_t *addr,const struct bt_le_conn_param *param)struct bt_conn *bt_conn_create_slave_le(const bt_addr_le_t *peer,const struct bt_le_adv_param *param)int bt_conn_set_security(struct bt_conn *conn, bt_security_t sec)bt_security_t bt_conn_get_security(struct bt_conn *conn)u8_t bt_conn_enc_key_size(struct bt_conn *conn)void bt_conn_cb_register(struct bt_conn_cb *cb)void bt_set_bondable(bool enable)int bt_conn_auth_cb_register(const struct bt_conn_auth_cb *cb)int bt_conn_auth_passkey_entry(struct bt_conn *conn, unsigned int passkey)int bt_conn_auth_cancel(struct bt_conn *conn)int bt_conn_auth_passkey_confirm(struct bt_conn *conn)int bt_conn_auth_pincode_entry(struct bt_conn *conn, const char *pin)int bt_le_read_rssi(u16_t handle,int8_t *rssi)int bt_get_local_address(bt_addr_le_t *adv_addr)int bt_set_tx_pwr(int8_t power)bt_le_adv_parambt_databt_le_scan_parambt_le_conn_parambt_conn,并加入已经写好的adc代码bflb_adc_init(adc, &cfg); bflb_adc_channel_config(adc, chan, TEST_ADC_CHANNEL); for (uint32_t i = 0; i < 10; i++) { bflb_adc_start_conversion(adc); struct bflb_adc_result_s result; uint32_t raw_data = bflb_adc_read_raw(adc); bflb_adc_parse_result(adc, &raw_data, &result, 1); printf("pos chan %d,%d mv \r\n", result.pos_chan, result.millivolt); // char data[20]; // sprintf(data,"ADC result:%d",result.millivolt); // if(conn){ // bt_gatt_notify(conn,&attrs[1],data,sizeof(data)); // } bflb_mtimer_delay_ms(250); }

2023-05-19 上传