目录:

  1. System.Threading.Monitor:提供同步访问对象的机制;
  2. lock 是语法糖,是对 Monitor Enter 和 Exit 方法的一个封装
  3. lock 案例

1. Monitor

1. 基本方法

  1. public static void Enter(object obj);

    在指定对象上获取排他锁。
  2. public static void Exit(object obj);

    释放指定对象上的排他锁。

2. 使用例子

// 被 Monitor 保护的队列
private Queue<T> m_inputQueue = new Queue<T>(); // 给 m_inputQueue 加锁,并往 m_inputQueue 添加一个元素
public void Enqueue(T qValue)
{
// 请求获取锁,并阻塞其他线程获得该锁,直到获得锁
Monitor.Enter(m_inputQueue);
try
{
m_inputQueue.Enqueue(qValue);
}
finally
{
// 释放锁
Monitor.Exit(m_inputQueue);
}
}

2. lock

lock 是语法糖,是对Monitor的Enter和Exit的一个封装。

lock (m_inputQueue) {} 等价于

bool __lockWasTaken = false;
try
{
System.Threading.Monitor.Enter(m_inputQueue, ref __lockWasTaken);
}
finally
{
if (__lockWasTaken) System.Threading.Monitor.Exit(m_inputQueue);
}
  1. 当同步对共享资源的线程访问时,请锁定专用对象实例(例如,private readonly object balanceLock = new object();)或另一个不太可能被代码无关部分用作 lock 对象的实例。 避免对不同的共享资源使用相同的 lock 对象实例,因为这可能导致死锁或锁争用;
  2. 具体而言,避免将以下对象用作 lock 对象:

    1)this(调用方可能将其用作 lock)

    2)Type 实例(可以通过 typeof 运算符或反射获取)

    3)字符串实例,包括字符串文本,(这些可能是暂存的)。

    尽可能缩短持有锁的时间,以减少锁争用。
private readonly object balanceLock = new object();
private Queue<T> m_inputQueue = new Queue<T>(); public void Enqueue(T qValue)
{
lock (balanceLock)
{
m_inputQueue.Enqueue(qValue);
}
}

3. lock 案例

1. 数据库访问工厂单例模式

private static object _iBlockPortLockObj = new object();
private static IBlockPort _iBlockPort; /// <summary>
/// 卡口
/// </summary>
/// <returns></returns>
public static IBlockPort CreateBlockPort()
{
if (_iBlockPort == null)
{
lock (_iBlockPortLockObj)
{
if (_iBlockPort == null)
{
string className = AssemblyName + "." + db + "BlockPort";
_iBlockPort = (IBlockPort)Assembly.Load(AssemblyName).CreateInstance(className);
}
}
} return _iBlockPort;
}

2. 队列进出

public abstract class AbstractCache<T> where T : ICloneable
{
protected int queenLength = 30; // 保持队列的最大长度,主要可能考虑内存 /// <summary>
/// 过车缓存列表
/// </summary>
public List<T> listCache { get; set; } protected object _lockObj = new object(); /// <summary>
/// 初始化或重置缓存列表
/// </summary>
protected void RefreshListCache()
{
lock (_lockObj)
{
if (listCache == null)
{
listCache = new List<T>();
}
else
{
listCache.Clear();
}
}
} /// <summary>
/// 添加新的数据进队列,后续考虑做成环形队列减少开销
/// </summary>
/// <param name="list"></param>
protected void AddListToCache(List<T> list)
{
lock (_lockObj)
{
if (listCache == null) return; listCache.InsertRange(0, list);
if (listCache.Count > queenLength)
{
listCache.RemoveRange(queenLength, listCache.Count - queenLength);
}
}
} /// <summary>
/// 移除并返回过车缓存队列的最后一个元素
/// </summary>
/// <returns></returns>
public T DequeueLastCar()
{
T res = default; lock (_lockObj)
{
if (listCache != null && listCache.Count > 0)
{
int lastIndex = listCache.Count - 1;
res = (T)listCache[lastIndex].Clone();
listCache.RemoveAt(lastIndex);
}
} return res;
}
}
  1. 前提:在某项目上,view 的控件包括一个下拉框(可选idA、idB等)、一个图片 image;
  2. 数据逻辑设计:线程 A 定时根据下拉框的选择作为条件从第三方的数据库获取数据并添加进队列

    1)线程 B 定时从队列取出一个并展示到 image 控件

    2)当下拉框切换选择时,清空队列 [便于展示跟下拉框关联的图片]
  3. 问题:从第三方的数据库取数据需要 1s 左右,如果刚好出现这样的操作:线程 A 查数据库获取 idA 相关的数据(将持续 1s)-> 下拉框 idA 切换到 idB 并触发执行清空队列操作 -> 线程 A 将 idA 的数据添加到队列,将会出现下拉框切换 idB 之后依旧展示 idA 相关的数据。
  4. 解决:在线程 a 查数据库时就对队列加锁(同时去掉队列入队的锁,避免死锁),这样在获取数据的中途切换下拉框,就能等到获取完并加入队列后再清空。
  5. 导致新的问题:在获取的过程中,因队列被锁,导致无法线程 B 出队的操作被阻塞。
  6. 解决:入队和出队共用一个锁,从数据库获取数据和清空队列共用一个锁。
/// <summary>
/// 添加新的数据进队列,后续考虑做成环形队列减少开销
/// 清空、添加、取出一个数据,都需要加锁,但是由于添加的数据是从海康那边拿过来的,可能需要几秒的时间,
/// 可能会导致这样的结果:线程 A 查数据库(持续几秒)-> 线程 B 执行清空队列操作 -> 线程 A 将数据添加到队列
/// 因此将,锁直接移动到 lock {线程 A 查数据库、将数据添加到队列}
/// </summary>
/// <param name="list"></param>
protected void AddListToCache(List<T> list)
{
if (listCache == null) return; listCache.InsertRange(0, list);
if (listCache.Count > queenLength)
{
listCache.RemoveRange(queenLength, listCache.Count - queenLength);
}
} CancellationTokenSource source = new CancellationTokenSource(); /// <summary>
/// 定时获取 xx 数据
/// </summary>
public void GetPassCarInterval()
{
Task.Factory.StartNew(() =>
{
while (!source.IsCancellationRequested)
{
if (!string.IsNullOrWhiteSpace(xx))
{
lock (_lockObj)
{
// 从数据库获取数据
var list = GetPassCarInfo.GetLastBlockPortCarRecordBy(xx); AddListToCache(list);
}
} AutoReset.WaitOne(Common.GetDataTimespan);
}
}, TaskCreationOptions.LongRunning);
}

C# 应用 - 多线程 7) 处理同步数据之 Synchronized code regions (同步代码区域): Monitor 和 lock的更多相关文章

  1. 虎牙在全球 DNS 秒级生效上的实践 集群内通过 raft 协议同步数据,毫秒级别完成同步。

    https://mp.weixin.qq.com/s/9bEiE4QFBpukAfNOYhmusw 虎牙在全球 DNS 秒级生效上的实践 原创: 周健&李志鹏 阿里巴巴中间件 今天

  2. Java同步块(synchronized block)使用详解

    Java 同步块(synchronized block)用来标记方法或者代码块是同步的.Java同步块用来避免竞争.本文介绍以下内容: Java同步关键字(synchronzied) 实例方法同步 静 ...

  3. Java多线程学习---------超详细总结(java 多线程 同步 数据传递 )

    目录(?)[-] 一扩展javalangThread类 二实现javalangRunnable接口 三Thread和Runnable的区别 四线程状态转换 五线程调度 六常用函数说明 使用方式 为什么 ...

  4. Java多线程:线程同步与关键字synchronized

    一.同步的特性1. 不必同步类中所有的方法, 类可以同时拥有同步和非同步方法.2. 如果线程拥有同步和非同步方法, 则非同步方法可以被多个线程自由访问而不受锁的限制. 参见实验1:http://blo ...

  5. Java 多线程 死锁 隐性死锁 数据竞争 恶性数据竞争 错误解决深入分析 全方向举例

    在几乎所有编程语言中,由于多线程引发的错误都有着难以再现的特点,程序的死锁或其它多线程错误可能只在某些特殊的情形下才出现,或在不同的VM上运行同一个程序时错误表现不同.因此,在编写多线程程序时,事先认 ...

  6. java 多线程总结篇3之——生命周期和线程同步

    一.生命周期 线程的生命周期全在一张图中,理解此图是基本: 线程状态图 一.新建和就绪状态 当程序使用new关键字创建了一个线程之后,该线程就处于新建状态,此时它和其他的Java对象一样,仅仅由Jav ...

  7. Java多线程面试题:线程锁+线程池+线程同步等

    1.并发编程三要素? 1)原子性 原子性指的是一个或者多个操作,要么全部执行并且在执行的过程中不被其他操作打断,要么就全部都不执行. 2)可见性 可见性指多个线程操作一个共享变量时,其中一个线程对变量 ...

  8. centos6.5下部署sersync+rsync --daemon同步数据

    rsync --daemon端配置 [root@rsync-daemon etc]# /etc/init.d/iptables stop [root@rsync-daemon ~]# dos2unix ...

  9. 巧用 JuiceFS Sync 命令跨云迁移和同步数据

    近年来,云计算已成为主流,企业从自身利益出发,或是不愿意被单一云服务商锁定,或是业务和数据冗余,或是出于成本优化考虑,会尝试将部分或者全部业务从线下机房迁移到云或者从一个云平台迁移到另一个云平台,业务 ...

随机推荐

  1. k8s二进制部署 - coredns安装

    coredns的资源清单文件rabc.yaml apiVersion: v1 kind: ServiceAccount metadata: name: coredns namespace: kube- ...

  2. 部署开源IP管理工具phpIPAM

    一.安装环境程序: yum install httpd mariadb-server php php-cli php-gd php-common php-ldap php-pdo php-pear p ...

  3. leetcode31 下一个排列 字典序

    数字的字典序就是比大小 先想几个例子  123_>132  1243-> 1324 ,12453-> 12534 1.不可排的序列就是降序序列 2.两个相同长度的串比大小,前面相同, ...

  4. np.random.randint()的返回值

    返回的是数组而非int 比如返回x,y 为[1][2] 而非1,2 容易在只有一维一列时没有意识到 其他函数的返回值也要注意

  5. PyQt5 问题集

    PyQt5中遇到的一些问题 1.多线程中界面异步刷新 我这里需要给界面动态添加新的控件,但是多线程中似乎并不能直接更新页面? 对于逻辑和界面分离的情况,使用自定义信号的方式进行页面控件的动态添加.注意 ...

  6. 如何使用 Python 编写后端 API 接口

    如何使用 Python 编写后端 API 接口 get API Python3 # coding:utf-8 import json # ModuleNotFoundError: No module ...

  7. Enums & JavasScript & TypeScript

    Enums & JavasScript & TypeScript ES6 & TS https://www.typescriptlang.org/docs/handbook/e ...

  8. Docker使用指南

    上文简单介绍了docker,这边记录一下docker的使用. 一.Docker启停 1.启动docker systemctl start docker 2.关闭docker systemctl sto ...

  9. SSL (Secure Sockets Layer)

    本文转载自SSL (Secure Sockets Layer) TLS简介 The Transport Layer Security (TLS) protocol aims primarily to ...

  10. css选择器,过滤筛选

    $('.required:not(.final_price)').each(function() { if (!$(this).val()) { error_count ++; if ($(this) ...