依星源码资源网,依星资源网

 找回密码
 立即注册

QQ登录

只需一步,快速开始

【好消息,好消息,好消息】VIP会员可以发表文章赚积分啦 !
查看: 130|回复: 0

读取纯真IP数据库C++源代码

[复制链接] 主动推送

1万

主题

1万

帖子

1万

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
12061
发表于 2024-11-16 12:08:14 | 显示全部楼层 |阅读模式
读取纯真IP数据库C++源代码
头文件
  1. #pragma once;
  2. #include <string>

  3. const int INDEX_LENGTH = 7;        // 一个索引包含4字节的起始IP和3字节的IP记录偏移,共7字节
  4. const int IP_LENGTH = 4;
  5. const int OFFSET_LENGTH = 3;

  6. enum {
  7. REDIRECT_MODE_1 = 0x01,    // 重定向模式1 偏移量后无地区名
  8. REDIRECT_MODE_2 = 0x02,    // 重定向模式2 偏移量后有地区名
  9. };

  10. class CIpFinder
  11. {
  12. public:
  13.     CIpFinder();
  14.     CIpFinder(const char* pszFileName);
  15.     ~CIpFinder();

  16.     // 获取ip国家名、地区名
  17.     void GetAddressByIp(unsigned long ipValue, std::string& strCountry, std::string& strLocation) const;
  18.     void GetAddressByIp(const char* pszIp, std::string& strCountry, std::string& strLocation) const;
  19.     void GetAddressByOffset(unsigned long ulOffset, std::string& strCountry, std::string& strLocation) const;

  20.     unsigned long GetString(std::string& str, unsigned long indexStart) const;
  21.     unsigned long GetValue3(unsigned long indexStart) const;
  22.     unsigned long GetValue4(unsigned long indexStart) const;

  23.     // 转换
  24.     unsigned long IpString2IpValue(const char *pszIp) const;
  25.     void IpValue2IpString(unsigned long ipValue, char* pszIpAddress, int nMaxCount) const;
  26.     bool IsRightIpString(const char* pszIp) const;

  27.     // 输出数据
  28.     unsigned long OutputData(const char* pszFileName, unsigned long ulIndexStart = 0, unsigned long ulIndexEnd = 0) const;
  29.     unsigned long OutputDataByIp(const char* pszFileName, unsigned long ipValueStart, unsigned long ipValueEnd) const;
  30.     unsigned long OutputDataByIp(const char* pszFileName, const char* pszStartIp, const char* pszEndIp) const;

  31.     unsigned long SearchIp(unsigned long ipValue, unsigned long indexStart = 0, unsigned long indexEnd = 0) const;
  32.     unsigned long SearchIp(const char* pszIp, unsigned long indexStart = 0, unsigned long indexEnd = 0) const;

  33.     bool Open(const char* pszFileName);
  34. private:
  35.     FILE *m_fpIpDataFile;            // IP数据库文件
  36.     unsigned long m_indexStart;    // 起始索引偏移
  37.     unsigned long m_indexEnd;        // 结束索引偏移
  38. };
复制代码
实现文件
  1. #include "ip.h"
  2. // ============================================================================
  3. // ==============================================================================
  4. CIpFinder::CIpFinder()
  5. {
  6. }

  7. // ============================================================================
  8. // ==============================================================================
  9. CIpFinder::CIpFinder(const char *pszFileName)
  10. {
  11.     this->Open(pszFileName);
  12. }

  13. // ============================================================================
  14. //    打开数据库文件
  15. // ==============================================================================
  16. bool CIpFinder::Open(const char *pszFileName)
  17. {
  18.     m_fpIpDataFile = fopen(pszFileName, "rb");
  19.     if (!m_fpIpDataFile) {
  20.         return false;
  21.     }

  22.     // IP头由两个十六进制4字节偏移量构成,分别为索引开始,和索引结束
  23.     m_indexStart = this->GetValue4(0);
  24.     m_indexEnd = this->GetValue4(4);
  25.     return true;
  26. }

  27. // ============================================================================
  28. // ==============================================================================
  29. CIpFinder::~CIpFinder()
  30. {
  31.     fclose(m_fpIpDataFile);
  32. }

  33. // ============================================================================
  34. //    根据IP地址字符串返回其十六进制值(4字节)
  35. // ============================================================================
  36. unsigned long CIpFinder::IpString2IpValue(const char *pszIp) const
  37. {
  38.     if (!this->IsRightIpString(pszIp)) {
  39.         return 0;
  40.     }
  41.     //~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  42.     int nNum = 0;            // 每个段数值
  43.     const char *pBeg = pszIp;
  44.     const char *pPos = NULL;
  45.     unsigned long ulIp = 0; // 整个IP数值
  46.     //~~~~~~~~~~~~~~~~~~~~~~~~~~~~

  47.     pPos = strchr(pszIp, '.');
  48.     while (pPos != NULL) {
  49.         nNum = atoi(pBeg);
  50.         ulIp += nNum;
  51.         ulIp *= 0x100;
  52.         pBeg = pPos + 1;
  53.         pPos = strchr(pBeg, '.');
  54.     }

  55.     nNum = atoi(pBeg);
  56.     ulIp += nNum;
  57.     return ulIp;
  58. }

  59. // ============================================================================
  60. //    根据ip值获取字符串(由点分割)
  61. // ============================================================================
  62. void CIpFinder::IpValue2IpString(unsigned long ipValue,
  63.                                  char *pszIpAddress,
  64.                                  int nMaxCount) const
  65. {
  66.     if (!pszIpAddress) {
  67.         return;
  68.     }

  69.     _snprintf(pszIpAddress, nMaxCount, "%d.%d.%d.%d", (ipValue & 0xFF000000) >> 24,
  70.               (ipValue & 0x00FF0000) >> 16, (ipValue & 0x0000FF00) >> 8,ipValue & 0x000000FF);
  71.     pszIpAddress[nMaxCount - 1] = 0;
  72. }

  73. // ============================================================================
  74. //    根据指定IP(十六进制值),返回其在索引段中的位置(索引)
  75. //    ulIndexStart和ulIndexEnd可以指定搜索范围 均为0表示搜索全部
  76. // ============================================================================
  77. unsigned long CIpFinder::SearchIp(unsigned long ipValue,
  78.                                   unsigned long indexStart,
  79.                                   unsigned long indexEnd) const
  80. {
  81.     if (!m_fpIpDataFile) {
  82.         return 0;
  83.     }

  84.     if (0 == indexStart) {
  85.         indexStart = m_indexStart;
  86.     }

  87.     if (0 == indexEnd) {
  88.         indexEnd = m_indexEnd;
  89.     }

  90.     //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  91.     unsigned long indexLeft = indexStart;
  92.     unsigned long indexRight = indexEnd;

  93.     // 先除后乘是为了保证mid指向一个完整正确的索引
  94.     unsigned long indexMid = (indexRight - indexLeft) / INDEX_LENGTH / 2 * INDEX_LENGTH + indexLeft;

  95.     // 起始Ip地址(如172.23.0.0),他和Ip记录中的Ip地址(如172.23.255.255)构成一个Ip范围,在这个范围内的Ip都可以由这条索引来获取国家、地区
  96.     unsigned long ulIpHeader = 0;
  97.     //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

  98.     do
  99.     {
  100.         ulIpHeader = this->GetValue4(indexMid);
  101.         if (ipValue < ulIpHeader) {
  102.             indexRight = indexMid;
  103.             indexMid = (indexRight - indexLeft) /    INDEX_LENGTH / 2 * INDEX_LENGTH + indexLeft;
  104.         } else {
  105.             indexLeft = indexMid;
  106.             indexMid = (indexRight - indexLeft) /    INDEX_LENGTH / 2 * INDEX_LENGTH + indexLeft;
  107.         }
  108.     } while (indexLeft < indexMid);            // 注意是根据mid来进行判断

  109.     // 只要符合范围就可以,不需要完全相等
  110.     return indexMid;
  111. }

  112. // ============================================================================
  113. // ==============================================================================
  114. unsigned long CIpFinder::SearchIp(const char *pszIp,
  115.                                   unsigned long indexStart,
  116.                                   unsigned long indexEnd) const
  117. {
  118.     if (!this->IsRightIpString(pszIp)) {
  119.         return 0;
  120.     }
  121.     return this->SearchIp(this->IpString2IpValue(pszIp), indexStart,
  122.                           indexEnd);
  123. }

  124. // ==========================================================================================================
  125. //    从指定位置获取一个十六进制的数 (读取3个字节, 主要用于获取偏移量, 与效率紧密相关的函数,尽可能优化)
  126. // ==========================================================================================================
  127. unsigned long CIpFinder::GetValue3(unsigned long indexStart) const
  128. {
  129.     if (!m_fpIpDataFile) {
  130.         return 0;
  131.     }

  132.     //~~~~~~~~~~~~~~~~~~~~
  133.     int nVal[3];
  134.     unsigned long ulValue = 0;
  135.     //~~~~~~~~~~~~~~~~~~~~

  136.     fseek(m_fpIpDataFile, indexStart, SEEK_SET);
  137.     for (int i = 0; i < 3; i++) {

  138.         // 过滤高位,一次读取一个字符
  139.         nVal[i] = fgetc(m_fpIpDataFile) & 0x000000FF;
  140.     }

  141.     for (int j = 2; j >= 0; --j) {

  142.         // 因为读取多个16进制字符,叠加
  143.         ulValue = ulValue * 0x100 + nVal[j];
  144.     }
  145.     return ulValue;
  146. }

  147. // ==========================================================================================================
  148. //    从指定位置获取一个十六进制的数 (读取4个字节, 主要用于获取IP值, 与效率紧密相关的函数,尽可能优化)
  149. // ==========================================================================================================
  150. unsigned long CIpFinder::GetValue4(unsigned long indexStart) const
  151. {
  152.     if (!m_fpIpDataFile) {
  153.         return 0;
  154.     }

  155.     //~~~~~~~~~~~~~~~~~~~~
  156.     int nVal[4];
  157.     unsigned long ulValue = 0;
  158.     //~~~~~~~~~~~~~~~~~~~~

  159.     fseek(m_fpIpDataFile, indexStart, SEEK_SET);
  160.     for (int i = 0; i < 4; i++) {

  161.         // 过滤高位,一次读取一个字符
  162.         nVal[i] = fgetc(m_fpIpDataFile) & 0x000000FF;
  163.     }

  164.     for (int j = 3; j >= 0; --j) {

  165.         // 因为读取多个16进制字符,叠加
  166.         ulValue = ulValue * 0x100 + nVal[j];
  167.     }
  168.     return ulValue;
  169. }

  170. // ============================================================================
  171. //    从指定位置获取字符串
  172. // ============================================================================
  173. unsigned long CIpFinder::GetString(std::string &str, unsigned long indexStart) const
  174. {
  175.     if (!m_fpIpDataFile) {
  176.         return 0;
  177.     }

  178.     str.erase(0, str.size());

  179.     fseek(m_fpIpDataFile, indexStart, SEEK_SET);
  180.     //~~~~~~~~~~~~~~~~~~~~~~
  181.     int nChar = fgetc(m_fpIpDataFile);
  182.     unsigned long ulCount = 1;
  183.     //~~~~~~~~~~~~~~~~~~~~~~

  184.    

  185.     // 读取字符串,直到遇到0x00为止
  186.      while (nChar != 0x00) {

  187.         // 依次放入用来存储的字符串空间中
  188.         str += static_cast<char>(nChar);
  189.         ++ulCount;
  190.         nChar = fgetc(m_fpIpDataFile);
  191.     }

  192.     // 返回字符串长度
  193.     return ulCount;
  194. }

  195. // ============================================================================
  196. //    通过指定的偏移量来获取ip记录中的国家名和地区名,偏移量可由索引获取
  197. //    ulOffset为Ip记录偏移量
  198. // ============================================================================
  199. void CIpFinder::GetAddressByOffset(unsigned long ulOffset,
  200.                                    std::string &strCountry,
  201.                                    std::string &strLocation) const
  202. {
  203.     if (!m_fpIpDataFile) {
  204.         return;
  205.     }

  206.     // 略去4字节Ip地址
  207.     ulOffset += IP_LENGTH;
  208.     fseek(m_fpIpDataFile, ulOffset, SEEK_SET);

  209.     //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  210.     // 读取首地址的值
  211.     int nVal = (fgetc(m_fpIpDataFile) & 0x000000FF);
  212.     unsigned long ulCountryOffset = 0;    // 真实国家名偏移
  213.     unsigned long ulLocationOffset = 0; // 真实地区名偏移
  214.     //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

  215.     // 为了节省空间,相同字符串使用重定向,而不是多份拷贝
  216.     if (REDIRECT_MODE_1 == nVal) {

  217.         //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  218.         // 重定向1类型
  219.         unsigned long ulRedirect = this->GetValue3(ulOffset + 1); // 重定向偏移
  220.                                                                     ///
  221.         //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

  222.         fseek(m_fpIpDataFile, ulRedirect, SEEK_SET);

  223.         if ((fgetc(m_fpIpDataFile) & 0x000000FF) == REDIRECT_MODE_2) {

  224.             // 混合类型1,重定向1类型进入后遇到重定向2类型
  225.             // 0x01 1字节
  226.             // 偏移量 3字节 -----> 0x02 1字节
  227.             //                     偏移量 3字节 -----> 国家名
  228.             //                     地区名
  229.             ulCountryOffset = this->GetValue3(ulRedirect + 1);
  230.             this->GetString(strCountry, ulCountryOffset);
  231.             ulLocationOffset = ulRedirect + 4;
  232.         } else {

  233.             // 单纯的重定向模式1
  234.             // 0x01 1字节
  235.             // 偏移量 3字节 ------> 国家名
  236.             //                      地区名
  237.             ulCountryOffset = ulRedirect;
  238.             ulLocationOffset = ulRedirect + this->GetString(strCountry,
  239.                                                             ulCountryOffset);
  240.         }
  241.     } else if (REDIRECT_MODE_2 == nVal) {

  242.         // 重定向2类型
  243.         // 0x02 1字节
  244.         // 国家偏移 3字节 -----> 国家名
  245.         // 地区名
  246.         ulCountryOffset = this->GetValue3(ulOffset + 1);
  247.         this->GetString(strCountry, ulCountryOffset);

  248.         ulLocationOffset = ulOffset + 4;
  249.     } else {

  250.         // 最简单的情况 没有重定向
  251.         // 国家名
  252.         // 地区名
  253.         ulCountryOffset = ulOffset;
  254.         ulLocationOffset = ulCountryOffset + this->GetString(strCountry,
  255.                                                              ulCountryOffset);
  256.     }

  257.     // 读取地区
  258.     fseek(m_fpIpDataFile, ulLocationOffset, SEEK_SET);
  259.     if ((fgetc(m_fpIpDataFile) & 0x000000FF) == REDIRECT_MODE_2
  260.     || (fgetc(m_fpIpDataFile) & 0x000000FF) == REDIRECT_MODE_1) {

  261.         // 混合类型2(最复杂的情形,地区也重定向)
  262.         // 0x01 1字节
  263.         // 偏移量 3字节 ------> 0x02 1字节
  264.         //                      偏移量 3字节 -----> 国家名
  265.         //                      0x01 or 0x02 1字节
  266.         //                      偏移量 3字节 ----> 地区名 偏移量为0表示未知地区
  267.         ulLocationOffset = this->GetValue3(ulLocationOffset + 1);
  268.     }

  269.     this->GetString(strLocation, ulLocationOffset);
  270. }

  271. // ============================================================================
  272. //    根据十六进制ip获取国家名地区名
  273. // ============================================================================
  274. void CIpFinder::GetAddressByIp(unsigned long ipValue,
  275.                                std::string &strCountry,
  276.                                std::string &strLocation) const
  277. {
  278.     //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  279.     unsigned long ulIndexOffset = this->SearchIp(ipValue);
  280.     unsigned long ulRecordOffset = this->GetValue3(ulIndexOffset + IP_LENGTH);
  281.     //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

  282.     this->GetAddressByOffset(ulRecordOffset, strCountry, strLocation);
  283. }

  284. // ============================================================================
  285. //    根据ip字符串获取国家名地区名
  286. // ============================================================================
  287. void CIpFinder::GetAddressByIp(const char *pszIp,
  288.                                std::string &strCountry,
  289.                                std::string &strLocation) const
  290. {
  291.     if (!this->IsRightIpString(pszIp)) {
  292.         return;
  293.     }
  294.     this->GetAddressByIp(this->IpString2IpValue(pszIp), strCountry, strLocation);
  295. }

  296. // ============================================================================
  297. //    将ip数据导出,start和end界定导出范围, 可通过SearchIp来获取
  298. // ============================================================================
  299. unsigned long CIpFinder::OutputData(const char *pszFileName,
  300.                                     unsigned long indexStart,
  301.                                     unsigned long indexEnd) const
  302. {
  303.     if (!m_fpIpDataFile || !pszFileName) {
  304.         return 0;
  305.     }

  306.     //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  307.     FILE *fpOut = fopen(pszFileName, "wb");
  308.     //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

  309.     if (!fpOut) {
  310.         return 0;
  311.     }

  312.     if (0 == indexStart) {
  313.         indexStart = m_indexStart;
  314.     }

  315.     if (0 == indexEnd) {
  316.         indexEnd = m_indexEnd;
  317.     }

  318.     //~~~~~~~~~~~~~~~~~~~~~~~~
  319.     char szEndIp[255];
  320.     char szStartIp[255];
  321.     std::string strCountry;
  322.     std::string strLocation;
  323.     unsigned long ulCount = 0;
  324.     unsigned long ipValueEnd = 0;
  325.     unsigned long ipValueStart = 0;
  326.     //~~~~~~~~~~~~~~~~~~~~~~~~

  327.     for (unsigned long i = indexStart; i < indexEnd; i += INDEX_LENGTH) {

  328.         // 获取IP段的起始IP和结束IP, 起始IP为索引部分的前4位16进制
  329.         // 结束IP在IP信息部分的前4位16进制中,靠索引部分指定的偏移量找寻
  330.         ipValueStart = this->GetValue4(i);
  331.         ipValueEnd = this->GetValue4(this->GetValue3(i + IP_LENGTH));

  332.         // 导出IP信息,格式是 起始IP/t结束IP/t国家位置/t地域位置/n
  333.         this->IpValue2IpString(ipValueStart, szStartIp, sizeof(szStartIp));
  334.         this->IpValue2IpString(ipValueEnd, szEndIp, sizeof(szEndIp));
  335.         this->GetAddressByOffset(this->GetValue3(i + IP_LENGTH), strCountry,
  336.                                  strLocation);
  337.         fprintf(fpOut, "%s/t%s/t%s/t%s/n", szStartIp, szEndIp,
  338.                 strCountry.c_str(), strLocation.c_str());
  339.         ulCount++;
  340.     }

  341.     fclose(fpOut);

  342.     // 返回导出总条数
  343.     return ulCount;
  344. }

  345. // ============================================================================
  346. //    通过ip值界定导出范围
  347. // ==============================================================================
  348. unsigned long CIpFinder::OutputDataByIp(const char *pszFileName,
  349.                                         unsigned long ipValueStart,
  350.                                         unsigned long ipValueEnd) const
  351. {
  352.     //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  353.     unsigned long indexStart = this->SearchIp(ipValueStart);
  354.     unsigned long indexEnd = this->SearchIp(ipValueEnd);
  355.     //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

  356.     return this->OutputData(pszFileName, indexStart, indexEnd);
  357. }

  358. // ============================================================================
  359. //    通过ip字符串界定导出范围
  360. // ==============================================================================
  361. unsigned long CIpFinder::OutputDataByIp(const char *pszFileName,
  362.                                         const char *pszStartIp,
  363.                                         const char *pszEndIp) const
  364. {
  365.     if (!this->IsRightIpString(pszStartIp) || !this->IsRightIpString(pszEndIp)) {
  366.         return 0;
  367.     }
  368.     //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  369.     unsigned long ipValueEnd = this->IpString2IpValue(pszEndIp);
  370.     unsigned long ipValueStart = this->IpString2IpValue(pszStartIp);
  371.     unsigned long indexEnd = this->SearchIp(ipValueEnd);
  372.     unsigned long indexStart = this->SearchIp(ipValueStart);
  373.     //~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

  374.     return this->OutputData(pszFileName, indexStart, indexEnd);
  375. }

  376. // ============================================================================
  377. //    判断给定IP字符串是否是合法的ip地址
  378. // ==============================================================================
  379. bool CIpFinder::IsRightIpString(const char* pszIp) const
  380. {
  381.     if (!pszIp) {
  382.         return false;
  383.     }

  384.     int nLen = strlen(pszIp);
  385.     if (nLen < 7) {

  386.         // 至少包含7个字符"0.0.0.0"
  387.         return false;
  388.     }

  389.     for (int i = 0; i < nLen; ++i) {
  390.         if (!isdigit(pszIp[i]) && pszIp[i] != '.') {
  391.             return false;
  392.         }

  393.         if (pszIp[i] == '.') {
  394.             if (0 == i) {
  395.                 if (!isdigit(pszIp[i + 1])) {
  396.                     return false;
  397.                 }
  398.             } else if (nLen - 1 == i) {
  399.                 if (!isdigit(pszIp[i - 1])) {
  400.                     return false;
  401.                 }
  402.             } else {
  403.                 if (!isdigit(pszIp[i - 1]) || !isdigit(pszIp[i + 1])) {
  404.                     return false;
  405.                 }
  406.             }
  407.         }
  408.     }
  409.     return true;
  410. }
复制代码


扫码关注微信公众号,及时获取最新资源信息!下载附件优惠VIP会员6折;永久VIP4折
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

免责声明:
1、本站提供的所有资源仅供参考学习使用,版权归原著所有,禁止下载本站资源参与商业和非法行为,请在24小时之内自行删除!
2、本站所有内容均由互联网收集整理、网友上传,并且以计算机技术研究交流为目的,仅供大家参考、学习,请勿任何商业目的与商业用途。
3、若您需要商业运营或用于其他商业活动,请您购买正版授权并合法使用。
4、论坛的所有内容都不保证其准确性,完整性,有效性,由于源码具有复制性,一经售出,概不退换。阅读本站内容因误导等因素而造成的损失本站不承担连带责任。
5、用户使用本网站必须遵守适用的法律法规,对于用户违法使用本站非法运营而引起的一切责任,由用户自行承担
6、本站所有资源来自互联网转载,版权归原著所有,用户访问和使用本站的条件是必须接受本站“免责声明”,如果不遵守,请勿访问或使用本网站
7、本站使用者因为违反本声明的规定而触犯中华人民共和国法律的,一切后果自己负责,本站不承担任何责任。
8、凡以任何方式登陆本网站或直接、间接使用本网站资料者,视为自愿接受本网站声明的约束。
9、本站以《2013 中华人民共和国计算机软件保护条例》第二章 “软件著作权” 第十七条为原则:为了学习和研究软件内含的设计思想和原理,通过安装、显示、传输或者存储软件等方式使用软件的,可以不经软件著作权人许可,不向其支付报酬。若有学员需要商用本站资源,请务必联系版权方购买正版授权!
10、本网站如无意中侵犯了某个企业或个人的知识产权,请来信【站长信箱312337667@qq.com】告之,本站将立即删除。
郑重声明:
本站所有资源仅供用户本地电脑学习源代码的内含设计思想和原理,禁止任何其他用途!
本站所有资源、教程来自互联网转载,仅供学习交流,不得商业运营资源,不确保资源完整性,图片和资源仅供参考,不提供任何技术服务。
本站资源仅供本地编辑研究学习参考,禁止未经资源商正版授权参与任何商业行为,违法行为!如需商业请购买各资源商正版授权
本站仅收集资源,提供用户自学研究使用,本站不存在私自接受协助用户架设游戏或资源,非法运营资源行为。
 
在线客服
点击这里给我发消息 点击这里给我发消息 点击这里给我发消息
售前咨询热线
312337667

微信扫一扫,私享最新原创实用干货

QQ|免责声明|小黑屋|依星资源网 ( 鲁ICP备2021043233号-3 )|网站地图

GMT+8, 2025-1-18 15:49

Powered by Net188.com X3.4

邮箱:312337667@qq.com 客服QQ:312337667(工作时间:9:00~21:00)

快速回复 返回顶部 返回列表