-
חיפשתי ברשת פונקציה לחישוב גיל, הנובע מתאריך לידה בפורמט Date, וזה מה שמצאתי.
setAge(BirthDate:Date) { var ageDifMs = Date.now() - BirthDate.getTime(); var ageDate = new Date(ageDifMs); // miliseconds from epoch return Math.abs(ageDate.getUTCFullYear() - 1970); }
בדקתי ואכן הפונקציה מחזירה Number של הגיל.
אני צריך לעדכן מידי פעם גילאים שהשתנו ולשם כך אני קורא לפונקציה הבאה:
updateAllAges(){ for (let i = 0; i < this.DB.length; i++) { this.DB[i].Age =DBtype.setAge(this.DB[i].BirthDate); } }
כאשר setAge היא פונקציה שהובאה למעלה, אלא שהפכתי אותה לסטטית ומשויכת למחלקה DBtype.
הוספתי לכפתור את ה-Click הבא:
<button (click)="updateAllAges() type="button">Click Me!</button>
ברמת הקומפילציה הכל עובד חלק (אנגולר), אבל כאשר אני משנה את התאריך ולוחץ על הכפתור אני מקבל בקונסול של הדפדפן את השגיאה הבאה:
core.js:6456 ERROR TypeError: BirthDate.getTime is not a function at Function.setAge (myFile.ts:3)
שאלתי היא למה getTime אינו פונקציה, וכיצד ניתן לפתור זאת?
תודה רבה לחברי האתר הנהדרים. -
@yyy למיטב הבנתי
this.DB
מייצג מערך של אובייקטים שמקורם במסד נתונים כלשהו ושמכך ניתן להניח שthis.DB[i].BirthDate
מייצג תאריך בפורמט JSON.
כדי לעשות שימוש במתודות שנמצאות בPrototype של Date, נדרשת תחילה המרה של התאריך שמיוצג כString לאובייקט מסוג Date. -
@רפאל אוף!!
אני מוכן לקבל את הרעיון ש TS בסך הכל נועדה לפתור חלק מהבעיות בזמן הכתיבה, ושבזמן הריצה אנחנו עדיין במערב הפרוע של JS, אבל בכל זאת הודעת השגיאה הייתה מטעה אותי, ומסתמא הייתי שובר את הראש זמן רב עד שהייתי קולט את הבעיה, TS רק גורמת להסתיר עוד יותר את השגיאה... -
@odeddvir הענין הזה הוא אחד הדברים שמפריעים לי במיוחד ב-TS. כלומר, הוא מאוד עוזר לך לשלוט בקוד שאתה כותב (כשאני כותב TS, הקוד באמת יותר מסודר לי בראש בצורה די משמעותית), אבל מצד שני, כשאתה עובר לעולם המעשה אתה צריך באמת לשים לב טוב מאוד שאתה לא משתמש בקוד באופן לא תקין (הדוגמה למעלה ממחישה זאת היטב), שאז מנגנון ה-Types והקומפילציה לא ממש עוזר.
-
@odeddvir ניתן לטפל בעניין ע"י שימוש נכון בתחביר.
אפשרות אחת היא שימוש בUnion type, לדוגמא string | Date מייצג ערך שיכול להיות Date או String.
interface Person { name: string; dob: string | Date; }
Union type מאפשר לך לגשת למאפיינים שמשותפים לכל הסוגים המרכיבים את הUnion ללא כל בעיה, אולם כדי לגשת למאפייינים היחודיים לאחד מהסוגים המרכיבים את הUnion נדרשת המרה של הUnion לסוג הספיציפי הדרוש ע"י בדיקת JS שהסוג אכן תואם והמהדר של Typescript ידייק וישנה את הסוג המוצהר (הUnion) לסוג הספיציפי, לדוגמא:
person.dob.getDay() // Error: getDay does not exist on type string if (typeof person.dob === 'object') { person.dob.getDay() // Date בתוך התנאי הסוג הוא }
דרך נוספת היא ליצור שני מודלים, הראשון יייצג את הערך היוצא לשרת (Outbound), והשני יייצג את הערך שחוזר מהשרת (Inbound) לדוגמא:
interface PersonBase { name: string; } interface ResponsePerson extends PersonBase { dbo: string; } class Person implements PersonBase { constructor(public name: string, public dbo: Date) { } public static fromPersonResponse({ name, dbo }: ResponsePerson): Person { return new Person(name, new Date(dbo)) } }
-
@רפאל אמר בgetTime אינו פונקציה:
@yyy על מנת להוכיח שאני טועה (כתמיד) נסה להוסיף את השורה הבאה לקוד, ווודא שהתוצאה שלילית.
console.log(typeof this.DB[0].BirthDate === 'string')
זה מחזיר true.
אני מבין שהשגיאה היא בשורה
var ageDifMs = Date.now() - BirthDate.getTime();
בכדי לנסות לפתור את הבעיה
עשיתי את הדבר הבאvar ageDifMs = Date.now() - JSON.parse(BirthDate).getTime();
בקיצור להפוך את זה לאובייקט.
הבעיה שכאן כבר יש שגיאה שהטייפ לא מתאים:Argument of type 'Date' is not assignable to parameter of type 'string'
כי הצהרתי שזה Date, ועכשיו אני מתייחס לזה כ-string.
בקיצור בלגן.
איך מתקדמים? -
@odeddvir אמר בgetTime אינו פונקציה:
הייתה לי הרגשה שזה קשור לתוהו ובוהו של JS
@odeddvir אמר בgetTime אינו פונקציה:
ושבזמן הריצה אנחנו עדיין במערב הפרוע של JS
-
@רפאל סתם להחכים, הרי תיאורטית TypeScript יכלו לייצר קוד JS טהור שבודק התאמה לטיפוס (אולי זה יקר מבחינת ביצועים) וזורק שגיאה, זה לא הפואנטה של TS בכלל, או שיש מקרים שהקו'ד המיוצר נותן טיפה קשיחות ביחס לקוד JS רגיל (נפקא מינה בדיוק למקרים כאלה שההודעות שגיאה יהיו חכמות יותר)?