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