為您解碼網(wǎng)站建設(shè)的點點滴滴
發(fā)表日期:2019-09 文章編輯:小燈 瀏覽次數(shù):1993
HTTP為條件更新提供了相應(yīng)的報頭,我們按照分析條件獲取的方式來分析條件更新在HTTP請求/回復(fù)過程中的實現(xiàn)??蛻舳说谝淮蜗蚍?wù)端發(fā)起針對某個資源的請求,服務(wù)端除了將資源數(shù)據(jù)作為回復(fù)消息主體返回之外,會將與資源關(guān)聯(lián)并且能夠可以用于對其進行對等性判斷的某個值作為回復(fù)的ETag報頭,這與條件獲取時一致的。
客戶端通過回復(fù)獲得請求的資源和ETag報頭值。對于資源修改操作,客戶端直接針對獲取的資源進行相應(yīng)的修改,并將修改后的資源以HTTP請求的方式向服務(wù)端提交;對于資源刪除操作,則可以指定被刪除資源的唯一標(biāo)識直接向服務(wù)端發(fā)送刪除的請求。而之前獲取的ETag指將會作為請求消息的If-Match報頭。
服務(wù)端接收到資源修改/刪除請求后先獲取到現(xiàn)有的資源的ETag值,并將此值與請求消息的If-Match報頭值進行比較。如果兩者不一致,則表明試圖被修改/刪除的資源已經(jīng)被修改了,在這種情況下會直接回復(fù)一個HTTP狀態(tài)為“412 (Precondition Failed)”的空消息。條件更新同時支持針對PUT、POST和DELETE這三種方法的HTTP請求。
服務(wù)端進行條件更新檢測,以及客戶端對If-Match請求報頭的設(shè)置都可以通過當(dāng)前的WebOperationContext來完成。如下面的代碼片斷所示,表示入棧請求上下文的IncomingWebRequestContext類型具有如下四個CheckConditionalUpdate方法重載用于進行添加更新檢測。
1: public class IncomingWebRequestContext
2: {
3: //其他成員
4: public void CheckConditionalUpdate(Guid entityTag);
5: public void CheckConditionalUpdate(int entityTag);
6: public void CheckConditionalUpdate(long entityTag);
7: public void CheckConditionalUpdate(string entityTag);
8: }
實現(xiàn)在CheckConditionalUpdate方法中的條件更新檢測具有這樣的邏輯:對于HTTP方法為PUT的請求,如果If-Match報頭值不為“*”,則直接拋出HTTP狀態(tài)為PreconditionFailed的WebFaultException異常;對于HTTP方法為POST和DELETE的請求來說,如果If-Match報頭值為“*”或者包含指定的entityTag則驗證通過,否則同樣則直接拋出HTTP狀態(tài)為PreconditionFailed的WebFaultException異常。
表示出棧請求上下文的OutgoingWebRequestContext類型具有如下一個IfMatch屬性,客戶端可以通過該屬性對請求消息的If-Match報頭進行設(shè)置。
1: public class OutgoingWebRequestContext
2: {
3: //其他成員
4: public string IfMatch { get; set; }
5: }
我們同樣通過對EmployeesService進行相應(yīng)的改造來模擬如何通過添加更新實現(xiàn)對相同資源的并發(fā)操作問題,這次我們修改的是用于獲取指定ID員工信息的Get操作和用于修改員工信息的Update操作。Get操作在返回與指定員工ID匹配的Employee對象之前我們將該對象的哈希碼作為了回復(fù)消息的ETag報頭(Employee類型重寫了GetHashCode方法)。
1: public class EmployeesService : IEmployees
2: {
3: //其他成員
4: public Employee Get(string id)
5: {
6: Employee employee = employees.FirstOrDefault(e => e.Id == id);
7: if (null == employee)
8: {
9: throw new WebFaultException(HttpStatusCode.NotFound);
10: }
11: WebOperationContext.Current.OutgoingResponse.SetETag(employee.GetHashCode());
12: return employee;
13: }
14: public void Update(Employee employee)
15: {
16: var existing = employees.FirstOrDefault(e => e.Id == employee.Id);
17: if (null == existing)
18: {
19: throw new WebFaultException(HttpStatusCode.NotFound);
20: }
21: //模擬并發(fā)修改
22: existing.Name += Guid.NewGuid().ToString();
23:?
24: WebOperationContext.Current.IncomingRequest.CheckConditionalUpdate(existing.GetHashCode());
25: employees.Remove(existing);
26: employees.Add(employee);
27: WebOperationContext.Current.OutgoingResponse.SetETag(employee.GetHashCode());
28:?
29: }
30: }
Update方法中我們通過手工修改相應(yīng)員工的Name屬性的方式來模擬針對相同員工信息的并發(fā)修改。在真正實施修改之前調(diào)用當(dāng)前IncomingWebRequestContext的CheckConditionalUpdate方法進行條件更新檢測,而作為參數(shù)傳入的ETag值為代表目前員工的Employee對象的哈希碼。方法的最后我們對回復(fù)消息的ETag報頭作了更新。
我們通過手工創(chuàng)建HTTP請求的方式對上述的兩個服務(wù)操作進行調(diào)用。如下面的代碼片斷所示,我們首先通過創(chuàng)建的HttpWebRequest對象調(diào)用Get操作獲得ID為001的員工信息并將其打印出來。然后創(chuàng)建調(diào)用Update操作的HttpWebRequest,并對HTTP方法(POST)和內(nèi)容類型(application/xml)進行了相應(yīng)的設(shè)置。我們之前針對員工獲取請求得到ETag報頭和員工數(shù)據(jù)作為本次請求的If-Match報頭和主體。如果調(diào)用GetResponse方法拋出WebException異常,并且其回復(fù)狀態(tài)為PreconditionFailed,則表明試圖修改的員工信息已被另一個用戶修改過了,所以我么打印“服務(wù)端數(shù)據(jù)已發(fā)生變化”字樣。
1: Uri address = new Uri("http://127.0.0.1:3721/employees/001");
2: var request = (HttpWebRequest)HttpWebRequest.Create(address);
3: request.Method = "GET";
4: var response = (HttpWebResponse)request.GetResponse();
5: string employee;
6: using (StreamReader reader = new StreamReader(response.GetResponseStream(), Encoding.UTF8))
7: {
8: employee = reader.ReadToEnd();
9: Console.WriteLine("獲取員工信息:");
10: Console.WriteLine(employee + "\n");
11: }
12: try
13: {
14: address = new Uri("http://127.0.0.1:3721/employees/");
15: request = (HttpWebRequest)HttpWebRequest.Create(address);
16: request.Method = "POST";
17: request.ContentType = "application/xml";
18: byte[] buffer = Encoding.UTF8.GetBytes(employee);
19: request.GetRequestStream().Write(Encoding.UTF8.GetBytes(employee), 0, buffer.Length);
20: request.Headers.Add(HttpRequestHeader.IfMatch, response.Headers[HttpResponseHeader.ETag]);
21: Console.WriteLine("修改員工信息:");
22: request.GetResponse();
23: }
24: catch (WebException ex)
25: {
26: response = ex.Response as HttpWebResponse;
27: if (null == response)
28: {
29: throw;
30: }
31: if (response.StatusCode == HttpStatusCode.PreconditionFailed)
32: {
33: Console.WriteLine("服務(wù)端數(shù)據(jù)已發(fā)生變化");
34: }
35: else
36: {
37: throw;
38: }
39: }
在服務(wù)成功寄宿的情況下調(diào)用這段程序會在控制臺上輸出如下的結(jié)果。由于并發(fā)錯誤的發(fā)生,員工信息其實并沒有被真正修改。
1: 獲取員工信息:
2: <Employee xmlns="http://www.artech.com/" xmlns:i="http://www.w3.org/2001/XMLSchema-instance"><Department>開發(fā)部</Department><Grade>G7</Grade><Id>001</Id><Name>張三</Name></Employee>
3:?
4: 修改員工信息:
5: 服務(wù)端數(shù)據(jù)已發(fā)生變化
日期:2018-04 瀏覽次數(shù):6839
日期:2017-02 瀏覽次數(shù):3511
日期:2017-09 瀏覽次數(shù):3747
日期:2017-12 瀏覽次數(shù):3593
日期:2018-12 瀏覽次數(shù):4902
日期:2016-12 瀏覽次數(shù):4661
日期:2017-07 瀏覽次數(shù):13709
日期:2017-12 瀏覽次數(shù):3584
日期:2018-06 瀏覽次數(shù):4335
日期:2018-05 瀏覽次數(shù):4515
日期:2017-12 瀏覽次數(shù):3625
日期:2017-06 瀏覽次數(shù):4048
日期:2018-01 瀏覽次數(shù):4019
日期:2016-12 瀏覽次數(shù):3975
日期:2018-08 瀏覽次數(shù):4486
日期:2017-12 瀏覽次數(shù):3795
日期:2016-09 瀏覽次數(shù):6554
日期:2018-07 瀏覽次數(shù):3275
日期:2016-12 瀏覽次數(shù):3295
日期:2018-10 瀏覽次數(shù):3447
日期:2018-10 瀏覽次數(shù):3552
日期:2018-09 瀏覽次數(shù):3641
日期:2018-02 瀏覽次數(shù):3668
日期:2015-05 瀏覽次數(shù):3591
日期:2018-09 瀏覽次數(shù):3377
日期:2018-06 瀏覽次數(shù):3500
日期:2017-02 瀏覽次數(shù):3937
日期:2018-02 瀏覽次數(shù):4402
日期:2018-02 瀏覽次數(shù):4266
日期:2016-12 瀏覽次數(shù):3638
Copyright ? 2013-2018 Tadeng NetWork Technology Co., LTD. All Rights Reserved.