首先是从main函数开发:

int main(itn argc,char* argv[])

{

pthread_t thread;

int count;

int status;

        client_threads = CLIENT_THREADS;

for(count = 0;count <client_threads,count++)

{

status = pthread_create(&thread,NULL,client_routine,(void*)count);

if(status != 0)

err_abort(status,"Create client thread");

}

status = pthread_mutex_lock(&client_mutex);

if(status != 0)

err_abort(status,"Lock client mutex");

while(client_threads>0)

{

status = pthread_cond_wait(&clients_done,&client_mutex);

if(status != 0)

err_abort(status,"Wait for client to finish");

}

status = pthread_mutex_unlock(&client_mutex);

if(status != 0)

err_abort(status,"Unlock client mutex");

printf("All client done\n");

tty_server_request(REQ_QUIT,1,NULL,NULL);

return 0;

}

这个main函数在中间for循环中,调用pthread_create函数创建4个线程,传递给线程的数据位count(0,1,2,3).调用client_routine函数。

void *client_routine(void *arg)

{

int my_number = (int)arg,loops;

char prompt[32];

char string[128],formatted[128];

int status;

sprintf(prompt,"Client 5d>",my_number);

while(1)

{

tty_server_request(REQ_READ,1,prompt,string);

if(strlen(string) ==0)

break;

for(loops =0;loops <4;loops++)

{

sprintf(formatted,"(%d#%d)%s",my_number,loops,string);

tty_server_request(REQ_WRITE,0,NULL,formatted);

sleep(1);

}

}

status = pthread_mutex_lock(&client_mutex);

if(status != 0)

err_abort(status,"Lock client mutex");

client_threads--;

if(client_threads <= 0)

{

status = pthread_cond_signal(&clients_done);

if(status != 0)

err_abort(status,"Signale client done");

}

status = pthread_mutex_unlock(7client_mutex);

if(status != 0)

err_abort(status,"Unlock client mutex");

return NULL;

}

pthread_cond_wait() 用于堵塞当前线程,等待别的线程使用pthread_cond_signal()或pthread_cond_broadcast来唤醒它。

pthread_cond_wait() 必须与pthread_mutex
配套使用。pthread_cond_wait()函数一进入wait状态就会自己主动release mutex。当其它线程通过pthread_cond_signal()或pthread_cond_broadcast。把该线程唤醒,使pthread_cond_wait()通过(返回)时,该线程又自己主动获得该mutex。

  pthread_cond_signal函数的作用是发送一个信号给另外一个正在处于堵塞等待状态的线程,使其脱离堵塞状态,继续运行.假设没有线程处在堵塞等待状态,pthread_cond_signal也会成功返回。
  使用pthread_cond_signal一般不会有“惊群现象”产生,他最多仅仅给一个线程发信号。假如有多个线程正在堵塞等待着这个条件变量的话,那么是依据各等待线程优先级的高低确定哪个线程接收到信号開始继续运行。假设各线程优先级同样。则依据等待时间的长短来确定哪个线程获得信号。但不管怎样一个pthread_cond_signal调用最多发信一次。
  可是pthread_cond_signal在多处理器上可能同一时候唤醒多个线程,当你仅仅能让一个线程处理某个任务时。其他被唤醒的线程就须要继续 wait,并且规范要求pthread_cond_signal至少唤醒一个pthread_cond_wait上的线程。事实上有些实现为了简单在单处理器上也会唤醒多个线程. 
   另外。某些应用,如线程池,pthread_cond_broadcast唤醒所有线程,但我们通常仅仅须要一部分线程去做运行任务,所以其他的线程须要继续wait.所以强烈推荐对pthread_cond_wait() 使用while循环来做条件推断.

在这里假设没有线程。则通知等待创建线程唤醒wait在此条件变量上的等待。

上面所有代码:

#include <pthread.h>

#include <math.h>

#include "errors.h"





#define CLIENT_THREADS 4





#define REQ_READ 1

#define REQ_WRITE 2

#define REQ_QUIT 3





typedef struct request_tag

{

struct request_tag *next;

int operation;

int synchronous;

int done_flag;

pthread_cond_t done;

char prompt[32];

char text[128];

}request_t;





typedef struct tty_server_tag

{

request_t *first;

request_t *last;

int running;

pthread_mutex_t mutex;

pthread_cond_t request;

}tty_server_t;





tty_server_t tty_server = 

{

NULL,NULL,0,PTHREAD_MUTEX_INITIALIZER,PTHREAD_COND_INITIALIZER

};





int client_threads;

pthread_mutex_t client_mutex = PTHREAD_MUTEX_INITIALIZER;

pthread_cond_t clients_done = PTHREAD_COND_INITIALIZER;





void *tty_server_routine(void *arg)

{

static pthread_mutex_t prompt_mutex = PTHREAD_MUTEX_INITIALIZER;

request_t *request;

int operation,len;

int status;

while(1)

{

status = pthread_mutex_lock(&tty_server.mutex);

if(status != 0)

err_abort(status,"Lock server mutex");

while(tty_server.first ==NULL)

{

status = pthread_cond_wait(&tty_server.request,&tty_server.mutex);

if(status != 0)

{

err_abort(status,"Wait for request");

}

request = tty_server.first;

tty_server.first= request->next;

if(tty_server.first == NULL)

{

tty_server.last = NULL;

}

status = pthread_mutex_unlock(&tty_server.mutex);

if(status != 0)

err_abrort(status,"Unlock server mutex");

operation = request->operation;

switch(operation)

{

case REQ_QUIT:

break;

case REQ_READ:

if(strlen(request->prompt)>0)

printf(request->prompt);

if(fgets(request->text,128,stdin)==NULL)

request->text[0]='\0';

len = strlen(request->text);

if(len>0&&request->text[len-1]=='\n')

request->text[len-1]='\0';

break;

case REQ_WRITE:

puts(request->text);

break;

default:

break;

}

if(request->synchronous)

{

status = pthread_mutex_lock(&tty_server.mutex);

if(status != 0)

err_abort(status,"Lock server mutex");

request->done_flag =1;

status = pthread_cond__signal(&request->done);

if(status != 0)

err_abort(status,"Unlock server mutex");

}else

free(request);

if(operation ==REQ_QUIT)

break;

}

return NULL;

}





void tty_server_request(int operation,int sync,const char* prompt,char* string)

{

request_t *request;

int status;





status =pthread_mutex_lock(&tty_server.mutex);

if(status != 0)

err_abort(status,"Lock server mutex");

if(!tty_server.running)

{

pthread_t thread;

pthread_attr_t detached_attr;





status = pthread_attr_init(&detached_attr);

if(status != 0)

err_abort(status,"Init attributes object");

status = pthread_attr_setdetachstate(&detached_attr,PTHREAD_CREATE_DETACHED);

if(status != 0)

err_abort(status,"Set detach state");

tty_server.running = 1;

status = pthread_create(&thread,&detached_attr,tty_server_routine,NULL);

if(status != 0)

err_abort(status,"Create server");

pthread_attr_destroy(&detached_attr);

}

request = (request_t*)malloc(sizeof(request_t));

if(request ==NULL)

errno_abort("Allocate Request");

request->next = NULL;

request->operation = operation;

request->synchronous = sync;

if(sync)

{

request->done_flag = 0;

status = pthread_cond_init(&request->done,NULL);

if(status != 0)

err_abort(status ,"Init request condition");

}

if(prompt != NULL)

strncpy(request->prompt,prompt,32);

else 

request->prompt[0]='\0';

if(operation == REQ_WRITE &&string != NULL)

strncpy(request->text,string,128);

else

request->text[0]='\0';

if(tty_server.first == NULL)

{

tty_server.first = request;

tty_server.last = request;

}else

{

(tty_server.last)->next = request;

tty_server.last= request;

}

status = pthread_cond_signal(&tty_server.request);

if(status != 0)

err_abort(status,"Wake server");

if(sync)

{

while(!request->done_flag)

{

status = pthread_cond_wait(&request->done,&tty_server.mutex);

if(status != 0)

err_abort(status,"Wait for sync request");

}

if(operation == REQ_READ)

{

if(strlen(request->text)>0)

strcpy(string,request->text);

else 

string[0]='\0';

}

status = pthread_cond_destroy(&request->done);

if(status != 0)

err_abort(status,"destroy request condition");

free(request);

}

status = pthread_mutex_unlock(&tty-server.mutex);

if(status != 0)

err_abort(status,"Unlock mutex");

}

client、server端编程的更多相关文章

  1. Python的XMLRPC机制:实现跨进程间、client/server端通信

    SimpleXMLRPCServer模块式python语言的一个基于 xml 格式的进程间通信的基础框架. SimpleXMLRPCServer是一个单线程的服务器,这意味着,如果几个客户端同时发出多 ...

  2. client / server端用户的登录

    # 客户端 import socket import hashlib import json import os import struct sk = socket.socket() # 实例化 sk ...

  3. Linux下的C Socket编程 -- server端的继续研究

    Linux下的C Socket编程(四) 延长server的生命周期 在前面的一个个例子中,server在处理完一个连接后便会立即结束掉自己,然而这种server并不科学啊,server应该是能够一直 ...

  4. Linux下的C Socket编程 -- server端的简单示例

    Linux下的C Socket编程(三) server端的简单示例 经过前面的client端的学习,我们已经知道了如何创建socket,所以接下来就是去绑定他到具体的一个端口上面去. 绑定socket ...

  5. UDP也需要现有Server端,然后再有Client端

    UDP编程: DatagramSocket(邮递员):对应数据报的Socket概念,不需要创建两个socket,不可使用输入输出流. DatagramPacket(信件):数据包,是UDP下进行传输数 ...

  6. HBase 协处理器编程详解第一部分:Server 端代码编写

    Hbase 协处理器 Coprocessor 简介 HBase 是一款基于 Hadoop 的 key-value 数据库,它提供了对 HDFS 上数据的高效随机读写服务,完美地填补了 Hadoop M ...

  7. Socket编程, 在server端read()函数调用后显示错误:Transport endpoint is not connected (犯了低级错误)

    for(;;){ socklen_t len = sizeof(client_address); connfd = accept(listenfd, (struct sockaddr *)&c ...

  8. 从零开始学习Node.js例子四 多页面实现数学运算 续二(client端和server端)

    1.server端 支持数学运算的服务器,服务器的返回结果用json对象表示. math-server.js //通过监听3000端口使其作为Math Wizard的后台程序 var math = r ...

  9. 应用java多线程实现server端与多client之间的通信

    应用多线程来实现server与多线程之间的通信的基本步骤 1.server端创建ServerSocket,循环调用accept()等待client链接 2.client创建一个Socket并请求和se ...

随机推荐

  1. LuoguP2763 试题库问题(最大流)

    建图同_____ 代码: #include<queue> #include<cstdio> #include<cstring> #include<algori ...

  2. 手机浏览器,微信浏览器对background-color不显示的问题

    PC上的浏览器可以正常显示,但是到了手机上就不显示了,古怪的问题花了我一晚上都没解决. 今天突然想到会不会是某些特立独行的了浏览器为了彰显个性,采用不同别人的解析方式呢? 我的原来CSS是这么写的: ...

  3. Mysql学习总结(16)——Mysql之数据库设计规范

    一.三大范式 1.第一范式:消除一个字段包含多个数据库值,消除一个记录包含重复的组(单独的一列包含多个项目),即可满足1NF. 2.第二范式:消除部分依赖性即可转化为2NF.部分依赖性表示一个记录中包 ...

  4. TextView- 内容过长省略号设定

    <LinearLayout android:layout_width="fill_parent" android:layout_height="wrap_conte ...

  5. [NOI.AC#40]Erlang

    链接 题解 显然,最多抽2个集合 如果一直抽一个,前提是该集合有重复的,答案是不同元素的个数+1 如果抽两个,那么最坏情况下,在一个集合中抽到某一个数的次数是这个集合不同元素的个数(因为抽不到重复的) ...

  6. Android HttpLoggingInterceptor的用法简介

    该拦截器用于记录应用中的网络请求的信息. 示例 OkHttpClient client = new OkHttpClient(); HttpLoggingInterceptor logging = n ...

  7. 【习题 7-1 UVA-208】Firetruck

    [链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 预处理一下终点能到达哪些点. 暴力就好. 输出结果的时候,数字之间一个空格.. [代码] /* 1.Shoud it use lon ...

  8. 整理wmic使用,不重启变环境变量 .

    整理wmic使用,不重启变环境变量 . 利用wmic修改是直接生效的:(e:\tools 是新添加的目录) wmic ENVIRONMENT where "name='path' and u ...

  9. Solr的关键特性

    1.基于标准的开放接口:Solr搜索服务器支持通过XML.JSON和HTTP查询和获取结果. 2.易管理:Solr可以通过HTML页面管理,Solr配置通过XML完成. 3.可伸缩性:能够有效地复制到 ...

  10. 对DataTable进行过滤筛选的一些方法Select,dataview

    当你从数据库里取出一些数据,然后要对数据进行整合,你很容易就会想到: DataTable dt = new DataTable();//假设dt是由"SELECT C1,C2,C3 FROM ...