منذ ظهور أطر عمل الجافاسكريبت، أصبحت عبارتي Tw-way data binding و One-way data binding من أكثر العبارات والمصطلحات التقنية التي نسمعها ونقرؤها في الدورات والدروس المتعلقة بهذه الأطر (مثال: Angular و React).
ما معنى Data Binding ؟
الترجمة اللغوية الحرفية لمصطلح Data binding هو ربط البيانات. واصطلاحا هو الآلية التي يتم بها تنظيم ربط أو تدفق البيانات بين نموذج البيانات (Model) وواجهة المستخدم (User Interface).
المقارنة بين آليتي تدفق البيانات أحادي الإتجاه (One-way) وثنائي الإتجاه (Two-way) طفت على السطح وبقوة في الأربع سنوات الأخيرة في ظل المقارنات بين إطار العمل Angular و مكتبة React.js. الأول يتبنى آلية Two-way data binding والثاني يعتمد آلية One-way data binding.
تدفق البيانات ثنائي الإتجاه في إطار العمل Angular
لطالما عرف إطار العمل أنغولار باعتماده على نمط Two-way data binding منذ إصداره الأول. هذا النمط ببساطة يتجلى في أن أي تغيير يطرأ على قيم نماذج البيانات (Data Model). يؤدي ضمنيا لتغيير واجهة المستخدم (تحديثها) والعكس صحيح، أي أن الزائر عندما يقوم بتغيير قيمة حقل ما (Input) فإن ذلك يعني تغييرا لنموذج البيانات المقابل لهذا الحقل.
لنأخذ هذا المثال البسيط لتوضيح المسألة أكثر، ليكن هذاهو الكود الخاص بواجهة المستخدم :
<div ng-app="myApp" ng-controller="myController"> Name: <input ng-model="name"> <h1>{{name}}</h1> </div>
1
2
3
4
<div ng-app="myApp" ng-controller="myController">
Name: <input ng-model="name">
<h1>{{name}}</h1>
</div>
وهذا الكود الخاص بال Controller :
function MyController($scope) { $scope.name = "Ahmed"; }
1
2
3
function MyController($scope) {
$scope.name = "Ahmed";
}
كل تغيير يطرأ على مستوى واجهة المستخدم من خلال الحقل <input> سيقابله تغيير تلقائي في قيمة $scope.name داخل ال Controller، وبالتالي سنرى أن قيمة {{name}} داخل الوسم <h1> تتغير بدورها.
تدفق البيانات أحادي الإتجاه في React.js
في عالم React.js تختلف الأمور قليلا. في المثال أعلاه، نستطيع كما قلنا إجراء تحديثات على مستوى HTML ليتم مزامنتها أوتوماتيكيا مع كود الجافاسكريبت (Data Model).
هذا لا يحدث مع مكتبة React.js.
بكل تأكيد نستطيع أخذ قيمة الحقل وتحديث حالة المكون (Component) أو واجهة المستخدم، ولكن الإختلاف أن العملية هنا ليست ضمنية (Implecit) أو تلقائية لأنه يجب علينا أن نقوم بذلك بشكل صريح (Explecit). الطريقة هي أننا نقوم بالإستماع لحدث onChange الذي يتم إطلاقه عندما يطرأ تغيير على قيمة <input> وبعد ذلك نقوم بتغيير حالة واجهة المستخدم (UI State) بشكل يدوي عن طريق الواجهة البرمجية التي تتيحها لنا React.js.
لاحظوا المثال التالي :
class App extends React.Component { constructor() { super(); this.handleChange = this.handleChange.bind(this); this.state = { name: 'Ahmed', }; } handleChange(event) { this.setState({ name: event.target.value, }); } render() { return ( <div> <input type="text" value={this.state.name} onChange={this.handleChange} /> <h1>{this.state.name}</h1> </div> ); } }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
class App extends React.Component {
constructor() {
super();
this.handleChange = this.handleChange.bind(this);
this.state = {
name: 'Ahmed',
};
}
handleChange(event) {
this.setState({
name: event.target.value,
});
}
render() {
return (
<div>
<input type="text" value={this.state.name} onChange={this.handleChange} />
<h1>{this.state.name}</h1>
</div>
);
}
}
نقوم باستخدام الدالة setState() لتغيير قيمة name. وكما نعلم فإنه كلما قمنا بتغيير الحالة عن طريق setState() فإن دالة render الخاصة بالمكون يتم إعادة استدعاؤها من قبل React.js من أجل تحديث واجهة المستخدم
React.js حريص على أن يكون State هو “المصدر الوحيد للحقيقة” (Single source of truth)، فكل ما تقوم به واجهة المستخدم هو إطلاق أحداث Events عندما يطرأ تغيير عليها حتى يعلم المكون متى عليه أن يقوم بتحديث نفسه بواسطة أسلحته الخاصة (()setState).
يمكن لهذا الرسم أن يجعل الأمور أكثر وضوحا:
بينما في حالة AngularJs، كانت الأمور ستبدو على هذا النحو :
النهاية
أرجو أن يكون هذا الموضوع أكثر وضوحا بالنسبة لكم الآن.
لم أرد الخوض في بعض التفاصيل كثيرا، كأن أشرح مثلا أن إصدار Angular 2 فما فوق يدعم كلا النمطين، لأن هدفي الوحيد من خلال هذا الدرس هو شرح الفرق بين هذين الشكلين من الآليات التي تعتمدها معظم أطر عمل الجافاسكريبت لتنظيم العلاقة بين واجهة المستخدم والبيانات. وإنما استخدمت مثالي React.js و AngularJS فقط لشعبيتهما عند معظم مطوري الجافاسكريبت حول العالم.
ما معنى Data Binding ؟
الترجمة اللغوية الحرفية لمصطلح Data binding هو ربط البيانات. واصطلاحا هو الآلية التي يتم بها تنظيم ربط أو تدفق البيانات بين نموذج البيانات (Model) وواجهة المستخدم (User Interface).
المقارنة بين آليتي تدفق البيانات أحادي الإتجاه (One-way) وثنائي الإتجاه (Two-way) طفت على السطح وبقوة في الأربع سنوات الأخيرة في ظل المقارنات بين إطار العمل Angular و مكتبة React.js. الأول يتبنى آلية Two-way data binding والثاني يعتمد آلية One-way data binding.
تدفق البيانات ثنائي الإتجاه في إطار العمل Angular
لطالما عرف إطار العمل أنغولار باعتماده على نمط Two-way data binding منذ إصداره الأول. هذا النمط ببساطة يتجلى في أن أي تغيير يطرأ على قيم نماذج البيانات (Data Model). يؤدي ضمنيا لتغيير واجهة المستخدم (تحديثها) والعكس صحيح، أي أن الزائر عندما يقوم بتغيير قيمة حقل ما (Input) فإن ذلك يعني تغييرا لنموذج البيانات المقابل لهذا الحقل.
لنأخذ هذا المثال البسيط لتوضيح المسألة أكثر، ليكن هذاهو الكود الخاص بواجهة المستخدم :
<div ng-app="myApp" ng-controller="myController"> Name: <input ng-model="name"> <h1>{{name}}</h1> </div>
1
2
3
4
<div ng-app="myApp" ng-controller="myController">
Name: <input ng-model="name">
<h1>{{name}}</h1>
</div>
وهذا الكود الخاص بال Controller :
function MyController($scope) { $scope.name = "Ahmed"; }
1
2
3
function MyController($scope) {
$scope.name = "Ahmed";
}
كل تغيير يطرأ على مستوى واجهة المستخدم من خلال الحقل <input> سيقابله تغيير تلقائي في قيمة $scope.name داخل ال Controller، وبالتالي سنرى أن قيمة {{name}} داخل الوسم <h1> تتغير بدورها.
تدفق البيانات أحادي الإتجاه في React.js
في عالم React.js تختلف الأمور قليلا. في المثال أعلاه، نستطيع كما قلنا إجراء تحديثات على مستوى HTML ليتم مزامنتها أوتوماتيكيا مع كود الجافاسكريبت (Data Model).
هذا لا يحدث مع مكتبة React.js.
بكل تأكيد نستطيع أخذ قيمة الحقل وتحديث حالة المكون (Component) أو واجهة المستخدم، ولكن الإختلاف أن العملية هنا ليست ضمنية (Implecit) أو تلقائية لأنه يجب علينا أن نقوم بذلك بشكل صريح (Explecit). الطريقة هي أننا نقوم بالإستماع لحدث onChange الذي يتم إطلاقه عندما يطرأ تغيير على قيمة <input> وبعد ذلك نقوم بتغيير حالة واجهة المستخدم (UI State) بشكل يدوي عن طريق الواجهة البرمجية التي تتيحها لنا React.js.
لاحظوا المثال التالي :
class App extends React.Component { constructor() { super(); this.handleChange = this.handleChange.bind(this); this.state = { name: 'Ahmed', }; } handleChange(event) { this.setState({ name: event.target.value, }); } render() { return ( <div> <input type="text" value={this.state.name} onChange={this.handleChange} /> <h1>{this.state.name}</h1> </div> ); } }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
class App extends React.Component {
constructor() {
super();
this.handleChange = this.handleChange.bind(this);
this.state = {
name: 'Ahmed',
};
}
handleChange(event) {
this.setState({
name: event.target.value,
});
}
render() {
return (
<div>
<input type="text" value={this.state.name} onChange={this.handleChange} />
<h1>{this.state.name}</h1>
</div>
);
}
}
نقوم باستخدام الدالة setState() لتغيير قيمة name. وكما نعلم فإنه كلما قمنا بتغيير الحالة عن طريق setState() فإن دالة render الخاصة بالمكون يتم إعادة استدعاؤها من قبل React.js من أجل تحديث واجهة المستخدم
React.js حريص على أن يكون State هو “المصدر الوحيد للحقيقة” (Single source of truth)، فكل ما تقوم به واجهة المستخدم هو إطلاق أحداث Events عندما يطرأ تغيير عليها حتى يعلم المكون متى عليه أن يقوم بتحديث نفسه بواسطة أسلحته الخاصة (()setState).
يمكن لهذا الرسم أن يجعل الأمور أكثر وضوحا:
بينما في حالة AngularJs، كانت الأمور ستبدو على هذا النحو :
النهاية
أرجو أن يكون هذا الموضوع أكثر وضوحا بالنسبة لكم الآن.
لم أرد الخوض في بعض التفاصيل كثيرا، كأن أشرح مثلا أن إصدار Angular 2 فما فوق يدعم كلا النمطين، لأن هدفي الوحيد من خلال هذا الدرس هو شرح الفرق بين هذين الشكلين من الآليات التي تعتمدها معظم أطر عمل الجافاسكريبت لتنظيم العلاقة بين واجهة المستخدم والبيانات. وإنما استخدمت مثالي React.js و AngularJS فقط لشعبيتهما عند معظم مطوري الجافاسكريبت حول العالم.