iOS
...
Data Objects
การใช้ ParseSwift SDK สำหรับการค้นหาข้อมูลสัมพันธ์บน iOS
12 นาที
การค้นหาที่สัมพันธ์กัน บทนำ ใน คู่มือก่อนหน้า https //www back4app com/docs/ios/parse swift sdk/data objects/query cookbook เราได้อธิบายรายละเอียดเกี่ยวกับวิธีการทำการค้นหาหลายประเภทในฐานข้อมูล back4app ในคู่มือนี้เราจะมุ่งเน้นไปที่ประเภทการค้นหาที่เฉพาะเจาะจงซึ่งเกี่ยวข้องกับวัตถุที่มีความสัมพันธ์ ข้อกำหนดเบื้องต้น ในการทำตามบทเรียนนี้ คุณจะต้องมี แอป ที่สร้างขึ้นบน back4app แอป ios พื้นฐานเพื่อทดสอบการค้นหา เป้าหมาย ค้นหาข้อมูลที่สัมพันธ์กันที่จัดเก็บในฐานข้อมูล back4app โดยใช้ parseswift sdk parseswift sdk 1 การทบทวนอย่างรวดเร็วเกี่ยวกับคลาส query\<u> การค้นหาใด ๆ ที่ดำเนินการในฐานข้อมูล back4app จะทำผ่านคลาสทั่วไป query\<u> query\<u> พารามิเตอร์ทั่วไป u u (ซึ่งสอดคล้องกับ parseobject parseobject โปรโตคอล) เป็นประเภทข้อมูลของวัตถุที่เราพยายามดึงข้อมูลจากฐานข้อมูล เมื่อมีประเภทข้อมูลเช่น myobject myobject , เราจะดึงวัตถุเหล่านี้จากฐานข้อมูล back4app ในลักษณะดังต่อไปนี้ 1 import parseswift 2 3 struct myobject parseobject { 4 5 } 6 7 let query = myobject query() 8 9 // executes the query asynchronously 10 query find { result in 11 // handle the result (of type result<\[myobject], parseerror>) 12 } คุณสามารถอ่านเพิ่มเติมเกี่ยวกับ query\<u> query\<u> คลาส ที่นี่ในเอกสารทางการ https //github com/parse community/parse swift 2 บันทึกข้อมูลบางอย่างในฐานข้อมูล back4app ก่อนที่เราจะเริ่มดำเนินการสอบถาม จำเป็นต้องตั้งค่าข้อมูลบางอย่างในฐานข้อมูล back4app เราจะเก็บวัตถุห้าประเภท author 1 struct author parseobject { 2 3 var name string? 4 } book 1 struct book parseobject { 2 3 var title string? 4 var publisher publisher? 5 var publishingdate date? 6 } isbd 1 struct isbd parseobject { 2 3 var isbn string? 4 var book pointer\<book>? 5 } publisher 1 struct publisher parseobject { 2 3 var name string? 4 } bookstore 1 struct bookstore parseobject { 2 3 var name string? 4 } นอกจากนี้ เพื่อสร้างคำถามสำหรับข้อมูลเชิงสัมพันธ์ เราจะดำเนินการตามความสัมพันธ์ต่อไปนี้ 1 1 ความสัมพันธ์ระหว่าง หนังสือ หนังสือ และ isbd isbd 1\ n ความสัมพันธ์ระหว่าง หนังสือ หนังสือ และ ผู้จัดพิมพ์ ผู้จัดพิมพ์ m\ n ความสัมพันธ์ระหว่าง หนังสือ หนังสือ และ ผู้เขียน ผู้เขียน m\ n ความสัมพันธ์ระหว่าง ร้านหนังสือ ร้านหนังสือ และ หนังสือ หนังสือ เราจะดำเนินการเก็บข้อมูลบางอย่างในฐานข้อมูล back4app ตอนนี้ ขั้นตอนนี้สามารถดำเนินการได้โดยใช้ swift swift หรือโดยตรงจากคอนโซลของแอปของคุณบนแพลตฟอร์ม back4app swift //after setting up your xcode project, calling the following method should save some sample data on your back4app database 1 import parseswift 2 3 func savesampledata() { 4 do { 5 // authors 6 let aaronwriter = try author(name "aaron writer") save() 7 let beatricenovelist = try author(name "beatrice novelist") save() 8 let caseycolumnist = try author(name "casey columnist") save() 9 10 // publishers 11 let acaciapublishings = try publisher(name "acacia publishings") save() 12 let birchdistributions = try publisher(name "birch distributions") save() 13 14 // books with their corresponding isbd 15 let alovestorybook = try book( 16 title "a love story", 17 publisher acaciapublishings, 18 publishingdate date(string "05/07/1998") 19 ) save() 20 relation? add("authors", objects \[aaronwriter]) save() // establishes the m\ n relatin between book and author 21 22 // fetches the isbd associated to alovestorybook and establishes the 1 1 relation 23 if let book = alovestorybook, var isbd = try book fetch(includekeys \["isbd"]) isbd { 24 isbd book = try pointer\<book>(book) 25 = try isbd save() 26 } else { 27 fatalerror() 28 } 29 30 let benevolentelvesbook = try book( 31 title "benevolent elves", 32 publisher birchdistributions, 33 publishingdate date(string "11/30/2008") 34 ) save() 35 relation? add("authors", objects \[beatricenovelist]) save() // establishes the m\ n relatin between book and author 36 37 // fetches the isbd associated to benevolentelvesbook and establishes the 1 1 relation 38 if let book = benevolentelvesbook, var isbd = try book fetch(includekeys \["isbd"]) isbd { 39 isbd book = try pointer\<book>(book) 40 = try isbd save() 41 } else { 42 fatalerror() 43 } 44 45 let canyoubelieveitbook = try book( 46 title "can you believe it", 47 publisher birchdistributions, 48 publishingdate date(string "08/21/2018") 49 ) save() 50 relation? add("authors", objects \[aaronwriter, caseycolumnist]) save() // establishes the m\ n relatin between book and author 51 52 // fetches the isbd associated to canyoubelieveitbook and establishes the 1 1 relation 53 if let book = canyoubelieveitbook, var isbd = try book fetch(includekeys \["isbd"]) isbd { 54 isbd book = try pointer\<book>(book) 55 = try isbd save() 56 } else { 57 fatalerror() 58 } 59 60 // book store 61 guard let safealovestorybook = alovestorybook, 62 let safebenevolentelvesbook = benevolentelvesbook, 63 let safecanyoubelieveitbook = canyoubelieveitbook 64 else { 65 throw nserror( 66 domain bundle main description, 67 code 0, 68 userinfo \[nslocalizeddescriptionkey "failed to unwrapp stored books "] 69 ) 70 } 71 72 // saves the stores together with their 1\ n relation with book's 73 let booksoflove = try bookstore(name "books of love") save() 74 = try booksoflove relation? add("books", objects \[safealovestorybook]) save() 75 76 let fantasybooks = try bookstore(name "fantasy books") save() 77 = try fantasybooks relation? add("books", objects \[safebenevolentelvesbook]) save() 78 79 let generalbooks = try bookstore(name "general books") save() 80 = try generalbooks relation? add("books", objects \[safealovestorybook, safecanyoubelieveitbook]) save() 81 82 } catch let error as parseerror { 83 print("error \n", error message) 84 } catch { 85 print("error \n", error localizeddescription) 86 } 87 } back4app's console //a quick way to insert elements on your back4app database is via the console located in your app’s api section once you are there, you can start running javascript code to save the sample data 1 // authors 2 const aaronwriter = new parse object('author'); 3 aaronwriter set('name', 'aaron writer'); 4 await aaronwriter save(); 5 6 const beatricenovelist = new parse object('author'); 7 beatricenovelist set('name', 'beatrice novelist'); 8 await beatricenovelist save(); 9 10 const caseycolumnist = new parse object('author'); 11 caseycolumnist set('name', 'casey columnist'); 12 await caseycolumnist save(); 13 14 // publishers 15 const acaciapublishings = new parse object('publisher'); 16 acaciapublishings set('name', 'acacia publishings'); 17 await acaciapublishings save(); 18 19 const birchdistributions = new parse object('publisher'); 20 birchdistributions set('name', 'birch distributions'); 21 await birchdistributions save(); 22 23 // books with their corresponding isbd 24 const alovestoryisbd = new parse object('isbd'); 25 alovestoryisbd set('isbn', '9781401211868'); 26 await alovestoryisbd save(); 27 28 const alovestorybook = new parse object('book'); 29 alovestorybook set('title', 'a love story'); 30 alovestorybook set('publisher', acaciapublishings); 31 alovestorybook set('publishingdate', new date('05/07/1998')); 32 alovestorybook set('isbd', alovestoryisbd); 33 const bookarelation = alovestorybook relation("authors"); 34 bookarelation add(aaronwriter); 35 await alovestorybook save(); 36 alovestoryisbd set('book', alovestorybook topointer()); 37 await alovestoryisbd save(); 38 39 const benevolentelvesisbd = new parse object('isbd'); 40 benevolentelvesisbd set('isbn', '9781401211868'); 41 await benevolentelvesisbd save(); 42 43 const benevolentelvesbook = new parse object('book'); 44 benevolentelvesbook set('title', 'benevolent elves'); 45 benevolentelvesbook set('publisher', birchdistributions); 46 benevolentelvesbook set('publishingdate', new date('11/31/2008')); 47 benevolentelvesbook set('isbd', benevolentelvesisbd); 48 const bookbrelation = benevolentelvesbook relation("authors"); 49 bookbrelation add(beatricenovelist); 50 await benevolentelvesbook save(); 51 benevolentelvesisbd set('book', benevolentelvesbook topointer()); 52 await benevolentelvesisbd save(); 53 54 const canyoubelieveitisbd = new parse object('isbd'); 55 canyoubelieveitisbd set('isbn', '9781401211868'); 56 await canyoubelieveitisbd save(); 57 58 const canyoubelieveitbook = new parse object('book'); 59 canyoubelieveitbook set('title', 'can you believe it?'); 60 canyoubelieveitbook set('publisher', birchdistributions); 61 canyoubelieveitbook set('publishingdate', new date('08/21/2018')); 62 canyoubelieveitbook set('isbd', canyoubelieveitisbd); 63 const bookcrelation = canyoubelieveitbook relation("authors"); 64 bookcrelation add(aaronwriter); 65 bookcrelation add(caseycolumnist); 66 await canyoubelieveitbook save(); 67 canyoubelieveitisbd set('book', canyoubelieveitbook topointer()); 68 await canyoubelieveitisbd save(); 69 70 // book store 71 const booksoflovestore = new parse object('bookstore'); 72 booksoflovestore set('name', 'books of love'); 73 const bookstorearelation = booksoflovestore relation("books"); 74 bookstorearelation add(alovestorybook); 75 await booksoflovestore save(); 76 77 const fantasybooksstore = new parse object('bookstore'); 78 fantasybooksstore set('name', 'fantasy books'); 79 const bookstorebrelation = fantasybooksstore relation("books"); 80 bookstorebrelation add(benevolentelvesbook); 81 await fantasybooksstore save(); 82 83 const generalbooksstore = new parse object('bookstore'); 84 generalbooksstore set('name', 'general books'); 85 const bookstorecrelation = generalbooksstore relation("books"); 86 bookstorecrelation add(alovestorybook); 87 bookstorecrelation add(canyoubelieveitbook); 88 await generalbooksstore save(); 3 สอบถามข้อมูล เมื่อฐานข้อมูลมีข้อมูลตัวอย่างให้ทำงานด้วยแล้ว เราจะเริ่มดำเนินการสอบถามประเภทต่างๆ ที่เกี่ยวข้องกับความสัมพันธ์ที่กล่าวถึงก่อนหน้านี้ การสอบถามที่เกี่ยวข้องกับความสัมพันธ์ 1 1 เมื่อมีประเภทข้อมูลสองประเภทที่แชร์ความสัมพันธ์ 1 1 (ในกรณีนี้คือ หนังสือ หนังสือ และ isbd isbd ) เราสามารถดึงข้อมูลจากอีกฝ่ายหนึ่งได้ดังนี้ วิธีที่เราใช้ในการดำเนินการความสัมพันธ์ใน หนังสือ หนังสือ ช่วยให้เราสามารถดึงข้อมูลที่เกี่ยวข้อง isbd isbd ได้โดยการเรียกใช้ include( ) include( ) ในการสอบถาม มาดึง isbd isbd จากหนังสือ a love story 1 let alovestorybookquery = book query("title" == "a love story") include("isbd") // note how we include the isbd with the include( ) method 2 3 let book = try? alovestorybookquery first() // retrieves synchronously the book including its isbd 4 5 alovestorybookquery first { result in // retrieves asynchronously the book including its isbd 6 // handle the result (of type result\<book, parseerror>) 7 } ในทางกลับกัน การสอบถามเพื่อดึงข้อมูล หนังสือ หนังสือ ที่เกี่ยวข้องกับ isbd isbd ที่กำหนดนั้นดำเนินการในลักษณะดังนี้ โดยการดูที่การดำเนินการของ isbd isbd , เราสังเกตว่าความสัมพันธ์ถูกแทนที่ด้วย หนังสือ หนังสือ (ประเภท pointer<หนังสือ> pointer<หนังสือ> ) ตัวชี้นี้ให้ชุดของวิธีการและคุณสมบัติเพื่อดึงข้อมูลเกี่ยวกับวัตถุที่มันชี้ไป ในที่นี้เราจะเรียกใช้ fetch( ) fetch( ) ในคุณสมบัติ หนังสือ หนังสือ เพื่อดึงข้อมูลที่เกี่ยวข้อง หนังสือ หนังสือ 1 let someisbd isbd 2 3 let book book? = try? someisbd book? fetch() // retrieves synchronously the book asscociated to someisbd 4 5 someisbd book? fetch { result in // retrieves asynchronously the book asscociated to someisbd 6 // handle the result (of type result\<book, parseerror>) 7 } เราควรสังเกตว่าการนำไปใช้สำหรับ 1 1 ความสัมพันธ์นี้ไม่ใช่เอกลักษณ์ ขึ้นอยู่กับกรณีการใช้งานของคุณ คุณสามารถนำเสนอ 1 1 ความสัมพันธ์ในหลายวิธี การค้นหาที่เกี่ยวข้องกับความสัมพันธ์ 1\ n ในสถานการณ์ที่เราต้องการค้นหาหนังสือทั้งหมดที่เผยแพร่โดยผู้จัดพิมพ์ที่กำหนด เราจำเป็นต้องดึงข้อมูลผู้จัดพิมพ์ก่อน ตัวอย่างเช่น เราจะดึงข้อมูลวัตถุที่เกี่ยวข้องกับผู้จัดพิมพ์ acacia publishings ขึ้นอยู่กับสถานการณ์ กระบวนการนี้อาจแตกต่างกัน 1 do { 2 // using the object's objectid 3 let acaciapublishings = try publisher(objectid "some object id") fetch() 4 5 // or 6 // using a query 7 let acaciapublishings = try publisher query("name" == "acacia publishings") first() // returns (synchronously) the first publisher with name 'acacia publishings' the constraint is constructed using the == operator provided by the parseswift sdk 8 9 // to be completed below 10 } catch { 11 // hanlde the error (of type parseerror) 12 } ตอนนี้ที่เรามีการเข้าถึง acaciapublishings acaciapublishings , เราสามารถสร้างการค้นหาเพื่อดึงหนังสือที่เกี่ยวข้องได้ เราจะดำเนินการสร้างการค้นหาโดยการสร้างคลาส query\<book> query\<book> ในกรณีนี้ คลาสนี้ถูกสร้างขึ้นโดยใช้วิธีสถิต query( ) query( ) ที่จัดเตรียมโดย book book วัตถุ อาร์กิวเมนต์ (แบบหลายค่า) สำหรับวิธีนี้คือ queryconstraint queryconstraint วัตถุมาตรฐาน ดังนั้น หนังสือที่เรากำลังมองหาจึงถูกดึงออกมาด้วยโค้ดต่อไปนี้ 1 do { 2 let acaciapublishings = try publisher query("name" == "acacia publishings") first() // returns the first publisher with name 'acacia publishings' 3 4 let constraint queryconstraint = try "publisher" == publisher 5 let query = book query(constraint) // creates the query to retrieve all book objects where its publisher field equalt to 'acaciapublishings' 6 7 let books \[book] = try query find() // executes the query synchronously 8 9 // books should contain only one element the book 'a love story' 10 } catch { 11 // hanlde the error (of type parseerror) 12 } การนำเสนอแบบอะซิงโครนัสสำหรับโค้ดด้านบนอาจเขียนในลักษณะดังต่อไปนี้ 1 // we retrieve the publisher with name 'acacia publishings' 2 publisher query("name" == "acacia publishings") first { result in 3 switch result { 4 case success(let publisher) 5 guard let constraint queryconstraint = try? "publisher" == publisher else { fatalerror() } 6 7 // then, we retrieve the books with the corresponding constraint 8 book query(constraint) find { result in 9 switch result { 10 case success(let books) 11 // books should contain only one element the book 'a love story' 12 break 13 case failure(let error) 14 // handle the error (of type parseerror) 15 break 16 } 17 } 18 19 case failure(let error) 20 // handle the error (of type parseerror) 21 break 22 } 23 } การค้นหาที่เกี่ยวข้องกับความสัมพันธ์ m\ n (กรณีที่ 1) เพื่อแสดงกรณีนี้ เราพิจารณาสถานการณ์ต่อไปนี้; เราต้องการแสดงรายการร้านค้าทั้งหมดที่มีหนังสือที่ตีพิมพ์หลังจากวันที่กำหนด (เช่น 01/01/2010 ) ก่อนอื่นเราต้องการการค้นหาชั่วคราวเพื่อเลือกหนังสือ จากนั้นเราจะสร้างการค้นหาหลักเพื่อแสดงรายการร้านค้า ดังนั้นเราจึงเตรียมการค้นหาครั้งแรกสำหรับหนังสือ 1 let booksquery = book query("publishingdate" > date(string "01/01/2010")) // we construct the date constraint using the > operator provided by the parseswift sdk 2 3 do { 4 let books = try booksquery find() 5 // to be completed below 6 } catch let error as parseerror { 7 // handle any potential error 8 } catch { 9 // handle any potential error 10 } เราจึงสร้างการค้นหาของร้านค้าโดยใช้ booksquery booksquery ’s results วิธีการ containedin( array ) containedin( array ) คืนค่าข้อจำกัดที่เราต้องการสำหรับกรณีนี้ 1 let booksquery = book query("publishingdate" > date(string "01/01/2010")) // we construct the date constraint using the > operator provided by the parseswift sdk 2 3 do { 4 let books = try booksquery find() 5 6 // here is where we construct the stores' query with the corresponding constraint 7 let storesquery = bookstore query(try containedin(key "books", array books)) 8 9 let stores = try storesquery find() 10 11 // stores should containt only one element the 'general books' bookstore 12 } catch let error as parseerror { 13 // handle any potential error 14 } catch { 15 // handle any potential error 16 } ในลักษณะเดียวกัน เราสามารถดำเนินการกระบวนการนี้แบบอะซิงโครนัส 1 let booksquery = book query("publishingdate" > date(string "01/01/2010")) // we construct the date constraint using the > operator provided by the parseswift sdk 2 3 booksquery find { result in 4 switch result { 5 case success(let books) 6 guard let constraint = try? containedin(key "books", array books) else { fatalerror() } 7 let storesquery = bookstore query(constraint) 8 9 storesquery find { result in 10 switch result { 11 case success(let stores) 12 13 case failure(let error) 14 // handle the error (of type parseerror) 15 } 16 } 17 case failure(let error) 18 // handle the error (of type parseerror) 19 } 20 } การค้นหาที่เกี่ยวข้องกับความสัมพันธ์ m\ n (กรณีที่ 2) สมมติว่าเราต้องการเลือกทุกร้านค้าที่มีหนังสือที่เขียนโดยผู้เขียนที่กำหนด เช่น aaron writer เพื่อให้บรรลุเป้าหมายนี้ เราต้องการคำถามเพิ่มเติมอีกสองคำถาม คำถาม ( query\<author> query\<author> ) เพื่อดึงวัตถุที่เกี่ยวข้องกับผู้เขียน aaron writer คำถาม ( query\<book> query\<book> ) เพื่อเลือกหนังสือทั้งหมดที่เขียนโดย aaron writer คำถามหลัก ( query\<bookstore> query\<bookstore> ) เพื่อเลือกร้านค้าที่เรากำลังมองหา ขั้นตอนในการดำเนินการคำถามเหล่านี้มีความคล้ายคลึงกับคำถามก่อนหน้านี้ 1 let authorquery = author query("name" == "aaron writer") // the first query to retrieve the data object associated to 'aaron writer' 2 3 do { 4 let aaronwriter = try authorquery first() 5 6 let booksquery = book query(try containedin(key "authors", array \[aaronwriter])) // the second query to retrieve the books written by 'aaron writer' 7 8 let books = try booksquery find() 9 10 let storesquery = bookstore query(try containedin(key "books", array books)) // the main query to select the stores where the author ('aaron writer') has his books available 11 12 let stores = try storesquery find() 13 14 // stores should contain two items 'books of love' and 'general books' 15 } catch let error as parseerror { 16 // handle the error 17 } catch { 18 // handle the error 19 } บทสรุป ด้วย parseswift sdk parseswift sdk , เราสามารถสร้างคำถามเชิงสัมพันธ์ที่อนุญาตให้เราเลือกไอเท็มตามประเภทของความสัมพันธ์ที่พวกเขามีกับประเภทข้อมูลอื่น