1.简单介绍
asp.net core web api 是 asp.net core mvc 的一个功能。asp.net core mvc 包含了对 web api 的支持。可以构建多种客户端的 http 服务。asp.net core web api可用于在 .net core 上构建 restful 应用程序。
框架包含对 http 内容协商的支持,内置支持以 json 或 xml 格式化的数据。编写自定义格式化程序已添加对自有格式的支持。
使用链接生成对超媒体的支持。启用对跨资源共享(cors)的支持,以便 web api 可以在多个 web应用程序之间共享。
例如,新建一个 api 项目,默认包含一个 valuescontroller:
[route("api/[controller]")]
    [apicontroller]
    public class valuescontroller : controllerbase
    {
        // get api/values
        [httpget]
        public actionresult<ienumerable<string>> get()
        {
            return new string[] { "value1", "value2" };
        }
        // get api/values/5
        [httpget("{id}")]
        public actionresult<string> get(int id)
        {
            return "value";
        }
        // post api/values
        [httppost]
        public void post([frombody] string value)
        {
        }
        // put api/values/5
        [httpput("{id}")]
        public void put(int id, [frombody] string value)
        {
        }
        // delete api/values/5
        [httpdelete("{id}")]
        public void delete(int id)
        {
        }
    }默认包含了 get,post,put,delete 这几个 http 请求操作。这里直接使用特性路由,在控制器上注明 [route("api/[controller]")]。调试并打开浏览器输入 https://localhost:5000/api/values ,会返回数据。
2.自定义格式化(format)
asp.net core mvc内建支持对相应数据的格式,用来修正格式或生成客户端指定的格式。
1.特定格式的操作结果
某些操作结果的类型是特定的格式,比如 jsonresult 或 contentresult 。操作可以总是返回格式为特定格式的具体结果。比如返回 jsonresult 时将返回 json 格式化数据,而不管客户端要求的是什么格式。
操作并不要求返回任何特定的类型, mvc 支持任何对象作为返回值。如果操作返回的是 iactionresult 的某个实现,并且控制器继承自 controller ,那么可以使用更多辅助方法。如果不是,则将使用合适的 ioutputformatter 实现序列化对象。
若要从继承 controller 基类的控制器返回特定格式的数据,可以使用内置的辅助方法 json 来返回 json 格式, 使用 content 来返回 纯文本。操作方法的返回类型必须是指定的结果类型(如 jsonresult)或 iactionresult。
[httpget]
        public jsonresult get()
        {
            return json(new user());
        }以上代码 content-type 将返回 application/json。
要想返回纯文本格式的数据,则使用 contentresult 以及 content 辅助方法:
[httpget]
        public contentresult get()
        {
            return content("result");
        }以上代码 content-type 将返回 test/plan 。 也可以使用一个字符串相应类型来实现这个行为:
[httpget]
        public string get()
        {
            return "result";
        }对于具有多个返回类型或选项的复杂操作,请选择 iactionresult 作为返回类型。
2.配置格式化程序
如果应用程序想支持默认 json 之外的格式,可以在 project.json 文件中添加这些额外的依赖项,并配置 mvc 来支持。输入和输出的格式是可以隔离的。输入格式通过使用模型绑定,输出格式通过格式化响应。
3.添加对 xml 格式的支持
要添加对 xml 格式的支持,需要先安装 microsoft.aspnetcore.mvc.formatters.xml 包,然后在 configureservices 中配置 xmlserializerformatters :
services.addmvc()
                .addxmlserializerformatters()
                .setcompatibilityversion(compatibilityversion.version_2_1);或者可以只添加输出格式:
services.addmvc(options => {
                options.outputformatters.add(new xmlserializeroutputformatter());
            })
                //.addxmlserializerformatters()
                .setcompatibilityversion(compatibilityversion.version_2_1);这两种方法都将使用 system.xml.serialization.xmlserializer 序列化结果。还可以通过添加其他相关格式来使用 system.runtime.serialization.datacontractserializer :
services.addmvc(options => {
                options.outputformatters.add(new xmldatacontractserializeroutputformatter());
            })
                //.addxmlserializerformatters()
                .setcompatibilityversion(compatibilityversion.version_2_1);4.强制特定格式化
若是想为某个操作限制响应格式,则可以使用 [produces] 过滤器。[produces] 过滤器可以为这个控制器或 action 指定响应格式:
[httpget("{id}", name = "get")]
        [produces("application/json")]
        public string get(int id)
        {
            return "value";
        }对于一些特殊情况,可能不想使用内建的格式化实现。默认情况下,返回类型为 string 时将格式化为 text/plain 。这种行为可以通过移除 textoutputformatter 来改变:
public void configureservices(iservicecollection services)
        {
            services.addmvc(options => {
                options.outputformatters.removetype<textoutputformatter>();
                options.outputformatters.removetype<httpnocontentoutputformatter>();
            }).setcompatibilityversion(compatibilityversion.version_2_1);
        }上面移除了 textoutputformatter 和 httpnocontentoutputformatter,当返回类型为 string 时,将返回 406 not acceptable。如果存在 xml 格式化程序,则将格式化响应结果。
移除了 httpnocontentoutputformatter ,当返回某个返回类型为模型对象的操作返回 null 时,将返回 204 no content 响应。json 格式将简单地返回主体信息为 null 的响应,而 xml 格式将返回一个空的带有 xsi:nil="true" 属性的 xml 元素。
5.响应格式 url 映射
客户端可以在 url 中请求特定的格式,比如在请求字符串或路径中,或者通过使用特定格式的文件扩展名(比如 .xml 或 .json),需要在 api 所使用的路由中指定:
[formatfilter]
    public class usercontroller : controller
    {
        // get: api/user
        [httpget]
        [route("[controller]/[action]/{id}.{format?}")]
        public iactionresult getbyid(int id)
        {
            return content("xxx");
        }
}这个路由配置将允许使用可选的文件扩展名来指定格式。[formatfilter] 特性将在 routedata 中检查该格式的值是否存在,并创建响应时将响应数据映射到相应的格式:
routeformatter
/user/getbyid/5 :默认输出格式
/user/getbyid/5.json :json格式(如果配置过)
/user/getbyid/5.xml; :xml格式(如果配置过)
6.自定义格式化程序 protocol buffers (简称 protobuf)
protocol buffers 是一种轻便高效的结构化数据存储格式,可用于结构化数据串行化,或者说序列化。它很适合做数据存储或 rpc(远程过程调用协议)数据交换格式。可用于通讯协议,数据存储等领域的语言无关,平台无关,可扩展的序列化结构数据格式。比如实现一个程序返回 protobuf 格式:
创建 api 项目,添加 protobuf-net 引用。
添加 protobufformatter 类:
public class protobufformatter:outputformatter
    {
        public string contenttype { get; private set; }
        public protobufformatter()
        {
            contenttype = "application/proto";
            supportedmediatypes.add(mediatypeheadervalue.parse("application/proto"));
        }
        public override task writeresponsebodyasync(outputformatterwritecontext context)
        {
            if (context == null)
            {
                throw new argumentnullexception(nameof(context));
            }
            var response = context.httpcontext.response;
            serializer.serialize(response.body,context.object);
            return task.fromresult(0);
        }
    }继承 outputformatter ,然后实现 writeresponsebodyasync 方法,初始化时赋值 contenttype ,并添加支持 mediatype。在 writeresponsebodyasync 方法中获取 response ,调用 protobuf-net 的 serializer.serialize 方法将 object 序列化至输出内容。 protobuf 在序列化时必须指定顺序,添加 user 类,实现 protobuf 实体:
[protocontract]
    public class user
    {
        [protomember(1)]
        public int id { get; set; }
        [protomember(2)]
        public string name { get; set; }
        [protomember(3)]
        public int age { get; set; }
    }类上需要添加 [protocontract] 特性,字段上需要 protomember 特性,并指定顺序。然后修改 usercontroller:
[route("api/[controller]")]
    [apicontroller]
    [formatfilter]
    public class usercontroller : controller
    {
        private ienumerable<user> users;
        public usercontroller()
        {
            users = new user[] {
                new user(){ id=1,name="bob",age = 20},
                new user(){ id=2,name="tom",age = 22}
            };
        }
        // get: api/user
        [httpget]
        [produces("application/proto")]
        public ienumerable<user> get()
        {
            return users;
        }
}修改 configureservices :
public void configureservices(iservicecollection services)
        {
            services.addmvc(options => {
                options.outputformatters.add(new protobufformatter());
            }).setcompatibilityversion(compatibilityversion.version_2_1);
        }运行程序,会返回一个二进制文件。
创建一个控制台,检查序列化:
class program
    {
        static void main(string[] args)
        {
            httpclient client = new httpclient();
            var stream = client.getstreamasync("https://localhost:44358/api/user").result;
            var users = serializer.deserialize<list<user>>(stream);
            foreach (var user in users)
            {
                console.writeline($"id:{user.id} - name:{user.name} - age:{user.age}");
            }
            console.readkey();
        }
    }到此这篇关于asp.net core web api的文章就介绍到这了。希望对大家的学习有所帮助,也希望大家多多支持代码网。
 
             我要评论
我要评论 
                                             
                                             
                                             
                                             
                                            
发表评论