Mongoose에 스키마와 모델이 모두있는 이유는 무엇입니까?


93

두 가지 유형의 객체는 서로 너무 가까워서 둘 다 중복 된 느낌을줍니다. 스키마와 모델 을 모두 갖는 이유는 무엇입니까 ?

답변:


61

종종 이러한 유형의 질문에 답하는 가장 쉬운 방법은 예를 사용하는 것입니다. 이 경우 누군가 이미 나를 위해 해주었습니다. :)

여기를보세요 :

http://rawberg.com/blog/nodejs/mongoose-orm-nested-models/

편집 : (댓글에서 언급했듯이) 원본 게시물이 더 이상 존재하지 않는 것 같아서 아래에서 재현하고 있습니다. 돌아 오거나 방금 이동 한 경우 알려주십시오.

몽구스의 모델 내에서 스키마를 사용하는 방법에 대한 적절한 설명과이를 수행하려는 이유를 제공하고 스키마가 구조 등에 관한 모든 것이있는 동안 모델을 통해 작업을 푸시하는 방법을 보여줍니다.

원본 게시물 :

모델 내부에 스키마를 삽입하는 간단한 예부터 시작하겠습니다.

var TaskSchema = new Schema({
    name: String,
    priority: Number
});

TaskSchema.virtual('nameandpriority')
    .get( function () {
        return this.name + '(' + this.priority + ')';
    });

TaskSchema.method('isHighPriority', function() {
    if(this.priority === 1) {
        return true;
    } else {
        return false;
    }
}); 

var ListSchema = new Schema({
    name: String,
    tasks: [TaskSchema]
});

mongoose.model('List', ListSchema);

var List = mongoose.model('List');

var sampleList = new List({name:'Sample List'});

TaskSchema작업에있을 수있는 기본 정보 로 새 개체를 만들었습니다 . Mongoose 가상 속성 은 태스크의 이름과 우선 순위를 편리하게 결합하도록 설정됩니다. 여기에 게터 만 지정했지만 가상 세터도 지원됩니다.

또한 isHighPriority이 설정에서 메서드가 작동하는 방식을 보여주기 위해 호출되는 간단한 작업 메서드도 정의했습니다 .

에서 ListSchema정의 당신은 작업 키의 배열을 유지하도록 구성하는 방법을 알 수 있습니다 TaskSchema개체를. 태스크 키는 DocumentArray포함 된 Mongo 문서를 처리하기위한 특별한 방법을 제공 하는 인스턴스가 됩니다.

지금은 ListSchema객체를 mongoose.model로 전달 하고 TaskSchema는 그대로 두었습니다. 기술적 TaskSchema으로는 자체 컬렉션에 저장하지 않기 때문에 공식 모델로 변환 할 필요 가 없습니다. 나중에 어떤 작업을해도 아무런 해를 끼치 지 않는 방법을 보여 드리며 특히 여러 파일을 스패닝하기 시작할 때 모든 모델을 동일한 방식으로 구성하는 데 도움이 될 수 있습니다.

List모델 설정의 그것에 몇 가지 작업을 추가하고 저장 몽고에 수 있습니다.

var List = mongoose.model('List');
var sampleList = new List({name:'Sample List'});

sampleList.tasks.push(
    {name:'task one', priority:1}, 
    {name:'task two', priority:5}
);

sampleList.save(function(err) {
    if (err) {
        console.log('error adding new list');
        console.log(err);
    } else {
        console.log('new list successfully saved'); 
    }
});

List모델 인스턴스의 작업 속성 ( simpleList)은 일반 자바 스크립트 배열처럼 작동하며 푸시를 사용하여 새 작업을 추가 할 수 있습니다. 주목해야 할 중요한 것은 작업이 일반 JavaScript 객체로 추가된다는 것입니다. 즉각적으로 직관적이지 않을 수있는 미묘한 차이입니다.

Mongo 셸에서 새 목록과 작업이 mongo에 저장되었는지 확인할 수 있습니다.

db.lists.find()
{ "tasks" : [
    {
        "_id" : ObjectId("4dd1cbeed77909f507000002"),
        "priority" : 1,
        "name" : "task one"
    },
    {
        "_id" : ObjectId("4dd1cbeed77909f507000003"),
        "priority" : 5,
        "name" : "task two"
    }
], "_id" : ObjectId("4dd1cbeed77909f507000001"), "name" : "Sample List" }

이제를 사용하여 작업 ObjectId을 풀업 Sample List하고 반복 할 수 있습니다 .

List.findById('4dd1cbeed77909f507000001', function(err, list) {
    console.log(list.name + ' retrieved');
    list.tasks.forEach(function(task, index, array) {
        console.log(task.name);
        console.log(task.nameandpriority);
        console.log(task.isHighPriority());
    });
});

마지막 코드를 실행하면 포함 된 문서에 메서드가 없다는 오류가 표시 isHighPriority됩니다. 현재 버전의 Mongoose에서는 포함 된 스키마의 메서드에 직접 액세스 할 수 없습니다. 문제를 해결하기 위한 공개 티켓 이 있으며 Mongoose Google 그룹에 질문을 제기 한 후 manimal45는 현재 사용할 수있는 유용한 해결 방법을 게시했습니다.

List.findById('4dd1cbeed77909f507000001', function(err, list) {
    console.log(list.name + ' retrieved');
    list.tasks.forEach(function(task, index, array) {
        console.log(task.name);
        console.log(task.nameandpriority);
        console.log(task._schema.methods.isHighPriority.apply(task));
    });
});

해당 코드를 실행하면 명령 줄에 다음 출력이 표시됩니다.

Sample List retrieved
task one
task one (1)
true
task two
task two (5)
false

이 해결 방법을 염두에두고를 TaskSchemaMongoose 모델로 전환 해 보겠습니다 .

mongoose.model('Task', TaskSchema);

var Task = mongoose.model('Task');

var ListSchema = new Schema({
    name: String,
    tasks: [Task.schema]
});

mongoose.model('List', ListSchema);

var List = mongoose.model('List');

TaskSchema정의는 내가 그것을 왼쪽 있도록 이전과 동일합니다. 모델이 된 후에도 점 표기법을 사용하여 기본 Schema 객체에 액세스 할 수 있습니다.

새 목록을 만들고 그 안에 두 개의 Task 모델 인스턴스를 포함 해 보겠습니다.

var demoList = new List({name:'Demo List'});

var taskThree = new Task({name:'task three', priority:10});
var taskFour = new Task({name:'task four', priority:11});

demoList.tasks.push(taskThree.toObject(), taskFour.toObject());

demoList.save(function(err) {
    if (err) {
        console.log('error adding new list');
        console.log(err);
    } else {
        console.log('new list successfully saved'); 
    }
});

Task 모델 인스턴스를 List toObject에 포함 할 때 데이터를에서 List.tasks DocumentArray예상 하는 일반 JavaScript 객체로 변환하도록 요청합니다 . 이런 식으로 모델 인스턴스를 저장하면 포함 된 문서에 ObjectIds.

전체 코드 예제는 요점으로 사용할 수 있습니다 . 이러한 해결 방법이 Mongoose가 계속해서 발전함에 따라 원활하게 진행되기를 바랍니다. 저는 아직 Mongoose와 MongoDB에 익숙하지 않으므로 의견에 더 나은 솔루션과 팁을 자유롭게 공유하십시오. 행복한 데이터 모델링!


3
일반적으로 링크가 작동을 멈출 수 있으므로 (이 경우와 같이) SO에 게시 된 질문에 대한 답변으로 베어 링크를 제출하지 않는 것이 좋습니다. 적어도 링크하는 기사의 관련 섹션을 복사 / 붙여 넣기 및 인용하십시오.
Behrang Saeedzadeh 2014 년

1
수행 - 그것은, 구글 캐시에 여전히 비교적 간단했다
아담 Comerford

1
기록을 위해, 내장 된 문서의 방법의 문제가 해결되었습니다 github.com/LearnBoost/mongoose/issues/249#ref-commit-e18077a
다코타

5
나는 다른 사람의 퍼레이드에 비를 내리려고하지 않지만,이 답변은 튜토리얼과 비슷합니다. 방법에 대한 대답이지만 이유는 아닙니다. 적은까지 투표가 있더라도, 나는 다음과 같은 대답이 훨씬 더 도움이 발견 : stackoverflow.com/a/22950402/26331을
aaaidan

2
나는 그 대답을 보았습니다 (그리고 그것을 찬성했습니다), 이것은 2 년 전에 대답하고 받아 들였습니다. 더 나은 답을 찾을 수있어서 기쁩니다. 누구의 퍼레이드에도 비가 내리지 않으며 2015 년 2 월부터 질문에 대한 댓글에서 언급 한 답변에 대한 링크가 있으므로 직접 연결할 필요가 없다고 생각했습니다.
Adam Comerford

54

스키마 는 MongoDB 컬렉션에 저장 될 문서의 구조를 정의하는 객체입니다. 모든 데이터 항목에 대한 유형 및 유효성 검사기를 정의 할 수 있습니다.

모델 은 명명 된 컬렉션에 쉽게 액세스 할 수있는 개체로, 컬렉션을 쿼리하고 스키마를 사용하여 해당 컬렉션에 저장 한 문서의 유효성을 검사 할 수 있습니다. 스키마, 연결 및 컬렉션 이름을 결합하여 생성됩니다.

원래 Valeri Karpov, MongoDB 블로그가 작성했습니다.


5

나는 받아 들여진 대답이 실제로 제기 된 질문에 대한 답이라고 생각하지 않습니다. 대답은 Mongoose가 개발자에게 Schema 및 Model 변수를 모두 제공하도록 결정한 이유를 설명하지 않습니다 . 개발자 의 필요성을 제거한 프레임 워크의 예데이터 스키마를 정의하는 것은 django입니다. 개발자는 models.py 파일에 모델을 작성하고 스키마를 관리하기 위해 프레임 워크에 맡깁니다. 장고에 대한 나의 경험을 고려할 때 그들이 이것을하는 이유에 대해 가장 먼저 떠오르는 이유는 사용 편의성입니다. 아마도 더 중요한 것은 DRY (반복하지 마십시오) 원칙입니다. 모델을 변경할 때 스키마를 업데이트 할 필요가 없습니다. django가 자동으로 수행합니다! Rails는 또한 데이터 스키마를 관리합니다. 개발자는 스키마를 직접 편집하지 않고 스키마를 조작하는 마이그레이션을 정의하여 변경합니다.

Mongoose가 스키마를 분리하고 모델은 두 개의 스키마에서 모델을 빌드하려는 인스턴스라는 것을 이해할 수있는 한 가지 이유입니다. 이러한 시나리오는 관리 할 가치가있는 것보다 더 복잡 할 수 있습니다. 하나의 모델에서 관리하는 두 개의 스키마가있는 경우 왜 하나의 스키마가 아닌가?

아마도 원래의 질문은 전통적인 관계형 데이터베이스 시스템의 유물 일 것입니다. NoSQL / Mongo 세계에서는 스키마가 MySQL / PostgreSQL보다 약간 더 유연하므로 스키마를 변경하는 것이 더 일반적입니다.


스키마 대 모델만으로는 충분하지 않은 것처럼 일치하는 TypeScript 인터페이스유지 하려고 할 때 더 많은 중복이 발생하고 GraphQL 스키마를 만들 때 더 많이 발생합니다.
Dan Dascalescu

0

이유를 이해하기 위해? 몽구스가 실제로 무엇인지 이해해야합니까?

mongoose는 MongoDB 및 Node JS를위한 객체 데이터 모델링 라이브러리로, 더 높은 수준의 추상화를 제공합니다. 따라서 Express와 Node 간의 관계와 약간 비슷하므로 Express는 일반 노드에 대한 추상화 계층이고 Mongoose는 일반 MongoDB 드라이버에 대한 추상화 계층입니다.

객체 데이터 모델링 라이브러리는 데이터베이스와 상호 작용할 Javascript 코드를 작성하는 방법 일뿐입니다. 따라서 일반 MongoDB 드라이버를 사용하여 데이터베이스에 액세스 할 수 있습니다. 잘 작동합니다.

하지만 대신 Mongoose를 사용합니다. 그 이유는 바로 사용할 수있는 훨씬 더 많은 기능을 제공하여 애플리케이션을 더 빠르고 간단하게 개발할 수 있기 때문입니다.

따라서 Mongoose의 일부 기능은 데이터 및 관계를 모델링 할 수있는 스키마, 쉬운 데이터 유효성 검사, 간단한 쿼리 API, 미들웨어 등을 제공합니다.

Mongoose에서 스키마는 데이터를 모델링하는 곳입니다. 여기서 데이터의 구조, 기본값 및 유효성 검사를 설명합니다. 그런 다음 해당 스키마를 가져 와서 모델을 만듭니다. 모델은 기본적으로 스키마를 둘러싼 래퍼입니다. 문서를 작성, 삭제, 업데이트 및 읽기 위해 실제로 데이터베이스와 인터페이스 할 수 있습니다.

여기에 이미지 설명 입력

스키마에서 모델을 생성 해 보겠습니다.

const tourSchema = new mongoose.Schema({
  name: {
    type: String,
    required: [true, 'A tour must have a name'],
    unique: true,
  },
  rating: {
    type: Number,
    default: 4.5,
  },
  price: {
    type: Number,
    required: [true, 'A tour must have a price'],
  },
});
//tour model
const Tour = mongoose.model('Tour', tourSchema);

대류에 따라 모델 이름의 첫 글자는 대문자 여야합니다.

몽구스와 스키마를 사용하여 생성 한 모델의 인스턴스를 생성 해 보겠습니다. 또한 데이터베이스와 상호 작용합니다.

const testTour = new Tour({ // instance of our model
  name: 'The Forest Hiker',
  rating: 4.7,
  price: 497,
});
 // saving testTour document into database
testTour
  .save()
  .then((doc) => {
    console.log(doc);
  })
  .catch((err) => {
    console.log(err);
  });

그래서 schama와 modle mongoose를 모두 가지고 있으면 우리 삶이 더 쉬워집니다.

당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.