Sinapse プラットフォームで利用可能な現場で実証済みの DLR プラグイン上に構築された JCOBridge は、JVM と CLR の世界の統合において最高のパフォーマンスを保証します。
- CLRタイプの取得
- CLRオブジェクトをインスタンス化する
- 静的メソッドを呼び出す
- インスタンスメソッドを呼び出す
- 静的プロパティを取得/設定する
- インスタンスのプロパティを取得/設定する
- デリゲートの設定
- イベントの購読/購読解除
- WPF コントロールを AWT/Swing ウィンドウに統合します
- WinForms コントロールを AWT/Swing ウィンドウに統合する
- 複雑な .NET グラフィカル ユーザー インターフェイス オブジェクトを AWT/Swing ウィンドウに統合
- ユーザー インターフェイス コントロール、プロパティ、およびイベントの管理
- JVMクラスの取得
- JVMオブジェクトをインスタンス化する
- 静的メソッドを呼び出す
- インスタンスメソッドを呼び出す
- 静的フィールドを取得/設定する
- インスタンスフィールドの取得/設定
- 動的アクセスを使用して、Java 言語で行う場合と同様にシームレスな方法でコードを記述します。
- 特定のインターフェイスを使用してメソッドとフィールドを直接管理します
JCOBridge (JVM-CLR Object Bridge) を使用すると、CLR/.NET 言語から Java や scala などの JVM ネイティブ言語を実行したり、その逆を行うことができ、ライブラリやコンポーネントをインポートして使用したり、グラフィカル ユーザー インターフェイスを 1 つから管理したりすることができます。プログラミングの世界をもう一つの世界へ。詳細については、www.jcobridge.com をご覧ください。
例を調べるには、次の手順を実行する必要があります。
このリポジトリでは、JCObridge がサポートするさまざまなプログラミング言語のサンプル コードを見つけることができます。サンプルは、関連プロジェクトを含む JVM と CLR という 2 つのメイン フォルダーに編成されています。外部コードの実行時コンパイルは行われず、実行のみが行われるため、コードを実行する前に両方の世界の例をコンパイルする必要があります。
プロジェクトのクロス プラットフォーム GUI では、AWT を使用して Windows および Linux ホスト上で .NET Core のクロスプラットフォーム グラフィカル ユーザー インターフェイスを作成する方法を示します。 Swing を使用するには、コード内のコントロールを好みのコントロールに変更するだけです。
これは、.NET アプリケーションから JavaClass.java で定義された単純なクラスを呼び出す基本的な例です。 /JVM/java/src/JavaClass.java には単純なクラスがあります。
public class JavaClass {
/**
* This simple method return the "Hello World!!" string
* * @return "Hello World!!" string
*/
public String helloWorld ()
{
return "Hello World from Java!!" ;
}
/**
* This simple method return the sum of two double
* @param a
* @param b
* @return a + b
*/
public double add ( double a , double b )
{
return a + b ;
}
/**
* This simple method return the sin of a double
* @param a
* @return sin of a
*/
public double sin ( double a )
{
return Math . sin ( a );
}
}
CLRJavaClassUseExampleprogram.cs には、単純な .NET C# アプリケーションがあります。
using MASES . LicenseManager . Common ;
using MASES . JCBridge . C2JBridge ;
using System ;
namespace JavaClassUseExample
{
class Program
{
static void Main ( string [ ] args )
{
new TestClass ( ) . Execute ( ) ;
}
class TestClass : SetupJVMWrapper
{
public override string ClassPath { get { return @"....JVMOutput" ; } }
public void Execute ( )
{
double a = 2 ;
double b = 3 ;
double c = Math . PI / 2 ;
var javaClass = DynJVM . JavaClass . @new ( ) ;
string hello = javaClass . helloWorld ( ) ;
double result = javaClass . add ( a , b ) ;
double sin = javaClass . sin ( c ) ;
Console . WriteLine ( "{0} {1} + {2} = {3} and sin({4:0.0000000}) = {5:0.00000000}" , hello , a , b , result , c , sin ) ;
}
}
}
}
コードを実行すると、次の出力が得られます。
Hello World from Java!! 2 + 3 = 5 and sin(3,1415927) = 1,00000000
この例は、環境パラメーターが .NET TestClassクラスで構成されているJava クラス使用例の拡張です。
class TestClass : SetupJVMWrapper
{
// the following line setup the classpath where JVM will search for classes
// during runtime it is possible to dynamically add other path using a call like DynJVM.JVMHelper.addPath(<the path to add>);
public override string ClassPath { get { return @"C:Program FilesMASES GroupJCOBCore;....JVMJavaOutput" ; } }
// uncomment the following line and set the correct JRE if the automatic search system fails
// public override string JVMPath { get { return @"C:Program FilesJavajre1.8.0_121binserverjvm.dll"; } }
// the following code adds all possible switch to the starting JVM.
// for a complete list see Oracle documentation: https://docs.oracle.com/javase/8/docs/technotes/tools/windows/java.html
public override IEnumerable < KeyValuePair < string , string > > JVMOptions
{
get
{
var dict = new Dictionary < string , string > ( ) ;
dict . Add ( "-Xmx128M" , null ) ; // this line adds a complete argument
// dict.Add(property, value); // this line adds an argument like -Dproperty = value
return dict ;
}
}
// the following code adds initial packages to the import statement.
public override IEnumerable < string > JVMPackages
{
get
{
var list = new List < string > ( ) ;
list . Add ( "java.lang" ) ; // this line adds java.lang.* like you do with "import java.lang.*" in Java
return list ;
}
}
// uncomment and set the following line when you need features of JDK like the use of the compiler
// public override string JDKHome { get { return @"C:Program FilesJavajdk1.8.0_121"; } }
public void Execute ( )
{
double a = 2 ;
double b = 3 ;
double c = Math . PI / 2 ;
var javaClass = DynJVM . JavaClass . @new ( ) ;
string hello = javaClass . helloWorld ( ) ;
double result = javaClass . add ( a , b ) ;
double sin = javaClass . sin ( c ) ;
Console . WriteLine ( "{0} {1} + {2} = {3} and sin({4:0.0000000}) = {5:0.00000000}" , hello , a , b , result , c , sin ) ;
Console . WriteLine ( "Press Enter to exit" ) ;
Console . ReadLine ( ) ;
}
}
これは、次の操作を実行する JCOBridge の機能を調べる、より複雑なアプリケーションです。 - 動的 JVM ラッパーを使用して .NET 環境で Java コードを実行 - 共有オブジェクトの管理 - Java 仮想マシン側での CLR オブジェクトの登録 - 登録されたオブジェクトの使用JVM 側から - JVM 登録クラスのメソッドを呼び出し、操作を CLR オブジェクトに反映させる - JVM 側でダイアログを作成し、.NET コードから使用する
このライブラリには、JVM から呼び出される double および string 操作を提供する単一のクラスが含まれています。
Java グラフィカル アプリケーションでのユーザー インターフェイスの統合を示すために使用される完全なロジックを備えた Windows フォーム パネル。
Java グラフィカル アプリケーションでのユーザー インターフェイスの統合を示すために使用される完全なロジックを備えた WPF パネル。
これは、CSharpClass.cs で定義された単純なクラスを Java アプリケーションから呼び出す基本的な例です。 CLRCSharpClassCSharpClass.cs には単純なクラスがあります。
using System ;
namespace MASES . CLRTests
{
public class CSharpClass
{
/// <summary>The method <c>HelloWorld</c> return the "Hello World!!" string</summary>
public String HelloWorld ( )
{
return "Hello World from C#!!" ;
}
/// <summary>The method <c>Add</c> return the sum of two double</summary>
public double Add ( double a , double b )
{
return a + b ;
}
/// <summary>The method <c>Add</c> return the sin of a double</summary>
public double Sin ( double a )
{
return Math . Sin ( a ) ;
}
}
}
/JVM/src/JavaClass.java には、単純な Java アプリケーションがあります。
import java . io . IOException ;
import org . mases . jcobridge .*;
public class CSharpClassUseExample {
public static void main ( String [] args ) {
try {
try {
try {
JCOBridge . Initialize ( "" );
} catch ( JCException e ) {
e . printStackTrace ();
}
} catch ( IOException e ) {
e . printStackTrace ();
}
//declare and create JCOBridge instance
JCOBridge bridge ;
bridge = JCOBridge . CreateNew ();
// adds the path where extarnal assemblies where found
bridge . AddPath ( "../CLR/Output/" );
// add REFERENCES to the .dll file
bridge . AddReference ( "CSharpClass" );
// GENERATE Object
JCObject CSharpObject = ( JCObject ) bridge . NewObject ( "MASES.CLRTests.CSharpClass" );
double a = 2 ;
double b = 3 ;
double c = Math . PI / 2 ;
//Invoke the C# class methods
String hello = ( String ) CSharpObject . Invoke ( "HelloWorld" );
double result = ( double ) CSharpObject . Invoke ( "Add" , a , b );
double sin = ( double ) CSharpObject . Invoke ( "Sin" , c );
System . out . println ( String . format ( "%s %.0f + %.0f = %.0f and sin(%.8f) = %.8f" , hello , a , b , result , c , sin ));
} catch ( JCException jce ) {
jce . printStackTrace ();
System . out . println ( "Exiting" );
return ;
}
}
}
コードを実行すると、次の出力が得られます。
Hello World from C#!! 2 + 3 = 5 and sin(3,14159265) = 1,00000000
この少し複雑な例では、2 つの .NET ライブラリから取得した 2 つの異なる複雑なコントロールを awt Java ユーザー インターフェイスに統合します。最初のコントロールは Windows フォームで、2 番目のコントロールは WPF オブジェクトです。 JVMJavasrcAWTWinFormsWPF.java のアプリケーションは、コントロールの参照と生成から、.NET イベント リスナーの登録、.NET イベント コールバック管理に至るまでの完全なプロセスを公開します。
import java . awt . Frame ;
import java . io . IOException ;
import org . mases . jcobridge .*;
public class AWTWinFormsWPF implements IJCVoidEventEmit {
public static void main ( String args []) {
new AWTWinFormsWPF (). createAndShow ();
}
int cycle = 0 ;
java . awt . TextArea gTextArea ;
// WPF
JCControl gControlWpfControl = null ;
// FORMS
JCControl gControlFormsControl = null ;
void createAndShow () {
try {
// LOGGER
IJCEventLog logger = null ;
try {
try {
JCOBridge . Initialize ( "" );
} catch ( JCException e ) {
e . printStackTrace ();
}
logger = new JCFileEventLog ( "WinFormsWPF.txt" );
} catch ( IOException e ) {
e . printStackTrace ();
}
JCOBridge bridge ;
bridge = JCOBridge . CreateNew ();
bridge . RegisterEventLog ( logger );
// adds the path where extarnal assemblies where found
bridge . AddPath ( "../../CLR/Output/" );
// add REFERENCES
bridge . AddReference ( "WPFTestControl" );
bridge . AddReference ( "WinFormsTestControl" );
// GENERATE CONTROLS
gControlWpfControl = bridge . GetControl ( "MASES.CLRTests.WPFTestControl.TestControl" );
gControlFormsControl = bridge . GetControl ( "MASES.CLRTests.WinFormsTestControl.TestControl" );
// CONFIGURE CONTROLS
gControlWpfControl . RegisterEventListener ( "FromComboBox" , this );
gControlWpfControl . RegisterEventListener ( "FromTextBox" , this );
gControlFormsControl . RegisterEventListener ( "FromComboBox" , this );
gControlFormsControl . RegisterEventListener ( "FromTextBox" , this );
Frame dialog = new Frame ();
gTextArea = new java . awt . TextArea ();
gTextArea . setText ( "This is an AWT TextArea" );
java . awt . GridLayout layout = new java . awt . GridLayout ( 2 , 2 );
dialog . setLayout ( layout );
dialog . add ( gControlWpfControl );
dialog . add ( gControlFormsControl );
dialog . add ( gTextArea );
dialog . validate ();
dialog . setTitle ( "WinForms-WPF AWT integration" );
dialog . setVisible ( true );
dialog . setSize ( 200 , 200 );
} catch ( JCException jce ) {
jce . printStackTrace ();
System . console (). readLine ( "Please press enter" );
System . out . println ( "Exiting" );
return ;
}
}
@ Override
public void EventRaised ( Object ... args ) {
System . out . println ( "EventRaised" );
if ( args [ 1 ] instanceof JCObject ) {
JCObject obj = ( JCObject ) args [ 1 ];
System . out . println ();
try {
if ( obj != null ) {
gTextArea . setText ( "Text area: event: " + obj . toString () + " Content: " + obj . Get ( "Content" ));
}
} catch ( JCException e ) {
e . printStackTrace ();
}
}
}
}
.NET CLR から呼び出される double および string 操作を提供する単一のクラス。
2 つのメソッドを含み、共有グローバル変数とオブジェクトの登録方法と使用方法を表示するクラス
import org . mases . jcobridge .*;
import java . awt .*;
public class GlobalVariableTest
{
public static void createGlobal () throws JCException
{
Dialog dialog = new Dialog (( Dialog ) null );
JCOBridge . RegisterJVMGlobal ( "SharedDialog" , dialog );
}
public static void testMyCLRClass ( Integer a , Integer b ) throws JCException
{
JCObject resultGetCLRObject = ( JCObject ) JCOBridge . GetCLRGlobal ( "MyCLRClass" );
resultGetCLRObject . Invoke ( "Add" , a , b );
}
}
createGlobal メソッドは、グローバル awt ダイアログを作成し、CLR 側からシームレスに使用できるように登録します。 testMyCLRClass は、登録された CLR グローバル オブジェクトの使用方法を示します。CLR の例では、.NET 側からこのオブジェクトを作成し、この関数を呼び出してそれを使用し、ホット操作が JVM と CLR の間で透過的に反映されることを示します。
単純な Scala クラスは、JVMScalascalaclasssrcmainscalaScalaClass.class の CLR から使用されるように定義されています。 JVMScala でコンパイルとバッチ スクリプトを呼び出して実行します。
バッチ スクリプトをコンパイルして実行する前に、Scala バイナリをインストールする必要があります。
import java . lang . _
final class ScalaClass ( aString : String , val anInteger : Int ) {
def this () {
this ( "defaultString" , - 1 )
}
def this ( aBool : Boolean ) {
this ( "defaultString" , - 1 )
}
val scalaString = "This is a Scala String"
def add ( x : Int , y : Int ): Int = x + y
def stringConcat ( args : Array [ String ]): String =
{
return args . mkString ( ", " )
}
}
CLRScalaClassUseExampleProgram.cs には、定義された ScalaClass を使用する単純なアプリケーションがあります。
using CommonTest ;
using MASES . JCOBridge . C2JBridge ;
using MASES . LicenseManager . Common ;
using System ;
using System . IO ;
namespace ScalaClassUseExample
{
class TestClass : BaseTestClass
{
public override string GetProjectClassPath ( )
{
#if ! JCOBRIDGE_CORE
return @"....JVMScalaOutput" ;
#else
return @"......JVMScalaOutput" ;
#endif
}
public override string ClassPath
{
get
{
return new ClassPathBuilder ( GetProjectClassPath ( ) + @"*" , @"C:Program Files (x86)scalalib*.jar" ) . Build ( ) ;
}
}
public override void Execute ( )
{
int a = 10 ;
int b = 15 ;
var scalaClass = DynJVM . ScalaClass . @new ( ) ;
var result = scalaClass . add ( a , b ) ;
Console . WriteLine ( "{0} + {1} = {2}" , a , b , result ) ;
string [ ] strings = new string [ ] { "One" , "Two" , "Three" } ;
var concatString = scalaClass . stringConcat ( strings ) ;
Console . WriteLine ( "{0} = {1}" , string . Concat ( strings ) , concatString ) ;
Console . WriteLine ( "Press Enter to exit" ) ;
Console . ReadLine ( ) ;
}
}
class Program
{
static void Main ( string [ ] args )
{
try
{
new TestClass ( ) . Execute ( ) ;
}
catch ( Exception e )
{
Console . WriteLine ( e . Message ) ;
Console . WriteLine ( "Press any key." ) ;
Console . ReadKey ( ) ;
}
}
}
}
Scala では、必要なライブラリはすべてベース パスに明示的に追加されます。
この例では、Scala 言語から JCOBridge 経由で .NET オブジェクトを呼び出します。バッチ スクリプトをコンパイルして実行する前に、Scala バイナリをインストールする必要があります。
import java . util . Iterator
import org . mases . jcobridge . _
object Main extends App {
try
{
JCOBridge . Initialize ();
}
catch
{
// catch to avoid problem with Trial mode of JCOBridge
case jce : JCException => System . out . println ( jce . getMessage )
}
val bridge = JCOBridge . CreateNew ()
// adds a new reference to WPF
bridge . AddReference ( "PresentationFramework" )
// get MessageBox type
val msgType = bridge . GetType ( "System.Windows.MessageBox" )
// invoke static method to show a message box on the screen
msgType . Invoke ( "Show" , "Please press enter to continue" )
// get .NET type
val enumType = bridge . GetType ( "System.Environment" )
// invokes static method
val genObj = enumType . Invoke ( "GetLogicalDrives" )
// retrieve the iterator
val iteratorObj = genObj . asInstanceOf [ JCObject ]. iterator
// iterate on all object and print the value
while ( iteratorObj . hasNext ) println ( iteratorObj . next )
// invoke static method to show a message box on the screen
msgType . Invoke ( "Show" , "Please press enter" )
// event callback example
val tObj = bridge . NewObject ( "System.Timers.Timer" ); // create the timer object
val timerObj = tObj . asInstanceOf [ JCObject ];
// register an event handler when the Timer elaps
timerObj . RegisterEventListener ( "Elapsed" , new ScalaJCVoidEventEmit ());
// set Interval property
timerObj . Set ( "Interval" , 1000 ); // set properties
// enable the Timer
timerObj . Set ( "Enabled" , true ); // start timer
// invoke static method to show a message box on the screen
msgType . Invoke ( "Show" , "Please press enter" )
}
final class ScalaJCVoidEventEmit () extends JCVoidEventEmit {
override def EventRaised ( args : Object *) : Unit =
{
// scala seems to have a problem to translate var args argument into JVM bytecode. This method is needed to avoid compilation problems
}
// this method defines exactly the signature expected from the event
def EventRaised ( sender : Object , arg : Object ) : Unit =
{
println ( "Timer Elapsed" )
}
}