临界区

来源:互联网 发布:年会酷炫的抽奖js 编辑:程序博客网 时间:2024/04/27 22:20
求助编辑

目录

简介
程序调度法则
线程同步问题
其他问题

编辑本段简介

  不论是硬件临界资源,还是软件临界资源,多个进程必须互斥地对它进行访问。每个进程中访问临界资源的那段代码称为临界区(Critical Section)(临界资源是一次仅允许一个进程使用的共享资源)。每次只准许一个进程进入临界区,进入后不允许其他进程进入。不论是硬件临界资源,还是软件临界资源,多个进程必须互斥地对它进行访问。
  多个进程中涉及到同一个临界资源的临界区称为相关临界区。

编辑本段程序调度法则

  进程进入临界区的调度原则是:
  1、如果有若干进程要求进入空闲的临界区,一次仅允许一个进程进入。
  2、任何时候,处于临界区内的进程不可多于一个。如已有进程进入自己的临界区,则其它所有试图进入临界区的进程必须等待。
  3、进入临界区的进程要在有限时间内退出,以便其它进程能及时进入自己的临界区。
  4、如果进程不能进入自己的临界区,则应让出CPU,避免进程出现“忙等”现象。

编辑本段线程同步问题

  如果有多个线程试图同时访问临界区,那么在有一个线程进入后其他所有试图访问此临界区的线程将被挂起,并一直持续到进入临界区的线程离开。临界区在被释放后,其他线程可以继续抢占,并以此达到用原子方式操作共享资源的目的。
  临界区在使用时以CRITICAL_SECTION结构对象保护共享资源,并分别用EnterCriticalSection()和LeaveCriticalSection()函数去标识和释放一个临界区。所用到的CRITICAL_SECTION结构对象必须经过InitializeCriticalSection()的初始化后才能使用,而且必须确保所有线程中的任何试图访问此共享资源的代码都处在此临界区的保护之下。否则临界区将不会起到应有的作用,共享资源依然有被破坏的可能。
  下面通过一段代码展示了临界区在保护多线程访问的共享资源中的作用。通过两个线程来分别对全局变量g_cArray[10]进行写入操作,用临界区结构对象g_cs来保持线程的同步,并在开启线程前对其进行初始化。为了使实验效果更加明显,体现出临界区的作用,在线程函数对共享资源g_cArray[10]的写入时,以Sleep()函数延迟1毫秒,使其他线程同其抢占CPU的可能性增大。如果不使用临界区对其进行保护,则共享资源数据将被破坏(参见图1(a)所示计算结果),而使用临界区对线程保持同步后则可以得到正确的结果(参见图1(b)所示计算结果)。
  图一

图一

  代码实现清单附下:
  // 临界区结构对象
  CRITICAL_SECTION g_cs;
  // 共享资源
  char g_cArray[10];
  UINT ThreadProc10(LPVOID pParam)
  {
  // 进入临界区
  EnterCriticalSection(&g_cs);
  // 对共享资源进行写入操作
  for (int i = 0; i < 10; i++)
  {
  g_cArray[i] = a;
  Sleep(1);
  }
  // 离开临界区
  LeaveCriticalSection(&g_cs);
  return 0;
  }
  UINT ThreadProc11(LPVOID pParam)
  {
  // 进入临界区
  EnterCriticalSection(&g_cs);
  // 对共享资源进行写入操作
  for (int i = 0; i < 10; i++)
  {
  g_cArray[10 - i - 1] = b;
  Sleep(1);
  }
  // 离开临界区
  LeaveCriticalSection(&g_cs);
  return 0;
  }
  ……
  void CSample08View::OnCriticalSection()
  {
  // 初始化临界区
  InitializeCriticalSection(&g_cs);
  // 启动线程
  AfxBeginThread(ThreadProc10, NULL);
  AfxBeginThread(ThreadProc11, NULL);
  // 等待计算完毕
  Sleep(300);
  // 报告计算结果
  CString sResult = CString(g_cArray);
  AfxMessageBox(sResult);
  }

编辑本段其他问题

  在使用临界区时,一般不允许其运行时间过长,只要进入临界区的线程还没有离开,其他所有试图进入此临界区的线程都会被挂起而进入到等待状态,并会在一定程度上影响程序的运行性能。尤其需要注意的是不要将等待用户输入或是其他一些外界干预的操作包含到临界区。如果进入了临界区却一直没有释放,同样也会引起其他线程的长时间等待。换句话说,在执行了EnterCriticalSection()语句进入临界区后无论发生什么,必须确保与之匹配的LeaveCriticalSection()都能够被执行到。可以通过添加结构化异常处理代码来确保LeaveCriticalSection()语句的执行。虽然临界区同步速度很快,但却只能用来同步本进程内的线程,而不可用来同步多个进程中的线程。
原创粉丝点击
热门问题 老师的惩罚 人脸识别 我在镇武司摸鱼那些年 重生之率土为王 我在大康的咸鱼生活 盘龙之生命进化 天生仙种 凡人之先天五行 春回大明朝 姑娘不必设防,我是瞎子 origin注册邮箱填错了怎么办 58同城手机输入不合法怎么办 银行卡密码输入错误三次怎么办 私密相册系统升级后打不开怎么办 由于志愿没填好孩子没书读怎么办 文具店不开了货怎么办 华为手机通讯录联系人重复怎么办 vcf文件用表格打开乱码怎么办 表格打出来太小怎么办 企业列入经营异常名录怎么办 小米电视滚动字幕模糊怎么办 海信电视浑的看不清怎么办 电视打开特别暗看不清怎么办 诈骗电话按了键怎么办 上海油电混合送沪牌以后怎么办? 车子被前夫砸了怎么办 老的标书丢了怎么办 拍牌照的标书掉了怎么办 上海拍到车牌后怎么办 杭州4s店车牌怎么办 天津车牌有指标想买车怎么办 上海大牌拍中了怎么办 买新车牌下不了怎么办 临沂上小学没报上名怎么办 早教中心倒闭了怎么办 企业税没交联系不上法人怎么办 企业被拉黑法人联系不到怎么办 支票根写错了怎么办 月结客户不付款怎么办 退市整理期过了怎么办 老板跑了财务负责人怎么办 在南京加入嘀嘀代驾怎么办 苹果盗刷支付宝怎么办 街电支付宝掉了怎么办 转转上号器没有苹果版的怎么办 安卓6不支持的应用怎么办 银行卡被存款机吞了怎么办 去银行卡存款机器故障怎么办 银行卡存款没有存进去怎么办 被黑网站黑钱了怎么办 饿了吗账户异常怎么办