الأربعاء، 17 أكتوبر 2012

التسلسل في جافا

مقدمة لتسلسل الكائن "Object Serialization"


يتم استخدام تسلسل الكائن "Object Serialization" في جافا  لتمرير أو بمعنى آخر تحويل كائنات جافا "Java objects" إلى ملف أو قاعدة بيانات أو عملية  أو شبكة أو أي نظام آخر. التسلسل يسوي الكائنات إلى وحدات البايت المتدفقة "stream of bytes" مرتبة أو متسلسلة. ويمكن بعد ذلك لوحدات البايت المتدفقة "stream of bytes" أن تُقرأ في وقت لاحق، أو في بيئة أخرى، لإعادتها الى الكائنات الأصلية.
تسلسل جافا لا يمكن أن يتم لحقول عابرة أو ثابتة. بالمناسبة الحقل العابر "field transient" يمنع " object state " من ان يتم كتابتها في التدفق "stream" و من استعادتها عند عملية إلغاء التسلسل "deserialization". جافا يقدم فئات لدعم كتابة الكائنات في التدفق "stream" و استعادة الكائنات من التدفق "stream". يمكن فقط للكائنات التي تدعم الواجهة  java.io.Serializable  أو الواجهة  java.io.Externalizable  الكتابة في التدفق "stream".

الواجهة العامة Serializable


الواجهة Serializable ليس لديها مناهج أو حقول. (واجهة ماركر)
فقط كائنات الفئات التي تقوم بتنفيذ واجهة java.io.Serializable يمكنها التسلسل أو فك التسلسل

الحقول العابرة "Transient Fields" والتسلسل في جافا "Java Serialization" 


الكلمة الأساسية العابرة "transient keyword" هي تعديل يُطبق على المتغيرات المثيلة "instance variables" في الفئة. وهي تحدد أن المتغير ليست جزءا من الحالة "state" المستمرة للكائن، وبالتالي لن يتم حفظها خلال التسلسل.
يمكنك استخدام الكلمة العابرة لوصف المتغيرات المؤقتة ، أو المتغيرات التي تحتوي على المعلومات المحلية، مثل معرف العملية "process ID" أو مرور الزمن "time lapse".

الإدخال والإخراج لجداول كائن "Object Streams"


ObjectOutputStream هي فئة تدفق الإخراج الأولية التي تطبق الواجهة ObjectOutput لتسلسل الكائنات. ObjectInputStream هي فئة تدفق الإدخال الأولية التي تطبق الواجهة ObjectInput deserializing لفك التسلسل للكائنات.
هذه الجداول ذات المستوى العال مرطبة  بتدفق السلاسل ذات المستوى المنخفض ، مثل FileInputStream أو FileOutputStream.
الجداول ذات المستوى المنخفض "low-level streams" تقوم بمعالجة بايت من البيانات "bytes of data". المنهج writeObject يحفظ حالة الفئة عن طريق كتابة الحقول الفردية في ObjectOutputStream. يستخدم المنهج readObject لإلغاء تسلسل كائن من تدفق إدخال الكائن "object input stream".

الحالة 1 : فيما يلي مثال يوضح تسلسل الكائن في ملف


PersonDetails هي الفئة النوات التي تطبق الواجهة Serializable

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
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
import java.io.Serializable;
 
public class PersonDetails implements Serializable {
 
 
 
      private String name;
 
      private int age;
 
      private String sex;
 
      public PersonDetails(String name, int age, String sex) {
 
            this.name = name;
 
            this.age = age;
 
            this.sex = sex;
 
      }
 
      public int getAge() {
 
            return age;
 
      }
 
      public void setAge(int age) {
 
            this.age = age;
 
      }
 
      public String getName() {
 
            return name;
 
      }
 
      public void setName(String name) {
 
            this.name = name;
 
      }
 
      public String getSex() {
 
            return sex;
 
      }
 
      public void setSex(String sex) {
 
            this.sex = sex;
 
      }
 
}


GetPersonDetails هي الفئة التي تستخدم لإلغاء تسلسل الكائن من الملف (person.txt).

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
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
import java.io.FileInputStream;
 
import java.io.IOException;
 
import java.io.ObjectInputStream;
 
import java.util.ArrayList;
 
import java.util.List;
 
 
 
public class GetPersonDetails {
 
 
 
      public static void main(String[] args) {
 
            String filename = "person.txt";
 
            List pDetails = null;
 
            FileInputStream fis = null;
 
            ObjectInputStream in = null;
 
            try {
 
                  fis = new FileInputStream(filename);
 
                  in = new ObjectInputStream(fis);
 
                  pDetails = (ArrayList) in.readObject();
 
                  in.close();
 
            } catch (IOException ex) {
 
                  ex.printStackTrace();
 
            } catch (ClassNotFoundException ex) {
 
                  ex.printStackTrace();
 
            }
 
            // print out the size
 
            System.out.println("Person Details Size: " + pDetails.size());
 
            System.out.println();
 
      }
 
}



PersonPersist هي الفئة التي تستخدم لتسلسل الكائن في الملف (person.txt).

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
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
public class PersonPersist {
 
 
 
      public static void main(String[] args) {
 
            String filename = "person.txt";
 
            PersonDetails person1 = new PersonDetails("hemanth", 10, "Male");
 
            PersonDetails person2 = new PersonDetails("bob", 12, "Male");
 
            PersonDetails person3 = new PersonDetails("Richa", 10, "Female");
 
            List list = new ArrayList();
 
            list.add(person1);
 
            list.add(person2);
 
            list.add(person3);
 
            FileOutputStream fos = null;
 
            ObjectOutputStream out = null;
 
            try {
 
                  fos = new FileOutputStream(filename);
 
                  out = new ObjectOutputStream(fos);
 
                  out.writeObject(list);
 
                  out.close();
 
                  System.out.println("Object Persisted");
 
            } catch (IOException ex) {
 
                  ex.printStackTrace();
 
            }
 
      }
 
}

الحالة 2 : فيما يلي مثال يوضح تسلسل الكائن في  قاعدة البيانات


PersonDetails يبقى نفسه كما هو مبين أعلاه
GetPersonDetails يبقى نفسه كما هو مبين أعلاه
1
2
3
4
5
6
7
8
9
Create SerialTest Table
 
create table SerialTest(
 
name BLOB,
 
viewname VARCHAR2(30)
 
);

إنشاء جدول SerialTest

1
2
3
4
5
6
7
8
9
Create SerialTest Table
 
create table SerialTest(
 
name BLOB,
 
viewname VARCHAR2(30)
 
);

PersonPersist هي الفئة التي تستخدم لتسلسل الكائن في جدول قاعدة البيانات SerialTest .


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
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
import java.io.ByteArrayInputStream;
 
import java.io.ByteArrayOutputStream;
 
import java.io.IOException;
 
import java.io.ObjectInputStream;
 
import java.io.ObjectOutputStream;
 
import java.sql.Connection;
 
import java.sql.DriverManager;
 
import java.sql.PreparedStatement;
 
import java.sql.ResultSet;
 
import java.sql.SQLException;
 
import java.sql.Statement;
 
public class PersonPersist {
 
 
 
      static String userid = "scott", password = "tiger";
 
      static String url = "jdbc:odbc:bob";
 
      static int count = 0;
 
      static Connection con = null;
 
      public static void main(String[] args) {
 
            Connection con = getOracleJDBCConnection();
 
            PersonDetails person1 = new PersonDetails("hemanth", 10, "Male");
 
            PersonDetails person2 = new PersonDetails("bob", 12, "Male");
 
            PersonDetails person3 = new PersonDetails("Richa", 10, "Female");
 
            PreparedStatement ps;
 
            try {
 
                  ps = con
 
                              .prepareStatement("INSERT INTO SerialTest VALUES (?, ?)");
 
                  write(person1, ps);
 
                  ps.execute();
 
                  write(person2, ps);
 
                  ps.execute();
 
                  write(person3, ps);
 
                  ps.execute();
 
                  ps.close();
 
                  Statement st = con.createStatement();
 
                  ResultSet rs = st.executeQuery("SELECT * FROM SerialTest");
 
                  while (rs.next()) {
 
                        Object obj = read(rs, "Name");
 
                        PersonDetails p = (PersonDetails) obj;
 
                        System.out.println(p.getName() + "\t" + p.getAge() + "\t"
 
                                    + p.getSex());
 
                  }
 
                  rs.close();
 
                  st.close();
 
            } catch (Exception e) {
 
            }
 
      }
 
      public static void write(Object obj, PreparedStatement ps)
 
                  throws SQLException, IOException {
 
            ByteArrayOutputStream baos = new ByteArrayOutputStream();
 
            ObjectOutputStream oout = new ObjectOutputStream(baos);
 
            oout.writeObject(obj);
 
            oout.close();
 
            ps.setBytes(1, baos.toByteArray());
 
            ps.setInt(2, ++count);
 
      }
 
      public static Object read(ResultSet rs, String column)
 
                  throws SQLException, IOException, ClassNotFoundException {
 
            byte[] buf = rs.getBytes(column);
 
            if (buf != null) {
 
                  ObjectInputStream objectIn = new ObjectInputStream(
 
                              new ByteArrayInputStream(buf));
 
                  return objectIn.readObject();
 
            }
 
            return null;
 
      }
 
      public static Connection getOracleJDBCConnection() {
 
            try {
 
                  Class.forName("sun.jdbc.odbc.JdbcOdbcDriver");
 
            } catch (java.lang.ClassNotFoundException e) {
 
                  System.err.print("ClassNotFoundException: ");
 
                  System.err.println(e.getMessage());
 
            }
 
            try {
 
                  con = DriverManager.getConnection(url, userid, password);
 
            } catch (SQLException ex) {
 
                  System.err.println("SQLException: " + ex.getMessage());
 
            }
 
            return con;
 
      }
 
}


الحالة 3 : فيما يلي مثال يوضح تسلسل الكائن في  قاعدة البيانات  باستخدام قاعدة التشفير 64


PersonDetails يبقى نفسه كما هو مبين أعلاه

GetPersonDetails يبقى نفسه كما هو مبين أعلاه

إنشاء جدول SerialTest

1
2
3
4
5
6
7
create table SerialTest(
 
name BLOB,
 
viewname VARCHAR2(30)
 
);

PersonPersist هي الفئة التي تستخدم لتسلسل الكائن في جدول قاعدة البيانات SerialTest

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
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
import java.io.ByteArrayInputStream;
 
import java.io.ByteArrayOutputStream;
 
import java.io.IOException;
 
import java.io.ObjectInputStream;
 
import java.io.ObjectOutputStream;
 
import java.sql.Connection;
 
import java.sql.DriverManager;
 
import java.sql.PreparedStatement;
 
import java.sql.ResultSet;
 
import java.sql.SQLException;
 
import java.sql.Statement;
 
public class PersonPersist {
 
 
 
      static String userid = "scott", password = "tiger";
 
      static String url = "jdbc:odbc:bob";
 
      static int count = 0;
 
      static Connection con = null;
 
      static String s;
 
      public static void main(String[] args) {
 
            Connection con = getOracleJDBCConnection();
 
            PersonDetails person1 = new PersonDetails("hemanth", 10, "Male");
 
            PersonDetails person2 = new PersonDetails("bob", 12, "Male");
 
            PersonDetails person3 = new PersonDetails("Richa", 10, "Female");
 
            PreparedStatement ps;
 
            try {
 
                  ps = con
 
                              .prepareStatement("INSERT INTO SerialTest VALUES (?, ?)");
 
                  write(person1, ps);
 
                  ps.execute();
 
                  write(person2, ps);
 
                  ps.execute();
 
                  write(person3, ps);
 
                  ps.execute();
 
                  ps.close();
 
                  Statement st = con.createStatement();
 
                  ResultSet rs = st.executeQuery("SELECT * FROM SerialTest");
 
                  while (rs.next()) {
 
                        Object obj = read(rs, "Name");
 
                        PersonDetails p = (PersonDetails) obj;
 
                        System.out.println(p.getName() + "\t" + p.getAge() + "\t"
 
                                    + p.getSex());
 
                  }
 
                  rs.close();
 
                  st.close();
 
            } catch (Exception e) {
 
            }
 
      }
 
      public static void write(Object obj, PreparedStatement ps)
 
                  throws SQLException, IOException {
 
            ByteArrayOutputStream baos = new ByteArrayOutputStream();
 
            ObjectOutputStream oout = new ObjectOutputStream(baos);
 
            oout.writeObject(obj);
 
            oout.close();
 
            byte[] buf = baos.toByteArray();
 
            s = new sun.misc.BASE64Encoder().encode(buf);
 
            ps.setString(1, s);
 
            // ps.setBytes(1, Base64.byteArrayToBase64(baos.toByteArray()));
 
            ps.setBytes(1, baos.toByteArray());
 
            ps.setInt(2, ++count);
 
      }
 
      public static Object read(ResultSet rs, String column)
 
                  throws SQLException, IOException, ClassNotFoundException {
 
            byte[] buf = new sun.misc.BASE64Decoder().decodeBuffer(s);
 
            // byte[] buf = Base64.base64ToByteArray(new
 
            // String(rs.getBytes(column)));
 
            if (buf != null) {
 
                  ObjectInputStream objectIn = new ObjectInputStream(
 
                              new ByteArrayInputStream(buf));
 
                  Object obj = objectIn.readObject(); // Contains the object
 
                  PersonDetails p = (PersonDetails) obj;
 
                  System.out.println(p.getName() + "\t" + p.getAge() + "\t"
 
                              + p.getSex());
 
            }
 
            return null;
 
      }
 
      public static Connection getOracleJDBCConnection() {
 
            try {
 
                  Class.forName("sun.jdbc.odbc.JdbcOdbcDriver");
 
            } catch (java.lang.ClassNotFoundException e) {
 
                  System.err.print("ClassNotFoundException: ");
 
                  System.err.println(e.getMessage());
 
            }
 
            try {
 
                  con = DriverManager.getConnection(url, userid, password);
 
            } catch (SQLException ex) {
 
                  System.err.println("SQLException: " + ex.getMessage());
 
            }
 
            return con;
 
      }
 
}


أدناه هو برنامج يبين تسلسل الكائن JButton إلى ملف و تدفق مصفوفة بايت "Byte Array Stream". كما في السابق من اجل تسلسل theobject يجب تطبيق الواجهة Serializable.

PersonDetails هي الفئة النوات التي تطبق الواجهة Serializable

ليست هناك تعليقات:

إرسال تعليق