DocumentDB-데이터 모델링

DocumentDB와 같은 스키마없는 데이터베이스를 사용하면 데이터 모델에 대한 변경 사항을 매우 쉽게 수용 할 수 있지만 데이터에 대해 생각하는 데 시간을 투자해야합니다.

  • 많은 옵션이 있습니다. 당연히 JSON 개체 그래프 또는 JSON 텍스트의 원시 문자열로만 작업 할 수 있지만 컴파일 타임에 클래스를 정의하지 않고도 런타임에 속성에 바인딩 할 수있는 동적 개체를 사용할 수도 있습니다.

  • 실제 C # 개체 또는 호출 된 엔터티 (비즈니스 도메인 클래스 일 수 있음)로 작업 할 수도 있습니다.

관계

문서의 계층 구조를 살펴 보겠습니다. 필수 id, lastName 및 isRegistered와 같은 몇 가지 최상위 속성이 있지만 중첩 속성도 있습니다.

{ 
   "id": "AndersenFamily", 
   "lastName": "Andersen", 
	
   "parents": [ 
      { "firstName": "Thomas", "relationship": "father" }, 
      { "firstName": "Mary Kay", "relationship": "mother" } 
   ],
	
   "children": [ 
      { 
         "firstName": "Henriette Thaulow", 
         "gender": "female", 
         "grade": 5, 
         "pets": [ { "givenName": "Fluffy", "type": "Rabbit" } ] 
      } 
   ], 
	
   "location": { "state": "WA", "county": "King", "city": "Seattle"}, 
   "isRegistered": true 
}
  • 예를 들어 parent 속성은 대괄호로 표시된 JSON 배열로 제공됩니다.

  • 이 예제에서는 배열에 자식이 하나 뿐이지 만 자식을위한 또 다른 배열도 있습니다. 따라서 이것은 문서 내에서 일대 다 관계에 해당하는 모델을 만드는 방법입니다.

  • 배열의 각 요소가 단순한 값이거나 다른 복잡한 객체, 심지어 다른 배열 일 수있는 배열을 사용하기 만하면됩니다.

  • 따라서 한 패밀리에는 여러 부모와 여러 자식이있을 수 있으며 자식 개체를 보면 자식과 애완 동물 간의 일대 다 관계를위한 중첩 배열 인 애완 동물의 속성이 있습니다.

  • 위치 속성의 경우 주, 카운티 및 도시의 세 가지 관련 속성을 하나의 개체로 결합합니다.

  • 개체 배열을 포함하는 대신 이러한 방식으로 개체를 포함하는 것은 관계형 데이터베이스의 별도 테이블에있는 두 행간에 일대일 관계를 갖는 것과 유사합니다.

데이터 삽입

DocumentDB와 같은 문서 저장소에서 데이터 모델링을 시작할 때 엔티티를 JSON으로 표시되는 자체 포함 문서로 취급하십시오. 관계형 데이터베이스로 작업 할 때 우리는 항상 데이터를 정규화합니다.

  • 데이터 정규화에는 일반적으로 고객과 같은 엔티티를 가져와 연락처 세부 정보 및 주소와 같은 신중한 데이터 조각으로 나누는 작업이 포함됩니다.

  • 모든 연락처 세부 정보 및 주소와 함께 고객을 읽으려면 JOINS를 사용하여 런타임에 데이터를 효과적으로 집계해야합니다.

이제 문서 데이터베이스에서 자체 포함 된 엔터티와 동일한 데이터를 모델링하는 방법을 살펴 보겠습니다.

{
   "id": "1", 
   "firstName": "Mark", 
   "lastName": "Upston", 
	
   "addresses": [ 
      {             
         "line1": "232 Main Street", 
         "line2": "Unit 1", 
         "city": "Brooklyn", 
         "state": "NY", 
         "zip": 11229
      }
   ],
	
   "contactDetails": [ 
      {"email": "[email protected]"}, 
      {"phone": "+1 356 545-86455", "extension": 5555} 
   ]
}

보시다시피 고객의 모든 정보가 단일 JSON 문서에 포함 된 고객 레코드를 비정규 화했습니다.

NoSQL에는 무료 스키마가 있으므로 연락처 세부 정보와 주소를 다른 형식으로 추가 할 수도 있습니다. NoSQL에서는 단일 읽기 작업으로 데이터베이스에서 고객 레코드를 검색 할 수 있습니다. 마찬가지로 레코드 업데이트도 단일 쓰기 작업입니다.

다음은 .Net SDK를 사용하여 문서를 만드는 단계입니다.

Step 1− DocumentClient를 인스턴스화합니다. 그런 다음 myfirstdb 데이터베이스를 쿼리하고 MyCollection 컬렉션도 쿼리합니다. MyCollection 컬렉션은이 개인 변수 컬렉션에 저장하므로 클래스 전체에서 액세스 할 수 있습니다.

private static async Task CreateDocumentClient() { 
   // Create a new instance of the DocumentClient
	
   using (var client = new DocumentClient(new Uri(EndpointUrl), AuthorizationKey)) { 
      database = client.CreateDatabaseQuery("SELECT * FROM c WHERE c.id =
         'myfirstdb'").AsEnumerable().First(); 
			
      collection = client.CreateDocumentCollectionQuery(database.CollectionsLink,
         "SELECT * FROM c WHERE c.id = 'MyCollection'").AsEnumerable().First();  
			
      await CreateDocuments(client); 
   }

}

Step 2 − CreateDocuments 작업에서 일부 문서를 생성합니다.

private async static Task CreateDocuments(DocumentClient client) {
   Console.WriteLine(); 
   Console.WriteLine("**** Create Documents ****"); 
   Console.WriteLine();
	
   dynamic document1Definition = new {
      name = "New Customer 1", address = new { 
         addressType = "Main Office", 
         addressLine1 = "123 Main Street", 
         location = new { 
            city = "Brooklyn", stateProvinceName = "New York"
         }, 
         postalCode = "11229", countryRegionName = "United States" 
      }, 
   };
	
   Document document1 = await CreateDocument(client, document1Definition); 
   Console.WriteLine("Created document {0} from dynamic object", document1.Id); 
   Console.WriteLine(); 
}

첫 번째 문서는이 동적 개체에서 생성됩니다. 이것은 JSON처럼 보일 수 있지만 물론 그렇지 않습니다. 이것은 C # 코드이며 실제 .NET 개체를 생성하고 있지만 클래스 정의는 없습니다. 대신 속성은 개체가 초기화되는 방식에서 유추됩니다. 또한이 문서에 대해 Id 속성을 제공하지 않았 음을 알 수 있습니다.

Step 3 − 이제 CreateDocument를 살펴보면 데이터베이스와 컬렉션을 생성 할 때 본 것과 동일한 패턴으로 보입니다.

private async static Task<Document> CreateDocument(DocumentClient client,
   object documentObject) {
   var result = await client.CreateDocumentAsync(collection.SelfLink, documentObject); 
	
   var document = result.Resource; 
   Console.WriteLine("Created new document: {0}\r\n{1}", document.Id, document); 
	
   return result; 
}

Step 4− 이번에는 문서를 추가 할 컬렉션의 SelfLink를 지정하는 CreateDocumentAsync를 호출합니다. 이 경우 시스템 생성 속성이있는 새 문서를 나타내는 리소스 속성이있는 응답을 반환합니다.

다음 CreateDocuments 작업에서 세 개의 문서를 만들었습니다.

  • 첫 번째 문서에서 Document 개체는 리소스에서 상속되는 SDK의 정의 된 클래스이므로 모든 공통 리소스 속성이 있지만 스키마없는 문서 자체를 정의하는 동적 속성도 포함됩니다.

private async static Task CreateDocuments(DocumentClient client) {
   Console.WriteLine(); 
   Console.WriteLine("**** Create Documents ****"); 
   Console.WriteLine();
	
   dynamic document1Definition = new {
      name = "New Customer 1", address = new {
         addressType = "Main Office", 
         addressLine1 = "123 Main Street", 
         location = new {
            city = "Brooklyn", stateProvinceName = "New York" 
         }, 
         postalCode = "11229", 
         countryRegionName = "United States" 
      }, 
   };
	
   Document document1 = await CreateDocument(client, document1Definition); 
   Console.WriteLine("Created document {0} from dynamic object", document1.Id); 
   Console.WriteLine();
	
   var document2Definition = @" {
      ""name"": ""New Customer 2"", 
		
      ""address"": { 
         ""addressType"": ""Main Office"", 
         ""addressLine1"": ""123 Main Street"", 
         ""location"": { 
            ""city"": ""Brooklyn"", ""stateProvinceName"": ""New York"" 
         }, 
         ""postalCode"": ""11229"", 
         ""countryRegionName"": ""United States"" 
      } 
   }"; 
	
   Document document2 = await CreateDocument(client, document2Definition); 
   Console.WriteLine("Created document {0} from JSON string", document2.Id);
   Console.WriteLine();
	
   var document3Definition = new Customer {
      Name = "New Customer 3", 
		
      Address = new Address {
         AddressType = "Main Office", 
         AddressLine1 = "123 Main Street", 
         Location = new Location {
            City = "Brooklyn", StateProvinceName = "New York" 
         }, 
         PostalCode = "11229", 
         CountryRegionName = "United States" 
      }, 
   };
	
   Document document3 = await CreateDocument(client, document3Definition); 
   Console.WriteLine("Created document {0} from typed object", document3.Id); 
   Console.WriteLine(); 
}
  • 이 두 번째 문서는 원시 JSON 문자열로만 작동합니다. 이제 JavaScriptSerializer를 사용하여 문자열을 개체로 역 직렬화하는 CreateDocument에 대한 오버로드로 들어가서 첫 번째 문서를 만드는 데 사용한 것과 동일한 CreateDocument 메서드로 전달합니다.

  • 세 번째 문서에서는 애플리케이션에 정의 된 C # 개체 Customer를 사용했습니다.

이 고객을 살펴 보겠습니다. 여기에는 주소가 위치를 포함하는 자체 속성이있는 중첩 된 개체 인 Id 및 주소 속성이 있으며 이는 또 다른 중첩 개체입니다.

using Newtonsoft.Json; 

using System; 
using System.Collections.Generic; 
using System.Linq; 
using System.Text; 
using System.Threading.Tasks;

namespace DocumentDBDemo {
 
   public class Customer { 
      [JsonProperty(PropertyName = "id")] 
      public string Id { get; set; }
      // Must be nullable, unless generating unique values for new customers on client  
      [JsonProperty(PropertyName = "name")] 
      public string Name { get; set; }  
      [JsonProperty(PropertyName = "address")] 
      public Address Address { get; set; } 
   }
	
   public class Address {
      [JsonProperty(PropertyName = "addressType")] 
      public string AddressType { get; set; }  
		
      [JsonProperty(PropertyName = "addressLine1")] 
      public string AddressLine1 { get; set; }  
		
      [JsonProperty(PropertyName = "location")] 
      public Location Location { get; set; }  
		
      [JsonProperty(PropertyName = "postalCode")] 
      public string PostalCode { get; set; }  
		
      [JsonProperty(PropertyName = "countryRegionName")] 
      public string CountryRegionName { get; set; } 
   }
	
   public class Location { 
      [JsonProperty(PropertyName = "city")] 
      public string City { get; set; }  
		
      [JsonProperty(PropertyName = "stateProvinceName")]
      public string StateProvinceName { get; set; } 
   } 
}

울타리의 양쪽에서 적절한 규칙을 유지하기 위해 JSON 속성 속성도 있습니다.

따라서 중첩 된 자식 개체와 함께 New Customer 개체를 만들고 CreateDocument를 한 번 더 호출합니다. 고객 개체에는 Id 속성이 있지만 값을 제공하지 않았으므로 DocumentDB는 이전 두 문서에서와 마찬가지로 GUID를 기반으로 하나를 생성했습니다.

위의 코드가 컴파일되고 실행되면 다음과 같은 출력이 표시됩니다.

**** Create Documents ****  
Created new document: 575882f0-236c-4c3d-81b9-d27780206b2c 
{ 
  "name": "New Customer 1", 
  "address": { 
    "addressType": "Main Office", 
    "addressLine1": "123 Main Street", 
    "location": { 
      "city": "Brooklyn", 
      "stateProvinceName": "New York" 
    }, 
    "postalCode": "11229", 
    "countryRegionName": "United States" 
  }, 
  "id": "575882f0-236c-4c3d-81b9-d27780206b2c", 
  "_rid": "kV5oANVXnwDGPgAAAAAAAA==", 
  "_ts": 1450037545, 
  "_self": "dbs/kV5oAA==/colls/kV5oANVXnwA=/docs/kV5oANVXnwDGPgAAAAAAAA==/", 
  "_etag": "\"00006fce-0000-0000-0000-566dd1290000\"", 
  "_attachments": "attachments/" 
} 
Created document 575882f0-236c-4c3d-81b9-d27780206b2c from dynamic object  
Created new document: 8d7ad239-2148-4fab-901b-17a85d331056 
{ 
  "name": "New Customer 2", 
  "address": {
    "addressType": "Main Office", 
    "addressLine1": "123 Main Street", 
    "location": { 
      "city": "Brooklyn", 
      "stateProvinceName": "New York" 
    }, 
    "postalCode": "11229", 
    "countryRegionName": "United States" 
  }, 
  "id": "8d7ad239-2148-4fab-901b-17a85d331056", 
  "_rid": "kV5oANVXnwDHPgAAAAAAAA==", 
  "_ts": 1450037545, 
  "_self": "dbs/kV5oAA==/colls/kV5oANVXnwA=/docs/kV5oANVXnwDHPgAAAAAAAA==/", 
  "_etag": "\"000070ce-0000-0000-0000-566dd1290000\"", 
  "_attachments": "attachments/" 
} 
Created document 8d7ad239-2148-4fab-901b-17a85d331056 from JSON string  
Created new document: 49f399a8-80c9-4844-ac28-cd1dee689968 
{ 
  "id": "49f399a8-80c9-4844-ac28-cd1dee689968", 
  "name": "New Customer 3", 
  "address": { 
    "addressType": "Main Office", 
    "addressLine1": "123 Main Street", 
    "location": { 
      "city": "Brooklyn", 
      "stateProvinceName": "New York" 
    }, 
    "postalCode": "11229", 
    "countryRegionName": "United States" 
  }, 
  "_rid": "kV5oANVXnwDIPgAAAAAAAA==", 
  "_ts": 1450037546, 
  "_self": "dbs/kV5oAA==/colls/kV5oANVXnwA=/docs/kV5oANVXnwDIPgAAAAAAAA==/", 
  "_etag": "\"000071ce-0000-0000-0000-566dd12a0000\"", 
  "_attachments": "attachments/" 
}
Created document 49f399a8-80c9-4844-ac28-cd1dee689968 from typed object