cad.net 定点数表示DWG版本号
定点数好处
发现不少人组合了"大版本.小版本"然后判断字符串,
虽然效率上面不差多少,但是不太考究细节是不对的.(因为我变强了,要节约内存了注重效率了)
1,有序数组比起hashmap来说占用内存少,
并且序列化和反序列化也是有序的,顺序读写,载入即用.
2,数据太少会是全量缓存,二分法此时运行是很快的.
3,遇到了小数点问题,通过位移拆开它们到int32的高16bit和低16bit,
高位越大,值也是越大的,所以仍然是原有顺序的(需要考虑溢出),
例如: a<<16 | b
小版本号,我用了Q16.16定点数,稍微学习了一个新概念.
例如: 123.456 * (1<<16)
4,三个数组可以序列化成config,交给外部来维护,不过我不太建议.
工具类
public static class VersionTool {
// 年号.
// 1,不用存年号数组,索引+2000就是年号了,
// 2,官方是Acad2000i,用2001年顶替
// 3,官方没有Acad2003,但要满足1的规则,全部数组用上年数据补齐03年.
// static readonly int[] _years =>[
// 2000, 2001, 2002, 2003/*虚拟03年*/,
// 2004, 2005, 2006,
// 2007, 2008, 2009,
// 2010, 2011, 2012,
// 2013, 2014, 2015, 2016, 2017,
// 2018, 2019, 2020, 2021, 2022,
// 2023, 2024, 2025];
// 字段:public static readonly int[] Years;
// 构造:Years = _years;
// 异或判断数值是否一致
// Debug.Assert((_years.Length ^ DwgVers.Length ^ AcadVers.Length) == AcadVers.Length, "怎么长度不一样了捏?");
#if NET45_OR_GREATER
[MethodImpl(MethodImplOptions.AggressiveInlining)]
#endif
public static int IndexToYear(int index){
int year = index + 2000;
if (year < 2000) return -1;
if (year > 2000 + AcadVers.Length) return -1;
return year;
}
// 和二分法一样,
// 没有命中时将第一个大于搜索值的索引作为返回值并取反,
// 以提供没有命中的信息(负数)和最近值取反后再插入.
#if NET45_OR_GREATER
[MethodImpl(MethodImplOptions.AggressiveInlining)]
#endif
public static int YearToIndex(int year){
int index = year - 2000;
if (index < 0) return ~0;
if (index > AcadVers.Length - 1)
return ~(AcadVers.Length - 1);
return index;
}
// https://www.autodesk.com.cn/support/technical/article/caas/sfdcarticles/sfdcarticles/CHS/drawing-version-codes-for-autocad.html
// DWG签名的AC号.
static readonly int[] _dwgVers = [
1015, 1015, 1015, 1015/*补齐03年*/,
1018, 1018, 1018,
1021, 1021, 1021,
1024, 1024, 1024,
1027, 1027, 1027, 1027, 1027,
1032, 1032, 1032, 1032, 1032,
1032, 1032, 1032];
// CAD版本号,它与注册表和COM相关.
static readonly string[] _acadVers = [
"R15.0", "R15.1", "R15.2", "R15.2"/*补齐03年*/,
"R16.0", "R16.1", "R16.2",
"R17.0", "R17.1", "R17.2",
"R18.0", "R18.1", "R18.2",
"R19.0", "R19.1", "R20.0", "R20.1", "R21.0",
"R22.0", "R23.0", "R23.1", "R24.0", "R24.1",
"R24.2", "R24.3", "R25.0"];
// 运行中的CAD年号
public static readonly int CurrentYear;
// 运行中的CAD版本号,位移+位或.
public static readonly int AcadVerOR;
// 这几个只读数组,有序等长,通过二分法共同索引.
public static readonly int[] AcadVers;
public static readonly int[] DwgVers;
// 静态构造函数
static VersionTool() {
int major = Acap.Version.Major;
int minor = Acap.Version.Minor;
// Q16.16定点数
// 0.5*2^16得到小数部分,则为5*(0.1f * (1 << 16));
// 避免浮点数运算产生偏移,提前算后面出来,得6553.
// minor是单数,不会溢出,不需要unchecked
var fp = minor * 6553;
AcadVerOR = (int)( ((uint)major << 16) | (ushort)fp );
// 消除字符串从而不需要构造hashmap.
// 位移+位或,使得它们从小数映射到整数,
// 并且保持大小关系,仍然有序可以能够二分.
// 编译时:C#没有循环消除,反编译源码还能看见,
// 运行时:避免求一次循环,可以把结果再重新粘贴回代码.
// --就像快速平方根倒数,夹逼求浮点数,再转16进制,然后粘贴到代码上,成为魔法数
int[] v = new int[_acadVers.Length];
for(int i = 0; i < _acadVers.Length; i++){
var s = _acadVers[i];
int m = (s[1] - '0')*10 + (s[2] - '0');
int n = (s[4] - '0'); //s[0]='R',s[3]='.',所以到[4]
var f = n * 6553;
v[i] = (int)( ((uint)m << 16) | (ushort)f );
}
AcadVers = v;
_acadVers = null!; //积极释放
DwgVers = _dwgVers;
// 设定年份
int index = Array.BinarySearch(AcadVers, AcadVerOR);
CurrentYear = IndexToYear(index);
#if DEBUG
// 防止开发者忘记更新数组,提供一个断言来进行维护
Debug.Assert(index > 0, "运行中的CAD版本未记录,请维护DWG版本号/CAD版本号");
Debug.Assert(DwgVers.Length == AcadVers.Length, "怎么长度不一样了捏");
// 解码版本号
int major2 = AcadVerOR >> 16;
int minor2 = AcadVerOR & 0xFFFF;//位与掩码,获取低16bit
minor2 /= 6553;
#endif
}
}
二分法找共同索引例子
根据微软介绍,没有命中的话,返回值负数是有意义的,
取反之后就是最近值(第一个比它大,所以可以直接insert).
Array.BinarySearch
int yearToFind = 2015;
int index = VersionTool.YearToIndex(yearToFind);
if (index < 0) {
Console.WriteLine($"最近CAD版本:{VersionTool.AcadVers[~index]}");
return;
}
Console.WriteLine($"索引:{index}.");
Console.WriteLine($"年份:{VersionTool.IndexToYear(index)}.");
Console.WriteLine($"CAD版本:{VersionTool.AcadVers[index]}");
Console.WriteLine($"DWG版本:{VersionTool.DwgVers[index]}");
获取DWG文件版本
模拟打开DWG文件.
文件类型不要根据文件后缀名判断,那是给用户看的,要根据文件签名判断
调用时候都要try一层,因为win有磁盘权限/管理员权限/只读文件属性/局域网共享中断等各种意想不到的情况
如何知道文件已经被其他程序打开?
根据Acad官方博客,是用Append模式打开来触发
// 调用要加using和try
public static FileStream ReadWriteDWG(string dwgFile,
FileMode fm = FileMode.Open,
FileAccess fa = FileAccess.Read,
FileShare fs = FileShare.ReadWrite | FileShare.Delete) {
// 文件流
var fileStream = new FileStream(dwgFile, fm, fa, fs);
// 通过文件签名判断是否DWG,位于前7个字节:
// https://blog.csdn.net/jiangyb999/article/details/124497625
int signature = 7;
if(fileStream.Length < signature)
throw new FileFormatException($"惊!文件签名损坏:{dwgFile}");
// 栈内分配
Span<byte> bomBuffer = stackalloc byte[signature];
// 堆上分配,如果Span用不了就用它
// byte[] bomBuffer = new byte[signature];
signature = fileStream.Read(bomBuffer, 0, signature);
if(signature < bomBuffer.Length)
throw new FileFormatException($"惊!文件签名损坏:{dwgFile}");
if(!(bomBuffer[0]=='A' && bomBuffer[1]=='C' && bomBuffer[6]=='\0'))
throw new FileFormatException($"惊!文件类型不是DWG:{dwgFile}");
// "AC1015\0",提取数字部分,不想把类型转去string再转回int,直接展开写.
int dn = (bomBuffer[2] - '0') * 1000
+ (bomBuffer[3] - '0') * 100
+ (bomBuffer[4] - '0') * 10
+ (bomBuffer[5] - '0');
int index = Array.BinarySearch(VersionTool.DwgVers, dn);
if (index < 0) {
string ver = Encoding.ASCII.GetString(bomBuffer, 0, bomBuffer.Length - 1);
throw new FileFormatException($"惊!插件未记录版本:{ver},快联系作者");
}
// 插件已经记录,但是此时运行在低版本CAD中,
// DWG版本会高于当前运行的CAD
// 坏主意:打不开就去注册表,找找有没有高版本CAD,然后让它来打开,嘻嘻...
if (VersionTool.AcadVers[index] > VersionTool.AcadVerOR) {
throw new FileFormatException("惊!此DWG高于当前CAD版本");
}
return fileStream;
}
相关链接
e大博客收集的CAD版本号
https://www.cnblogs.com/edata/p/10802746.html
我在COM文章收集了注册表中全部的CAD(我之前的代码好像惨不忍睹)
https://www.cnblogs.com/JJBox/p/11381254.html
枚举类型,按位运算的原理
https://www.cnblogs.com/JJBox/p/12033641.html
(完)