猿掌柜 发布的文章 - 社畜猿
首页
🕒归档
📖留言板
💌关于
搜 索
1
我的小世界——博客上线啦
1,323 阅读
2
写于小程序上线之前
1,209 阅读
3
近期准备上线本站小程序
868 阅读
4
利用CSS3制作旋转3D立方体
745 阅读
5
RabbitMq C# .NET 接收广播 消费者 订阅者 简单使用 ~~
381 阅读
💯C#
🐘PHP
📝烂笔头
♉侃侃
登录
搜 索
标签搜索
React Native
rabbmitmq
thinkphp
Mysql
Typecho
socket
uniapp
tinkphp
php
echart
C语言
程序猿
累计撰写
130
篇文章
累计收到
17
条评论
首页
栏目
💯C#
🐘PHP
📝烂笔头
♉侃侃
页面
🕒归档
📖留言板
💌关于
用户登录
登录
找到
130
篇与
相关的结果
2021-09-28
c# 写入字节流文件
c# 写入字节流文件public void WriteData(byte[] data) { string path = AppDomain.CurrentDomain.BaseDirectory; path += "\\dat"; if (!string.IsNullOrEmpty(path)) { if (!Directory.Exists(path)) { Directory.CreateDirectory(path); } path = path + "\\" + filename + ".dat"; if (!File.Exists(path)) { FileStream fs = File.Create(path); fs.Close(); } if (File.Exists(path)) { FileStream file = new FileStream(path, FileMode.Append);//以追加的形式打开文件 file.Write(data, 0, data.Length);//写入byte[]型数据 file.Flush(); file.Close(); } } }其二FileStream fs = new FileStream(path, FileMode.Append); //初始化FileStream对象 BinaryWriter bw = new BinaryWriter(fs); //创建BinaryWriter对象 //FileStream file = new FileStream(path, FileMode.Append);//以追加的形式打开文件 foreach (var item in list) { bw.Write(item);//写入byte[]型数据 } //file.Flush(); bw.Close(); fs.Close();
2021年09月28日
32 阅读
1 评论
1 点赞
2021-09-13
雪花ID Snowflake .Net版—分布式唯一ID
先抄个雪花ID介绍,雪花算法:雪花ID是用一个64位的整形数字来做ID,对应.net中的long,数据库中的bigint,雪花算法的原始版本是scala版,用于生成分布式ID(纯数字,时间顺序),订单编号等。自增ID:对于数据敏感场景不宜使用,且不适合于分布式场景。 GUID:采用无意义字符串,数据量增大时造成访问过慢,且不宜排序。算法描述:最高位是符号位,始终为0,不可用。41位的时间序列,精确到毫秒级,41位的长度可以使用69年。时间位还有一个很重要的作用是可以根据时间进行排序。10位的机器标识,10位的长度最多支持部署1024个节点。12位的计数序列号,序列号即一系列的自增id,可以支持同一节点同一毫秒生成多个ID序号,12位的计数序列号支持每个节点每毫秒产生4096个ID序号。雪花ID严重依赖系统当前时间,当系统时间被人为反后调整时,算法会出问题,可能会出重复ID.Snowflake原算法是在检测到系统时间被回调后直接抛异常.本代码在时钟回拨后,会将生成的ID时间戳停留在最后一次时间戳上(每当序列溢出时会往前走一毫秒),等待系统时间追上后即可以避过时钟回拨问题.这种处理方式的优点是时钟回拨后不会异常,能一直生成出雪花ID,但缺点是雪花ID中的时间戳不是系统的当前时间,会是回拨前的最后记录的一次时间戳,但相差也不大.不知道有没有什么生产系统会对这个时间戳要求非常严格,无法使用这种补救方式的?当然停掉系统后的时钟回拨是无法处理的,这种还是会有可能出现重复ID的.介绍完毕,下面直接上源码吧,,本源码除了生成雪花ID外,还提供解析雪花ID的方法.using System; using System.Collections.Generic; using System.Diagnostics; using System.Linq; namespace snowflake_Id { class Program { static void Main(string[] args) { var ids = new List<long>(); var idwork = new SnowflakeId(15, 3); for (var i = 0; i < 100; i++) { var idTest = idwork.NextId(); ids.Add(idTest); } foreach (var id in ids) { var idStr = SnowflakeId.AnalyzeId(id); Console.WriteLine("雪花ID:" + id + "\t\b解析-> " + idStr); } return; var watch = Stopwatch.StartNew(); System.Threading.Tasks.Parallel.For(0, 8, i => { for (var j = 0; j < 100000; j++) { var id = idwork.NextId(); lock (ids) { ids.Add(id); } } }); Console.WriteLine(ids.Count+ "个Id生成完毕,耗时:" + watch.ElapsedMilliseconds); //return; System.Threading.Tasks.Parallel.For(0, ids.Count, (i, state) => { for (var j = i+1; j < ids.Count; j++) { if (ids[i] == ids[j]) { Console.WriteLine("有重复项:" + ids[i]); state.Stop(); } } }); Console.WriteLine("重复项检查完毕,耗时:" + watch.ElapsedMilliseconds); Console.WriteLine("fineshed"); } } } /// <summary> /// 雪花ID /// Twitter_Snowflake /// SnowFlake的结构如下(每部分用-分开) /// 0 - 0000000000 0000000000 0000000000 0000000000 0 - 00000 - 00000 - 000000000000 /// 1位标识,由于long基本类型在Java中是带符号的,最高位是符号位,正数是0,负数是1,所以id一般是正数,最高位是0 /// 41位时间截(毫秒级),注意,41位时间截不是存储当前时间的时间截,而是存储时间截的差值(当前时间截 - 开始时间截)得到的值), /// 41位的时间截,可以使用69年,年T = (1L << 41) / (1000L * 60 * 60 * 24 * 365) = 69 /// 这里的的开始时间截,一般是我们的id生成器开始使用的时间,由我们程序来指定的(如下下面程序IdWorker类的startTime属性)。 /// 10位的数据机器位,可以部署在1024个节点,包括5位datacenterId和5位workerId /// 12位序列,毫秒内的计数,12位的计数顺序号支持每个节点每毫秒(同一机器,同一时间截)产生4096个ID序号 /// 总共加起来刚好64位,为一个Long型。 /// SnowFlake的优点是,整体上按照时间自增排序,并且整个分布式系统内不会产生ID碰撞(由数据中心ID和机器ID作区分), /// 并且效率较高,经测试,SnowFlake单机每秒都能够产生出极限4,096,000个ID来 /// </summary> public class SnowflakeId { // 开始时间截((new DateTime(2020, 1, 1, 0, 0, 0, DateTimeKind.Utc)-Jan1st1970).TotalMilliseconds) private const long twepoch = 1577836800000L; // 机器id所占的位数 private const int workerIdBits = 5; // 数据标识id所占的位数 private const int datacenterIdBits = 5; // 支持的最大机器id,结果是31 (这个移位算法可以很快的计算出几位二进制数所能表示的最大十进制数) private const long maxWorkerId = -1L ^ (-1L << workerIdBits); // 支持的最大数据标识id,结果是31 private const long maxDatacenterId = -1L ^ (-1L << datacenterIdBits); // 序列在id中占的位数 private const int sequenceBits = 12; // 数据标识id向左移17位(12+5) private const int datacenterIdShift = sequenceBits + workerIdBits; // 机器ID向左移12位 private const int workerIdShift = sequenceBits; // 时间截向左移22位(5+5+12) private const int timestampLeftShift = sequenceBits + workerIdBits + datacenterIdBits; // 生成序列的掩码,这里为4095 (0b111111111111=0xfff=4095) private const long sequenceMask = -1L ^ (-1L << sequenceBits); // 数据中心ID(0~31) public long datacenterId // 工作机器ID(0~31) public long workerId // 毫秒内序列(0~4095) public long sequence // 上次生成ID的时间截 public long lastTimestamp /// <summary> /// 雪花ID /// </summary> /// <param name="datacenterId">数据中心ID</param> /// <param name="workerId">工作机器ID</param> public SnowflakeId(long datacenterId,long workerId ) { if (datacenterId > maxDatacenterId || datacenterId < 0) { throw new Exception(string.Format("datacenter Id can't be greater than or less than 0", maxDatacenterId)); } if (workerId > maxWorkerId || workerId < 0) { throw new Exception(string.Format("worker Id can't be greater than or less than 0", maxWorkerId)); } this.workerId = workerId; this.datacenterId = datacenterId; this.sequence = 0L; this.lastTimestamp = -1L; } /// <summary> /// 获得下一个ID /// </summary> /// <returns></returns> public long NextId() { lock (this) { long timestamp = GetCurrentTimestamp(); if (timestamp > lastTimestamp) //时间戳改变,毫秒内序列重置 { sequence = 0L; } else if (timestamp == lastTimestamp) //如果是同一时间生成的,则进行毫秒内序列 { sequence = (sequence + 1) & sequenceMask; if (sequence == 0) //毫秒内序列溢出 { timestamp = GetNextTimestamp(lastTimestamp); //阻塞到下一个毫秒,获得新的时间戳 } } else //当前时间小于上一次ID生成的时间戳,证明系统时钟被回拨,此时需要做回拨处理 { sequence = (sequence + 1) & sequenceMask; if (sequence > 0) { timestamp = lastTimestamp; //停留在最后一次时间戳上,等待系统时间追上后即完全度过了时钟回拨问题。 } else //毫秒内序列溢出 { timestamp = lastTimestamp + 1; //直接进位到下一个毫秒 } //throw new Exception(string.Format("Clock moved backwards. Refusing to generate id for milliseconds", lastTimestamp - timestamp)); } lastTimestamp = timestamp; //上次生成ID的时间截 //移位并通过或运算拼到一起组成64位的ID var id= ((timestamp - twepoch) << timestampLeftShift) | (datacenterId << datacenterIdShift) | (workerId << workerIdShift) | sequence; return id; } } /// <summary> /// 解析雪花ID /// </summary> /// <returns></returns> public static string AnalyzeId(long Id) { StringBuilder sb = new StringBuilder(); var timestamp = (Id >> timestampLeftShift) ; var time = Jan1st1970.AddMilliseconds(timestamp + twepoch); sb.Append(time.ToLocalTime().ToString("yyyy-MM-dd HH:mm:ss:fff")); var datacenterId = (Id ^ (timestamp << timestampLeftShift)) >> datacenterIdShift; sb.Append("_" + datacenterId); var workerId = (Id ^ ((timestamp << timestampLeftShift) | (datacenterId << datacenterIdShift))) >> workerIdShift; sb.Append("_" + workerId); var sequence = Id & sequenceMask; sb.Append("_" + sequence); return sb.ToString(); } /// <summary> /// 阻塞到下一个毫秒,直到获得新的时间戳 /// </summary> /// <param name="lastTimestamp">上次生成ID的时间截</param> /// <returns>当前时间戳</returns> private static long GetNextTimestamp(long lastTimestamp) { long timestamp = GetCurrentTimestamp(); while (timestamp <= lastTimestamp) { timestamp = GetCurrentTimestamp(); } return timestamp; } /// <summary> /// 获取当前时间戳 /// </summary> /// <returns></returns> private static long GetCurrentTimestamp() { return (long)(DateTime.UtcNow - Jan1st1970).TotalMilliseconds; } private static readonly DateTime Jan1st1970 = new DateTime(1970, 1, 1, 0, 0, 0, DateTimeKind.Utc); }
2021年09月13日
81 阅读
1 评论
2 点赞
2021-09-07
C# 保存程序运行日志
保存程序运行日志
2021年09月07日
28 阅读
1 评论
3 点赞
2021-09-06
c# 16进制字节数组转16进制字符串
16进制字节数组转16进制字符串
2021年09月06日
79 阅读
0 评论
0 点赞
2021-09-06
c# 标准时间与时间戳相互转换
c# 标准时间与时间戳相互转换
2021年09月06日
41 阅读
0 评论
3 点赞
2021-09-06
c# 时间戳转换16进制byte数组
c# 时间戳转换16进制byte数组
2021年09月06日
76 阅读
0 评论
1 点赞
2021-09-03
两周年啦!!!
时间过得真快啊在这边已经两年了刚开始过来搞web,头两个月确实搞web,前后端全包了后来20年疫情上班后,之前写上位机的同事被开了,然后又给分配了一个项目又开始学.net写上位机,。。。这一写就停不下来了~~~嗯,写写.net也挺好玩的。。。有新招来一个应届生,让他写Web不知道他啥时候能接手哦估计我这是要转向.net了~~~我的小电车怎么还在排产,难过~~~油价还是太贵了~~~6.81,比之前便宜了2毛~ 学习图像识别算法 学个JAVA?
2021年09月03日
32 阅读
0 评论
1 点赞
2021-08-25
又到八月末了~~
买了个小车
2021年08月25日
37 阅读
0 评论
0 点赞
2021-07-26
C#获取文件夹下的所有文件详解
C#获取文件夹下的所有文件详解
2021年07月26日
17 阅读
0 评论
1 点赞
2021-07-12
c# 连接mysql关闭清空缓冲池—记录
c# 连接mysql关闭清空缓冲池
2021年07月12日
22 阅读
0 评论
1 点赞
1
...
5
6
7
...
13