19 Ekim 2008 Pazar

Oracle Apex 3.1 Sunumu

18 Ekimde, OracleTURK grubuna, Oracle Akademi sponsorluğunda, Kadıköy Bireysel Eğitim Merkezinde gerçekleştirmiş olduğum, Oracle Application Express v3.1 sunumu ve workshop esnasında gösterdiğim bazı özelliklere ait kodlar için tıklayın.

For english:
I present the Oracle Application Express v3.1 to OracleTurk Group, at Kadıköy Bireysel Eğitim Merkezi, sponsored by Oracle Turkiye Akademi. To access this presentation click here.

26 Eylül 2008 Cuma

Asynchronous Ajax in Apex

We use Ajax for rapid query processing without submitting page. In apex, you can build ajax within your pages. For this purposes, we use "Application Process" under the Shared Components.

Sometimes, Application Process can not response rapidly. For example, if you query a count from very large table, it takes for minutes or hours. If you put a text field for table name and if user can query number of records of input big table, also if you perform this operation with ajax in apex as synchronously, the BROWSER will freeze and consumes %100 cpu time. Everything will go bad.

In this situation, you have to use "Asynchronous Ajax" opposite of Synchoronous, so you can more than one job on same time. To perform this, give an eye to this example:

Firstly, you can build your ajax javascript :


function getNumberOfRecords(){
var ajaxRequest = new htmldb_Get(null,&APP_ID.,'APPLICATION_PROCESS=GetCount',0);
ajaxRequest.add('P6_TABLE_NAME',html_GetElement('P6_TABLE_NAME').value);
ajaxRequest.GetAsync(f_AsyncReturn);
ajaxRequest = null;
}


This code is used to connect your apex page with your application process. ( By the way, Application Processes are need to created as On Demand ). In normally, synchronous method, you call get() method of ajaxRequest object. Pay attention to this, in asynchronous method, you need to call GetAsync() method by passing an Object. What is the f_AsyncReturn ? :


function f_AsyncReturn(){
if(p.readyState == 1){
$x('P6_TEXTAREA').value = '';
}else if(p.readyState == 2){
}else if(p.readyState == 3){
}else if(p.readyState == 4){
$x('P6_TEXT_AREA').value = p.responseText;
// you can add other things add this area
}else{return false;}
}


If you coded ajax in other languages such as php,jsp; probably you coded as above. There are 4 states to deal with ajax. One of them (state 1) is "Loading", one of them (state 4) is "Success". Result of ajax is written in 4. states. Before ajax Request is completed, you can write "Loading" message in first state.

The other operations are the same with Synchronous method. You need to create an Application Process, it's running time must be "On Demand" as well.

So, by performing this Asynchronous method, your browser will not be freezed. And, not consuming all of CPU time. But disadvantage of this ( may be not seemed apparently ), your job is take much time rather than synchronous method. But, it is worth!

1 Eylül 2008 Pazartesi

Running DDL Statements on Remote Database

You cannot run any DDL statement on a remote databases via using our custom remote sqls.

CREATE TABLE EMP@DB1 ...

This causes an error. Instead of this you can use the Dynamic Sql to produce ddl statement to execute it when it is called.

For example, There are two DB. A_DB and B_DB. We write this function in A_DB:


CREATE OR REPLACE PROCEDURE dyn_sql( v_Sql_in VARCHAR2 )
IS
curName INTEGER;
iRet INTEGER;
BEGIN
curName := DBMS_SQL.OPEN_CURSOR;
DBMS_SQL.PARSE(curName, v_sql_in, DBMS_SQL.NATIVE);
iret := DBMS_SQL.EXECUTE(curName);
DBMS_SQL.CLOSE_CURSOR(curName);
END;


When you call this procedure as "exec dyn_sql('CREATE TABLE EMP (EMPNO INTEGER)');" in B_DB database, an EMP table will be created in A_DB database.

31 Ağustos 2008 Pazar

About Returning Into Clause on Remote DB - ORA-22816

We use PL/SQL RETURNING INTO clause for returning value(s) after dml commands(INSERT,UPDATE,DELETE). Assume that you would insert row a employee table with employee_id value. Likely, you type this sql statement:


INSERT INTO TEMPLOYEES (EMP_ID,EMP_NAME) VALUES( SEQ_TEMPLOYEES.NEXTVAL,'FUAT SUNGUR');


After this statement you need this employee_id to add another table. You can use RETURNING INTO clause to get employee_id value after an insert statement as follows:


INSERT INTO TEMPLOYEES (EMP_ID,EMP_NAME) VALUES( SEQ_TEMPLOYEES.NEXTVAL,'FUAT SUNGUR') RETURNING EMP_ID INTO iEmpId;


After these, iEmpId variable has employee_id value.
There is no problem if you perform these on one database.

If you want to perform this between databases, likely you would get :

ORA-22816: unsupported feature with RETURNING clause


This error may be pothering and discouraging you to what do you want. We can use other tactic to perform our willings.

Assume that you have two databases A_DB, B_DB and in B_DB databases you want to add a row in Employee Table that table in A_DB. In this situation you take advantage of functions. Write a function that returns the sequence number of just added row.


FUNCTION INSERT_EMPLOYEE_AND_GET_SEQ(vEmpName_in IN VARCHAR2)
RETURN INTEGER
IS
iEmpId INTEGER;
BEGIN

INSERT INTO TEMPLOYEES (EMP_ID,EMP_NAME) VALUES (SEQ_TEMPLOYEES.NEXTVAL,vEmpName_in) RETURNING EMP_ID INTO iEmpId;
RETURN iEmpId;
EXCEPTION
WHEN OTHERS THEN NULL;
END;


Also you have add function parameter to whether the returning into is used or not. May be this sequence number is need only once and after you would use that sequence to insert employee. So, there are lots of need why you do it.

10 Ağustos 2008 Pazar

Google Syntax Highlighter for PL/SQL

Google Syntax Highlighter App. provides us to show code in our blogs or other html pages. This application shows code of C,Java,Jscript,C#,SQL,XML,Ruby,Phyton. But as you predicted, Oracle PL/SQL is missing. So, i decided to develop a code to highlight our Oracle/PLSQL code. Things you have to do them are as follows :

  • Download the Google Syntaxlighter to your pc: http://code.google.com/p/syntaxhighlighter/
  • Then upload the folder that is extracted from zip file, to your server.
  • For blogger, go to your template page and edit.
  • After first head tag and before the closed head tag you should write some javascript code to template page.
  • The template page should look alike this page :

    <link type="text/css" rel="stylesheet" href="http://xxx.googlepages.com/SyntaxHighlighter.css"></link>

    <script language="javascript" src="http://xxx.googlepages.com/shCore.js"></script>

    <script language="javascript" src="http://xxx.googlepages.com/shPlsql.js"></script>

    ....

    ....

    ....

    // dont modify if you dont want to change template except to show coding

    ....

    <script language="javascript">

    window.onload = function () {

    dp.SyntaxHighlighter.ClipboardSwf = 'http://afsungur.googlepages.com/clipboard.swf';



    dp.SyntaxHighlighter.BloggerMode();



    dp.SyntaxHighlighter.HighlightAll('code');

    }

    </script>


  • Then writing a post, you should write your pl sql code as follows:
    <pre name="code" class="plsql">



    ... your pl sql code ...



    </pre>



Download shBrushPlSql.js

(This is compatible with reserver words and keywords of Oracle Latest Version )

After all of these step, now you can test your plsql code. Like this :)


CREATE OR REPLACE PACKAGE BODY PKG_RULE_CONTROL IS

-- CREATED BY A. FUAT SUNGUR
-- afsungur@gmail.com
--
--
-- Kural ile ilgili islemleri gerceklestiren pakettir

------------------------------------------------------------------------------------------
-- DBMS_SCHEDULER tarafindan cagrilacak pl sql blogunun dinamik olarak uretilmesini saglar
-- RUN_JOB prosedurune gonderilecek parametreler dinamik olarak ayarlanir.
FUNCTION PRODUCE_JOB_SQL
(
vExpr_1 VARCHAR2,
vExpr_2 VARCHAR2,
nCond_Id NUMBER,
iRuleId NUMBER
) RETURN VARCHAR2 IS

vJobSql VARCHAR2(2000);
iSqlCode INTEGER;
vSqlErrMsg VARCHAR2(250);

BEGIN
-- CHR(39) = ' ( tek tirnak )
IF (vExpr_2 IS NOT NULL)
THEN
vJobSql := 'BEGIN PKG_JOB_CONTROL.RUN_JOB(' || CHR(39) ||
vExpr_1 || CHR(39) || ',' || CHR(39) || VExpr_2 ||
CHR(39) || ',' || nCond_Id || ',' || iRuleId ||
'); END;';
ELSE
vJobSql := 'BEGIN PKG_JOB_CONTROL.RUN_JOB(' || CHR(39) ||
vExpr_1 || CHR(39) || ',NULL,' || nCond_Id || ',' ||
iRuleId || '); END;';
END IF;
RETURN vJobSql;
EXCEPTION
WHEN OTHERS THEN
ROLLBACK;
INSERT INTO TERRORS
VALUES
(SEQ_TERRORS.NEXTVAL,
iSqlCode,
vSqlErrMsg,
CURRENT_TIMESTAMP,
'FUNCTION PKG_RULE_CONTROL.PRODUCE_JOB_SQL()');
COMMIT;
RETURN NULL;
END;
END;

3 Ağustos 2008 Pazar

Creating your own authentication method in Apex

With this operation, you can do your own login page instead of default apex login page.

Firstly, you need to create a package that deal with user login operations. PKG_APP_SECURITY package name is appropriate for this need.


CREATE OR REPLACE PACKAGE BODY PKG_APP_SECURITY
AS

PROCEDURE login

(p_uname IN VARCHAR2,
p_password IN VARCHAR2,
p_session_id IN VARCHAR2,
p_flow_page IN VARCHAR2)
IS
BEGIN
-- THIS PROVIDES AUTHENTICATION
wwv_flow_custom_auth_std.login (
p_uname => p_uname,
p_password => p_password,
p_session_id => p_session_id,
p_flow_page => p_flow_page || ':' || 1);

EXCEPTION
WHEN OTHERS
THEN RAISE;
END login;

PROCEDURE add_user (p_username IN VARCHAR2, p_password IN VARCHAR2)
AS
BEGIN
INSERT INTO P_USERS (username, PASSWORD)
VALUES (UPPER (p_username),
get_hash (TRIM (p_username), p_password));

COMMIT;
EXCEPTION
WHEN OTHERS
THEN ROLLBACK; RAISE;
END add_user;

FUNCTION get_hash (p_username IN VARCHAR2, p_password IN VARCHAR2)
RETURN VARCHAR2
AS
BEGIN
RETURN DBMS_OBFUSCATION_TOOLKIT.md5 (
input_string => UPPER (p_username) || '/' || UPPER (p_password));
END get_hash;

PROCEDURE valid_user2 (p_username IN VARCHAR2, p_password IN VARCHAR2)
AS
v_dummy VARCHAR2 (1);

BEGIN

SELECT '1' INTO v_dummy
FROM P_USERS
WHERE UPPER (username) = UPPER (p_username)
AND PASSWORD = get_hash (p_username, p_password);

EXCEPTION
WHEN NO_DATA_FOUND
THEN raise_application_error (-20000, 'Invalid username / password.');
END valid_user2;

FUNCTION valid_user (p_username IN VARCHAR2, p_password IN VARCHAR2)
RETURN BOOLEAN
AS
BEGIN
valid_user2 (UPPER (p_username), p_password);
RETURN TRUE;
EXCEPTION
WHEN OTHERS
THEN RETURN FALSE;
END valid_user;

END PKG_APP_SECURITY;

The P_USERS table has two columns, types of both columns are varchar2 and first of them is username and second of them is password.

Secondly, you should use an authentication schema. To perform this, in Shared Components, click Authentication Schemas. Then Create Button.

1-> Click "Create Schema" Button.
2-> Write your own authentication schema name, such as "MY AUTH METHOD".
3-> Then "Create Scheme" button.
4-> After scheme is created, click your authentication schema icon in authentication schemas page.
5-> Then Login Processing tab, in "Authentication Function" textarea, type this:
"RETURN PKG_APP_SECURITY.valid_user"
6-> If this method return false in application the authentication is not successfullied.
7-> Don't forget, you should set your own authentication schema to as current used.

Last thing we need to do, edit your login page (genrally page 101). Then, in processes tab ( in page processing tab ), click login process. And, type this to process textarea in source tab:
"pkg_app_security.login (P_UNAME => :P101_USERNAME, P_PASSWORD => :P101_PASSWORD, P_SESSION_ID => v('APP_SESSION'), P_FLOW_PAGE => :APP_ID );".

After performed these 3 steps, you can use your own login page with your own user tables in Apex Applications.

2 Ağustos 2008 Cumartesi

Installing Application Express 3.1 on Oracle 10g 10.2.0.1.0

In installing apex on Oracle 10g 10.2.0.1.0, i was getting a XDB Login Problem. I setup apex and entering the url path of apex, then browser ask me repeatedly xdb username and password. I tried lots of things that could not solve this problem. Especially this problem take place in Oracle 10g 10.2.0.1.0. I searched lots of web pages, forum, blogs then i found the solution.

Firstly, download the apex_3.1.1.zip in Oracle Web page. Then extract this file to Oracle Home.
When you type "dir /b" or "ls -la" in Oracle Home, you should see same this :


G:\oracle\product\10.2.0\db_3>dir /b
1907-b75f9673d7_orcl
admin
apex
assistants
BIN
cdata
cfgtoollogs
clone
com
config
....


So, type "cd apex" to enter this folder. Then run sqlplus as sysdba.
And execute the @apexins (apexins.sql).

Then, to change the ADMIN user password type the @apxxepwd.

After these, you should configure the Embedded PL SQL Gateway ( if you wont use the oracle http server like me).

Type @apex_epg_config to configure the plsql gateway.
Then, unlock the ANONYMOUS user, type "ALTER USER ANONYMOUS ACCOUNT UNLOCK".
After these, now we are at the last point.
To enable XML DB HTTP SERVER :
exec dbms_xdb.sethttpport(7780);
// if you enter a port of used service, you can't run the apex.
// so do not enter the Apache Tomcat Port : 8080

If XDB username and password is questioned, you should run this command:
( I found it after lots of searching )
EXEC DBMS_EPG.authorize_dad ( dad_name => 'APEX', user => 'ANONYMOUS' );

Now, you can try to login the Apex page via typing "http://localhost:7780/apex/apex_admin".

If it asks still XDB username and password, you should read this blog and comments carefully.


http://jeffkemponoracle.blogspot.com/2007/07/apex-30-via-embedded-plsql-gateway.html

If you encounter any problem, send an email or comments.

16 Temmuz 2008 Çarşamba

Materialized View

Oracle DW uygulamalarında sıkça rastlayabileceğimiz bir yapı olan Materialized Viewları, saklı tablo sonuçları olarak düşünebiliriz. Normal Viewlardan farkı, bir sql sorgusu olarak değil de sql sorgusu + data olarak saklanmasıdır. Yani MV diskte yer kaplar.

Peki neler yapılıyor MVlerle, milyonlarca kayıt bulunan tabloda belli sorguları önceden çalıştırıp bunun sonuçlarını saklama işlemini yapıyoruz.
Örneğin, bir satış tablomuz var, 5 milyar kaydımız var bu tabloda, 100 id'li müşterinin o günkü toplam harcamalarını öğrenmek istiyoruz. Bu öğrenme işi çok fazla kez tekrar ediliyor diyelim. 5 milyar kayıt içinde group by, sum() gibi sql özelliklerini kullanmak çok fazla kaynak kullanımına sebebiyet vereceğinden dolayı, bu işlemlerin önceden bir yerde tutulması bizim işimize yarayacaktır.

Yazının başında belirtildiği gibi, Data Warehouse sistemlerde özellikle kullanılan bir yapı, biraz real-time uygulamaların dışında gerçekleştirilebilir diyebiliriz. Örneğin salı günkü verilerden yola çıkarak, çarşamba günü rapor hazırlama işlemleri bu şekilde daha hızlı gerçekleştirilebilir. Salı akşamı sorgular çalıştırılır, MV'ler güncellenir, çarşamba günü sorgular çalıştırıldığında Ana tablo yerine bu MV'lerden ilgili bilgiler çekilmiş olur. Yani 5 milyar kayıt içinde arayacağımıza, çok daha az miktarda veri içinde arama yapmış olacağız. Bu da bizim kaynakları kullanma performansımızı arttıracak. Fakat bu durumda güncel verilerden yararlanmamış oluyoruz, sadece önceki güne ait verilerden yararlanmış oluyoruz. Bunu önleme yöntemleri de mevcut, MV'de.

MV'de bilinmesi gereken en önemli özelliklerden birisi de Query Rewriting özelliği. Bu özellik Session bazında ve MV oluştururken enable yapıldığı takdirde, MV 'in sql sorgusu çalıştırıldığında ana tablo yerine MV'den veriler çekilmiş oluyor. Şöyle ki, MV yi oluştururken "SELECT * FROM SALES" olarak oluşturduğumuz zaman, Query Rewriting özelliği enable olur ise, "SELECT * FROM SALES" sorgusu çalıştırıldığında, CBO ( Cost Based Optimizer ) bu sorgunun aslında MV'den çekilmesine karar verir ve yazılan sorguyu MV'yi işaret edecek şekilde ( örn : "SELECT * FROM MV_SALES" ) değiştirirr ve sorguyu yapan farkında olmadan MV kullanılmış olur.

Eğer bu özellik enable edilmez ise, MV'den yararlanmak için sorgu yazılırken MV direk işaret edilmelidir. "SELECT * FROM MV_SALE" gibi, aksi takdirde direk ana tablodan veriler çekilir.

3 Temmuz 2008 Perşembe

Tablespace - Datafile Oluşturma

Oracle'da verilerimiz fiziksel olarak datafile'larda saklanır ve bu datafile'ları barındıran lojiksel olarak Tablespace adını verdiğimiz yapılar bulunur. Tablespace'ler, data fileların toplu olarak yönetimini kolaylaştırmıştır. Bir Tablespacei read only yaparak, onu içeren dosyaları da read only yapmış oluruz, hepsini teker teker yapacağımıza.

Şimdi bu tablespace oluşturma işlemini manuel olarak yapalım.
İlk olarak konsoldan bir dosya oluşturalım.


root@fuat-laptop:/home/oracle/datafiles# touch fuat_3.dbf
root@fuat-laptop:/home/oracle/datafiles# ls -la


Daha sonra bu dosyaya yazma işlemlerinin gerçekleşebilmesi için:

root@fuat-laptop:/home/oracle/datafiles# chmod 777 fuat_3.dbf


Şimdi fuat_ts_2 adli tablespacimizi fuat_3.dbf dosyasini içerecek şekilde oluşturalım:

CREATE TABLESPACE FUAT_TS_1 DATAFILE '/home/oracle/datafiles/fuat_1.dbf' SIZE 10M;

Tablespace altered.


Kontrol etmek için:

root@fuat-laptop:/home/oracle/datafiles# ls -la
total 20536
drwxr-xr-x 2 root root 4096 2008-06-03 13:35 .
drwxrwxrwx 3 root root 4096 2008-06-03 10:11 ..
-rwxrwxrwx 1 root root 10493952 2008-06-03 11:49 fuat_3.dbf
...
...
...


Görüldüğü gibi fuat_3.dbf dosyası 10MB olarak oluşturuldu.

Şimdi, başka bir dosyayı var olan tablespace'e ekleme işlemini gerçekleştirelim.
Yine bir dosya oluşturalım, fuat_4.dbf diye, chmod ayarlarını yapalım,
ve sql plus 'ta su komutu yazalım:


alter tablespace fuat_ts_1 add datafile '/home/oracle/datafiles/fuat_4.dbf' SIZE 10M


Çok basit komutlarla datafile'larımızı ve tablespace'lerimizi oluşturmuş olduk.

2 Temmuz 2008 Çarşamba

Linux Üzerinde Oracle 10g XE ve Oracle SQL Developer

Merhaba, Linux'e uygulama kurmak bazı durumlarda gerçekten güç olabiliyor ( yeteri kadar bilgi sahibi değilseniz ). Çok araştırma yapmak, forumları gözden geçirmek, döküman okumak gerekiyor. Veya deneme/yanılma yöntemi ile de çözüm bulabiliyorsunuz. Aarama motorlarından arama yapıp bazılarının blogundan bunları elde etmek de mumkun oluyor.

Oracle 10g XE Linux'e kurulumu için aşağıdaki linki inceleyebilirsiniz:

http://www.oracle.com/technology/software/products/database/xe/htdocs/102xelinsoft.html


Ubuntu'ya kurulumu cok kolay, indirilecek .deb dosyasının çift tıklama ile yükleyebiliyorsunuz.

Tabi yükleme ile bitmiyor, configure etmemiz gereken bazı şeyler var. Bunun için aşağıdaki adresten yararlanabilrsiniz :

http://susedim.blogspot.com/2008/03/suse-103-zerine-oracle-xe-kurulumu.html


SQL Developer
http://www.oracle.com/technology/software/products/sql/index.html

rpm li dosyayı acma sekli linkte yazıyor. Icini acip icindeki klasoru masaustune kopyleyip, /opt/sqldeveloper/sqldeveloper.sh dosyasını çalıştırarak uygulamayı başlatabilirsiniz.

26 Haziran 2008 Perşembe

Linux Shell Scripting

Windowsta yazdığımız BAT dosyalarını yani toplu işlem yapma dosyalarının benzerini Linux için de yazabiliriz. Linux Shell Scripting denilen bu işlemde programlama diline yakın bir biçimde shell programlama yapabiliyorsunuz.

İlgili tutorial için : http://www.freeos.com/guides/lsst/index.html

MPI Api Kurulumu

MPi Apisinin Linux üzerinde kurulumu nasıl oluyor?
Bu apiyi, linux'e kurmak için epey uğraştım, sonunda başardım, okulda bazı arkadaşlar da benim gibi kuramayıp, okulun serverine uzaktan bağlanıp, kod derlemeye çalışmıştık.

İlk olarak buradan apiyi Linux işletim sistemine indiriyoruz. Daha sonra indirilen sıkıştırılmış dosyayı bir klasöre açıyoruz ve daha sonra terminal'den bu klasöre giriyoruz.

Terminal'e
"./configure CC=cc CXX=CC --without-fc -prefix=/directory/to/install/in" yazarak apinin configurasyonunu sağlıyoruz ( --without-fc FORTRAN uygulaması yazmayacağımızı belirtmek için ). Eğer hata çıkarsa, gcc nin yeni versiyonunun yüklenmesi gerekebilir. Bunu da "sudo apt-get install build-essential" dan yapabiliriz. Bu konfigurasyon işlemi sisteme bağlı olarak 5-10 dakika sürebilir, bu işlem bittikten sonra aynı klasörde terminale "make" daha sonra "make install" yazarak, apinin kurulmasını sağlıyoruz. Daha sonra terminale "mpicc" yazarak apinin yüklenip yüklenmediğini kontrol ediyoruz. Eğer şu şekilde bir hata alırsak :


root@fuat-laptop:/home/fuat/Desktop/lam-7.1.4# mpicc
The program 'mpicc' can be found in the following packages:
* lam4-dev
* libmpich1.0-dev
* libopenmpi-dev
* libmpich-mpd1.0-dev
* libmpich-shmem1.0-dev
Try: apt-get install
bash: mpicc: command not found


lam4-dev package'ını yüklememiz gerekbilir. Yüklemek için "sudo apt-get install lam4-dev" yazıyoruz terminal'e. Daha sonra mpicc yi tekrar deniyoruz ve "gcc no input files" hatası alırsak apinin doğru yüklendiğini görmüş oluyoruz.

Sırada mpirun var. Terminal'e mpirun yazıp mpirun uygulamasını kontrol ediyoruz, eğer yine bulunamazsa, "sudo apt-get install lam-runtime" yazarak bu package'in yüklenmesini sağlıyoruz.

Sıra geldi apiyi çalıştırmaya, "lamboot" yazıyoruz terminale, eğer şu şekild bir çıktı alırsak :



fuat@fuat-laptop:~/Desktop/lam-7.1.4/myexamples$ lamboot

LAM 7.1.2/MPI 2 C++/ROMIO - Indiana University



apimiz, başarıyla kurulmuştur...

25 Haziran 2008 Çarşamba

Linux'te Babylon Sözlük

Windows'tan aşina olduğumuz, yanımızdan ayıramadığımız bir parçadır Babylon. Şahsen, ingilizce kelime haznemi geliştirmede en faydası dokunan bir araç oldu bu uygulama. Peki bu araç sadece Windows'ta mı?

Sırf bu yüzden linux kullanmayanlar olduğunu biliyorum, şahsen benim de uzun süredir linux kullanmamamın sebeplerinden bir tanesi de buydu. Linux'te bu programı nasıl kullanacağız?

Babylon sadece Windows sürümleri için var fakat aynı işlemi gören bir Linux uygulaması da var : Stardict. Kelimelerin üstüne gelip kolayca farklı dilde olan anlamlarını görebilirsiniz.

Yapmamız gerekenler:

1 - İlk olarak Stardict adlı uygulamayı bilgisayarımıza kuruyoruz.
Ubuntu için : http://downloads.sourceforge.net/stardict/stardict_3.0.1-1_i386.deb

Diğer linux : http://linux.softpedia.com/progDownload/StarDict-Download-1162.html

2 - Daha sonra Babylon sözlüklerinin Stardicte kullanılması için Converter uygulaması kurmamız gerekiyor. Bu program .BGL uzantılı dosyaları, Stardict uygulamasının okuyabileceği .IFO uzantılı dosyaya çeviriyor. Programın adı : Dict Convert

http://linux.softpedia.com/get/Education/Dictconv-23446.shtml
adresinden programı indirebilirsiniz.

Programı indirdikten sonra zipli dosyayı bir klasöre açın ve daha sonra klasörün içine terminalden girdikten sonra :
"./configure" yazarak configurasyonu sağlayın. Eğer C++ Compiler hatası alırsanız terminale "sudo apt-get install build-essential" yazın ve ilgili paketlerin yüklenmesini sağlayın. Eğer libxml2 hatası alırsanız terminale "sudo apt-get install libxml2-dev" yazıp libxml2 paketinin yüklenmesini sağlayın. Daha sonra bu işlemi de sorunsuz hallederseniz, "make all install" yazarak programın kurulmasını sağlayın.

3 - Çevirme işleminin yapılması
http://www.babylon.com/category/30/Turkish.html adresinden türkçe sözlüklerden ilgilendiklerinizi indirin.
Daha sonra terminalde "dictconv babylonsozluk.bgl -o stardictsozluk.ifo" yazıp çevirme işlemini gerçekleştirin. Daha sonra bu komutu yazdığınız klasöre olusacak 3 dosyayı stardictsozluk.ifo,
stardictsozluk.idx, stardictsozluk.dict; stardict'in kurulduğu dizindeki dic klasörünün içine kopyalayın. Eğer nereye kurulduğunu bilmiyorsanız, "updatedb" yazın terminale, daha sonra "locate stardict" yazın ve stardict klasörünün nerede olduğunu öğrenmiş olun.

19 Haziran 2008 Perşembe

MPI ile Paralel Programlama

Message Passing Interface kısaca MPI, C programlama dili ile paralel programlamayı sağlayan bir C apisidir. Bu apiyi uygulamalarımızda kullanarak programlarımızı birçok işlemci üzerinde çalıştırmayı sağlayabiliriz.

Apinin bize sağladığı en önemli özellik processler arası haberleşmeyi sağlama imkanını sunması. Bu sayede zaten paralel programlama işlemini kusursuz yapmış oluyoruz. Genel kullanım alanları biyolojik modeller, uzay problemleri, dna modelleme gibi çok yoğun işlemleri gerektiren durumlarda kullanılıyor.

Bu apiyi kullanabilmek için en önemli şart, programımızı aynı işlemleri yapacak parçalara bölebilmek. Örneğin SMS Server'ınız var, 100 adet mesajı sıra ile okuyup, her sms'e bir cevap üreteceksiniz. Bunun için normal programlama ( seri programlama diyebiliriz aslında ) ile ne yaparız, 1 den 100 e kadar iterasyon şeklinde sms okuyup, cevaplandırırız. Fakat paralel programlama da ise örneğin 4 process kullansak, her birine 25 sms veririz. Her biri bir sms okur aynı anda, yani aynı anda 4 sms okur ( tam aynı anda olmaz aslında çünkü I/O cihazına aynı anda sadece bir process ulaşabilir ) gibi bir durum oluşur.

Fakat bilgisayar mimarisi ve işletim sistemleri konularından da bildiğimiz üzere bu işlemde 4 prcess kullanacağımız zaman 4 kat hızlanma gerçekleşmez. Çünkü programın bir kısmı yine seri olarak yapılacaktır ( zorunludur diyebiliriz). Bunun için Amdahl's Law incelenebilir.

En çok kullanılan durumlardan bir tanesi de matris üzerinde işlem yapma olayıdır. Örneğin matris çarpımı, matris toplamı veya 2 arrayin toplamı gibi işlemler paralel programlama ile çok daha hızlı bir şekilde yapılabilir.

Vereceğim linkte Game Of Life probleminin paralel programlama ile çözülmüş örneği var. Rar dosyası içerisinde Bilgisayar Mimarisi dersi için yapmış olduğum Game Of Life oyunun kodu ve raporu bulunmakta.

Yakın zamanda metodların kullanımı ile ilgili yazı da yazacağım.

Not : Program Linux üzerinde çalışmaktadır. Ve MPI apisi kurulu olmalıdır.
MPI Apisini indirip kurmak için : http://www.lam-mpi.org/using/docs/

Dosyayı indir

6 Haziran 2008 Cuma

JSP - Oracle - Array Gonderme

Bazı durumlarda JSP sayfamızdan Oracledaki procedurumuze array gondermek isteyebiliriz. Bunun icin ilk olarak yapmamız gereken Oracle'da bir array type tanımlamak.

CREATE OR REPLACE TYPE INTEGERARRAYTYPE_TY AS TABLE OF INTEGER

Daha sonra Procedurumuzun parametre kismina bu array tipinde bir degisken yazacagiz:


CREATE OR REPLACE FUNCTION INSERT_LINE(vLineCode IN VARCHAR, arrStations IN INTEGERARRAYTYPE_TY)

Daha sonra JSP sayfamızdan Array ile ilgili işlemleri yapmamız gerekiyor.
İlk olarak jsp sayfamızda ilgili sınıfları import edelim:


<%@page language="java" import="oracle.sql.ARRAY"%>
<%@page language="java" import="oracle.sql.ArrayDescriptor">


Daha sonra arrayimizi olusturup, import ettigimiz sinifa ait nesneleri olusturalim.


Integer arrStations[] = new Integer[3];
arrStations[0]= 0;
arrStations[1]= 1;
arrStations[2]= 2;

ArrayDescriptor desc1 = ArrayDescriptor.createDescriptor("INTEGERARRAYTYPE_TY",Conn);
ARRAY input1 = new ARRAY(desc1, Conn, arrStations);


ArrayDescriptor nesnesi olustururken, createDescriptor metodunu kullaniyoruz. Bu metod 2 parametre aliyor. İlki bizim Oracle da olusturdugumuz integer array tipi, digeri ise jspden oracle'a baglanirken kullandigimiz Connection sınıfından türetilmis Conn nesnesi.

Conn = DriverManager.getConnection("jdbc:oracle:thin:@127.0.0.1:1521:XE",
"IETT","****");

Buraya kadar olan islemlerde arrayimiz jsp sayfamizda olustu. Sirada, bunu gonderme islemi var.



CallableStatement cstmt = null;
String sql = "{call ? := INSERT_LINE(?,?)}";
try
{
cstmt = Conn.prepareCall(sql);
cstmt.setString(2,strLineCode_tr);
cstmt.setArray(3,input1);
cstmt.registerOutParameter(1,Types.INTEGER);
int b = cstmt.executeUpdate();
int sonuc = cstmt.getInt(1);

}
catch(Exception e) {}
finally {}


CallableStatement kullaniyoruz burada. Bu sinifin setArray fonksiyonunu cstmt nesnesinden cagiriyoruz. Bu metod 2 parametre aliyor. Ilki parametre sirasi, insert_line oracle fonksiyonunun 2.parametresi oluyor bu array. Return degeri ilk parametre oldugu icin dogal olarak sirasi 3 oluyor. setArray metodunun 2.parametresi ise ARRAY sınıfından türetilen nesne oluyor. Bu da yukarıda önceden ürettiğimiz ARRAY sınıfına ait input1 nesnesi.

Bu sekilde arrayimizi oracleda bir fonksiyona gondermis olduk. Biraz zahmetli bir ise benziyor. Eger daha performansli ve kolay bir yol bulursam yine paylasacagim.

1 Haziran 2008 Pazar

Google Map Api - Ders 8 - Bir koordinata Odaklan

Bu özellik zaten önceden bahsettiğim özelliklerden bir kaç fonksiyonu kullanılarak yapılabilir. 2 Textfield'a kullanıcıdan enlem ve boylam değerleri istenilerek haritada istenilen koordinata odaklanılabilir. Hareket halindeki nesne takibinde bu özellik işe yarayacaktır.

Kullanılan 2 fonskyionumuz var:


map.setCenter(new GLatLng(enlem,boylam));
map.setZoom(zoomSeviyesi);


İlgili Örnek için :

http://afsungur.googlepages.com/map_ornek_10.html

22 Mayıs 2008 Perşembe

Google Map Api - Ders 7 - PolyLine Oluşturmak

Bazı durumlarda 2 nokta arasındaki mesafeyi yolu belli etmek amacıyla çizgi çizeriz. Bu bazen 2 nokta ile de sınırlı kalmayabilir. Bir yerden baska bir yere giderken uğranılması gereken yerleri de belirtmek isteyebiliriz. Bu durumlarda apimizin GPolyLine sınıfından yararlanacağız.

Bunun için ilk olarak yapilmasi gereken bir GLatLng arrayi oluşturmak. Bu arrayin elemanlari birer koordinat objesi olacak ve bu arrayın pointerını GPolyLine sınıfının constructoruna parametre olarak göndereceğiz. Örnekte 10 noktalı yani 10 elemanlı bir GLatLng arrayi oluşturulmuş ve bu array GPolyLine sınıfının constructoruna parametre olarak gönderilmiştir.


var a = new GLatLng(41,29);
var b = new GLatLng(42,30);
koordinatlar[0] = a;
koordinatlar[1] = b;

polyOptions = {geodesic:true};
polyLine = new GPolyline(koordinatlar, "#ff0000", 5, 1, polyOptions);
map.addOverlay(polyLine);


GPolyLine sınıfının constructoru sırasıyla GLatLng arrayı, polyline çizgisinin rengi, kalınlığı, opaklığı ve çizgi özellikleri olacak şekilde 5 parametre alıyor. Polyoption ise geodesic ve clickable özellikleri var. True veya false değerlerini alabiliyor.

Örnek için :

http://afsungur.googlepages.com/map_ornek_9.html

Google Map Api - Ders 6 - Hareket eden işaretçiler oluşturmak

Belki de bu apinin en çok beğendim özelliği hareket eden işaretçiler oluşturmak. Aslında burada apinin ekstradan herhangi bir özelliğini kullanmayacağız. Sadece programlama mantığımızdan yola çıkarak temel bir kaç javascript özelliği ile hareket edebilen işaretçiler oluşturacağız.

Programımızı yine fonksiyonlara ayiriyoruz. En başta işaretçimizi oluşturduk bunda problem yok.
Şimdi 4 tane metodumuz olacak.

1-) startMarker() : Isaretciyi hareket icin tetikleyecek metod
2-) stopMarker() : Isaretcinin hareketini durduracak metod
3-) moveMarker() : Isaretciyi hareket ettirecek metod
4-) setCoords() : Isaretcinin hareket edecegi koordinatları random olarak oluşturacak metod. Ben random koordinatlar olusturdum, arzu eden buraya elle degerler girip isaretcinin bir yol üzerinde hareket etmesini sağlayabilir. Projemde ben bu sekilde kullanmıstım. Elle degerleri girip aracları belli yol üzerinde hareket ettirebiliyordum.

Javascript'te timer olaylarını biraz bilmek gerekiyor. startMarker metodunda yer alacak aşağıdaki kod , speedInterval milisaniye sonra moveMarker metodunun çalıştıracağını belirtiyor.


function startMarker()
{
timerID = setTimeout("moveMarker()", speedInterval);
}


Peki movemarket metodunda ne yapiliyor? Random olarak urettigimiz koordinatlara işaretçinin yönlendirilmesi yapiliyyor:


function moveMarker()
{
coordObj = new GLatLng(coordY[coordSayac],coordX[coordSayac]);
busMarker.setLatLng(coordObj);
coordSayac++;
timerID = setTimeout("moveMarker()", speedInterval);
}


Fonksyionun ilk satirinda random degerler ile doldurdugumuz coordY ve coordX arraylarindan ilgili indexteki degerler ile bir koordinat objesi oluşturuluyor ( coordObj ). Daha sonra işaretçi buraya yönlendiriliyor ( busMarker.setLatLng ). Daha sonra index değeri bir arttırılıp, aynı metod speedInterval milisaniye sonra tekrar çağırılıyor.

Örneğin tamamı için:

http://afsungur.googlepages.com/map_ornek_8.html

21 Mayıs 2008 Çarşamba

İnnovasyon Nedir, ne işe yarar?

TEB'in innovasyon ile ilgili hazırlamış olduğu güzel bir sunum:

Burada

17 Mayıs 2008 Cumartesi

Kardemir Karabükspor Bank Asya 1. Lig'de

Canım memleketim Karabük'ün biricik temsilcisi Kardemir Karabükspor'a Bank Asya 1. Lig'de başarılar dilerim.

13 Mayıs 2008 Salı

Google Map Api - Ders 5 - İşaretçi Eventleri ve Draggable işaretçiler

İşaretçiler üzerinde Event olaylarından bahsedeceğim. Ayrıca bu yazıda dinamik, hareket edebilen işaretçilerden de bahsedeceğim.

Bu özellik de apinin en çok kullanılan özelliklerinden birisi. İşaretçiler üzerinde yapılabilecek event işlemleri:

mousedown,mouseup,click,dragstart,drag,dragend.

Örneğin işaretçiye tıklandığında bir balon içerisinde bize bir mesaj yazılsın. Bunun için GEvent classının statik metodlarından yararlanıyoruz.

GEvent.addListener(busMarker, "click", function() {
// yapilacak islemler
// ...
// ...
}

Eğer işaretçi üzerine tıklarsanız yapılacak işlemler kısmındaki kod parçası çalışacaktır. Önceden dediğimiz gibi bir balon çıkmasını sağlayalım. Bunun için GMap2 classının openInfoWindowHtml metodundan yararlanacağız. Bu metodun parametrelerinin ilki GLatLng objesi, 2. si ise yazılacak mesaj.


var myHtml = <b>&quote Isareci uzerine tikladiniz... Bu bir html mesaji </b>&quote
var latlng = new GLatLng(41,29);
map.openInfoWindowHtml(latlng, myHtml);


Burada dikkat edilmesi gereken openInfoWindowHtml metoduna gönderilen latlang objesinin koordinatlarının marker ile aynı olması gerektiğidir. Eğer aynı olmaz ise siz işaretçiye tıklarsanız başka koordinat üzerinde bulut çıkabilir. Tabi bu yine size ve programınıza bağlı.

Tam bir örnek için : http://afsungur.googlepages.com/map_ornek_6.html

Draggable İşaretçiler
Bu işaretçiler önceden anlatılan statik işaretçiler gibi sabit bir şekilde yer almayabilirler. Kullanıcı isterse bu işaretçileri hareket ettirebilir. Bu hareket sayesinde harita üzerindeki bir çok farklı bölge üzerinde gezinebilir. Buraların koordinatları alınabilir, editlenebilir, veritabanına atılabilir vs. vs.. Bir çok şey yapılabilir. Bunun için ilk olarak işaretçimizin draggable olarak yani suruklenebilir olarak ayarlanmasi gerekiyor.


markeroptions = {draggable: true,icon: greenIcon };
busMarker = new GMarker(center,markeroptions);


şeklinde yeni draggable markerimizi oluşturabiliriz.

Peki draggable işaretçilerin ek özellikleri neler? Örneğin 3 yeni event listener draggable işaretçiler ile kullanılabiliyor.
dragstart,drag,dragend
dragstart -> sürükleme işlemi başladığı anda yapılacak işler
drag -> sürükleme esnasında yapılacak işler
dragend -> sürükleme bırakıldığında yapılacak işler için listener görevi görüyor.

yine GEvent classından yararlanıyoruz:

GEvent.addListener(busMarker,"dragstart",function() {
// surukleme basladiginda yapilacak isler
}

GEvent.addListener(busMarker,"drag",function() {
// surukleme devam ettiginde yapilacak isler
}

GEvent.addListener(busMarker,"dragend",function() {
// surukleme bittiginde yapilacak isler
}

Az sonra verecegim ornekte 2 isaretci olacak. Biri yesil, digeri mavi. Yesili hareket ettirdiginiz surece yesil isaretcinin koordinatlarını bir div bolgesinde goreceksiniz. Maviyi hareket ettirmeye basladiginiz ve suruklemeyi biraktiginiz anda belli bir div bölgesinde mesaj goreceksiniz.

Ornek 7 : http://afsungur.googlepages.com/map_ornek_7.html

Google Map Api - Ders4 - İşaretçinin Koordinatını Alma

Oluşturduğumuz işaretçilerin bazen koordinatlarını alıp işlememiz gerekebilir. Ders 3'teki işaretçimizin koordinatlarını alıp belli değer ile arttırarak yeni işaretçiler oluşturabiliriz. Yaratıcı düşünce size ait, yüzlerce binlerce şey yapabilirsiniz. Önemli olan temel komutları öğrenmek.


var temp = new GLatLng();
temp = busMarker.getLatLng();
enlem_degeri = temp.lat();
boylam_degeri = temp.lng();



Bir koordinat objesi ( GLatLng ) objesi oluşturduktan sonra buna işaretçimizin ( busMarker ) koordinatlarını atıyoruz. Daha sonra GLatLng sınıfının metodları olan lat(), lng() metodlarını temp objesinden çağırıyoruz. Eğer object oriented programlama biliyorsanız bu işleri anlamanız gerçekten çok kolay ! Daha sonra enlem ve boylam degerlerini sırasıyla çekiyoruz.

İşaretçinin konumunu belli bir div bölgesine yazdıran örnek yapalım:

Örnek : http://afsungur.googlepages.com/map_ornek_5.html

12 Mayıs 2008 Pazartesi

Google Map Api - Ders 3 - İşaretçiler

İşaretçiler, harita üzerinde bir koordinatı belirtmek için kullanılan bir simge olarak karşımıza çıkıyor. Örneğin bulunduğunuz evin koordinatını biliyorsanız bu map üzerinde tam evinizin üstüne bir işaretçi koyabilirsiniz. Eğer koordinatları bilmiyorsanız üzülmeyin, ileriki derslerde de bu konuya değineceğiz.

İşaretçileri genel olarak 2 kısma ayırabiliriz. Statik ve dinamik işaretçiler.
Bu derste belirtilen koordinata bir işaretçi eklemeyi göstereceğiz.

İşaretçi oluşturmak için GMarker sınıfından yararlanacağız. Bu sınıfın constructoru genel olarak 2 paremetre alır. İlki oluşacak işaretçinin koordinatı, 2.si ise işaretçinin özellikler ( rengi, büyüklüğü .. )

var busMarker = new GMarker(center,markeroptions);

şeklinde işaretçimizi oluşturabiliriz. Fakat center ve marker options nesnelerini bu kodu çalıştırmadan önce set etmemiz gerekiyor.

center nesnesi bir GLatLng obbjesi. Yani işaretçinin koordinatlarının belirlendiği obje. Harita üzerinde 41,29 noktasına bir işaretçi koymak istediğimizi düşünürsek, center nesnesini şu şekilde oluşturmamız gerekiyor:

var center = new GLatLng(41,29);

Center nesnemizi oluşturduk. Daha sonra markeroptions nesnemizi oluşturmamız gerekiyor. Peki bu nesne de neyin nesi?
Bu nesne işaretçinin özelliklerinin belirlenmesini sağlayan bir nesne. Örneğin işaretçimizin yeşil renkli bir Google Map standart işaretçisi olmasını istiyoruz.

İlk olarak bir icon nesnesi oluşturalım:

var greenIcon = new GIcon(G_DEFAULT_ICON);
greenIcon.image = "http://gmaps-samples.googlecode.com/svn/trunk/markers/green/blank.png";

bu image özelliğinde kullanılan png dosyasının pathini siz kendiniz de değiştirebilirsiniz hiç sorun değil.

icon nesnesinin image özelliğinden başka bir çok özelliği mevcut, mesela iconSize bunlardan birisi. greenIcon.iconSize = ... diyerek iconun büyüküğünü de ayarlamış oluyorsunuz.

greenIcon.shadow = "http://...."; diyerek iconun gölgesinin pathini ayarlayabiliyorsunuz.

iconumuzu oluşturduk diyelim, şimdi sıra geldi markeroptions nesnemizi oluşturmaya:

var markeroptions = {draggable: false,icon: greenIcon };

bu şekilde de markeroptions nesnemizi oluşturabiliyoruz. Bu marker nesnesi draggable özelliği false olduğundan dolayı sürüklenemez bir işaretçi yani statik. Tabi daha bir çok özelliği var bu markeroptions'un..

clickable: true/false
bounceGravity : marker sürüklenip bırakıldığında markerin yere düşme hızı. Adamlar bunu bile düşünmüş ben başka bişi demiyorum :D ( bu arada bu int değeri alıyo - default 1)
title : String


burada bir kaç tanesinden bahsettiğim özellikleri referans sayfasında bulabilirsiniz.

En son olarak da map nesnesine bu ürettiğimiz marker nesnesini harita üzerine konumlandırmasını söylüyoruz :

map.addOverlay(busMarker);

örnek : http://afsungur.googlepages.com/map_ornek_4.html

Gelecek derslerde işaretçilerin derinliklerine doğru inilecek... Bu işin ben zevkli olduğunu düşünüyorum.. İnsanın aklına yüzlerce fikir geliyor bu harita apisini kullandıktan sonra..

Google MAP APi Reference : http://code.google.com/apis/maps/documentation/reference.html

Google Map Api - Ders 2 - Harita Özellikleri

Google Map Api'de harita için kullandığımız Map nesnelerinin bir çok fonksiyonu var. Bunlardan bir kaç tane önemli olanı açıklamakta fayda var.

Örneğin oluşturduğumuz harita nesnesine eklenecek şu 2 metodu inceleyerek, önceki örneğimizle aradaki farkı görelim.

map.addControl(new GLargeMapControl());
map.addControl(new GMapTypeControl());

Bu 2 metod haritaya 2 özellik katıyor. İlki zoom olayını sağlama, bu şekilde zoom ölçüsünü kullanıcı kendi bir slider aracılığı ile belirleyebiliyor. 2. si ise harita özelliğini değiştirebilme imkanını sağlayan metod. Bu sayede haritayı isterseniz yol haritası, isterseniz uydu fotoğrafı veya her ikisini hibritleyerek gösterme imkanına sahip olabiliyorsunuz.

Örnek için : http://afsungur.googlepages.com/map_ornek_2.html

setMapType(type)
removeMapType(type) metodlarını da direk set etmek veya tipi silmek için kullanabilirsiniz.

Bir map nesnesi oluştururken GMap classının constructoruna ek parametreler göndererek harita özelliklerini değiştirebilirsiniz.



map = new GMap2(document.getElementById("map_canvas"),

{ size:new GSize(800,600),

draggableCursor:"move",

draggingCursor:"crosshair"

}

);
Örneğin yukarıda GMap2 constructoruna ek parametreler dizisi gönderdik.
size burada harita boyutunu belirtiyor. draggingCursor, haritayı drag ederken ki cursor şeklini belirtiyor.
Diğer özellikleri Google MAP Api reference 'dan edinebilirsiniz.

Örnek 3 : http://afsungur.googlepages.com/map_ornek_3.html

Google Map Api - Ders 1 - Haritayı Sayfaya Yükleme

Bunun için ilk olarak Google hesabımız
olması gerekiyor. Ve apiyi kullanabilmemiz için bir key almamız gerekiyor. Bu çok da önemli bir şey değil. Çünkü ben rasgele web sitesi ismi yazarak da bu keyi aldım. Localhostta harita sorunsuz çalışıyor. Fakat yine de gerçek veri yazabiliyor isek yazmak lazım.Aksi takdirde her sayfa açılışında alert alabiliriz.

http://code.google.com/apis/maps/signup.html adresinden keyi alabilirsiniz.

Key aldıktan sonra yapmanız gereken aşağıdaki kodu sayfanızda head tagleri arasına yerleştirmeniz.




<script src="http://maps.google.com/maps?file=api&v=2&key=ABQIAAAAzr2EBOXUKnm_jVnk0OJI7xSosDVG8KKPE1-m51RBrvYughuyMxQ-i1QfUnH94QxWIa6N4U6MouMmBA"

type="text/javascript">

</script>

Daha sonra yapmamız gereken ilk iş haritamızın yerleşecegi div bolgesini ayarlamak.

<div id="map_canvas" style="width: 500px; height: 300px"></div>


yazarak div bölgesini ayarlayabiliyoruz. Bu bölge herhangi bir tablonun gözü olabilir, sayfanın en uç noktası, en başı olabilir bu size kalmış. Daha sonra bu div bölgesine haritayı basmamız gerekiyor. Onun için artık javascript metodumuzu yazmamız gerek.


<script type="text/javascript">



function initialize() {

if (GBrowserIsCompatible()) {

var map = new GMap2(document.getElementById("map_canvas"));

map.setCenter(new GLatLng(41.028, 28.293), 13);

}

}

</script>



Yukarıdaki javascript kodunu head tagleri arasina yerlestiriyoruz. Burada yapılan iş map_canvas div bolgesine haritayı basmak ve harita ilk açıldığında merkez olarak alınacak koordinatın map.setCenter metodu ile belirlenmesi.

GMap Clasına ait olan setCenter metodu görüldüğü gibi GLatLng sınıfına ait bir obje almış. Bu objeyi bir koordinat objesi olarak nitelendirebiliriz.
Kodu su sekilde de yazabilirdik:
GLatLng classının constr. ilk parametresi enlem, 2.si boylam buna dikkat!!!

var a = new GLatLng(41.028,28.903);
map.setCenter(a,13);

Buradaki 13 sayisi ise zoom seviyesini gösteriyor. 1 en düşük değer oluyor, 14 en büyük değer oluyor. Bazı durumlarda 15 de geçerli olabiliyor. Çekilen uydu fotoğrafına göre...

Tabi unutmadan bu initialize metodunun sayfa acilirken hemen yuklenmesini istiyor isek body load kismina su sekilde koymamiz lazim:


<body onload="initialize()" onunload="GUnload()">



Bu sekilde haritamizi sayfamiza yuklemis oluyoruz.

Örnek 1 : http://afsungur.googlepages.com/a.html

Google Map Api

Bu sene okulumuzdaki bilgisayar projesi için İett Araç Takip sisteminin simülasyonunu gerçekleştirdim. Bunun için hayatımıza anlam katan Google'ın , Map Apisinden yararlanmam kaçınılmaz oldu. Gerçekten adamlar yapmışlar diyorsunuz. Uğraşmışlar, didinmişler bizler için developerlar için apiyi hazırlamışlar. Bir süredir bu api ile uğraşıyordum. Artık öğrendiklerimi paylaşmanın vakti geldi. Basit örnekler ile bu apiden nasıl yararlanacağımızdan projelerimize nasıl entegre edeceğimizden bahsedeceğim.
Görüşmek üzere..

12 Nisan 2008 Cumartesi

Java SMS Api v1.0

Hi everybody. At last, i've finished my SMS Api. May be it is not an API because it has only four classes that handle the SMSs.

After read the user guide carefully because installing and running this class some complicated. If you encounter any problem, sending an email that contains your problems, i want to help you to install and run this api.

May be there are some bugs, i have not checked them comprehensively. I will fix them later version.

Thank you,

Download this api :
http://afsungur.googlepages.com/SMSApi.rar

9 Nisan 2008 Çarşamba

XML Valid

Yazdığınız XML dosyalarınızı yine kendi yazdığınız DTD veya SCHEMA dosyalarıyla kontrol edebiliyorsunuz. XML Validator işleminin düzgün yapılıp yapılmadığını öğrenebileceğiniz kullanımı gayet kolay bir yazılım var:

Stylus Studio 2008 XML Enterprise Suite

indirmek için : www.stylusstudio.com/xml_download.html

Web üzerinden Schema Kontrolu ( biraz agir calissa da gereksinimleri karsilamakta ) :
http://tools.decisionsoft.com/schemaValidate/

6 Nisan 2008 Pazar

XML ve XSLT

XML, data change ve veri sakalamak için kullanabilecegimiz platform bagımsız oldukca elverisli bir teknoloji. Cok yaygın kullanımından dolayı bu yapıyı iyi anlamak gerekiyor. Çünkü yeri geldiğinde çok farklı cihazlar arasında veri alış-verişi yapılıyor ve bu veri alış-verişinin bir standart kullanılarak yapılması ilerde başka şahıslar tarafından bu işlemin geliştirilmesini sağlayabilir.

XML olmadığını düşünelim, Şöyle ki bir elektronik cihazdan bilgisayara veri okuyacaksınız. Elektronik cihaza bir kod embed ettiniz, bu kodda 5 tane arac bilgisi var diyelim ve bunları su sekilde cihaza kaydediyoruz:

Nesne1||Otobus||Plaka||34 OO 9899||Kapasite||200||Marka||Mercedes
Nesne2||Otobus||Plaka||34 aa 1121||Kapasite||400||Marka||Mercedes
Nesne3||Araba||Plaka||34 PP 3311||Kapasite||5||Marka||AUDI

Görüldüğü gibi bu kodda veya kod demeyelim, bu fileda araclara ait bilgiler var. Bilgiler karmasık bir düzende kaydedilmiş. Bunu okumak gercekten zor. Hadi biz bunu pcde kendi yazdıgımız bir prog.la okuduk diyelim bilgisayar tarafından, fakat başka birisi yine bunu okuyamayacak, aynı programın onda da olması gerekiyor. Bu yüzden bu işin bir standartlaştırmaya kavuşması gerekiyor. Belli kurallar dahilinde dataların saklanması gerekiyor ki bu işlem evrensel olsun, milyonlarca insan tarafından kullanılabilsin. O yüzden XML geliştirilmiş bu işin standartlaştırılması sağlanmıştır.

<?xml version="1.0"?>
<nesneler>
<nesne id="1">
<tip>Otobus</tip>
<plaka>34 00 9899</plaka>
<kapasite>200</kapasite>
<marka>Mercedes</marka>
</nesne>
<nesne id="2">
<tip>Otobus</tip>
<plaka>34 aa 1121</plaka>
<kapasite>400</kapasite>
<marka>Mercedes</marka>
</nesne>
<nesne id="3">
<tip>Araba</tip>
<plaka>34 PP 3311</plaka>
<kapasite>5</kapasite>
<marka>AUDI</marka>
</nesne>
</nesneler>


Yukarıdaki file'ı araclar.xml olarka kaydedip, çift tıkladığımızda ağaç yapısına benzer bir yapıyı göreceğiz. Çünkü şu ana kadar HTML'den yararlanmadık. Şimdi bu XML dosyasını HTML içinde XSLT ile görüntülemeye çalışalım.

İlk olarak XML File'ımızın 2.satırına yani <?xml version.. ile başlayan satırın bir altına şu satırı ekliyoruz :

<?xml-stylesheet type="text/xsl" href="araclar.xsl"?>

araclar.xsl dosyası araclar.xml dosyasıyla aynı klasörde bulunmakta ve araclar.xml dosyasının çift tıklanıp açıldığında anlamlı bir şekilde yani html gibi görüntülenmesini sağlayacak.

Peki araclar.xsl dosyasını nasıl oluşturuyoruz.

<?xml version="1.0"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:template match="/">

Yukarıdaki ifadelerden ilk ikisini tanıyoruz. 3. satırdaki ise XPath olayını belirtiyor. XPath ne? XPath aynı directory yapısı gibi çalışıyor. XML deki node'lara ulaşmamız için bir path tanımlıyoruz. Buradaki onu belirtiyor. Aşağıdaki satırlarda olayı daha iyi anlayacağız.


<html>
<body>
<h2> Vehicles </h2>
<table border="1">
<tr>
<th>Tip</th>
<th>Plaka</th>
<th>Kapasite</th>
<th>Marka</th>
</tr>

tablomuzu oluşturduk ve başlıklarını ekledik şimdi sıra geldi tablonun rowlarını doldurmaya :

<xsl:for-each select="nesneler/nesne">


directory'e ulaşır gibi xml de nodedan nodea atlıyoruz. nesneler nodundan nesne noduna atlıyoruz.

<xsl:sort select="kapasite" />


bilgilerin kapasite degerlerine gore sıralanmasını istedik. burada ascending sıralama gerceklesiyor. dilenirse order="" attribute ile bu deger degistirilebilir

<xsl:choose>

choose deyimi switch yapısına benziyor. when keywordleri switch yapısındaki case'lere, otherwise ise default kısmına denk düşüyor.

<xsl:when test="kapasite<100">
<tr>
<td bgcolor="red"><xsl:value-of select="tip" /></td>
<td><xsl:value-of select="plaka" /></td>
<td><xsl:value-of select="kapasite" /></td>
<td><xsl:value-of select="marka" /></td>
</tr>
</xsl:when>
<xsl:otherwise>
<tr>
<td bgcolor="yellow"><xsl:value-of select="tip" /></td>
<td><xsl:value-of select="plaka" /></td>
<td><xsl:value-of select="kapasite" /></td>
<td><xsl:value-of select="marka" /></td>
</tr>
</xsl:otherwise>
</xsl:choose>
</xsl:for-each>
</table>
</body>
</html>
</xsl:template>
</xsl:stylesheet>

burada yaptığımız eğer xml deki dataların yani aracların kapasiteleri 100 den kucuk ise tiplerinin arka planı "red" olarak yani kırmızı olarak, 100 den buyukse tiplerinin arka planı sarı olacak sekilde tablolarımızı doldurmus oluyoruz.

KOMPLE KOD:

araclar.xsl

<?xml version="1.0"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:template match="/">
<html>
<body>
<h2> Vehicles </h2>
<table border="1">
<tr>
<th>Tip</th>
<th>Plaka</th>
<th>Kapasite</th>
<th>Marka</th>
</tr>
<xsl:for-each select="nesneler/nesne">
<xsl:sort select="kapasite" />
<xsl:choose>
<xsl:when test="kapasite<100">
<tr>
<td bgcolor="red"><xsl:value-of select="tip" /></td>
<td><xsl:value-of select="plaka" /></td>
<td><xsl:value-of select="kapasite" /></td>
<td><xsl:value-of select="marka" /></td>
</tr>
</xsl:when>
<xsl:otherwise>
<tr>
<td bgcolor="yellow"><xsl:value-of select="tip" /></td>
<td><xsl:value-of select="plaka" /></td>
<td><xsl:value-of select="kapasite" /></td>
<td><xsl:value-of select="marka" /></td>
</tr>
</xsl:otherwise>
</xsl:choose>
</xsl:for-each>
</table>
</body>
</html>
</xsl:template>
</xsl:stylesheet>



31 Mart 2008 Pazartesi

Java Sms Api - Haber 2


Şimdilik Nokia S40 lar için çalışan ( s60 için deneyemedim :) ) SMS Api bitmis durumda. Gözden geçiriyorum. Eksiklikler, kodda hatalar var mı? Ben de sorunsuz çalışıyor.

Amacım projelerinize SMS özelliğini katabilmeniz. Gerçekten harika fikirler geliyor insanın aklına, çok güzel projeler geliyor. Cep telefonunu insanlar her zaman taşıyorlar, onlara her zaman ulaşmak mümkün oluyor. Telefonların bu güzelliğinden yararlanarak projeler geliştirmek gerekiyor artık.

Kısa bir süreliğine açık kaynak kodlu olmayacak, yani upload edeceğim jarlarda source codelar olmayacak :(

Ben pcsuite bluetoothla baglanıyorum, cablo ile de olması lazım diye
düsünüyorum deneme imkanım olmadığı için bir şey diyemeyeceğim. büyük
ihtimalle olur diye düşünüyorum, usb de bir nevi seri port,
baglandiginiz usb nin comport numarasını uygulamanıza girebilirseniz,
apiyi çalıştırırsınız diye umud ediyorum.


Gereken donanım ve yazılımlar,
-pc suite 6.82 or newer version
-bluetooth module or usb cable

az daha sabretmek lazım...

sabreden derviş :

http://afsungur.googlepages.com/SMSApi.rar



30 Mart 2008 Pazar

Java Thread Örneği - Yükleniyor Göstergesi

Java ile uygulama geliştirirken bir çok kez file'dan okuma veya database'e baglanma islemlerini gerceklestiririz. Ozellikle bazı zamanlarda saniyeler suren bir gecikme meydana gelebilir bu tip okumaları yaparken. Ve de bu isi threadsiz yapiyorsak, malesef kullanici dostu olamamisiz demek ki...

Örneğin, bir butonumuz var ve bu buton veritabanına bağlanıp bize bir Connection nesnesini handle edecek. Thread kullanmaz isek butona bastığımızda buton kullanıcının veritabanına baglanma islemlerini gerceklestirme icerisinde oldugundan tepki veremeyecek ve kullanıcı programın kilitlendiğini zannedecektir. Böyle bir durumu önlemek için thread kullanılabilir.

Threadler çalışan proseslerin birer parçasıdırlar. Ingilizce anlamı ipliktir. Islemci bir anda sadece bir instruction işleyebilir. Thread ile instruction işlendiği anda disk okuması yapılabilir veya ekrana görüntü basılabilir. Çünkü bu işlemleri direk yapan cpu değildir. Biz de threadin bu özelliğinden yararlanacağız.

2 clasımız var. birinden diğer threadli clasımızı çağıracağız. Thread kullanılmayan classta thread kullanılan clası şu şekilde çağıracağız bizim örneğimiz için:

ff_1 =new FeedbackFrame_1(this, conn, sp, smsControl, smsReader, smsSender);

ff_1 bir FeedbackFrame_1 clasına ait nesne ve bu clasın contructoru görüldüğü gibi bir çok obje alıyo. Bunları siz
uygulamanızın durumuna göre değiştirebilirsiniz. Peki FeedbackFrame_1.java dosyası nasıl bir dosya, içeriği ne:

import java.lang.reflect.*;
import java.awt.*;
import java.awt.event.*;
import java.sql.Connection;
import javax.swing.*;
import javax.comm.*;

/**
*
* @author Ahmet Fuat SUNGUR
* Sms kontrolunu saglamakla yukumludur. Soyleki serveri baslatmadan onceki ayarlarin bir thread
* aracigiyla yapilip, kullanici arayuzunu dondurmamak icin kullanilmis bir class'tir.
*/
public class FeedbackFrame_1 extends JFrame implements Runnable {

private SmsFlow sf;
private Thread t;
private JLabel label;
private int state;
private SerialPort sp;
private ComSettings comInstance;
private SmsControl sc;
private SmsReader sr;
private SmsSender ss;
private Connection conn;


public FeedbackFrame_1(SmsFlow sf,Connection conn, SerialPort sp,SmsControl sc, SmsReader sr, SmsSender ss) {
this.sf = sf;
this.sc = sc;
this.sr = sr;
this.ss = ss;
this.conn = conn;
this.sp = sp;
setupFrame();
t = new Thread(this);
t.start();
pack();
show();
}

private void setupFrame() {
label = new JLabel();
label.setPreferredSize(new Dimension(200,50));

Container c = getContentPane();
JButton stopButton = new JButton("Stop");
stopButton.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent ae) {
error();
}
});
c.add(label, BorderLayout.NORTH);
c.add(stopButton, BorderLayout.SOUTH);
this.setTitle("Baglantilar deneniyor...");
}

private void setText(final String s) {
try {
SwingUtilities.invokeAndWait(new Runnable() {
public void run() {
label.setText(s);
}
});
} catch (InterruptedException ie) {
error();
} catch (InvocationTargetException ite) {
error();
}
}

private void error() {
t.interrupt();
if (SwingUtilities.isEventDispatchThread())
closeDown();
else SwingUtilities.invokeLater(new Runnable() {
public void run() {
closeDown();
}
});
}

private void closeDown() {
sf.setupCancelled();
hide();
dispose();
}
public SmsSender getSmsSender()
{
return this.ss;
}
public SmsReader getSmsReader()
{
return this.sr;
}
public SmsControl getSmsControl()
{
return this.sc;
}

public void run() {
// Simulate connecting to server

try{
/////
// Streamlari ayarliyoruz. DataOutput, DataInput, OUTPUT Stream...
setText("Stream aciliyorr...");
try
{
Thread.sleep(1000);
}
catch (InterruptedException ie) {}
sc.setStreams();


// Ve GSM modemimizle konusmaya basliyoruz.
// ilk mesaji gonderdik.
setText("İlk mesajlar gonderiliyor...");
try
{
Thread.sleep(1000);
}
catch (InterruptedException ie) {}
sc.send("AT");
sc.expect("OK");

setText("Karakter mod destegi...");
try
{
Thread.sleep(1000);
}
catch (InterruptedException ie) {}
// Mesajlarin pdu formatinda degil de text formatinda olmasini istiyoruz.
sc.send("AT+CMGF=1");
sc.expect("OK");

// Ve simdi sira geldi mesajlari okumaya ve gelen mesajlari parse edip
// gerekli cevabi kullaniciya gondermeye.
// Simdi GSM modemimizdeki mesajlari okurken 1 okuyup, okudugumuz
// mesaji irdeleyip, yorumlayip, gereken cevabi uretip, bu cevabi
// bize mesaj gonderen kullaniciya geri iletmemiz gerekiyor.
// Burada bir producer consumer problemi var. Producer Reader kismi,
// Consumer Sender kismi oluyor. Yani bir mesaj ilk once okunacak, bu mesaj
// ile ilgili islemler yapilmadan 2. mesaj okunmaya gecilmeyecek.
// Bunu asmak icin threadlarden yararlandik ve producer consumer
// problemini çözmek için SmsProducerConsumer clasini olusturduk.
// SmsReader ve SmsSender classlari constructorlarinda bu SmsProducerConsumer
// classina ait nesneyi alirlar ve bu sayede gerekli senkronizasyon
// saglanmis olur.
/*SmsFirstSettings sfs = new SmsFirstSettings(smsControl);
sfs.start();
smsControl = sfs.getSmsControl(); */
setText("SMSReader ve SMSSender Class'lari hazirlaniyor...");
try
{
Thread.sleep(1000);
}
catch (InterruptedException ie) {}
SmsProducerConsumer smsProducerConsumer = new SmsProducerConsumer();
sr = new SmsReader(sp,conn,sf,sc,smsProducerConsumer);
ss = new SmsSender(sp,conn,sf,sc,smsProducerConsumer);
if (Thread.currentThread().isInterrupted())
return;

SwingUtilities.invokeLater(new Runnable() {
public void run() {
sf.setupDone();
hide();
dispose();
}
});
}
catch(Exception e)
{}
}

public SerialPort getSerialPort()
{
return sp;
}
}


Burada önemli olan clasın Runnable interfacını implement etmesi, bunu unutmuyoruz.

ff_1 = new FeedbackFrame(...)
kodunu yazdığımızda yukarıdaki classın ilk run methodu çağırılıyor. Bu da bu husus açısından önemli...
Daha sonra yaptiğiniz baglantilari bir sekilde get etmeniz gerekiyor. o yüzden bu feedback clasına constructordan gonderdiginiz parametreleri get edebilecek metodlar eklemelisiniz...

iyi çalışmalar..

29 Mart 2008 Cumartesi

Jar Oluşturmak

aa klasöründeki Class dosyalarından jar oluşturmak için Komut satırından :

jar -cf yourjarfile.jar aa/

yazınız. Peki jar ne işe yarar :

- Classlarınızı jar haline getirip, yani tek dosya haline getirip, ileride api olarak bunu sunabilirsiniz.
- Classlarınıza tek dosya da ulaşmanızı sağlar.
- Mobil Cihazlara java dosyalarının aktarımını sağlar.



27 Mart 2008 Perşembe

FORALL Performance

FORALL
Örnek olarak bir tablodan çektiğimiz verilerle başka bir tabloyu doldurmamız gerekiyor. Yapacağımız iş nedir? Her select işlemi ile aldığımız veriyi insert ile tabloya eklemek veya bulk collect ile bir arraya veriyi atayıp daha sonra o array içerisinde FOR ile dönerek tabloya verileri ekleyebiliriz.
Bize bu konuda yardımcı olacak başka bir yapı var: FORALL. FORALL keywordu, Toplu DML Statementlerde performans artışı sağlayan çok önemli bir keyword. Somut bir örnekle FORALL’ın performansını gösterelim.
Step_1 : FORALL keywordlu procedure

create or replace procedure step_1 is
TYPE array_type_id IS TABLE OF TEST_1.ID%TYPE;
TYPE array_type_name IS TABLE OF TEST_1.NAME%TYPE;

iteration_time INTEGER := 1000;
varIdArr array_type_id := array_type_id();
varNameArr array_type_name := array_type_name();

BEGIN
SELECT ID,NAME BULK COLLECT INTO varIdArr,varNameArr FROM TEST_1;
FORALL i in varIdArr.first..varIdArr.last
INSERT INTO TEST_2 VALUES(varIdArr(i),varNameArr(i));
end;


Step_2: FOR keywordlu procedure

create or replace procedure step_2 is
TYPE array_type_id IS TABLE OF TEST_1.ID%TYPE;
TYPE array_type_name IS TABLE OF TEST_1.NAME%TYPE;

iteration_time INTEGER := 1000;
varIdArr array_type_id := array_type_id();
varNameArr array_type_name := array_type_name();

BEGIN
SELECT ID,NAME BULK COLLECT INTO varIdArr,varNameArr FROM TEST_1;
FOR i in varIdArr.first..varIdArr.last
LOOP
INSERT INTO TEST_2 VALUES(varIdArr(i),varNameArr(i));
END LOOP;
end;


Test Sonuçları:

SQL> exec runstats_pkg.rs_stop;

Run1 ran in 9 hsecs
Run2 ran in 125 hsecs
run 1 ran in 7,2% of the time

Name Run1 Run2 Diff
LATCH.dml lock allocation 1 0 -1
STAT...calls to get snapshot s 10 9 -1
LATCH.list of block allocation 0 1 1
STAT...IMU Flushes 1 0 -1
STAT...session cursor cache co 1 0 -1
LATCH.Consistent RBA 1 2 1
LATCH.lgwr LWN SCN 1 2 1
LATCH.mostly latch-free SCN 1 2 1
LATCH.OS process allocation 0 1 1
LATCH.enqueue hash chains 1 2 1
STAT...bytes received via SQL* 1,912 1,910 -2
STAT...messages sent 0 2 2
LATCH.library cache lock 2 4 2
LATCH.enqueues 0 2 2
STAT...enqueue requests 2 0 -2
LATCH.undo global data 24 28 4
STAT...consistent changes 14 19 5
LATCH.redo writing 3 8 5
STAT...cleanout - number of kt 22 28 6
STAT...active txn count during 22 28 6
STAT...calls to kcmgcs 22 28 6
STAT...consistent gets - exami 22 28 6
STAT...consistent gets 847 838 -9
STAT...consistent gets from ca 847 838 -9
LATCH.messages 2 14 12
LATCH.channel operations paren 0 14 14
LATCH.checkpoint queue latch 0 16 16
STAT...redo ordering marks 0 26 26
STAT...calls to kcmgas 0 26 26
STAT...workarea memory allocat -49 0 49
LATCH.SQL memory manager worka 6 73 67
STAT...change write time 3 78 75
STAT...free buffer requested 4 81 77
LATCH.cache buffers lru chain 4 81 77
STAT...recursive cpu usage 6 112 106
STAT...CPU used when call star 6 117 111
STAT...CPU used by this sessio 6 117 111
STAT...DB time 10 124 114
STAT...Elapsed Time 12 127 115
LATCH.object queue header oper 12 243 231
LATCH.simulator lru latch 56 877 821
LATCH.simulator hash latch 59 889 830
STAT...redo entries 87 10,036 9,949
LATCH.redo allocation 91 10,040 9,949
STAT...recursive calls 3 10,007 10,004
STAT...execute count 11 10,015 10,004
STAT...session logical reads 968 11,070 10,102
STAT...db block gets from cach 121 10,232 10,111
STAT...db block gets 121 10,232 10,111
STAT...db block changes 185 20,163 19,978
LATCH.library cache pin 60 20,068 20,008
LATCH.library cache 75 20,083 20,008
LATCH.cache buffers chains 2,238 52,343 50,105
STAT...undo change vector size 25,852 642,176 616,324
STAT...redo size 209,100 2,471,572 2,262,472

Run1 latches total versus runs -- difference and pct
Run1 Run2 Diff Pct
2,691 104,847 102,156 2.57%

PL/SQL procedure successfully completed

SQL>
Görüldüğü gibi FORALL’ı kullandığımız zaman 9hsec’te gerçekleşen iş, bu keywordu kullanmadığımız zaman 125hsec süre içersinde gerçekleşmiş. Ayrıca LATCH oranlarını karşılaştırdığımızda da 1/50 gibi bir oran karşımıza çıkıyor. Ve bu istatistikler bize FORALL’ın Toplu DML İşlemlerinde (INSERT,DELETE,UPDATE) ne kadar önemli bilgi sağladığını göstermiş oluyor.

BULK COLLECT Performance

BULK COLLECT
Bu denememizde BULK COLLECT adlı bir yapıyla tanışmış oluyoruz. Bu yapı bize sorgu sonuçlarını bir cursora değil de bir array’e aktarmamızı sağlıyor. Normalde bir tablodaki her satıra ait bir colonu yazdırmak istesek bunu şu şekilde yaparız:

FOR REC IN ( SELECT NAME FROM EMP)
LOOP
DBMS_OUTPUT….
END LOOP;


İşte, BULK COLLECT kullanarak bir cursor içerisinde değil de bir array içerisinde dönmüş oluyor. Ve bu PLSQL Kodunun performansını arttırıyor.
Denememizi şu şekilde yapacağız. 2 procedurumuz var. Step_1 BULK COLLECT ile veri çekiyor. Step_2 ise bildiğimiz şekilde.
Step_1 Proceduru :
create or replace procedure step_1 is
TYPE array_type_id IS TABLE OF TEST_1.ID%TYPE;
TYPE array_type_name IS TABLE OF TEST_1.NAME%TYPE;

varIdArr array_type_id;
varNameArr array_type_name;
BEGIN
SELECT id,name bulk collect into varIdArr,varNameArr from test_1 WHERE ID < 10000;

for i in varIdArr.first..varIdArr.last
Loop
dbms_output.put_line('>ID :' || varIdArr(i) || ' - NAME :' || varNameArr(i) );
end loop;
end;


Step_2 Proceduru:

create or replace procedure step_2 is
BEGIN
FOR REC IN ( SELECT * FROM TEST_1 WHERE ID < 10000)
LOOP
DBMS_OUTPUT.put_line('>ID :' || REC.ID || ' - NAME :' || REC.NAME);
END LOOP;
END;


Testimizi Başlatmak için:

exec runstats_pkg.rs_start;
exec step_1;
exec runstats_pkg.rs_middle;
exec step_2;
exec runstats_pkg.rs_stop;


Test Sonuçları :
SQL> exec runstats_pkg.rs_stop;

Run1 ran in 1048 hsecs
Run2 ran in 1139 hsecs
run 1 ran in 92,01% of the time

Name Run1 Run2 Diff
LATCH.mostly latch-free SCN 4 3 -1
STAT...session cursor cache co 1 0 -1
STAT...recursive cpu usage 3 4 1
LATCH.library cache lock 28 27 -1
LATCH.object queue header oper 4 5 1
LATCH.library cache lock alloc 1 0 -1
LATCH.undo global data 17 18 1
STAT...redo entries 11 12 1
LATCH.Shared B-Tree 0 1 1
LATCH.active checkpoint queue 4 3 -1
LATCH.kwqbsn:qsga 0 1 1
LATCH.cache buffers lru chain 0 1 1
LATCH.lgwr LWN SCN 4 3 -1
STAT...free buffer requested 0 1 1
LATCH.KMG MMAN ready and start 3 4 1
LATCH.enqueues 164 162 -2
LATCH.resmgr:resource group CP 2 0 -2
LATCH.library cache pin 26,809 26,811 2
LATCH.enqueue hash chains 166 164 -2
STAT...bytes received via SQL* 1,105,839 1,105,837 -2
STAT...db block changes 36 33 -3
LATCH.redo allocation 22 19 -3
LATCH.redo writing 15 12 -3
STAT...heap block compress 7 4 -3
LATCH.qmn task queue latch 0 4 4
LATCH.messages 72 68 -4
STAT...active txn count during 11 15 4
STAT...cleanout - number of kt 11 15 4
STAT...consistent gets - exami 11 15 4
STAT...calls to kcmgcs 11 15 4
LATCH.shared pool 50 55 5
STAT...consistent changes 25 20 -5
STAT...db block gets 37 44 7
STAT...db block gets from cach 37 44 7
LATCH.channel operations paren 56 48 -8
STAT...parse time cpu 14 1 -13
STAT...CPU used by this sessio 60 47 -13
STAT...CPU used when call star 63 47 -16
LATCH.checkpoint queue latch 72 54 -18
STAT...parse time elapsed 30 2 -28
STAT...DB time 87 56 -31
STAT...workarea memory allocat 31 -2 -33
LATCH.SQL memory manager worka 274 207 -67
STAT...undo change vector size 2,180 2,248 68
STAT...Elapsed Time 1,056 1,140 84
LATCH.shared pool simulator 107 15 -92
LATCH.simulator lru latch 54 232 178
LATCH.simulator hash latch 54 232 178
STAT...redo size 2,972 3,216 244
STAT...no work - consistent re 777 4,109 3,332
STAT...table scan blocks gotte 777 4,109 3,332
STAT...consistent gets 800 4,136 3,336
STAT...consistent gets from ca 800 4,136 3,336
STAT...recursive calls 6 3,344 3,338
STAT...session logical reads 837 4,180 3,343
LATCH.cache buffers chains 1,778 8,426 6,648
STAT...session uga memory 196,392 0 -196,392
STAT...session pga memory 458,752 0 -458,752
STAT...table scan rows gotten 497,103 2,246,371 1,749,268

Run1 latches total versus runs -- difference and pct
Run1 Run2 Diff Pct
80,153 86,968 6,815 92.16%

PL/SQL procedure successfully completed

SQL>


Bu Testte asıl dikkat etmemiz gereken birkaç nokta var. Bunlardan en önemlisi Tom Kyte’a göre Total Latches olayı. Görüldüğü gibi 2. Procedurumuzda bu daha fazla. Yani daha fazla gecikme meydana gelmiş. Diğer bakılabilecek önemli nokta ise elapsed time, yine aynı şekilde 1. Procedure 2.procedur’a göre daha performanslı elapsed time açısından.

Java SMS Api

Okuldaki projemde kullandığım SMS applicationu bir Api haline getirip,açık kaynak kodlu olarak geliştirilmesini sağlamak amacıyla yakında bir api çıkarmayı düşünüyorum. Bu api ile uygulama geliştiriciler kendi pclerine bağladıkları cep telefonundan Java kodlarıyla sms gönderip alabilecek. Bu şekilde projelerinize SMS özelliğini bu şekilde ekleyebileceksiniz.

Şimdilik Nokia S40 destekleyecek gibi, ama daha sonra tüm cihazlar için geliştirmeyi düşünüyorum.

Edit : Şu anda SMS Gönderme işlemi tamam, okuma işlemi var. Bu biraz uzun süreceğe benziyor. Ama beklemeye değer...

sabreden derviş :

http://afsungur.googlepages.com/SMSApi.rar

26 Mart 2008 Çarşamba

Fantastik Forvet Gitti

Kendini GittiGidiyor.com'da Fantastik Forvet başlığıyla açık artırmaya çıkaran amatör küme oyuncusu
Engin Kodan, 6.100 YTL'ye yeni kulübünü buldu.




GittiGidiyor, 'Bilkentli'nin Donu'ndan sonra en ilginç açık artırmalardan birine sahne oldu. İstanbul amatör küme takımlarından İmesspor'da oynayan Engin Kodan, bir 3. lig takımında oynamak için kendini GittiGidiyor.com'da açık artırmaya çıkardı.

Fantastik Forvet
başlıklı artırmaya ilgi büyüktü. 21.755 tekil kullanıcı tarafından
izlenen artırmaya 4 kulüpten teklif geldi. 5.000 YTL ile açılan
artırma, İstanbul bölgesinden bir kulübün 6.100 YTL bedelle Fantastik Forvet'i almasıyla son buldu.

20 Mart 2008 Perşembe

ScribeFire

ScribeFire, Firefox'a eklenebilecek çok kullanışlı olduğunu düşündüğüm bir add-on. Bu araç ile blogspot wordpress gibi bloglarınıza tek tıklama ile yazı ekleyebiliyorsunuz. Veya firefox'ta açtığınız herhangi bir sayfayı bloglayabiliyorsunuz...

Ve daha nicesi...

http://www.scribefire.com/getting-started/


17 Mart 2008 Pazartesi

18 Mart Çanakkele Zaferi

Canlarını, vatan uğruna gözünü kırpmadan verebilen şehitlerimizi ve vatanı için kolunu,bacağını,ayağını feda etmiş nice gazilerimizi saygıyla anıyoruz.

http://tr.wikipedia.org/wiki/%C3%87anakkale_Sava%C5%9Flar%C4%B1

25 Şubat 2008 Pazartesi

BAT dosyası yazımı

Bu yazımda tek klikle birden cok dos komutu calismasini saglayan bat dosyalarinin olusturulmasindan bahsedecegim.
Birden cok islemi, gunun veya ayin belli zamanlarinda sık sık tekrar mi ediyorsunuz?
Bu sizin icin cok mu zaman alıyor?
Autorun.inf viruslerinde biktiniz mi?
size bat dosyasini oneriorum, bazi durumlarda bu viruslerden kurtulmanizi da sagliyor.

İlk olarak dos komutlarina ne kadar asinasiniz onu ogrenmek gerek:

http://en.wikipedia.org/wiki/List_of_DOS_commands
http://www.easydos.com/dosindex.html
http://www.computerhope.com/dosapp.htm

biraz komutlari ogrendikten sonra ornek bir uygulama yapalim.
!!dosya uzantilarini goruntule secenegimizin klasör seçeneklerinden seçili olmasi gerekiyor ( ne cumle oldu be! )

daha sonra, yeni metin belgesi olusturuyoruz, a.bat diye. Buna sag tiklayip duzenle diyoruz.
Daha sonra C:\aaa klasörümüzü D:\bbb klasörüne kopyalayacak bat dosyamizi yaziyoruz:

a.bat dosyasinin içeriği:
---------------
@echo off
echo Dosya kopyalama islemi baslayacak...
pause
c:
xcopy /E/Y aaa D:\bbb
echo Dosya kopyalam islemi tamamlandi...
pause
----------------
pause : devam etmek icin bir tusa basiniz. yazisini ekrana basip, sizden herhangi bir tusa basmanizi istiyor.
xcopy , klasörü komple kopyalamayı sagliyor
yardim icin dosta help xcopy yaZin.

...

tüm disklerime tek tıkla defrag attirmak icin sunu yazdim:
defrag_all_disks.bat:
---------------------
@echo off
echo Sistem Bakımı
defrag C: -f -v
defrag D: -f -v
defrag E: -f -v
defrag F: -f -v
defrag G: -f -v
defrag H: -f -v
pause
-------------------
artık tek tıkla tüm disklerime defrag(disk birleştiricisi) yaptırıyorum

autorun.inf virusleriyle de epey ugrasmıstım. bazen cozum elde edebildim ama yine de virusler gercek yuzlerini gosterdiler. biraz da ugrasılsa tam olarak cozum elde edilebilir.
attrib ile viruslerin korumasini kaldirip, virusun bulundugu klasore kendi urettiginiz file ı kopyleyin, daha sonra attrib +s +h +r ile kendi file'ınıza yani kendi autorun.inf inize gorunmezlik eklemeye calisin. cok zeki olmayan virusleri bu yolla sisteminizden def edebilrisiniz..

sonuc : bat dosyalari cok kullanisli, birden cok isi (batch) tek hamlede yapmaniza olanak sagliyor eger iyi dos bilginiz varsa

20 Şubat 2008 Çarşamba

Oracle Date Functions

Projemiz için lazım oldu, ben de yeni öğrendim bir çoğunu :

SELECT TO_CHAR(SYSDATE,'DD-MON-YYYY') FROM DUAL;
20-ŞUB-2008

SELECT TO_CHAR(SYSDATE,'DD-MON-YEAR') FROM DUAL;
20-ŞUB-TWO THOUSAND EIGHT

SELECT TO_CHAR(SYSDATE,'DD-MON-YYYY DAY') FROM DUAL;
20-ŞUB-2008 ÇARŞAMBA

SELECT TO_CHAR(SYSDATE,'DD-MM-YYYY DAY') FROM DUAL;
20-02-2008 ÇARŞAMBA

-------------------

Ayın ilk günü:

SELECT TO_CHAR(TRUNC(SYSDATE,'MONTH'),'DAY') FROM DUAL;
CUMA

Ayın Son günü:

SELECT TO_CHAR(LAST_DAY(SYSDATE),'DAY') FROM DUAL;
CUMA

En yakın Cuma gününün ayın hangi günü olduğu:
SELECT TO_CHAR(NEXT_DAY(SYSDATE,'CUMA'),'DD') FROM DUAL;

İki tarih arasındaki ay farkı
SELECT ROUND(MONTHS_BETWEEN(TO_DATE('31-08-2008','DD-MM-YYYY'),TO_DATE('25-05-2008','DD-MM-YYYY'))) FROM DUAL;

( yukarıda roundu kaldırırsak sonucu tam sayi olarak degil de kusurlu bir sayi olarak buluruz )

simdilik bu kadar...

17 Şubat 2008 Pazar

Bean

Java'da

Kullanmak istediğimiz class'ı jsp dosyalarımızın bulunduğu ana klasörden WEB-INF/classes klasörüne atmamız gerekiyor. Eğer böyle bir klasör yoksa, bu klasörü siz oluşturun. Mesela:

E:\Program Files\Apache Software Foundation\Tomcat 6.0\webapps\fuat\WEB-INF\classes

Daha sonra package'imizin adini alan klasörü de üstteki klasör içinde oluşturalım. Örnek IETT olsun.

E:\Program Files\Apache Software Foundation\Tomcat 6.0\webapps\fuat\WEB-INF\classes\iett\

daha sonra, bus.class dosyamızı üstteki klasöre atalım. 
Şimdi sıra geldi bu class dosyasını jsp dosyamıza include etmeye.
JSP dosyamızı açıp,

< jsp:usebean id="db" class="iett.bus">
<% db.setId(5); db.setPlate("ahmet"); out.print(db.getId()); out.print(db.getPlate()); %> 

yazdık mı, b.jsp olarak kaydedip çalıştırdık mı, bizden mutlusu yoktur.
( ha, tabi class olarak kaydettik ya o classtaki üstte kullandığımız setId, setPlate, getInfo metodları olacak tabi ! )



JSP'ye Giriş

Java Server Pages, Java dilinin web'e uyarlanmış versiyonudur. Servlet'lerden farklı olarak JSP'ler HTML ile iç içedirler. Sayfanın bir kısmında HTML, bir kısmında JSP kullanabilirsiniz.

İlk olarak Tomcat6'yı sistemimize kuruyoruz :

http://godel.cs.bilgi.edu.tr/apache/tomcat/tomcat-6/v6.0.16/bin/apache-tomcat-6.0.16.zip

Sonrasında, Tomcati kurduğumuz dizinde bin/ klasöründe tomcat6.exe yi çift tıklayarak serverımızı çalıştırıyoruz.
"E:\Program Files\Apache Software Foundation\Tomcat 6.0\bin"

daha sonra 
E:\Program Files\Apache Software Foundation\Tomcat 6.0\webapps\
klasörüne "fuat" adında bir klasör olusturuyoruz.

Bu klasör içerisine a.jsp adlı jsp dosyamızı kaydediyoruz.

a.jsp
<% out.println("AFS"); %>

E:\Program Files\Apache Software Foundation\Tomcat 6.0\webapps\fuat\a.jsp olarak kaydettik.

Browserımızı açtık ve şunu girdik

http://localhost:8080/fuat/a.jsp

ve jsp dosyamızı bu şekilde çalıştırmış olduk.
Eğer port değişikliği yapmak istiyorsanız, 
E:\Program Files\Apache Software Foundation\Tomcat 6.0\conf\server.xml adresinde 8080 olan ilgili port kisimlarini 8089,8091 gibi değerler ile değiştirebilirsiniz.

16 Şubat 2008 Cumartesi

IETT Araç Takip ve Yolcu Bilgilendirme Sistemi

Otobüs beklemekten çok sıkıldınız veya önünüzden bir taksi geçiyor "ya zaten binmemize gerek yok, şimdi otobüs gelir" mi diyorsunuz?

Artık bunları projemizle dert etmenize gerek kalmayacak:

IETT Araç Takip ve Yolcu Bilgilendirme Sistemi

Beklediğiniz hatta ait otobüsün, beklediğiniz durağa yaklaşık olarak kaç dakika sonra geleceğini hem web hem mobile uygulamayla artık öğrenebileceksiniz.

Çok yakında, sizlerle...

Hello World

Bir bilgisayar mühendisi adayı olarak, böyle başlamazsak mesleğimize ihanet etmiş oluruz :

BEGIN
               DBMS_OUTPUT.PUT_LINE('HELLO WORLD');
END;