一、概念介绍

1.1,什么是OData?

还是看OData官网的简单说明:

An open protocol to allow the creation and consumption of queryable and interoperable RESTful APIs in a simple and standard way.

这是一个开放的数据查询和服务协议,目前已经有众多厂商和平台支持,已经形成了完整的生态链,这应该是未来数据查询的标准,参见官网说明。

OData的意义还在于,它能够大大简小SOA架构里面服务的粒度,只需要提供一个OData数据源,而查询工作交给客户端去做即可,这将大大减少服务端服务方法定义的数量。

OData的版本现在已经是V4了,之前的很多类库都是基于V1-V3版本的。现在的V4版本已经很完善了,而且成为了工业标准,所以现在可以放心的在项目中使用了。

1.2,OData on .NET

OData 的前身是WCF Data Service,后来演变成跨平台的数据查询协议,现在,除了WCF支持OData,ASP.NET WebAPI 已经内置支持OData了,这将获得一种轻量级的,Rest架构的OData访问方案,本文将讲解如何在VS 2013上搭建一个OData 服务和客户端程序。

1.3,参考资源

在阅读本文之前,首先建议你参考下面的OData 学习资源,本文也是根据这里的资源整理而成,之所以要重新整理一次,是因为原文说的并不清楚,给出的示例程序有些小小的问题,而且国内有关OData的文章介绍非常少。

http://www.odata.org/blog/how-to-use-web-api-odata-to-build-an-odata-v4-service-without-entity-framework/
http://www.asp.net/web-api/overview/odata-support-in-aspnet-web-api/odata-v4/create-an-odata-v4-endpoint
http://www.asp.net/web-api/overview/odata-support-in-aspnet-web-api/odata-v4/create-an-odata-v4-client-app

本文的实例程序相关代码请在此下载。

二、OData WebAPI搭建

2.1,创建项目

新建一个ASP.NET WebAPI 项目,名字是 ODataWebApplication ,如下图:

注意选择一个空项目,并购选WebAPI,单击确定。

2.2,添加Asp.Net OData 支持

打开VS2013的程序包管理器控制台,在下面输入 Install-Package Microsoft.AspNet.OData 命令,如下图:

这里会添加很多附属的程序集文件,下面是一个详细的清单列表,如果你的符合下面的内容,就表示安装成功了:

键入“get-help NuGet”以查看所有可用的 NuGet 命令。PM> Install-Package Microsoft.AspNet.OData
正在尝试解析依赖项“Microsoft.AspNet.WebApi.Client (≥ 5.2.2)”。
正在尝试解析依赖项“Newtonsoft.Json (≥ 6.0.4)”。
正在尝试解析依赖项“Microsoft.AspNet.WebApi.Core (≥ 5.2.2 && < 5.3.0)”。
正在尝试解析依赖项“Microsoft.OData.Core (≥ 6.10.0 && < 7.0.0)”。
正在尝试解析依赖项“Microsoft.Spatial (= 6.10.0)”。
正在尝试解析依赖项“Microsoft.OData.Edm (= 6.10.0)”。
正在安装“Newtonsoft.Json 6.0.8”。
已成功安装“Newtonsoft.Json 6.0.8”。
正在安装“Microsoft.AspNet.WebApi.Client 5.2.3”。
您正在从 Microsoft 下载 Microsoft.AspNet.WebApi.Client,有关此程序包的许可协议在 http://www.microsoft.com/web/webpi/eula/net_library_eula_ENU.htm 上提供。请检查此程序包是否有其他依赖项,这些依赖项可能带有各自的许可协议。您若使用程序包及依赖项,即构成您接受其许可协议。如果您不接受这些许可协议,请从您的设备中删除相关组件。
已成功安装“Microsoft.AspNet.WebApi.Client 5.2.3”。
正在安装“Microsoft.AspNet.WebApi.Core 5.2.3”。
您正在从 Microsoft 下载 Microsoft.AspNet.WebApi.Core,有关此程序包的许可协议在 http://www.microsoft.com/web/webpi/eula/net_library_eula_ENU.htm 上提供。请检查此程序包是否有其他依赖项,这些依赖项可能带有各自的许可协议。您若使用程序包及依赖项,即构成您接受其许可协议。如果您不接受这些许可协议,请从您的设备中删除相关组件。
已成功安装“Microsoft.AspNet.WebApi.Core 5.2.3”。
正在安装“Microsoft.Spatial 6.10.0”。
您正在从 Microsoft Corporation 下载 Microsoft.Spatial,有关此程序包的许可协议在 http://go.microsoft.com/?linkid=9809688 上提供。请检查此程序包是否有其他依赖项,这些依赖项可能带有各自的许可协议。您若使用程序包及依赖项,即构成您接受其许可协议。如果您不接受这些许可协议,请从您的设备中删除相关组件。
已成功安装“Microsoft.Spatial 6.10.0”。
正在安装“Microsoft.OData.Edm 6.10.0”。
您正在从 Microsoft Corporation 下载 Microsoft.OData.Edm,有关此程序包的许可协议在 http://go.microsoft.com/?linkid=9809688 上提供。请检查此程序包是否有其他依赖项,这些依赖项可能带有各自的许可协议。您若使用程序包及依赖项,即构成您接受其许可协议。如果您不接受这些许可协议,请从您的设备中删除相关组件。
已成功安装“Microsoft.OData.Edm 6.10.0”。
正在安装“Microsoft.OData.Core 6.10.0”。
您正在从 Microsoft Corporation 下载 Microsoft.OData.Core,有关此程序包的许可协议在 http://go.microsoft.com/?linkid=9809688 上提供。请检查此程序包是否有其他依赖项,这些依赖项可能带有各自的许可协议。您若使用程序包及依赖项,即构成您接受其许可协议。如果您不接受这些许可协议,请从您的设备中删除相关组件。
已成功安装“Microsoft.OData.Core 6.10.0”。
正在安装“Microsoft.AspNet.OData 5.5.0”。
您正在从 Microsoft 下载 Microsoft.AspNet.OData,有关此程序包的许可协议在 http://www.microsoft.com/web/webpi/eula/net_library_eula_ENU.htm 上提供。请检查此程序包是否有其他依赖项,这些依赖项可能带有各自的许可协议。您若使用程序包及依赖项,即构成您接受其许可协议。如果您不接受这些许可协议,请从您的设备中删除相关组件。
已成功安装“Microsoft.AspNet.OData 5.5.0”。
正在从 ODataWebApplication 删除“Microsoft.AspNet.WebApi.Client.zh-Hans 5.0.0”。
已成功将“Microsoft.AspNet.WebApi.Client.zh-Hans 5.0.0”从 ODataWebApplication 中删除。
正在从 ODataWebApplication 删除“Microsoft.AspNet.WebApi.Core.zh-Hans 5.0.0”。
已成功将“Microsoft.AspNet.WebApi.Core.zh-Hans 5.0.0”从 ODataWebApplication 中删除。
正在从 ODataWebApplication 删除“Microsoft.AspNet.WebApi.Client 5.0.0”。
已成功将“Microsoft.AspNet.WebApi.Client 5.0.0”从 ODataWebApplication 中删除。
正在从 ODataWebApplication 删除“Newtonsoft.Json 5.0.6”。
已成功将“Newtonsoft.Json 5.0.6”从 ODataWebApplication 中删除。
正在将“Newtonsoft.Json 6.0.8”添加到 ODataWebApplication。
已成功将“Newtonsoft.Json 6.0.8”添加到 ODataWebApplication。
正在将“Microsoft.AspNet.WebApi.Client 5.2.3”添加到 ODataWebApplication。
已成功将“Microsoft.AspNet.WebApi.Client 5.2.3”添加到 ODataWebApplication。
正在从 ODataWebApplication 删除“Microsoft.AspNet.WebApi.Core 5.0.0”。
已成功将“Microsoft.AspNet.WebApi.Core 5.0.0”从 ODataWebApplication 中删除。
正在将“Microsoft.AspNet.WebApi.Core 5.2.3”添加到 ODataWebApplication。
已成功将“Microsoft.AspNet.WebApi.Core 5.2.3”添加到 ODataWebApplication。
正在将“Microsoft.Spatial 6.10.0”添加到 ODataWebApplication。
已成功将“Microsoft.Spatial 6.10.0”添加到 ODataWebApplication。
正在将“Microsoft.OData.Edm 6.10.0”添加到 ODataWebApplication。
已成功将“Microsoft.OData.Edm 6.10.0”添加到 ODataWebApplication。
正在将“Microsoft.OData.Core 6.10.0”添加到 ODataWebApplication。
已成功将“Microsoft.OData.Core 6.10.0”添加到 ODataWebApplication。
正在将“Microsoft.AspNet.OData 5.5.0”添加到 ODataWebApplication。
已成功将“Microsoft.AspNet.OData 5.5.0”添加到 ODataWebApplication。
正在将“Microsoft.AspNet.WebApi.Client.zh-Hans 5.2.3”添加到 ODataWebApplication。
正在安装“Microsoft.AspNet.WebApi.Client.zh-Hans 5.2.3”。
您正在从 Microsoft 下载 Microsoft.AspNet.WebApi.Client.zh-Hans,有关此程序包的许可协议在 http://www.microsoft.com/web/webpi/eula/net_library_eula_CHS.htm 上提供。请检查此程序包是否有其他依赖项,这些依赖项可能带有各自的许可协议。您若使用程序包及依赖项,即构成您接受其许可协议。如果您不接受这些许可协议,请从您的设备中删除相关组件。
已成功安装“Microsoft.AspNet.WebApi.Client.zh-Hans 5.2.3”。
已成功将“Microsoft.AspNet.WebApi.Client.zh-Hans 5.2.3”添加到 ODataWebApplication。
正在将“Microsoft.AspNet.WebApi.Core.zh-Hans 5.2.3”添加到 ODataWebApplication。
正在安装“Microsoft.AspNet.WebApi.Core.zh-Hans 5.2.3”。
您正在从 Microsoft 下载 Microsoft.AspNet.WebApi.Core.zh-Hans,有关此程序包的许可协议在 http://www.microsoft.com/web/webpi/eula/net_library_eula_CHS.htm 上提供。请检查此程序包是否有其他依赖项,这些依赖项可能带有各自的许可协议。您若使用程序包及依赖项,即构成您接受其许可协议。如果您不接受这些许可协议,请从您的设备中删除相关组件。
已成功安装“Microsoft.AspNet.WebApi.Core.zh-Hans 5.2.3”。
已成功将“Microsoft.AspNet.WebApi.Core.zh-Hans 5.2.3”添加到 ODataWebApplication。
正在卸载“Microsoft.AspNet.WebApi.Client.zh-Hans 5.0.0”。
已成功卸载“Microsoft.AspNet.WebApi.Client.zh-Hans 5.0.0”。
正在卸载“Microsoft.AspNet.WebApi.Core.zh-Hans 5.0.0”。
已成功卸载“Microsoft.AspNet.WebApi.Core.zh-Hans 5.0.0”。
正在卸载“Microsoft.AspNet.WebApi.Client 5.0.0”。
已成功卸载“Microsoft.AspNet.WebApi.Client 5.0.0”。
正在卸载“Newtonsoft.Json 5.0.6”。
已成功卸载“Newtonsoft.Json 5.0.6”。
正在卸载“Microsoft.AspNet.WebApi.Core 5.0.0”。
已成功卸载“Microsoft.AspNet.WebApi.Core 5.0.0”。PM>

2.3,添加Model和控制器

按照  http://www.odata.org/blog/how-to-use-web-api-odata-to-build-an-odata-v4-service-without-entity-framework/ 这个链接内容的文章,添加Model和控制器,具体过程请参考原文。在本篇文章的实例中,为了更好的重用Model,我将它放到了一个独立的Demo.Models 项目中。

注意,添加控制器的时候选择空的 WebAPI 控制器,不要选择带OData 的。

下面是添加完整后的项目目录结构:

2.4,解决程序集冲突

立刻运行这个项目,发现报下面的错误:

未能加载文件或程序集“System.Web.Http, Version=5.0.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35”或它的某一个依赖项。找到的程序集清单定义与程序集引用不匹配。 (异常来自 HRESULT:0x80131040)

此时需要在Web.config 文件中加入下面的配置内容:

<system.web> <compilation debug="true" targetFramework="4.5"/> <httpRuntime targetFramework="4.5"/> </system.web> <runtime> <assemblyBinding xmlns="urn:schemas-microsoft-com:asm.v1"> <dependentAssembly> <assemblyIdentity name="System.Web.Http" publicKeyToken="31BF3856AD364E35" culture="neutral"/> <bindingRedirect oldVersion="0.0.0.0-5.2.3.0" newVersion="5.2.3.0"/> </dependentAssembly> <dependentAssembly> <assemblyIdentity name="System.Net.Http.Formatting" publicKeyToken="31BF3856AD364E35" culture="neutral"/> <bindingRedirect oldVersion="0.0.0.0-5.2.3.0" newVersion="5.2.3.0"/> </dependentAssembly> </assemblyBinding> </runtime>

也可以在编译项目的时候,注意查看“输出窗口”,单击“警告”的文字内容,会有下面的提示询问,回答确定即可自动为你添加上面的内容:

2.5,正确的OData 服务程序

再次运行,程序不报错了,用谷歌浏览器来打开本程序,出现了下面的内容,就表示ASP.NET WebAPI OData V4 已经成功了:

{"@odata.context":"http://localhost:20491/$metadata#People","value":[{"ID":"001","Name":"Angel","Description":null},{"ID":"002","Name":"Clyde","Description":"Contrary to popular belief, Lorem Ipsum is not simply random text."},{"ID":"003","Name":"Elaine","Description":"It has roots in a piece of classical Latin literature from 45 BC, making Lorems over 2000 years old."}]
}

三、添加OData客户端

3.1,添加OData 控制台程序

按照下面链接文章的内容,新建一个控制台程序:

http://www.asp.net/web-api/overview/odata-support-in-aspnet-web-api/odata-v4/create-an-odata-v4-client-app

3.2,为VS添加OData客户端工具

按照上面链接文章的内容,添加此工具,如下图:

下载好该工具后让你安装运行,最后会让你重启动VS。

3.3,添加OData客户端T4程序

重新启动后,如果看到下面的内容,表示该工具安装成功了。

该工具运行后,会在项目下面添加一个OData Client T4 文件,添加后,运行该项目,会报下面的错误:

错误    1    正在运行转换: System.ArgumentException: The value "" is not a valid MetadataDocumentUri because is it not a valid absolute Uri. The MetadataDocumentUri must be set to an absolute Uri referencing the $metadata endpoint of an OData service. 在 Microsoft.VisualStudio.TextTemplating0CFD4DFE54AE767D51F4D19A4964CA0B33E6E87CF53FFF790646D6E3F14916C4544A62E4C781ABB55BC19EE5225FDBA7E28D9767E56040F8F9A9EF5278D6B7CA.GeneratedTextTransformation.set_MetadataDocumentUri(String value) 位置 c:\Users\dth\Documents\Visual Studio 2013\Projects\ODataWebApplication\Demo.ConsoleClient\ODataClient1.ttinclude:行号 125 在 Microsoft.VisualStudio.TextTemplating0CFD4DFE54AE767D51F4D19A4964CA0B33E6E87CF53FFF790646D6E3F14916C4544A62E4C781ABB55BC19EE5225FDBA7E28D9767E56040F8F9A9EF5278D6B7CA.GeneratedTextTransformation.ApplyParametersFromConfigurationClass() 位置 c:\Users\dth\Documents\Visual Studio 2013\Projects\ODataWebApplication\Demo.ConsoleClient\ODataClient1.ttinclude:行号 313 在 Microsoft.VisualStudio.TextTemplating0CFD4DFE54AE767D51F4D19A4964CA0B33E6E87CF53FFF790646D6E3F14916C4544A62E4C781ABB55BC19EE5225FDBA7E28D9767E56040F8F9A9EF5278D6B7CA.GeneratedTextTransformation.TransformText() 位置 c:\Users\dth\Documents\Visual Studio 2013\Projects\ODataWebApplication\Demo.ConsoleClient\ODataClient1.ttinclude:行号 58    c:\Users\dth\Documents\Visual Studio 2013\Projects\ODataWebApplication\Demo.ConsoleClient\ODataClient1.ttinclude    125    1    Demo.ConsoleClient

3.4,配置OData Client T4 信息

根据错误信息,找到T4文件错误的位置,将前面的OData WebAPI项目的地址,写在文件里面,如下所示:

// The URI of the metadata document. The value must be set to a valid service document URI or a local file path // eg : "http://services.odata.org/V4/OData/OData.svc/", "File:///C:/Odata.edmx", or @"C:\Odata.edmx" // ### Notice ### If the OData service requires authentication for accessing the metadata document, the value of // MetadataDocumentUri has to be set to a local file path, or the client code generation process will fail. public const string MetadataDocumentUri = "http://localhost:20491/";

3.5,生成OData Client 代理类

此时再次运行该T4文件,我们发现OData Client 代理类文件生成了,内容很多,这里就不贴了。

3.6,使用OData Client 代理类

添加下面的代码,调用OData Client 代理类并运行:

class Program { static void Main(string[] args) { // TODO: Replace with your local URI. string serviceUri = "http://localhost:20491/"; var container = new DefaultContainer(new Uri(serviceUri));foreach (var p in container.People) { Console.WriteLine("{0} {1} {2}", p.ID, p.Name, p.Description); }Console.Read(); } }

如果看到下面的运行结果,表示OData Client 程序成功了:

至此,OData WebAPI Serivce & Client 的工作就全部完成了。

四、不使用OData客户端工具访问OData 服务

4.1,封装OData Client 类库

经过前面的过程我们看到,依托于OData 客户端工具生成OData 代理类的过程还是比较麻烦的,当然好处也有,但缺点就是没有手工操控的那么灵活自由。

仔细研究下前面的代理类,我们发现这里关键依赖于  Microsoft.OData.Client 程序集的DataServiceContext 对象,将代理类进行抽取封装就可以完成我们手工的代理类了。

创建一个类库项目,新建一个 ODataV4ContextBase.cs 文件,

接着为该项目添加Nuget 依赖的包:

Install-Package Microsoft.OData.Client

然后在项目下增加了一个文件 packages.config,里面有如下内容:

<?xml version="1.0" encoding="utf-8"?>
<packages><package id="Microsoft.OData.Client" version="6.11.0" targetFramework="net40" /><package id="Microsoft.OData.Core" version="6.11.0" targetFramework="net40" /><package id="Microsoft.OData.Edm" version="6.11.0" targetFramework="net40" /><package id="Microsoft.Spatial" version="6.11.0" targetFramework="net40" />
</packages>

然后,编写ODataV4ContextBase 类的具体内容:

using Microsoft.OData.Client;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;namespace PWMIS.OData.Client
{/// <summary>/// OData V4 Version ASP.NET WebAPI OData RestFull Client Context/// <remarks>v1.0 2015.4.1 http://www.pwmis.com/sqlmap </remarks>/// </summary>public class ODataV4ContextBase : DataServiceContext{/// <summary>/// V4 OData Init/// </summary>/// <param name="serviceRoot">V4 OData ASP.NET WebAPI url base</param>public ODataV4ContextBase(string  serviceRoot): base(new System.Uri( serviceRoot), ODataProtocolVersion.V4){if (!serviceRoot.EndsWith("/"))serviceRoot = serviceRoot + "/";GeneratedEdmModel gem = new GeneratedEdmModel(serviceRoot);this.Format.LoadServiceModel = gem.GetEdmModel;this.Format.UseJson();}public IQueryable<T> CreateNewQuery<T>(string name) where T : class{return base.CreateQuery<T>(name);}class GeneratedEdmModel{private string ServiceRootUrl;public GeneratedEdmModel(string serviceRootUrl){this.ServiceRootUrl = serviceRootUrl;}public Microsoft.OData.Edm.IEdmModel GetEdmModel(){string metadataUrl = ServiceRootUrl + "$metadata";return LoadModelFromUrl(metadataUrl);}private Microsoft.OData.Edm.IEdmModel LoadModelFromUrl(string metadataUrl){System.Xml.XmlReader reader = CreateXmlReaderFromUrl(metadataUrl);try{return Microsoft.OData.Edm.Csdl.EdmxReader.Parse(reader);}finally{((System.IDisposable)(reader)).Dispose();}}private static System.Xml.XmlReader CreateXmlReaderFromUrl(string inputUri){return System.Xml.XmlReader.Create(inputUri);}}}
}

4.2 编写ODataClient客户端

在解决方案里面添加一个WinForm项目,在项目里面添加一个ODataContainer.cs 文件,内容如下:

class ODataContainer : PWMIS.OData.Client.ODataV4ContextBase{public ODataContainer(string serviceRoot):base(serviceRoot){}public IQueryable<Person> People{get{return base.CreateNewQuery<Person>("People");} }}

非常的简单,这里只是添加了一个属性 People。

然后,在窗体代码中调用:

private void button1_Click(object sender, EventArgs e){string serviceUri = "http://localhost:20491/";var container = new ODataContainer(serviceUri);var query = container.People.Where(p => p.Description!=null);this.dataGridView1.DataSource = query.ToList();}

最后运行该程序,出现下面的界面,就表示成功了:

至此,一个不依赖于EF的全内存的OData 应用程序就完全做好了,更多OData的研究,请大家一起来做吧。

感谢支持 PDF.NET SOD框架,相关代码请在此下载。

注:本文说的 ODataV4ContextBase OData客户端基类程序已经集成到了PDF.NET框架中,在上面的下载里面即可看到。

《SOD框架“企业级”应用数据架构实战》

【转】OData – the best way to REST–实例讲解ASP.NET WebAPI OData (V4) Service Client相关推荐

  1. AspNet.WebAPI.OData.ODataPQ实现WebAPI的分页查询服务-(个人拙笔)

    AspNet.WebAPI.OData.ODataPQ 这是针对 Asp.net WebAPI OData 协议下,查询分页.或者是说 本人在使用Asp.Net webAPI 做服务接口时写的一个分页 ...

  2. java实现页面高效刷新_selenium高效应对Web页面元素刷新的实例讲解

    当我们在页面上进行selenium.type()或者selenium.click()操作的时候,往往需要需要等待一个元素的出现,对于一般的网页,当我们进入一个新页面的时候,往往会使用selenium. ...

  3. php 返回一个json对象,PHP给前端返回一个JSON对象的实例讲解

    解决问题:用php做后台时,如何给前端发起的AJAX请求返回一个JSON格式的"对象": 说明:我本身是一个前端,工作久了之后发现要是不掌握一门后端开发语言的话,总感觉有点无力.最 ...

  4. python简易版实例_Python3之简单搭建自带服务器的实例讲解

    WEB开发,我们先从搭建一个简单的服务器开始,Python自带服务模块,且python3相比于python2有很大不同, 在Python2.6版本里,/usr/bin/lib/python2.6/ 目 ...

  5. 手摸手教你数据可视化!(附实例讲解)

    ↑↑↑关注后"星标"Datawhale 每日干货 & 每月组队学习,不错过 Datawhale干货 作者:CrescentAI,华南理工大学,Datawhale优秀学习者 ...

  6. 【Python基础】手把手教你数据可视化!(附实例讲解)

    点击上方"小白学视觉",选择加"星标"或"置顶" 重磅干货,第一时间送达 作者:CrescentAI,华南理工大学,Datawhale优秀学 ...

  7. php脚本函数,PHP执行系统命令函数实例讲解

    命令注入 命令注入(Command Injection),对一些函数的参数没有做过滤或过滤不严导致的,可以执行系统或者应用指令(CMD命令或者 bash 命令)的一种注入攻击手段. 常见的执行系统命令 ...

  8. 实例讲解之校园网病毒该如何铲除

    实例讲解之校园网病毒该如何铲除 转载于:https://blog.51cto.com/wilsondy/163994

  9. httpclient 调取接口_使用HttpClient调用接口的实例讲解

    一,编写返回对象 public class HttpResult { // 响应的状态码 private int code; // 响应的响应体 private String body; get/se ...

最新文章

  1. github 视觉测量_教你如何提高双目立体视觉系统的精度
  2. 精读《手写 SQL 编译器 - 文法介绍》
  3. 中石油训练赛 - Flow Finder(树上模拟)
  4. 阿里云 centos mysql_在阿里云的CentOS环境中安装配置MySQL的教程
  5. 小程序 后台返回的对象数组(每个数组添加一个新的属性值)
  6. php 打包网站在线压缩为zip
  7. 阶段3 1.Mybatis_02.Mybatis入门案例_2.mybatis入门案例中的设计模式分析
  8. truecrypt linux用法,TrueCrypt介绍及入门使用讲解【翻译】
  9. FreeRTOS 正点原子教程学习笔记
  10. 木疙瘩动画效果视频学习
  11. STM32F407VET6+cubemx+FSMC+ST7789
  12. 软件架构设计说明书该怎么写?
  13. 计算机组装diy,电脑diy,详细教您如何组装电脑
  14. php的优秀案例,单页Web设计优秀案例_php
  15. APS与ERP及MES的关系和接口
  16. 每周跑一下Python脚本,轻松生成工作日志模板
  17. pix2pixHD:High-Resolution Image Synthesis and Semantic Manipulation with Conditional GANs
  18. 卫士处刑者冠军css3边,流放之路职业开荒路 决斗者冠军双持刀刃乱舞
  19. 数据库安全防护之防止被黑客攻击的策略
  20. 正面硬刚Beats!这款耳机从美国红回中国,细腻音质千元内无敌手!

热门文章

  1. Leetcode 950. Reveal Cards In Increasing Order
  2. php解决与处理网站高并发 大流量访问的方法
  3. 安卓学习日记:初识Android Studio · java环境配置和AS安装
  4. 2016年4月 之 《C程序设计语言》
  5. SharePoint 2010 RBS 安装和配置遇到的一个问题
  6. 【转贴】利用 Javascript 获取 URL 参数(适合IE、FF)
  7. fir fpga 不同截止频率_学习FPGA将来的出路在哪里?
  8. 微星刀锋 无法进入bios_夏天来了 微星主板风扇转速调节攻略
  9. linux命令行提示符居中,linux命令行学习(54):修改提示符
  10. yum如何安装特定版本的gcc_linux下如何升级python